mirror of
https://github.com/postmannen/ctrl.git
synced 2025-01-08 13:09:16 +00:00
3a31ced938
implemented reading and embedding the content of a local file with {{CTRL_FILE}} fixed error with not found filename in copy test, remove debug logs seedfile: removed deletion, and changed file permissions to 600 created flags for profiling renamed startup.subscriber to startup.startProcess created a separate method for helloPublisher removed processKind, og removed not needed file check in copy request removed sams from channels removed publisher channel on subject, and messages to publish are now directly published from the newMessagesCh removed no longer needed compression and serialization flags. all messaging are using zstd for compression, and cbor for serializing added functions for handling cbor serializing and zstd compression, and swapped out json marshaling of jetstream message data with cbor and zstd added flag for max jetstream messages to keep on broker per subject
654 lines
16 KiB
Go
654 lines
16 KiB
Go
package ctrl
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/fsnotify/fsnotify"
|
|
natsserver "github.com/nats-io/nats-server/v2/server"
|
|
)
|
|
|
|
var logging = flag.Bool("logging", false, "set to true to enable the normal logger of the package")
|
|
var persistTmp = flag.Bool("persistTmp", false, "set to true to persist the tmp folder")
|
|
|
|
var tstSrv *server
|
|
var tstConf *Configuration
|
|
var tstNats *natsserver.Server
|
|
var tstTempDir string
|
|
|
|
func TestMain(m *testing.M) {
|
|
flag.Parse()
|
|
|
|
if *persistTmp {
|
|
tstTempDir = "tmp"
|
|
} else {
|
|
tstTempDir = os.TempDir()
|
|
}
|
|
|
|
// NB: Forcing this for now.
|
|
tstTempDir = "tmp"
|
|
|
|
tstNats = newNatsServerForTesting(42222)
|
|
if err := natsserver.Run(tstNats); err != nil {
|
|
natsserver.PrintAndDie(err.Error())
|
|
}
|
|
|
|
tstSrv, tstConf = newServerForTesting("127.0.0.1:42222", tstTempDir)
|
|
tstSrv.Start()
|
|
|
|
exitCode := m.Run()
|
|
|
|
tstSrv.Stop()
|
|
tstNats.Shutdown()
|
|
|
|
os.Exit(exitCode)
|
|
}
|
|
|
|
func newServerForTesting(addressAndPort string, testFolder string) (*server, *Configuration) {
|
|
|
|
// Start ctrl instance
|
|
// ---------------------------------------
|
|
// tempdir := t.TempDir()
|
|
|
|
// Create the config to run a ctrl instance.
|
|
//tempdir := "./tmp"
|
|
conf := newConfigurationDefaults()
|
|
if *logging {
|
|
conf.LogLevel = "warning"
|
|
}
|
|
conf.BrokerAddress = addressAndPort
|
|
conf.NodeName = "central"
|
|
conf.CentralNodeName = "central"
|
|
conf.ConfigFolder = testFolder
|
|
conf.SubscribersDataFolder = testFolder
|
|
conf.SocketFolder = testFolder
|
|
conf.SubscribersDataFolder = testFolder
|
|
conf.DatabaseFolder = testFolder
|
|
conf.StartProcesses.IsCentralErrorLogger = true
|
|
conf.StartProcesses.IsCentralAuth = true
|
|
conf.EnableDebug = false
|
|
conf.LogLevel = "none"
|
|
|
|
ctrlServer, err := NewServer(&conf, "test")
|
|
if err != nil {
|
|
log.Fatalf(" * failed: could not start the ctrl instance %v\n", err)
|
|
}
|
|
|
|
return ctrlServer, &conf
|
|
}
|
|
|
|
// Start up the nats-server message broker for testing purposes.
|
|
func newNatsServerForTesting(port int) *natsserver.Server {
|
|
// Start up the nats-server message broker.
|
|
nsOpt := &natsserver.Options{
|
|
Host: "127.0.0.1",
|
|
Port: port,
|
|
JetStream: true,
|
|
}
|
|
|
|
ns, err := natsserver.NewServer(nsOpt)
|
|
if err != nil {
|
|
log.Fatalf(" * failed: could not start the nats-server %v\n", err)
|
|
}
|
|
|
|
return ns
|
|
}
|
|
|
|
// Write message to socket for testing purposes.
|
|
func writeMsgsToSocketTest(conf *Configuration, messages []Message, t *testing.T) {
|
|
js, err := json.Marshal(messages)
|
|
if err != nil {
|
|
t.Fatalf("writeMsgsToSocketTest: %v\n ", err)
|
|
}
|
|
|
|
socket, err := net.Dial("unix", filepath.Join(conf.SocketFolder, "ctrl.sock"))
|
|
if err != nil {
|
|
t.Fatalf(" * failed: could to open socket file for writing: %v\n", err)
|
|
}
|
|
defer socket.Close()
|
|
|
|
_, err = socket.Write(js)
|
|
if err != nil {
|
|
t.Fatalf(" * failed: could not write to socket: %v\n", err)
|
|
}
|
|
|
|
}
|
|
|
|
func TestRequest(t *testing.T) {
|
|
if !*logging {
|
|
log.SetOutput(io.Discard)
|
|
}
|
|
|
|
type containsOrEquals int
|
|
const (
|
|
REQTestContains containsOrEquals = iota
|
|
REQTestEquals containsOrEquals = iota
|
|
fileContains containsOrEquals = iota
|
|
)
|
|
|
|
type viaSocketOrCh int
|
|
const (
|
|
viaSocket viaSocketOrCh = iota
|
|
viaCh viaSocketOrCh = iota
|
|
)
|
|
|
|
type test struct {
|
|
info string
|
|
message Message
|
|
want []byte
|
|
containsOrEquals
|
|
viaSocketOrCh
|
|
}
|
|
|
|
// Web server for testing.
|
|
{
|
|
h := func(w http.ResponseWriter, r *http.Request) {
|
|
w.Write([]byte("web page content"))
|
|
}
|
|
http.HandleFunc("/", h)
|
|
|
|
go func() {
|
|
http.ListenAndServe(":10080", nil)
|
|
}()
|
|
}
|
|
|
|
tests := []test{
|
|
{
|
|
info: "hello test",
|
|
message: Message{
|
|
ToNode: "errorCentral",
|
|
FromNode: "errorCentral",
|
|
Method: ErrorLog,
|
|
MethodArgs: []string{},
|
|
MethodTimeout: 5,
|
|
Data: []byte("error data"),
|
|
// ReplyMethod: REQTest,
|
|
Directory: "error_log",
|
|
FileName: "error.results",
|
|
}, want: []byte("error data"),
|
|
containsOrEquals: fileContains,
|
|
viaSocketOrCh: viaCh,
|
|
},
|
|
{
|
|
info: "hello test",
|
|
message: Message{
|
|
ToNode: "central",
|
|
FromNode: "central",
|
|
Method: Hello,
|
|
MethodArgs: []string{},
|
|
MethodTimeout: 5,
|
|
// ReplyMethod: REQTest,
|
|
Directory: "test",
|
|
FileName: "hello.results",
|
|
}, want: []byte("Received hello from \"central\""),
|
|
containsOrEquals: fileContains,
|
|
viaSocketOrCh: viaCh,
|
|
},
|
|
{
|
|
info: "cliCommand test, echo gris",
|
|
message: Message{
|
|
ToNode: "central",
|
|
FromNode: "central",
|
|
Method: CliCommand,
|
|
MethodArgs: []string{"bash", "-c", "echo gris"},
|
|
MethodTimeout: 5,
|
|
ReplyMethod: Test,
|
|
}, want: []byte("gris"),
|
|
containsOrEquals: REQTestEquals,
|
|
viaSocketOrCh: viaCh,
|
|
},
|
|
{
|
|
info: "cliCommand test via socket, echo sau",
|
|
message: Message{
|
|
ToNode: "central",
|
|
FromNode: "central",
|
|
Method: CliCommand,
|
|
MethodArgs: []string{"bash", "-c", "echo sau"},
|
|
MethodTimeout: 5,
|
|
ReplyMethod: Test,
|
|
}, want: []byte("sau"),
|
|
containsOrEquals: REQTestEquals,
|
|
viaSocketOrCh: viaSocket,
|
|
},
|
|
{
|
|
info: "cliCommand test, echo sau, result in file",
|
|
message: Message{
|
|
ToNode: "central",
|
|
FromNode: "central",
|
|
Method: CliCommand,
|
|
MethodArgs: []string{"bash", "-c", "echo sau"},
|
|
MethodTimeout: 5,
|
|
ReplyMethod: File,
|
|
Directory: "test",
|
|
FileName: "file1.result",
|
|
}, want: []byte("sau"),
|
|
containsOrEquals: fileContains,
|
|
viaSocketOrCh: viaCh,
|
|
},
|
|
{
|
|
info: "cliCommand test, echo several, result in file continous",
|
|
message: Message{
|
|
ToNode: "central",
|
|
FromNode: "central",
|
|
Method: CliCommand,
|
|
MethodArgs: []string{"bash", "-c", "echo giraff && echo sau && echo apekatt"},
|
|
MethodTimeout: 5,
|
|
ReplyMethod: File,
|
|
Directory: "test",
|
|
FileName: "file2.result",
|
|
}, want: []byte("sau"),
|
|
containsOrEquals: fileContains,
|
|
viaSocketOrCh: viaCh,
|
|
},
|
|
{
|
|
info: "httpGet test, localhost:10080",
|
|
message: Message{
|
|
ToNode: "central",
|
|
FromNode: "central",
|
|
Method: HttpGet,
|
|
MethodArgs: []string{"http://localhost:10080"},
|
|
MethodTimeout: 5,
|
|
ReplyMethod: Test,
|
|
}, want: []byte("web page content"),
|
|
containsOrEquals: REQTestContains,
|
|
viaSocketOrCh: viaCh,
|
|
},
|
|
// TODO: Check out this one why it fails, and also why I'm checking for REQHttpGet here ??
|
|
//{
|
|
// info: "opProcessList test",
|
|
// message: Message{
|
|
// ToNode: "central",
|
|
// FromNode: "central",
|
|
// Method: REQOpProcessList,
|
|
// MethodArgs: []string{},
|
|
// MethodTimeout: 5,
|
|
// ReplyMethod: REQTest,
|
|
// }, want: []byte("central.REQHttpGet"),
|
|
// containsOrEquals: REQTestContains,
|
|
// viaSocketOrCh: viaCh,
|
|
//},
|
|
}
|
|
|
|
// Range over the tests defined, and execute them, one at a time.
|
|
for _, tt := range tests {
|
|
switch tt.viaSocketOrCh {
|
|
case viaCh:
|
|
tstSrv.newMessagesCh <- tt.message
|
|
|
|
case viaSocket:
|
|
msgs := []Message{tt.message}
|
|
writeMsgsToSocketTest(tstConf, msgs, t)
|
|
|
|
}
|
|
|
|
switch tt.containsOrEquals {
|
|
case REQTestEquals:
|
|
result := <-tstSrv.errorKernel.testCh
|
|
resStr := string(result)
|
|
resStr = strings.TrimSuffix(resStr, "\n")
|
|
result = []byte(resStr)
|
|
|
|
if !bytes.Equal(result, tt.want) {
|
|
t.Fatalf(" \U0001F631 [FAILED] :%v : want: %v, got: %v\n", tt.info, string(tt.want), string(result))
|
|
}
|
|
t.Logf(" \U0001f600 [SUCCESS] : %v\n", tt.info)
|
|
|
|
case REQTestContains:
|
|
result := <-tstSrv.errorKernel.testCh
|
|
resStr := string(result)
|
|
resStr = strings.TrimSuffix(resStr, "\n")
|
|
result = []byte(resStr)
|
|
|
|
if !strings.Contains(string(result), string(tt.want)) {
|
|
t.Fatalf(" \U0001F631 [FAILED] :%v : want: %v, got: %v\n", tt.info, string(tt.want), string(result))
|
|
}
|
|
t.Logf(" \U0001f600 [SUCCESS] : %v\n", tt.info)
|
|
|
|
case fileContains:
|
|
resultFile := filepath.Join(tstConf.SubscribersDataFolder, tt.message.Directory, string(tt.message.FromNode), tt.message.FileName)
|
|
|
|
found, err := findStringInFileTest(string(tt.want), resultFile)
|
|
if err != nil || found == false {
|
|
t.Fatalf(" \U0001F631 [FAILED] : %v: %v\n", tt.info, err)
|
|
|
|
}
|
|
|
|
t.Logf(" \U0001f600 [SUCCESS] : %v\n", tt.info)
|
|
}
|
|
}
|
|
|
|
// --- Other REQ tests that does not fit well into the general table above.
|
|
|
|
checkREQTailFileTest(tstConf, t, tstTempDir)
|
|
checkMetricValuesTest(tstSrv, t)
|
|
checkErrorKernelMalformedJSONtest(tstConf, t)
|
|
t.Log("*******starting with checkREQCopySrc\n")
|
|
checkREQCopySrc(tstConf, t, tstTempDir)
|
|
}
|
|
|
|
// Check the tailing of files type.
|
|
func checkREQTailFileTest(conf *Configuration, t *testing.T, tmpDir string) error {
|
|
// Create a file with some content.
|
|
fp := filepath.Join(tmpDir, "test.file")
|
|
fh, err := os.OpenFile(fp, os.O_APPEND|os.O_RDWR|os.O_CREATE|os.O_SYNC, 0660)
|
|
if err != nil {
|
|
return fmt.Errorf(" * failed: unable to open temporary file: %v", err)
|
|
}
|
|
defer fh.Close()
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
// Write content to the file with specified intervals.
|
|
go func() {
|
|
for i := 1; i <= 10; i++ {
|
|
_, err = fh.Write([]byte("some file content\n"))
|
|
if err != nil {
|
|
fmt.Printf(" * failed: writing to temporary file: %v\n", err)
|
|
}
|
|
fh.Sync()
|
|
time.Sleep(time.Millisecond * 500)
|
|
|
|
// Check if we've received a done, else default to continuing.
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
// no done received, we're continuing.
|
|
}
|
|
|
|
}
|
|
}()
|
|
|
|
s := `[
|
|
{
|
|
"directory": "tail-files",
|
|
"fileName": "fileName.result",
|
|
"toNode": "central",
|
|
"methodArgs": ["` + fp + `"],
|
|
"method":"tailFile",
|
|
"ACKTimeout":5,
|
|
"retries":3,
|
|
"methodTimeout": 10
|
|
}
|
|
]`
|
|
|
|
writeToSocketTest(conf, s, t)
|
|
|
|
resultFile := filepath.Join(conf.SubscribersDataFolder, "tail-files", "central", "fileName.result")
|
|
|
|
// Wait n times for result file to be created.
|
|
n := 50
|
|
for i := 0; i <= n; i++ {
|
|
_, err := os.Stat(resultFile)
|
|
if os.IsNotExist(err) {
|
|
time.Sleep(time.Millisecond * 100)
|
|
continue
|
|
}
|
|
|
|
if os.IsNotExist(err) && i >= n {
|
|
cancel()
|
|
return fmt.Errorf(" \U0001F631 [FAILED] : checkREQTailFileTest: no result file created for request within the given time")
|
|
}
|
|
}
|
|
|
|
cancel()
|
|
|
|
_, err = findStringInFileTest("some file content", resultFile)
|
|
if err != nil {
|
|
return fmt.Errorf(" \U0001F631 [FAILED] : checkREQTailFileTest: %v", err)
|
|
}
|
|
|
|
t.Logf(" \U0001f600 [SUCCESS] : checkREQTailFileTest\n")
|
|
return nil
|
|
}
|
|
|
|
// Check the file copier.
|
|
func checkREQCopySrc(conf *Configuration, t *testing.T, tmpDir string) error {
|
|
testFiles := 5
|
|
|
|
for i := 1; i <= testFiles; i++ {
|
|
|
|
// Create a file with some content.
|
|
srcFileName := fmt.Sprintf("copysrc%v.file", i)
|
|
srcfp := filepath.Join(tmpDir, srcFileName)
|
|
fh, err := os.OpenFile(srcfp, os.O_APPEND|os.O_RDWR|os.O_CREATE|os.O_SYNC, 0660)
|
|
if err != nil {
|
|
t.Fatalf(" \U0001F631 [FAILED] : checkREQCopySrc: unable to open temporary file: %v", err)
|
|
}
|
|
defer fh.Close()
|
|
|
|
// Write content to the file.
|
|
|
|
_, err = fh.Write([]byte("some file content\n"))
|
|
if err != nil {
|
|
t.Fatalf(" \U0001F631 [FAILED] : checkREQCopySrc: writing to temporary file: %v\n", err)
|
|
}
|
|
|
|
dstFileName := fmt.Sprintf("copydst%v.file", i)
|
|
dstfp := filepath.Join(tmpDir, dstFileName)
|
|
|
|
s := `[
|
|
{
|
|
"toNode": "central",
|
|
"method":"copySrc",
|
|
"methodArgs": ["` + srcfp + `","central","` + dstfp + `","20","10"],
|
|
"ACKTimeout":5,
|
|
"retries":3,
|
|
"methodTimeout": 10,
|
|
"fileName": "filecopy2.log"
|
|
}
|
|
]`
|
|
|
|
writeToSocketTest(conf, s, t)
|
|
|
|
// Wait n times for result file to be created.
|
|
n := 50
|
|
for i := 0; i <= n; i++ {
|
|
_, err := os.Stat(dstfp)
|
|
if os.IsNotExist(err) {
|
|
time.Sleep(time.Millisecond * 100)
|
|
continue
|
|
}
|
|
|
|
if os.IsNotExist(err) && i >= n {
|
|
t.Fatalf(" \U0001F631 [FAILED] : checkREQCopySrc: no result file created for request within the given time")
|
|
}
|
|
}
|
|
|
|
t.Logf(" \U0001f600 [SUCCESS] : src=%v, dst=%v", srcfp, dstfp)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func checkMetricValuesTest(ctrlServer *server, t *testing.T) error {
|
|
mfs, err := ctrlServer.metrics.promRegistry.Gather()
|
|
if err != nil {
|
|
return fmt.Errorf("error: promRegistry.gathering: %v", mfs)
|
|
}
|
|
|
|
if len(mfs) <= 0 {
|
|
return fmt.Errorf("error: promRegistry.gathering: did not find any metric families: %v", mfs)
|
|
}
|
|
|
|
found := false
|
|
for _, mf := range mfs {
|
|
if mf.GetName() == "ctrl_processes_total" {
|
|
found = true
|
|
|
|
m := mf.GetMetric()
|
|
|
|
if m[0].Gauge.GetValue() <= 0 {
|
|
return fmt.Errorf("error: promRegistry.gathering: did not find any running processes in metric for processes_total : %v", m[0].Gauge.GetValue())
|
|
}
|
|
}
|
|
}
|
|
|
|
if !found {
|
|
return fmt.Errorf("error: promRegistry.gathering: did not find specified metric processes_total")
|
|
}
|
|
|
|
t.Logf(" \U0001f600 [SUCCESS] : checkMetricValuesTest")
|
|
|
|
return nil
|
|
}
|
|
|
|
// Check errorKernel
|
|
func checkErrorKernelMalformedJSONtest(conf *Configuration, t *testing.T) error {
|
|
|
|
// JSON message with error, missing brace.
|
|
m := `[
|
|
{
|
|
"directory": "some dir",
|
|
"fileName":"someext",
|
|
"toNode": "somenode",
|
|
"data": ["some data"],
|
|
"method": "errorLog"
|
|
missing brace here.....
|
|
]`
|
|
|
|
writeToSocketTest(conf, m, t)
|
|
|
|
resultFile := filepath.Join(conf.SubscribersDataFolder, "errorLog", "errorCentral", "error.log")
|
|
|
|
// Wait n times for error file to be created.
|
|
n := 50
|
|
for i := 0; i <= n; i++ {
|
|
_, err := os.Stat(resultFile)
|
|
if os.IsNotExist(err) {
|
|
time.Sleep(time.Millisecond * 100)
|
|
continue
|
|
}
|
|
|
|
if os.IsNotExist(err) && i >= n {
|
|
return fmt.Errorf(" \U0001F631 [FAILED] : checkErrorKernelMalformedJSONtest: no result file created for request within the given time")
|
|
}
|
|
}
|
|
|
|
// Start checking if the result file is being updated.
|
|
chUpdated := make(chan bool)
|
|
go checkFileUpdated(resultFile, chUpdated)
|
|
|
|
// We wait 5 seconds for an update, or else we fail.
|
|
ticker := time.NewTicker(time.Second * 5)
|
|
|
|
for {
|
|
select {
|
|
case <-chUpdated:
|
|
// We got an update, so we continue to check if we find the string we're
|
|
// looking for.
|
|
found, err := findStringInFileTest("error: malformed json", resultFile)
|
|
if !found && err != nil {
|
|
return fmt.Errorf(" \U0001F631 [FAILED] : checkErrorKernelMalformedJSONtest: %v", err)
|
|
}
|
|
|
|
if !found && err == nil {
|
|
continue
|
|
}
|
|
|
|
if found {
|
|
t.Logf(" \U0001f600 [SUCCESS] : checkErrorKernelMalformedJSONtest")
|
|
return nil
|
|
}
|
|
case <-ticker.C:
|
|
return fmt.Errorf(" * failed: did not get an update in the errorKernel log file")
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check if file are getting updated with new content.
|
|
func checkFileUpdated(fileRealPath string, fileUpdated chan bool) {
|
|
watcher, err := fsnotify.NewWatcher()
|
|
if err != nil {
|
|
log.Println("Failed fsnotify.NewWatcher")
|
|
return
|
|
}
|
|
defer watcher.Close()
|
|
|
|
done := make(chan bool)
|
|
go func() {
|
|
//Give a true value to updated so it reads the file the first time.
|
|
fileUpdated <- true
|
|
for {
|
|
select {
|
|
case event := <-watcher.Events:
|
|
log.Println("event:", event)
|
|
if event.Op&fsnotify.Write == fsnotify.Write {
|
|
log.Println("modified file:", event.Name)
|
|
//testing with an update chan to get updates
|
|
fileUpdated <- true
|
|
}
|
|
case err := <-watcher.Errors:
|
|
log.Println("error:", err)
|
|
}
|
|
}
|
|
}()
|
|
|
|
err = watcher.Add(fileRealPath)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
<-done
|
|
}
|
|
|
|
// Check if a file contains the given string.
|
|
func findStringInFileTest(want string, fileName string) (bool, error) {
|
|
// Wait n seconds for the results file to be created
|
|
n := 50
|
|
|
|
for i := 0; i <= n; i++ {
|
|
_, err := os.Stat(fileName)
|
|
if os.IsNotExist(err) {
|
|
time.Sleep(time.Millisecond * 100)
|
|
continue
|
|
}
|
|
|
|
if os.IsNotExist(err) && i >= n {
|
|
return false, fmt.Errorf(" * failed: no result file created for request within the given time\n")
|
|
}
|
|
}
|
|
|
|
fh, err := os.Open(fileName)
|
|
if err != nil {
|
|
return false, fmt.Errorf(" * failed: could not open result file: %v", err)
|
|
}
|
|
|
|
result, err := io.ReadAll(fh)
|
|
if err != nil {
|
|
return false, fmt.Errorf(" * failed: could not read result file: %v", err)
|
|
}
|
|
|
|
found := strings.Contains(string(result), want)
|
|
if !found {
|
|
return false, nil
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
// Write message to socket for testing purposes.
|
|
func writeToSocketTest(conf *Configuration, messageText string, t *testing.T) {
|
|
socket, err := net.Dial("unix", filepath.Join(conf.SocketFolder, "ctrl.sock"))
|
|
if err != nil {
|
|
t.Fatalf(" * failed: could to open socket file for writing: %v\n", err)
|
|
}
|
|
defer socket.Close()
|
|
|
|
_, err = socket.Write([]byte(messageText))
|
|
if err != nil {
|
|
t.Fatalf(" * failed: could not write to socket: %v\n", err)
|
|
}
|
|
|
|
}
|