mirror of
https://github.com/postmannen/ctrl.git
synced 2025-01-05 20:09:16 +00:00
434 lines
13 KiB
Go
434 lines
13 KiB
Go
// The structure of how to add new method types to the system.
|
|
// -----------------------------------------------------------
|
|
// All methods need 3 things:
|
|
// - A type definition
|
|
// - The type needs a getKind method
|
|
// - The type needs a handler method
|
|
// Overall structure example shown below.
|
|
//
|
|
// ---
|
|
// type methodCommandCLICommand struct {
|
|
// commandOrEvent CommandOrEvent
|
|
// }
|
|
//
|
|
// func (m methodCommandCLICommand) getKind() CommandOrEvent {
|
|
// return m.commandOrEvent
|
|
// }
|
|
//
|
|
// func (m methodCommandCLICommand) handler(s *server, message Message, node string) ([]byte, error) {
|
|
// ...
|
|
// ...
|
|
// ackMsg := []byte(fmt.Sprintf("confirmed from node: %v: messageID: %v\n---\n%s---", node, message.ID, out))
|
|
// return ackMsg, nil
|
|
// }
|
|
//
|
|
// ---
|
|
// You also need to make a constant for the Method, and add
|
|
// that constant as the key in the map, where the value is
|
|
// the actual type you want to map it to with a handler method.
|
|
// You also specify if it is a Command or Event, and if it is
|
|
// ACK or NACK.
|
|
// Check out the existing code below for more examples.
|
|
|
|
package steward
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"time"
|
|
)
|
|
|
|
// Method is used to specify the actual function/method that
|
|
// is represented in a typed manner.
|
|
type Method string
|
|
|
|
// ------------------------------------------------------------
|
|
// The constants that will be used throughout the system for
|
|
// when specifying what kind of Method to send or work with.
|
|
const (
|
|
// Execute a CLI command in for example bash or cmd.
|
|
// This is a command type, so the output of the command executed
|
|
// will directly showed in the ACK message received.
|
|
// The data field is a slice of strings where the first string
|
|
// value should be the command, and the following the arguments.
|
|
CLICommand Method = "CLICommand"
|
|
// Execute a CLI command in for example bash or cmd.
|
|
// This is an event type, where a message will be sent to a
|
|
// node with the command to execute and an ACK will be replied
|
|
// if it was delivered succesfully. The output of the command
|
|
// ran will be delivered back to the node where it was initiated
|
|
// as a new message.
|
|
// The data field is a slice of strings where the first string
|
|
// value should be the command, and the following the arguments.
|
|
CLICommandRequest Method = "CLICommandRequest"
|
|
// Execute a CLI command in for example bash or cmd.
|
|
// This is an event type, where a message will be sent to a
|
|
// node with the command to execute and an ACK will be replied
|
|
// if it was delivered succesfully. The output of the command
|
|
// ran will be delivered back to the node where it was initiated
|
|
// as a new message.
|
|
// The NOSEQ method will process messages as they are recived,
|
|
// and the reply back will be sent as soon as the process is
|
|
// done. No order are preserved.
|
|
// The data field is a slice of strings where the first string
|
|
// value should be the command, and the following the arguments.
|
|
CLICommandRequestNOSEQ Method = "CLICommandRequestNOSEQ"
|
|
// Will generate a reply for a CLICommandRequest.
|
|
// This type is normally not used by the user when creating
|
|
// a message. It is used in creating the reply message with
|
|
// request messages. It is also used when defining a process to
|
|
// start up for receiving the CLICommand request messages.
|
|
// The data field is a slice of strings where the first string
|
|
// value should be the command, and the following the arguments.
|
|
CLICommandReply Method = "CLICommandReply"
|
|
// Send text logging to some host.
|
|
// A file with the full subject+hostName will be created on
|
|
// the receiving end.
|
|
// The data field is a slice of strings where the values of the
|
|
// slice will be written to the log file.
|
|
TextLogging Method = "TextLogging"
|
|
// Send Hello I'm here message.
|
|
SayHello Method = "SayHello"
|
|
// Error log methods to centralError node.
|
|
ErrorLog Method = "ErrorLog"
|
|
// Echo request will ask the subscriber for a
|
|
// reply generated as a new message, and sent back to where
|
|
// the initial request was made.
|
|
ECHORequest Method = "ECHORequest"
|
|
// Will generate a reply for a ECHORequest
|
|
ECHOReply Method = "ECHOReply"
|
|
)
|
|
|
|
// The mapping of all the method constants specified, what type
|
|
// it references, and the kind if it is an Event or Command, and
|
|
// if it is ACK or NACK.
|
|
// Allowed values for the commandOrEvent field are:
|
|
// - CommandACK
|
|
// - CommandNACK
|
|
// - EventACK
|
|
// - EventNack
|
|
func (m Method) GetMethodsAvailable() MethodsAvailable {
|
|
ma := MethodsAvailable{
|
|
methodhandlers: map[Method]methodHandler{
|
|
CLICommand: methodCLICommand{
|
|
commandOrEvent: CommandACK,
|
|
},
|
|
CLICommandRequest: methodCLICommandRequest{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
CLICommandRequestNOSEQ: methodCLICommandRequestNOSEQ{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
CLICommandReply: methodCLICommandReply{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
TextLogging: methodTextLogging{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
SayHello: methodSayHello{
|
|
commandOrEvent: EventNACK,
|
|
},
|
|
ErrorLog: methodErrorLog{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
ECHORequest: methodEchoRequest{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
ECHOReply: methodEchoReply{
|
|
commandOrEvent: EventACK,
|
|
},
|
|
},
|
|
}
|
|
|
|
return ma
|
|
}
|
|
|
|
// getHandler will check the methodsAvailable map, and return the
|
|
// method handler for the method given
|
|
// as input argument.
|
|
func (m Method) getHandler(method Method) methodHandler {
|
|
ma := m.GetMethodsAvailable()
|
|
mh := ma.methodhandlers[method]
|
|
|
|
return mh
|
|
}
|
|
|
|
// The structure that works as a reference for all the methods and if
|
|
// they are of the command or event type, and also if it is a ACK or
|
|
// NACK message.
|
|
type MethodsAvailable struct {
|
|
methodhandlers map[Method]methodHandler
|
|
}
|
|
|
|
// Check if exists will check if the Method is defined. If true the bool
|
|
// value will be set to true, and the methodHandler function for that type
|
|
// will be returned.
|
|
func (ma MethodsAvailable) CheckIfExists(m Method) (methodHandler, bool) {
|
|
mFunc, ok := ma.methodhandlers[m]
|
|
if ok {
|
|
// fmt.Printf("******THE TOPIC EXISTS: %v******\n", m)
|
|
return mFunc, true
|
|
} else {
|
|
// fmt.Printf("******THE TOPIC DO NOT EXIST: %v******\n", m)
|
|
return nil, false
|
|
}
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Subscriber method handlers
|
|
// ------------------------------------------------------------
|
|
|
|
type methodHandler interface {
|
|
handler(proc process, message Message, node string) ([]byte, error)
|
|
getKind() CommandOrEvent
|
|
}
|
|
|
|
// -----
|
|
|
|
type methodCLICommand struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommand) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommand) handler(proc process, message Message, node string) ([]byte, error) {
|
|
c := message.Data[0]
|
|
a := message.Data[1:]
|
|
cmd := exec.Command(c, a...)
|
|
//cmd.Stdout = os.Stdout
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
log.Printf("error: execution of command failed: %v\n", err)
|
|
}
|
|
|
|
ackMsg := []byte(fmt.Sprintf("confirmed from node: %v: messageID: %v\n---\n%s---", node, message.ID, out))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// -----
|
|
|
|
type methodTextLogging struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodTextLogging) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodTextLogging) handler(proc process, message Message, node string) ([]byte, error) {
|
|
sub := Subject{
|
|
ToNode: string(message.ToNode),
|
|
CommandOrEvent: proc.subject.CommandOrEvent,
|
|
Method: message.Method,
|
|
}
|
|
|
|
logFile := filepath.Join(proc.configuration.SubscribersDataFolder, string(sub.name())+"-"+string(message.FromNode))
|
|
f, err := os.OpenFile(logFile, os.O_APPEND|os.O_RDWR|os.O_CREATE, os.ModeAppend)
|
|
if err != nil {
|
|
log.Printf("error: methodEventTextLogging.handler: failed to open file: %v\n", err)
|
|
return nil, err
|
|
}
|
|
defer f.Close()
|
|
|
|
for _, d := range message.Data {
|
|
_, err := f.Write([]byte(d))
|
|
f.Sync()
|
|
if err != nil {
|
|
log.Printf("error: methodEventTextLogging.handler: failed to write to file: %v\n", err)
|
|
}
|
|
}
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// -----
|
|
|
|
type methodSayHello struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodSayHello) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodSayHello) handler(proc process, message Message, node string) ([]byte, error) {
|
|
|
|
log.Printf("<--- Received hello from %#v\n", message.FromNode)
|
|
|
|
// send the message to the procFuncCh which is running alongside the process
|
|
// and can hold registries and handle special things for an individual process.
|
|
proc.procFuncCh <- message
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// ---
|
|
|
|
type methodErrorLog struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodErrorLog) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodErrorLog) handler(proc process, message Message, node string) ([]byte, error) {
|
|
log.Printf("<--- Received error from: %v, containing: %v", message.FromNode, message.Data)
|
|
return nil, nil
|
|
}
|
|
|
|
// ---
|
|
|
|
type methodEchoRequest struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodEchoRequest) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodEchoRequest) handler(proc process, message Message, node string) ([]byte, error) {
|
|
log.Printf("<--- ECHO REQUEST received from: %v, containing: %v", message.FromNode, message.Data)
|
|
|
|
// Create a new message for the reply, and put it on the
|
|
// ringbuffer to be published.
|
|
newMsg := Message{
|
|
ToNode: message.FromNode,
|
|
Data: []string{""},
|
|
Method: ECHOReply,
|
|
Timeout: 3,
|
|
Retries: 3,
|
|
}
|
|
proc.newMessagesCh <- []subjectAndMessage{newSAM(newMsg)}
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// ---
|
|
|
|
type methodEchoReply struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodEchoReply) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodEchoReply) handler(proc process, message Message, node string) ([]byte, error) {
|
|
log.Printf("<--- ECHO Reply received from: %v, containing: %v", message.FromNode, message.Data)
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// ---
|
|
|
|
type methodCLICommandRequest struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommandRequest) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommandRequest) handler(proc process, message Message, node string) ([]byte, error) {
|
|
log.Printf("<--- CLICommand REQUEST received from: %v, containing: %v", message.FromNode, message.Data)
|
|
|
|
c := message.Data[0]
|
|
a := message.Data[1:]
|
|
cmd := exec.Command(c, a...)
|
|
//cmd.Stdout = os.Stdout
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
log.Printf("error: execution of command failed: %v\n", err)
|
|
}
|
|
|
|
time.Sleep(time.Second * time.Duration(message.MethodTimeout))
|
|
|
|
// Create a new message for the reply, and put it on the
|
|
// ringbuffer to be published.
|
|
newMsg := Message{
|
|
ToNode: message.FromNode,
|
|
Data: []string{string(out)},
|
|
Method: CLICommandReply,
|
|
Timeout: 3,
|
|
Retries: 3,
|
|
}
|
|
fmt.Printf("** %#v\n", newMsg)
|
|
proc.newMessagesCh <- []subjectAndMessage{newSAM(newMsg)}
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// --- methodCLICommandRequestNOSEQ
|
|
|
|
type methodCLICommandRequestNOSEQ struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommandRequestNOSEQ) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
// The NOSEQ method will process messages as they are recived,
|
|
// and the reply back will be sent as soon as the process is
|
|
// done. No order are preserved.
|
|
func (m methodCLICommandRequestNOSEQ) handler(proc process, message Message, node string) ([]byte, error) {
|
|
log.Printf("<--- CLICommand REQUEST received from: %v, containing: %v", message.FromNode, message.Data)
|
|
|
|
go func() {
|
|
|
|
c := message.Data[0]
|
|
a := message.Data[1:]
|
|
cmd := exec.Command(c, a...)
|
|
//cmd.Stdout = os.Stdout
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
log.Printf("error: execution of command failed: %v\n", err)
|
|
}
|
|
|
|
time.Sleep(time.Second * time.Duration(message.MethodTimeout))
|
|
|
|
// Create a new message for the reply, and put it on the
|
|
// ringbuffer to be published.
|
|
newMsg := Message{
|
|
ToNode: message.FromNode,
|
|
Data: []string{string(out)},
|
|
Method: CLICommandReply,
|
|
Timeout: 3,
|
|
Retries: 3,
|
|
}
|
|
fmt.Printf("** %#v\n", newMsg)
|
|
proc.newMessagesCh <- []subjectAndMessage{newSAM(newMsg)}
|
|
|
|
}()
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|
|
|
|
// ---
|
|
|
|
type methodCLICommandReply struct {
|
|
commandOrEvent CommandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommandReply) getKind() CommandOrEvent {
|
|
return m.commandOrEvent
|
|
}
|
|
|
|
func (m methodCLICommandReply) handler(proc process, message Message, node string) ([]byte, error) {
|
|
fmt.Printf("### %v\n", message.Data)
|
|
|
|
ackMsg := []byte("confirmed from: " + node + ": " + fmt.Sprint(message.ID))
|
|
return ackMsg, nil
|
|
}
|