2021-02-01 11:13:38 +01:00
|
|
|
package steward
|
2021-01-29 14:22:36 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/gob"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
|
|
|
|
"github.com/nats-io/nats.go"
|
|
|
|
)
|
|
|
|
|
2021-02-03 09:06:37 +01:00
|
|
|
// RunSubscriber will start a subscribing process.
|
|
|
|
// TODO: Right now the only thing a subscriber can do is ro receive commands,
|
|
|
|
// check if there are more things a subscriber should be able to do.
|
2021-01-29 14:22:36 +01:00
|
|
|
func (s *server) RunSubscriber() {
|
|
|
|
|
|
|
|
// Create a channel to put the data received in the subscriber callback
|
|
|
|
// function
|
|
|
|
reqMsgCh := make(chan Message)
|
|
|
|
|
|
|
|
// Subscribe will start up a Go routine under the hood calling the
|
|
|
|
// callback function specified when a new message is received.
|
2021-02-03 10:23:50 +01:00
|
|
|
subject := fmt.Sprintf("%s.%s.%s.%s", s.nodeName, "command", "shellcommand", "shell")
|
2021-02-01 13:41:04 +01:00
|
|
|
_, err := s.natsConn.Subscribe(subject, listenForMessage(s.natsConn, reqMsgCh, s.nodeName))
|
2021-01-29 14:22:36 +01:00
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("error: Subscribe failed: %v\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do some further processing of the actual data we received in the
|
|
|
|
// subscriber callback function.
|
|
|
|
for {
|
|
|
|
msg := <-reqMsgCh
|
|
|
|
fmt.Printf("%v\n", msg)
|
|
|
|
switch msg.MessageType {
|
|
|
|
case eventReturnAck:
|
2021-02-03 10:23:50 +01:00
|
|
|
// Since the command to execute is at the first position in the
|
|
|
|
// slice we need to slice it out. The arguments are at the
|
|
|
|
// remaining positions.
|
2021-01-29 14:22:36 +01:00
|
|
|
c := msg.Data[0]
|
|
|
|
a := msg.Data[1:]
|
|
|
|
cmd := exec.Command(c, a...)
|
|
|
|
cmd.Stdout = os.Stdout
|
|
|
|
err := cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("error: execution of command failed: %v\n", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Listen for message will send an ACK message back to the sender,
|
|
|
|
// and put the received incomming message on the reqMsg channel
|
|
|
|
// for further processing.
|
|
|
|
func listenForMessage(natsConn *nats.Conn, reqMsgCh chan Message, node string) func(req *nats.Msg) {
|
|
|
|
return func(req *nats.Msg) {
|
|
|
|
message := Message{}
|
|
|
|
|
|
|
|
// Create a buffer to decode the gob encoded binary data back
|
|
|
|
// to it's original structure.
|
|
|
|
buf := bytes.NewBuffer(req.Data)
|
|
|
|
gobDec := gob.NewDecoder(buf)
|
|
|
|
err := gobDec.Decode(&message)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("error: gob decoding failed: %v\n", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put the data recived on the channel for further processing
|
|
|
|
reqMsgCh <- message
|
|
|
|
|
|
|
|
// Send a confirmation message back to the publisher
|
|
|
|
natsConn.Publish(req.Reply, []byte("confirmed from: "+node+": "+fmt.Sprint(message.ID)))
|
|
|
|
}
|
|
|
|
}
|