went back to monitoring via pings; starting terminal interface over the weekend
parent
530413e9a1
commit
37ad511131
Binary file not shown.
Binary file not shown.
@ -0,0 +1,25 @@
|
|||||||
|
general outline
|
||||||
|
|
||||||
|
want:
|
||||||
|
- reserved ips
|
||||||
|
- only really matters for server itself
|
||||||
|
- convienent for the devices but not extremly neccesarry
|
||||||
|
- seems to be that now the local network addresses stay the same
|
||||||
|
- vlan for the server and embedded devices
|
||||||
|
- server/devices can still access http(s?) port for updates
|
||||||
|
- just want to close devices ssh port
|
||||||
|
- way to connect to vlan remotely (vpn tunnel)
|
||||||
|
- ssh into any device once connected to the server
|
||||||
|
- subdomain/static ip for server to test with autoconnect?
|
||||||
|
- can test with reserved ips
|
||||||
|
- basically endpoint.forelight.com would forward to the provided ip:port combo to allow remote client testing and hardcoded urls for dns resolution
|
||||||
|
- local dns option?
|
||||||
|
- ask about laptop dual boot
|
||||||
|
- fedora on like 50 gigs with a barebones i3wm
|
||||||
|
- any auth issues?
|
||||||
|
|
||||||
|
reserve static ip for the server
|
||||||
|
mac address
|
||||||
|
ip address request
|
||||||
|
|
||||||
|
|
Binary file not shown.
After Width: | Height: | Size: 239 KiB |
Binary file not shown.
After Width: | Height: | Size: 239 KiB |
@ -0,0 +1,70 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sync"
|
||||||
|
"log"
|
||||||
|
)
|
||||||
|
|
||||||
|
// this package creates coordinators responsible for keeping track of active clients and invoking managers
|
||||||
|
|
||||||
|
type Coordinator struct {
|
||||||
|
Type string // ["reactor","tui"]
|
||||||
|
IncomingClients <-chan *Client
|
||||||
|
*Managers
|
||||||
|
Err chan error
|
||||||
|
}
|
||||||
|
|
||||||
|
type Managers struct {
|
||||||
|
Directory map[uint32](chan<- bool)
|
||||||
|
sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCoordinator(t string,ch chan *Client, err chan error) *Coordinator {
|
||||||
|
d := make(map[uint32](chan<- bool))
|
||||||
|
m := &Managers{Directory:d}
|
||||||
|
c := &Coordinator{Type: t,IncomingClients: ch,Err:err}
|
||||||
|
c.Managers = m
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
func FindNewManager(c *Client,ch chan bool, err chan error) {
|
||||||
|
switch c.Type {
|
||||||
|
case "reactor":
|
||||||
|
NewReactorManager(c,ch,err)
|
||||||
|
case "tui":
|
||||||
|
NewTUIManager(c,ch,err)
|
||||||
|
default:
|
||||||
|
log.Fatal(fmt.Sprintf("ERROR %v NOT FOUND",c.Type))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Coordinator) Start() {
|
||||||
|
// on start we need to create channel listener
|
||||||
|
// on each new connection we want to check its id against our mapping
|
||||||
|
go c.Listen()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Coordinator) Listen() {
|
||||||
|
for {
|
||||||
|
cl := <-c.IncomingClients
|
||||||
|
go c.ClientHandler(cl)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Coordinator) ClientHandler(cl *Client) {
|
||||||
|
// (creates and) notifies manager of client connection
|
||||||
|
c.Managers.Lock()
|
||||||
|
defer c.Managers.Unlock()
|
||||||
|
if m, exists := c.Managers.Directory[cl.Id]; exists {
|
||||||
|
// manager in memory
|
||||||
|
m <-true
|
||||||
|
} else {
|
||||||
|
// create channel and manager
|
||||||
|
ch := make(chan bool)
|
||||||
|
FindNewManager(cl, ch,c.Err)
|
||||||
|
c.Managers.Directory[cl.Id] = ch
|
||||||
|
// will block until manager is ready
|
||||||
|
ch <-true
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,110 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"log"
|
||||||
|
"sync"
|
||||||
|
"context"
|
||||||
|
"FRMS/internal/pkg/system"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
pb "FRMS/internal/pkg/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
// the goal here is to set up a gRPC server to respond to client pings with their IP and to establish a new manager for that specific client
|
||||||
|
|
||||||
|
// going to rename shit to be more general
|
||||||
|
type Listener struct { // exporting for easy use in the short term
|
||||||
|
// Reactor map[uint32]*ReactorManager this will go in eventual "coordinator" struct
|
||||||
|
Ip string
|
||||||
|
Port int
|
||||||
|
*Coordinators
|
||||||
|
Err chan error
|
||||||
|
pb.UnimplementedHandshakeServer
|
||||||
|
}
|
||||||
|
|
||||||
|
type Coordinators struct {
|
||||||
|
Channel map[string](chan<- *Client)
|
||||||
|
sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
type Client struct {
|
||||||
|
Ip string
|
||||||
|
Port int
|
||||||
|
Id uint32
|
||||||
|
Model string
|
||||||
|
Type string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClient(ip, model, t string, port int, id uint32) *Client {
|
||||||
|
return &Client{Ip:ip,Port:port,Id:id,Model:model,Type:t}
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetIp(e string) (string, error) {
|
||||||
|
return system.GetIp(e)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewListener(ifconfig string,ch chan error) (*Listener, error) {
|
||||||
|
//m := make(map[uint32]*ReactorManager)
|
||||||
|
var ip string
|
||||||
|
var err error
|
||||||
|
if ip, err = GetIp(ifconfig); err != nil {
|
||||||
|
return &Listener{}, err
|
||||||
|
}
|
||||||
|
m := make(map[string](chan<- *Client))
|
||||||
|
c := &Coordinators{Channel:m}
|
||||||
|
l := &Listener{Ip:ip,Err:ch}
|
||||||
|
l.Coordinators = c
|
||||||
|
return l, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *Listener) Start() {
|
||||||
|
// start grpc server and implement reciever
|
||||||
|
if err := l.Register(); err != nil {
|
||||||
|
l.Err <- err
|
||||||
|
}
|
||||||
|
// listener started and grpc handler registered
|
||||||
|
fmt.Printf("Started listener on %v:%v\n",l.Ip,l.Port)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *Listener) Register() error {
|
||||||
|
// creates a gRPC service and binds it to our handler
|
||||||
|
lis, err := net.Listen("tcp", fmt.Sprintf("%v:0",l.Ip)) // by binding to :0 we should get assigned an empty port
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
l.Port = lis.Addr().(*net.TCPAddr).Port // getting the port we were assigned
|
||||||
|
grpcServer := grpc.NewServer()
|
||||||
|
pb.RegisterHandshakeServer(grpcServer, l)
|
||||||
|
go grpcServer.Serve(lis)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *Listener) ClientDiscoveryHandler(ctx context.Context, ping *pb.ClientDiscoveryRequest) (*pb.ClientDiscoveryResponse, error) {
|
||||||
|
// incoming reactor ping need to spawn coord
|
||||||
|
c := NewClient(ping.GetIp(),ping.GetModel(),ping.GetClientType(),int(ping.GetPort()),ping.GetId())
|
||||||
|
fmt.Printf("%v Client %v has connected from %v:%v\n",c.Type,c.Id,c.Ip,c.Port)
|
||||||
|
go l.ConnectClient(c)
|
||||||
|
// we dont handle any actual logic about the creation so we just respon true if the request was received
|
||||||
|
return &pb.ClientDiscoveryResponse{Success:true}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *Listener) ConnectClient(c *Client){
|
||||||
|
// send to reactor coordinator for ease
|
||||||
|
l.Coordinators.Lock()
|
||||||
|
defer l.Coordinators.Unlock()
|
||||||
|
switch c.Type {
|
||||||
|
case "reactor","tui":
|
||||||
|
if ch, exists := l.Coordinators.Channel[c.Type]; exists {
|
||||||
|
ch <-c
|
||||||
|
} else {
|
||||||
|
ch := make(chan *Client)
|
||||||
|
newC := NewCoordinator(c.Type, ch, l.Err)
|
||||||
|
go newC.Start()
|
||||||
|
l.Coordinators.Channel[c.Type] = ch
|
||||||
|
ch <-c
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
log.Fatal("Error! client %v not supported!",c.Type)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,135 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
"math"
|
||||||
|
"sync"
|
||||||
|
"errors"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
"google.golang.org/grpc/credentials/insecure"
|
||||||
|
)
|
||||||
|
|
||||||
|
// this package will implement a boilerplate manager
|
||||||
|
// manager connects to client on start and returns the gRPC connection to make gRPC clients
|
||||||
|
|
||||||
|
type Manager struct {
|
||||||
|
*Client // gives access to c.Ip c.Id etc
|
||||||
|
Hb time.Duration
|
||||||
|
Active active
|
||||||
|
Err chan error
|
||||||
|
}
|
||||||
|
|
||||||
|
type active struct{
|
||||||
|
sync.Mutex
|
||||||
|
bool
|
||||||
|
int
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewManager(c *Client, err chan error) *Manager {
|
||||||
|
hb := time.Duration(1) //hb to
|
||||||
|
m := &Manager{Hb:hb,Err:err}
|
||||||
|
m.Client = c
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Manager) Start() {
|
||||||
|
// establish connection with client and start pinging at set intervals
|
||||||
|
if !m.Activate() {
|
||||||
|
// manager already running
|
||||||
|
m.Err <-errors.New("Manager already running!")
|
||||||
|
} // if we get here, manager is atomically activated and we can ensure start wont run again
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Manager) Exit() {
|
||||||
|
// exit function to eventually allow saving to configs
|
||||||
|
if !m.Deactivate() {
|
||||||
|
m.Err <-errors.New("Manager already disabled!")
|
||||||
|
}
|
||||||
|
fmt.Printf("Manager %v exiting\n", m.Id)
|
||||||
|
}
|
||||||
|
|
||||||
|
// reactor manager atomic operations
|
||||||
|
|
||||||
|
func (m *Manager) IsActive() bool {
|
||||||
|
m.Active.Lock()
|
||||||
|
defer m.Active.Unlock()
|
||||||
|
return m.Active.bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Manager) Activate() bool {
|
||||||
|
m.Active.Lock()
|
||||||
|
defer m.Active.Unlock()
|
||||||
|
alive := m.Active.bool
|
||||||
|
if alive {
|
||||||
|
return false
|
||||||
|
} else {
|
||||||
|
m.Active.bool = true
|
||||||
|
m.Active.int = 0
|
||||||
|
return m.Active.bool
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Manager) Deactivate() bool {
|
||||||
|
m.Active.Lock()
|
||||||
|
defer m.Active.Unlock()
|
||||||
|
alive := m.Active.bool
|
||||||
|
if alive {
|
||||||
|
m.Active.bool = false
|
||||||
|
return true
|
||||||
|
} else {
|
||||||
|
return m.Active.bool
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// connection stuff
|
||||||
|
|
||||||
|
func (m *Manager) Timeout() int {
|
||||||
|
// keeps track of and generates timeout [0-1.2s) over span of ~2.5s
|
||||||
|
// returns 0 on TO elapse
|
||||||
|
m.Active.Lock()
|
||||||
|
defer m.Active.Unlock()
|
||||||
|
if m.Active.int < 9 {
|
||||||
|
v := int(5 * math.Pow(float64(2), float64(m.Active.int)))
|
||||||
|
m.Active.int += 1
|
||||||
|
return v
|
||||||
|
} else {
|
||||||
|
// exceeded retries
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Manager) Connect() *grpc.ClientConn{
|
||||||
|
// establish initial gRPC connection with client
|
||||||
|
var opts []grpc.DialOption
|
||||||
|
var conn *grpc.ClientConn
|
||||||
|
opts = append(opts,grpc.WithTransportCredentials(insecure.NewCredentials()))
|
||||||
|
for {
|
||||||
|
if !m.IsActive() {
|
||||||
|
fmt.Printf("Aborting connection attempt\n")
|
||||||
|
return &grpc.ClientConn{}
|
||||||
|
}
|
||||||
|
var err error
|
||||||
|
conn, err = grpc.Dial(fmt.Sprintf("%v:%v",m.Ip,m.Port),opts...)
|
||||||
|
// begin error handling
|
||||||
|
code := status.Code(err)
|
||||||
|
if code != 0 { // != OK
|
||||||
|
if code == (5 | 14) { // == unavailable or not found
|
||||||
|
to := m.Timeout()
|
||||||
|
if to == 0 {
|
||||||
|
fmt.Printf("Client not responding\n")
|
||||||
|
m.Exit()
|
||||||
|
return&grpc.ClientConn{}
|
||||||
|
}
|
||||||
|
fmt.Printf("gRPC endpoint currently unavailable, retrying in %v ms\n",to)
|
||||||
|
time.Sleep(time.Duration(to) * time.Millisecond)
|
||||||
|
} else {
|
||||||
|
fmt.Printf("ERR GRPC: %v\n",code)
|
||||||
|
m.Err <-err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return conn
|
||||||
|
}
|
@ -0,0 +1,67 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
"context"
|
||||||
|
"sync"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
pb "FRMS/internal/pkg/grpc"
|
||||||
|
)
|
||||||
|
|
||||||
|
// this package will implement a reactor coordinator and associated go routines
|
||||||
|
|
||||||
|
type ReactorManager struct {
|
||||||
|
*Manager
|
||||||
|
Devs *Devices
|
||||||
|
ClientConnections <-chan bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type Devices struct {
|
||||||
|
mu sync.Mutex
|
||||||
|
D map[int]Device
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewReactorManager(c *Client,ch chan bool,err chan error) {
|
||||||
|
d := new(Devices)
|
||||||
|
r := &ReactorManager{Devs:d,ClientConnections:ch}
|
||||||
|
r.Manager = NewManager(c, err)
|
||||||
|
go r.Listen()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ReactorManager) Listen() {
|
||||||
|
for {
|
||||||
|
c := <-r.ClientConnections
|
||||||
|
if c {
|
||||||
|
r.Start()
|
||||||
|
} else {
|
||||||
|
r.Exit()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ReactorManager) Start() {
|
||||||
|
r.Manager.Start()
|
||||||
|
conn := r.Connect()
|
||||||
|
go r.Monitor(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! ", r.Id)
|
||||||
|
r.Exit()
|
||||||
|
}
|
||||||
|
for _,v := range resp.GetDevices() {
|
||||||
|
fmt.Printf("%v is %v, ",v.GetType(),v.GetStatus())
|
||||||
|
}
|
||||||
|
fmt.Print("\n")
|
||||||
|
time.Sleep(r.Hb * time.Second) // time between sensor pings
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,80 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
// implement tui specific manager to be called for each client conn
|
||||||
|
|
||||||
|
type TUIManager struct {
|
||||||
|
*Manager // embedded manager for access to methods and client
|
||||||
|
Sys *KnownReactors
|
||||||
|
ClientConnections <-chan bool
|
||||||
|
Err chan error
|
||||||
|
Hb time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
type KnownReactors struct {
|
||||||
|
Reactors map[uint32]*Reactor
|
||||||
|
sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
type Reactor struct {
|
||||||
|
Devices map[string]*Device
|
||||||
|
}
|
||||||
|
|
||||||
|
type Device struct {
|
||||||
|
Status DeviceStatus
|
||||||
|
Type string
|
||||||
|
Addr int
|
||||||
|
}
|
||||||
|
|
||||||
|
type DeviceStatus uint32
|
||||||
|
|
||||||
|
const (
|
||||||
|
READY DeviceStatus = iota
|
||||||
|
ACTIVE
|
||||||
|
DISABLED
|
||||||
|
)
|
||||||
|
|
||||||
|
func (d DeviceStatus) String() string {
|
||||||
|
return [...]string{"Ready","Active","Disabled"}[d]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d Device) String() string {
|
||||||
|
return fmt.Sprintf("%v is %v at %x",d.Type,d.Status,d.Addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
type DeviceManager interface{
|
||||||
|
// GetStatus() uint32 UNSUPPORTED: arguable memory benifit but until we support 100s of sensors across 10s of tui clients im not implementing it
|
||||||
|
PrintSatus() string
|
||||||
|
GetType() string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewTUIManager(c *Client,ch chan bool, err chan error) {
|
||||||
|
k := new(KnownReactors)
|
||||||
|
m := NewManager(c, err)
|
||||||
|
hb := time.Duration(5)
|
||||||
|
t := &TUIManager{Hb: hb,Sys: k,Err: err, ClientConnections: ch}
|
||||||
|
t.Manager = m
|
||||||
|
go t.Listen()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TUIManager) Listen() {
|
||||||
|
for {
|
||||||
|
c := <-t.ClientConnections
|
||||||
|
if c {
|
||||||
|
t.Start()
|
||||||
|
} else {
|
||||||
|
t.Exit()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *TUIManager) Start() {
|
||||||
|
t.Manager.Start()
|
||||||
|
//conn := t.Conn()
|
||||||
|
//go t.Monitor(conn)
|
||||||
|
}
|
Loading…
Reference in New Issue