mirror of
https://github.com/postmannen/ctrl.git
synced 2025-01-05 20:09:16 +00:00
115 lines
3.8 KiB
Go
115 lines
3.8 KiB
Go
package steward
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/gob"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
)
|
|
|
|
// --- Message
|
|
|
|
type Message struct {
|
|
// Label is a word without white spaces that can be used to
|
|
// desribe what kind of service the message is for. For
|
|
// example syslog, web_log, prometheus_scraper, metrics.
|
|
// The intentention is that it can be used to add some extra
|
|
// information that can be used on the subscriber side to for
|
|
// example create specific folders using the Format name to
|
|
// logically group data recevied.
|
|
Label string `json:"label" yaml:"label"`
|
|
// The node to send the message to
|
|
ToNode node `json:"toNode" yaml:"toNode"`
|
|
// The Unique ID of the message
|
|
ID int `json:"id" yaml:"id"`
|
|
// The actual data in the message
|
|
Data []string `json:"data" yaml:"data"`
|
|
// method, what is this message doing, etc. CLI, syslog, etc.
|
|
Method Method `json:"method" yaml:"method"`
|
|
FromNode node
|
|
// Normal Reply wait timeout
|
|
Timeout int `json:"timeout" yaml:"timeout"`
|
|
// Normal Resend retries
|
|
Retries int `json:"retries" yaml:"retries"`
|
|
// The timeout of the new message created via a request event.
|
|
RequestTimeout int `json:"requestTimeout" yaml:"requestTimeout"`
|
|
// The retries of the new message created via a request event.
|
|
RequestRetries int `json:"requestRetries" yaml:"requestRetries"`
|
|
// Timeout for long a process should be allowed to operate
|
|
MethodTimeout int `json:"methodTimeout" yaml:"methodTimeout"`
|
|
|
|
// PreviousMessage are used for example if a reply message is
|
|
// generated and we also need a copy of thedetails of the the
|
|
// initial request message
|
|
PreviousMessage *Message
|
|
|
|
// done is used to signal when a message is fully processed.
|
|
// This is used for signaling back to the ringbuffer that we are
|
|
// done with processing a message, and the message can be removed
|
|
// from the ringbuffer and into the time series log.
|
|
done chan struct{}
|
|
}
|
|
|
|
// gobEncodePayload will encode the message structure into gob
|
|
// binary format.
|
|
func gobEncodeMessage(m Message) ([]byte, error) {
|
|
var buf bytes.Buffer
|
|
gobEnc := gob.NewEncoder(&buf)
|
|
err := gobEnc.Encode(m)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error: gob.Encode failed: %v", err)
|
|
}
|
|
|
|
return buf.Bytes(), nil
|
|
}
|
|
|
|
// --- Subject
|
|
|
|
type node string
|
|
|
|
// subject contains the representation of a subject to be used with one
|
|
// specific process
|
|
type Subject struct {
|
|
// node, the name of the node
|
|
ToNode string `json:"node" yaml:"toNode"`
|
|
// messageType, command/event
|
|
CommandOrEvent CommandOrEvent `json:"commandOrEvent" yaml:"commandOrEvent"`
|
|
// method, what is this message doing, etc. CLICommand, Syslog, etc.
|
|
Method Method `json:"method" yaml:"method"`
|
|
// messageCh is used by publisher kind processes to read new messages
|
|
// to be published. The content on this channel have been routed here
|
|
// from routeMessagesToPublish in *server.
|
|
// This channel is only used for publishing processes.
|
|
messageCh chan Message
|
|
// Label, single word used to describe the data content. For example
|
|
// syslog, metrics, etc.
|
|
Label string
|
|
}
|
|
|
|
// newSubject will return a new variable of the type subject, and insert
|
|
// all the values given as arguments. It will also create the channel
|
|
// to receive new messages on the specific subject.
|
|
func newSubject(method Method, node string) Subject {
|
|
// Get the CommandOrEvent type for the Method.
|
|
ma := method.GetMethodsAvailable()
|
|
coe, ok := ma.methodhandlers[method]
|
|
if !ok {
|
|
log.Printf("error: no CommandOrEvent type specified for the method\n")
|
|
os.Exit(1)
|
|
}
|
|
|
|
return Subject{
|
|
ToNode: node,
|
|
CommandOrEvent: coe.getKind(),
|
|
Method: method,
|
|
messageCh: make(chan Message),
|
|
}
|
|
}
|
|
|
|
// subjectName is the complete representation of a subject
|
|
type subjectName string
|
|
|
|
func (s Subject) name() subjectName {
|
|
return subjectName(fmt.Sprintf("%s.%s.%s", s.ToNode, s.Method, s.CommandOrEvent))
|
|
}
|