2021-06-14 04:49:47 +00:00
package steward
import (
2021-06-14 15:07:31 +00:00
"bufio"
2021-06-16 04:03:09 +00:00
"encoding/json"
2021-06-14 04:49:47 +00:00
"flag"
"fmt"
2021-06-15 20:21:18 +00:00
"log"
2021-06-14 04:49:47 +00:00
"os"
2021-06-15 20:21:18 +00:00
"reflect"
"strconv"
2021-06-16 04:03:09 +00:00
"strings"
2021-06-16 19:11:50 +00:00
"time"
2021-06-14 04:49:47 +00:00
2021-06-24 09:35:41 +00:00
"github.com/gdamore/tcell/v2"
2021-06-14 04:49:47 +00:00
"github.com/rivo/tview"
2021-06-29 06:21:42 +00:00
"github.com/RaaLabs/steward"
2021-06-14 04:49:47 +00:00
)
type Stew struct {
stewardSocket string
}
func NewStew ( ) ( * Stew , error ) {
stewardSocket := flag . String ( "stewardSocket" , "/usr/local/steward/tmp/steward.sock" , "specify the full path of the steward socket file" )
flag . Parse ( )
_ , err := os . Stat ( * stewardSocket )
if err != nil {
return nil , fmt . Errorf ( "error: specify the full path to the steward.sock file: %v" , err )
}
s := Stew {
stewardSocket : * stewardSocket ,
}
return & s , nil
}
func ( s * Stew ) Start ( ) error {
2021-06-29 05:13:48 +00:00
c := newPageMessage ( )
2021-06-23 12:55:39 +00:00
err := c . start ( )
2021-06-14 04:49:47 +00:00
if err != nil {
return fmt . Errorf ( "error: console failed: %v" , err )
}
return nil
}
2021-06-14 15:07:31 +00:00
// ---------------------------------------------------
2021-06-29 05:13:48 +00:00
type pageMessage struct {
2021-06-24 09:35:41 +00:00
flex * tview . Flex
msgInputForm * tview . Form
2021-06-25 05:51:02 +00:00
msgOutputForm * tview . TextView
2021-06-24 09:35:41 +00:00
logForm * tview . TextView
app * tview . Application
2021-06-23 12:55:39 +00:00
}
2021-06-29 05:13:48 +00:00
func newPageMessage ( ) * pageMessage {
p := pageMessage { }
p . app = tview . NewApplication ( )
2021-06-23 12:55:39 +00:00
2021-06-29 05:13:48 +00:00
p . msgInputForm = tview . NewForm ( )
p . msgInputForm . SetBorder ( true ) . SetTitle ( "Request values" ) . SetTitleAlign ( tview . AlignLeft )
2021-06-24 09:35:41 +00:00
2021-06-29 05:13:48 +00:00
p . msgOutputForm = tview . NewTextView ( )
p . msgOutputForm . SetBorder ( true ) . SetTitle ( "Message output" ) . SetTitleAlign ( tview . AlignLeft )
p . msgOutputForm . SetChangedFunc ( func ( ) {
2021-06-25 05:51:02 +00:00
// Will cause the log window to be redrawn as soon as
// new output are detected.
2021-06-29 05:13:48 +00:00
p . app . Draw ( )
2021-06-25 05:51:02 +00:00
} )
2021-06-14 15:07:31 +00:00
2021-06-29 05:13:48 +00:00
p . logForm = tview . NewTextView ( )
p . logForm . SetBorder ( true ) . SetTitle ( "Log/Status" ) . SetTitleAlign ( tview . AlignLeft )
p . logForm . SetChangedFunc ( func ( ) {
2021-06-24 09:35:41 +00:00
// Will cause the log window to be redrawn as soon as
// new output are detected.
2021-06-29 05:13:48 +00:00
p . app . Draw ( )
2021-06-23 12:55:39 +00:00
} )
// Create a flex layout.
2021-06-24 09:35:41 +00:00
//
// First create the outer flex layout.
2021-06-29 05:13:48 +00:00
p . flex = tview . NewFlex ( ) . SetDirection ( tview . FlexRow ) .
2021-06-24 09:35:41 +00:00
// Add the top windows with columns.
AddItem ( tview . NewFlex ( ) . SetDirection ( tview . FlexColumn ) .
2021-06-29 05:13:48 +00:00
AddItem ( p . msgInputForm , 0 , 10 , false ) .
AddItem ( p . msgOutputForm , 0 , 10 , false ) ,
2021-06-24 09:35:41 +00:00
0 , 10 , false ) .
// Add the bottom log window.
AddItem ( tview . NewFlex ( ) .
2021-06-29 05:13:48 +00:00
AddItem ( p . logForm , 0 , 2 , false ) ,
2021-06-24 09:35:41 +00:00
0 , 2 , false )
2021-06-29 05:13:48 +00:00
return & p
2021-06-23 12:55:39 +00:00
}
// Will start the console.
2021-06-29 05:13:48 +00:00
func ( p * pageMessage ) start ( ) error {
2021-06-17 06:28:10 +00:00
// Check that the message struct used within stew are up to date, and
// consistent with the fields used in the main Steward message file.
// If it throws an error here we need to update the msg struct type,
// or add a case for the field to except.
2021-06-23 12:55:39 +00:00
err := compareMsgAndMessage ( )
2021-06-17 06:28:10 +00:00
if err != nil {
log . Printf ( "%v\n" , err )
os . Exit ( 1 )
}
2021-06-29 05:13:48 +00:00
p . drawMsgForm ( )
2021-06-14 04:49:47 +00:00
2021-06-29 05:13:48 +00:00
if err := p . app . SetRoot ( p . flex , true ) . SetFocus ( p . msgInputForm ) . EnableMouse ( true ) . Run ( ) ; err != nil {
2021-06-15 20:21:18 +00:00
panic ( err )
2021-06-14 15:07:31 +00:00
}
2021-06-14 04:49:47 +00:00
2021-06-15 20:21:18 +00:00
return nil
}
2021-06-14 15:07:31 +00:00
2021-06-17 06:28:10 +00:00
// Will check and compare all the fields of the main message struct
// used in Steward, and the message struct used in Stew that they are
// equal.
// If they are not equal an error will be returned to the user with
// the name of the field that was missing in the Stew message struct.
//
// Some of the fields in the Steward Message struct are used by the
// system for control, and not needed when creating an initial message
// template, and we can add case statements for those fields below
// that we do not wan't to check.
2021-06-23 12:55:39 +00:00
func compareMsgAndMessage ( ) error {
2021-06-29 06:21:42 +00:00
stewardMessage := steward . Message { }
2021-06-23 12:55:39 +00:00
stewMsg := msg { }
2021-06-17 06:28:10 +00:00
2021-06-23 12:55:39 +00:00
stewardRefVal := reflect . ValueOf ( stewardMessage )
stewRefVal := reflect . ValueOf ( stewMsg )
2021-06-17 06:28:10 +00:00
2021-06-23 12:55:39 +00:00
// Loop trough all the fields of the Message struct.
2021-06-17 06:28:10 +00:00
for i := 0 ; i < stewardRefVal . NumField ( ) ; i ++ {
found := false
for ii := 0 ; ii < stewRefVal . NumField ( ) ; ii ++ {
if stewardRefVal . Type ( ) . Field ( i ) . Name == stewRefVal . Type ( ) . Field ( ii ) . Name {
found = true
break
}
}
// Case statements for the fields we don't care about for
// the message template.
if ! found {
switch stewardRefVal . Type ( ) . Field ( i ) . Name {
case "ID" :
// Not used in message template.
2021-06-17 07:36:42 +00:00
case "FromNode" :
2021-06-17 06:28:10 +00:00
// Not used in message template.
case "PreviousMessage" :
// Not used in message template.
case "done" :
// Not used in message template.
default :
return fmt . Errorf ( "error: %v within the steward Message struct were not found in the stew msg struct" , stewardRefVal . Type ( ) . Field ( i ) . Name )
}
}
}
return nil
}
2021-06-29 05:13:48 +00:00
func ( p * pageMessage ) drawMsgForm ( ) error {
2021-06-17 06:28:10 +00:00
m := msg { }
2021-06-14 15:07:31 +00:00
2021-06-17 06:28:10 +00:00
// Loop trough all the fields of the Message struct, and create
// a an input field or dropdown selector for each field.
// If a field of the struct is not defined below, it will be
// created a "no defenition" element, so it we can easily spot
// Message fields who miss an item in the form.
2021-06-23 12:55:39 +00:00
//
// INFO: The reason that reflect are being used here is to have
// a simple way of detecting that we are creating form fields
// for all the fields in the struct. If we have forgot'en one
// it will create a "no case" field in the console, to easily
// detect that a struct field are missing a defenition below.
2021-06-28 05:28:14 +00:00
2021-06-21 01:11:50 +00:00
mRefVal := reflect . ValueOf ( m )
2021-06-15 20:21:18 +00:00
for i := 0 ; i < mRefVal . NumField ( ) ; i ++ {
var err error
values := [ ] string { "1" , "2" }
2021-06-14 15:07:31 +00:00
2021-06-28 05:28:14 +00:00
fieldName := mRefVal . Type ( ) . Field ( i ) . Name
switch fieldName {
2021-06-15 20:21:18 +00:00
case "ToNode" :
// Get nodes from file.
values , err = getNodeNames ( "nodeslist.cfg" )
if err != nil {
return err
}
2021-06-24 09:35:41 +00:00
item := tview . NewDropDown ( )
item . SetLabelColor ( tcell . ColorIndianRed )
2021-06-28 05:28:14 +00:00
item . SetLabel ( fieldName ) . SetOptions ( values , nil )
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddFormItem ( item )
2021-06-24 09:35:41 +00:00
//c.msgForm.AddDropDown(mRefVal.Type().Field(i).Name, values, 0, nil).SetItemPadding(1)
2021-06-15 20:21:18 +00:00
case "ID" :
2021-06-24 09:35:41 +00:00
// This value is automatically assigned by steward.
2021-06-15 20:21:18 +00:00
case "Data" :
value := ` "bash","-c","..." `
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , value , 30 , nil , nil )
2021-06-15 20:21:18 +00:00
case "Method" :
2021-06-29 06:21:42 +00:00
var m steward . Method
2021-06-15 20:21:18 +00:00
ma := m . GetMethodsAvailable ( )
values := [ ] string { }
2021-06-29 06:21:42 +00:00
for k := range ma . Methodhandlers {
2021-06-15 20:21:18 +00:00
values = append ( values , string ( k ) )
}
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddDropDown ( fieldName , values , 0 , nil ) . SetItemPadding ( 1 )
2021-06-15 20:21:18 +00:00
case "ReplyMethod" :
2021-06-29 06:21:42 +00:00
var m steward . Method
rm := m . GetReplyMethods ( )
2021-06-15 20:21:18 +00:00
values := [ ] string { }
2021-06-16 19:38:33 +00:00
for _ , k := range rm {
2021-06-15 20:21:18 +00:00
values = append ( values , string ( k ) )
}
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddDropDown ( fieldName , values , 0 , nil ) . SetItemPadding ( 1 )
2021-06-15 20:21:18 +00:00
case "ACKTimeout" :
value := 30
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , fmt . Sprintf ( "%d" , value ) , 30 , validateInteger , nil )
2021-06-15 20:21:18 +00:00
case "Retries" :
value := 1
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , fmt . Sprintf ( "%d" , value ) , 30 , validateInteger , nil )
2021-06-15 20:21:18 +00:00
case "ReplyACKTimeout" :
value := 30
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , fmt . Sprintf ( "%d" , value ) , 30 , validateInteger , nil )
2021-06-15 20:21:18 +00:00
case "ReplyRetries" :
value := 1
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , fmt . Sprintf ( "%d" , value ) , 30 , validateInteger , nil )
2021-06-15 20:21:18 +00:00
case "MethodTimeout" :
value := 120
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , fmt . Sprintf ( "%d" , value ) , 30 , validateInteger , nil )
2021-06-15 20:21:18 +00:00
case "Directory" :
value := "/some-dir/"
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , value , 30 , nil , nil )
2021-06-15 20:21:18 +00:00
case "FileExtension" :
value := ".log"
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddInputField ( fieldName , value , 30 , nil , nil )
2021-06-17 07:36:42 +00:00
case "Operation" :
2021-06-23 12:55:39 +00:00
// Prepare the selectedFunc that will be triggered for the operations
// when a field in the dropdown is selected.
// This selectedFunc will generate the sub fields of the selected
// operation, and will also remove any previously drawn operation
// sub fields from the current form.
2021-06-21 01:11:50 +00:00
values := [ ] string { "ps" , "startProc" , "stopProc" }
selectedFunc := func ( option string , optionIndex int ) {
2021-06-23 12:55:39 +00:00
// Prepare the map structure that knows what tview items the
// operation should contain.
// Then we can pick from this map later, to know what fields
// to draw, and also which fields to delete if another operation
// is selected from the dropdown.
type formItem struct {
label string
formItem tview . FormItem
}
operationFormItems := map [ string ] [ ] formItem { }
operationFormItems [ "ps" ] = nil
operationFormItems [ "startProc" ] = func ( ) [ ] formItem {
formItems := [ ] formItem { }
// Get values values to be used for the "Method" dropdown.
2021-06-29 06:21:42 +00:00
var m steward . Method
2021-06-23 12:55:39 +00:00
ma := m . GetMethodsAvailable ( )
values := [ ] string { }
2021-06-29 06:21:42 +00:00
for k := range ma . Methodhandlers {
2021-06-23 12:55:39 +00:00
values = append ( values , string ( k ) )
2021-06-21 01:11:50 +00:00
}
2021-06-23 12:55:39 +00:00
// Create the individual form items, and append them to the
2021-06-28 05:28:14 +00:00
// formItems slice to be drawn later.
2021-06-23 12:55:39 +00:00
{
label := "startProc Method"
item := tview . NewDropDown ( )
2021-06-24 09:35:41 +00:00
item . SetLabel ( label ) . SetOptions ( values , nil ) . SetLabelWidth ( 25 )
2021-06-23 12:55:39 +00:00
formItems = append ( formItems , formItem { label : label , formItem : item } )
}
{
label := "startProc AllowedNodes"
item := tview . NewInputField ( )
2021-06-24 09:35:41 +00:00
item . SetLabel ( label ) . SetFieldWidth ( 30 ) . SetLabelWidth ( 25 )
2021-06-23 12:55:39 +00:00
formItems = append ( formItems , formItem { label : label , formItem : item } )
2021-06-21 01:11:50 +00:00
}
2021-06-23 12:55:39 +00:00
return formItems
} ( )
operationFormItems [ "stopProc" ] = func ( ) [ ] formItem {
formItems := [ ] formItem { }
// RecevingNode node `json:"receivingNode"`
// Method Method `json:"method"`
// Kind processKind `json:"kind"`
// ID int `json:"id"`
// Get nodes from file.
values , err = getNodeNames ( "nodeslist.cfg" )
if err != nil {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v: error: unable to get nodes\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) )
2021-06-23 12:55:39 +00:00
return nil
}
{
label := "stopProc ToNode"
item := tview . NewDropDown ( )
2021-06-24 09:35:41 +00:00
item . SetLabel ( label ) . SetOptions ( values , nil ) . SetLabelWidth ( 25 )
2021-06-23 12:55:39 +00:00
formItems = append ( formItems , formItem { label : label , formItem : item } )
2021-06-21 01:11:50 +00:00
2021-06-23 12:55:39 +00:00
}
// Get values values to be used for the "Method" dropdown.
2021-06-29 06:21:42 +00:00
var m steward . Method
2021-06-23 12:55:39 +00:00
ma := m . GetMethodsAvailable ( )
values := [ ] string { }
2021-06-29 06:21:42 +00:00
for k := range ma . Methodhandlers {
2021-06-23 12:55:39 +00:00
values = append ( values , string ( k ) )
}
// Create the individual form items, and append them to the
// formItems slice.
{
label := "stopProc Method"
item := tview . NewDropDown ( )
2021-06-24 09:35:41 +00:00
item . SetLabel ( label ) . SetOptions ( values , nil ) . SetLabelWidth ( 25 )
2021-06-23 12:55:39 +00:00
formItems = append ( formItems , formItem { label : label , formItem : item } )
}
processKind := [ ] string { "publisher" , "subscriber" }
{
label := "stopProc processKind"
item := tview . NewDropDown ( )
2021-06-24 09:35:41 +00:00
item . SetLabel ( label ) . SetOptions ( processKind , nil ) . SetLabelWidth ( 25 )
2021-06-23 12:55:39 +00:00
formItems = append ( formItems , formItem { label : label , formItem : item } )
}
{
label := "stopProc ID"
item := tview . NewInputField ( )
2021-06-24 09:35:41 +00:00
item . SetLabel ( label ) . SetFieldWidth ( 30 ) . SetLabelWidth ( 25 )
2021-06-23 12:55:39 +00:00
formItems = append ( formItems , formItem { label : label , formItem : item } )
}
return formItems
} ( )
itemDraw := func ( label string ) {
// Delete previously drawn sub operation form items.
for _ , vSlice := range operationFormItems {
for _ , v := range vSlice {
2021-06-29 05:13:48 +00:00
i := p . msgInputForm . GetFormItemIndex ( v . label )
2021-06-23 12:55:39 +00:00
if i > - 1 {
2021-06-29 05:13:48 +00:00
p . msgInputForm . RemoveFormItem ( i )
2021-06-23 12:55:39 +00:00
}
}
}
// Get and draw the form items to the form.
formItems := operationFormItems [ label ]
for _ , v := range formItems {
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddFormItem ( v . formItem )
2021-06-23 12:55:39 +00:00
}
}
switch option {
case "ps" :
2021-06-24 09:35:41 +00:00
itemDraw ( "ps" )
2021-06-23 12:55:39 +00:00
case "startProc" :
itemDraw ( "startProc" )
case "stopProc" :
itemDraw ( "stopProc" )
2021-06-21 01:11:50 +00:00
default :
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v: error: missing menu item for %v\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) , option )
2021-06-21 01:11:50 +00:00
}
}
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddDropDown ( fieldName , values , 0 , selectedFunc ) . SetItemPadding ( 1 )
2021-06-15 20:21:18 +00:00
default :
2021-06-17 06:28:10 +00:00
// Add a no definition fields to the form if a a field within the
// struct were missing an action above, so we can easily detect
// if there is missing a case action for one of the struct fields.
2021-06-29 05:13:48 +00:00
p . msgInputForm . AddDropDown ( "error: no case for: " + fieldName , values , 0 , nil ) . SetItemPadding ( 1 )
2021-06-14 15:07:31 +00:00
}
2021-06-14 04:49:47 +00:00
}
2021-06-23 12:55:39 +00:00
// Add Buttons below the message fields. Like Generate and Exit.
2021-06-29 05:13:48 +00:00
p . msgInputForm .
2021-06-23 12:55:39 +00:00
// Add a generate button, which when pressed will loop through all the
// message form items, and if found fill the value into a msg struct,
// and at last write it to a file.
//
// TODO: Should also add a write directly to socket here.
2021-06-25 05:51:02 +00:00
AddButton ( "generate to console" , func ( ) {
2021-06-23 12:55:39 +00:00
// ---
2021-06-29 06:21:42 +00:00
opCmdStartProc := steward . OpCmdStartProc { }
opCmdStopProc := steward . OpCmdStopProc { }
2021-06-23 12:55:39 +00:00
// ---
2021-06-25 05:51:02 +00:00
// fh, err := os.Create("message.json")
// if err != nil {
// log.Fatalf("error: failed to create test.log file: %v\n", err)
// }
// defer fh.Close()
2021-06-29 05:13:48 +00:00
p . msgOutputForm . Clear ( )
fh := p . msgOutputForm
2021-06-15 20:21:18 +00:00
2021-06-16 04:03:09 +00:00
m := msg { }
// Loop trough all the form fields
2021-06-29 05:13:48 +00:00
for i := 0 ; i < p . msgInputForm . GetFormItemCount ( ) ; i ++ {
fi := p . msgInputForm . GetFormItem ( i )
2021-06-16 04:03:09 +00:00
label , value := getLabelAndValue ( fi )
switch label {
case "ToNode" :
2021-06-25 05:51:02 +00:00
if value == "" {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: missing ToNode \n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) )
2021-06-25 05:51:02 +00:00
return
}
2021-06-29 06:21:42 +00:00
m . ToNode = steward . Node ( value )
2021-06-16 04:03:09 +00:00
case "Data" :
2021-06-16 14:14:43 +00:00
// Split the comma separated string into a
// and remove the start and end ampersand.
2021-06-16 04:03:09 +00:00
sp := strings . Split ( value , "," )
2021-06-16 19:11:50 +00:00
2021-06-16 14:14:43 +00:00
var data [ ] string
for _ , v := range sp {
2021-06-16 19:11:50 +00:00
// Check if format is correct, return if not.
2021-06-16 14:14:43 +00:00
pre := strings . HasPrefix ( v , "\"" )
2021-06-16 19:11:50 +00:00
suf := strings . HasSuffix ( v , "\"" )
if ! pre || ! suf {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: missing or malformed format for command, should be \"cmd\",\"arg1\",\"arg2\" ...\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) )
2021-06-16 19:11:50 +00:00
return
2021-06-16 14:14:43 +00:00
}
2021-06-16 19:11:50 +00:00
// Remove leading and ending ampersand.
v = v [ 1 : ]
v = strings . TrimSuffix ( v , "\"" )
2021-06-16 14:14:43 +00:00
data = append ( data , v )
}
m . Data = data
2021-06-16 04:03:09 +00:00
case "Method" :
2021-06-29 06:21:42 +00:00
m . Method = steward . Method ( value )
2021-06-16 04:03:09 +00:00
case "ReplyMethod" :
2021-06-29 06:21:42 +00:00
m . ReplyMethod = steward . Method ( value )
2021-06-16 04:03:09 +00:00
case "ACKTimeout" :
v , _ := strconv . Atoi ( value )
m . ACKTimeout = v
case "Retries" :
v , _ := strconv . Atoi ( value )
m . Retries = v
case "ReplyACKTimeout" :
v , _ := strconv . Atoi ( value )
m . ReplyACKTimeout = v
case "ReplyRetries" :
v , _ := strconv . Atoi ( value )
m . ReplyRetries = v
case "MethodTimeout" :
v , _ := strconv . Atoi ( value )
m . MethodTimeout = v
case "Directory" :
m . Directory = value
case "FileExtension" :
m . FileExtension = value
2021-06-17 07:36:42 +00:00
case "Operation" :
2021-06-23 12:55:39 +00:00
// We need to check what type of operation it is, and pick
// the correct struct type, and fill it with values
switch value {
case "ps" :
//TODO
case "startProc" :
m . Operation = & opCmdStartProc
case "stopProc" :
m . Operation = & opCmdStopProc
default :
m . Operation = nil
}
case "startProc Method" :
2021-06-25 05:51:02 +00:00
if value == "" {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: missing startProc Method\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) )
2021-06-25 05:51:02 +00:00
return
}
2021-06-29 06:21:42 +00:00
opCmdStartProc . Method = steward . Method ( value )
2021-06-23 12:55:39 +00:00
case "startProc AllowedNodes" :
// Split the comma separated string into a
// and remove the start and end ampersand.
sp := strings . Split ( value , "," )
2021-06-29 06:21:42 +00:00
var allowedNodes [ ] steward . Node
2021-06-23 12:55:39 +00:00
for _ , v := range sp {
// Check if format is correct, return if not.
2021-06-25 05:51:02 +00:00
pre := strings . HasPrefix ( v , "\"" ) || ! strings . HasPrefix ( v , "," ) || ! strings . HasPrefix ( v , "\"," ) || ! strings . HasPrefix ( v , ",\"" )
2021-06-23 12:55:39 +00:00
suf := strings . HasSuffix ( v , "\"" )
if ! pre || ! suf {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: malformed format for command, should be \"cmd\",\"arg1\",\"arg2\" ...\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) )
2021-06-23 12:55:39 +00:00
return
}
// Remove leading and ending ampersand.
v = v [ 1 : ]
v = strings . TrimSuffix ( v , "\"" )
2021-06-29 06:21:42 +00:00
allowedNodes = append ( allowedNodes , steward . Node ( v ) )
2021-06-23 12:55:39 +00:00
}
opCmdStartProc . AllowedNodes = allowedNodes
2021-06-17 06:28:10 +00:00
default :
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: did not find case defenition for how to handle the \"%v\" within the switch statement\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) , label )
2021-06-15 20:21:18 +00:00
}
}
2021-06-16 04:03:09 +00:00
msgs := [ ] msg { }
msgs = append ( msgs , m )
2021-06-25 05:51:02 +00:00
msgsIndented , err := json . MarshalIndent ( msgs , "" , " " )
if err != nil {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: jsonIndent failed: %v\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) , err )
2021-06-25 05:51:02 +00:00
}
_ , err = fh . Write ( msgsIndented )
if err != nil {
2021-06-29 05:13:48 +00:00
fmt . Fprintf ( p . logForm , "%v : error: write to fh failed: %v\n" , time . Now ( ) . Format ( "Mon Jan _2 15:04:05 2006" ) , err )
2021-06-25 05:51:02 +00:00
}
2021-06-15 20:21:18 +00:00
} ) .
2021-06-23 12:55:39 +00:00
// Add exit button.
2021-06-15 20:21:18 +00:00
AddButton ( "exit" , func ( ) {
2021-06-29 05:13:48 +00:00
p . app . Stop ( )
2021-06-15 20:21:18 +00:00
} )
2021-06-29 05:13:48 +00:00
p . app . SetFocus ( p . msgInputForm )
2021-06-14 04:49:47 +00:00
return nil
}
2021-06-16 14:14:43 +00:00
// Will return the Label And the text Value of an input or dropdown form field.
2021-06-16 04:03:09 +00:00
func getLabelAndValue ( fi tview . FormItem ) ( string , string ) {
var label string
var value string
switch v := fi . ( type ) {
case * tview . InputField :
value = v . GetText ( )
label = v . GetLabel ( )
case * tview . DropDown :
label = v . GetLabel ( )
_ , value = v . GetCurrentOption ( )
}
return label , value
}
2021-06-16 14:14:43 +00:00
// Check if number is int.
2021-06-15 20:21:18 +00:00
func validateInteger ( text string , ch rune ) bool {
if text == "-" {
return true
}
_ , err := strconv . Atoi ( text )
return err == nil
}
2021-06-14 15:07:31 +00:00
// getNodes will load all the node names from a file, and return a slice of
// string values, each representing a unique node.
func getNodeNames ( filePath string ) ( [ ] string , error ) {
fh , err := os . Open ( filePath )
if err != nil {
return nil , fmt . Errorf ( "error: unable to open node file: %v" , err )
2021-06-14 04:49:47 +00:00
}
2021-06-14 15:07:31 +00:00
defer fh . Close ( )
nodes := [ ] string { }
2021-06-14 04:49:47 +00:00
2021-06-14 15:07:31 +00:00
scanner := bufio . NewScanner ( fh )
for scanner . Scan ( ) {
node := scanner . Text ( )
nodes = append ( nodes , node )
2021-06-14 04:49:47 +00:00
}
2021-06-14 15:07:31 +00:00
return nodes , nil
2021-06-14 04:49:47 +00:00
}
2021-06-23 12:55:39 +00:00
// -----------------------
// Creating a copy of the real Message struct here to use within the
// field specification, but without the control kind of fields from
// the original to avoid changing them to pointer values in the main
// struct which would be needed when json marshaling to omit those
// empty fields.
type msg struct {
// The node to send the message to
2021-06-29 06:21:42 +00:00
ToNode steward . Node ` json:"toNode" yaml:"toNode" `
2021-06-23 12:55:39 +00:00
// The actual data in the message
Data [ ] string ` json:"data" yaml:"data" `
// Method, what is this message doing, etc. CLI, syslog, etc.
2021-06-29 06:21:42 +00:00
Method steward . Method ` json:"method" yaml:"method" `
2021-06-23 12:55:39 +00:00
// ReplyMethod, is the method to use for the reply message.
// By default the reply method will be set to log to file, but
// you can override it setting your own here.
2021-06-29 06:21:42 +00:00
ReplyMethod steward . Method ` json:"replyMethod" yaml:"replyMethod" `
2021-06-23 12:55:39 +00:00
// From what node the message originated
ACKTimeout int ` json:"ACKTimeout" yaml:"ACKTimeout" `
// Resend retries
Retries int ` json:"retries" yaml:"retries" `
// The ACK timeout of the new message created via a request event.
ReplyACKTimeout int ` json:"replyACKTimeout" yaml:"replyACKTimeout" `
// The retries of the new message created via a request event.
ReplyRetries int ` json:"replyRetries" yaml:"replyRetries" `
// Timeout for long a process should be allowed to operate
MethodTimeout int ` json:"methodTimeout" yaml:"methodTimeout" `
// Directory is a string that can be used to create the
//directory structure when saving the result of some method.
// For example "syslog","metrics", or "metrics/mysensor"
// The type is typically used in the handler of a method.
Directory string ` json:"directory" yaml:"directory" `
// FileExtension is used to be able to set a wanted extension
// on a file being saved as the result of data being handled
// by a method handler.
FileExtension string ` json:"fileExtension" yaml:"fileExtension" `
// operation are used to give an opCmd and opArg's.
Operation interface { } ` json:"operation,omitempty" `
}