|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
"log"
|
|
|
|
"context"
|
|
|
|
"sync"
|
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
"google.golang.org/grpc/credentials/insecure"
|
|
|
|
pb "FRMS/internal/pkg/grpc"
|
|
|
|
)
|
|
|
|
|
|
|
|
// this package will implement a reactor coordinator and associated go routines
|
|
|
|
|
|
|
|
type ReactorManager struct {
|
|
|
|
*Manager
|
|
|
|
Devs *Devices
|
|
|
|
*System
|
|
|
|
}
|
|
|
|
|
|
|
|
type Devices struct {
|
|
|
|
mu sync.Mutex
|
|
|
|
D map[int]Device
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewReactorManager(c *Client,sys *System,err chan error) GeneralManager {
|
|
|
|
d := new(Devices)
|
|
|
|
r := &ReactorManager{Devs:d}
|
|
|
|
r.Manager = NewManager(err)
|
|
|
|
r.System = sys
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *ReactorManager) Start(cl *Client) {
|
|
|
|
r.Manager.Start(cl)
|
|
|
|
conn := r.Connect()
|
|
|
|
empty := &grpc.ClientConn{}
|
|
|
|
if conn != empty {
|
|
|
|
go r.UpdateReactor(r.Id,true)
|
|
|
|
go r.Monitor(conn)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *ReactorManager) Exit() {
|
|
|
|
r.Manager.Exit()
|
|
|
|
go r.UpdateReactor(r.Id,false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *ReactorManager) GetPort() int {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *ReactorManager) Connect() *grpc.ClientConn {
|
|
|
|
// establish gRPC conection with reactor
|
|
|
|
var opts []grpc.DialOption
|
|
|
|
var conn *grpc.ClientConn
|
|
|
|
opts = append(opts,grpc.WithTransportCredentials(insecure.NewCredentials()))
|
|
|
|
|
|
|
|
for {
|
|
|
|
if !r.IsActive() {
|
|
|
|
log.Fatal("No longer active, aborting connection attempt\n")
|
|
|
|
return &grpc.ClientConn{}
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
conn, err = grpc.Dial(fmt.Sprintf("%v:%v",r.Ip,r.Port),opts...)
|
|
|
|
// error handling
|
|
|
|
code := status.Code(err)
|
|
|
|
if code != 0 { // != OK
|
|
|
|
if code == (5 | 14) { // unavailable or not found
|
|
|
|
to := r.Timeout()
|
|
|
|
if to == 0 {
|
|
|
|
log.Printf("Client not responding\n")
|
|
|
|
return &grpc.ClientConn{}
|
|
|
|
}
|
|
|
|
log.Printf("Client currently down, retrying in %v ms\n",to)
|
|
|
|
time.Sleep(time.Duration(to) * time.Millisecond)
|
|
|
|
|
|
|
|
} else {
|
|
|
|
log.Fatal("GRPC ERROR: %v",code)
|
|
|
|
r.Err <- err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return conn
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *ReactorManager) Monitor(conn *grpc.ClientConn) {
|
|
|
|
defer conn.Close()
|
|
|
|
client := pb.NewMonitoringClient(conn)
|
|
|
|
for r.IsActive() {
|
|
|
|
req := &pb.ReactorStatusRequest{Id:r.Id}
|
|
|
|
resp, err := client.GetReactorStatus(context.Background(),req)
|
|
|
|
code := status.Code(err)
|
|
|
|
if code != 0 { // if != OK
|
|
|
|
fmt.Printf("Reactor %v down! Code: %v\n", r.Id,code)
|
|
|
|
r.Exit()
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
for _,v := range resp.GetDevices() {
|
|
|
|
go r.System.UpdateReactorDevice(r.Id, int(v.GetAddr()), v.GetType(), v.GetStatus(), v.GetData())
|
|
|
|
}
|
|
|
|
time.Sleep(r.Hb) // time between sensor pings
|
|
|
|
}
|
|
|
|
}
|