mirror of
https://github.com/postmannen/ctrl.git
synced 2025-01-07 12:59:15 +00:00
9dda8ac68d
renamed HandlerFunc to Handler added so you can define requests without a method, and just have a procFunc. added startup of central key processes split up configuration flags for central acl's and keys.
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.IsCentralAcl = 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)
|
|
}
|
|
|
|
}
|