1
0
Fork 0
mirror of https://github.com/postmannen/ctrl.git synced 2025-01-18 21:59:30 +00:00
ctrl/tui.go

635 lines
18 KiB
Go
Raw Normal View History

package steward
import (
"bufio"
"encoding/json"
"fmt"
2022-01-11 14:55:35 +01:00
"io/ioutil"
"log"
"os"
2022-01-09 09:36:48 +01:00
"path/filepath"
"reflect"
"strconv"
"strings"
"time"
"github.com/gdamore/tcell/v2"
"github.com/rivo/tview"
)
2022-01-11 14:55:35 +01:00
// ---------------------------------------------------------------------
// Main structure
// ---------------------------------------------------------------------
2022-01-08 04:19:51 +01:00
type tui struct {
}
2022-01-08 04:19:51 +01:00
func newTui() (*tui, error) {
s := tui{}
return &s, nil
}
type slide struct {
name string
key tcell.Key
primitive tview.Primitive
}
2022-01-08 04:19:51 +01:00
func (s *tui) Start() error {
pages := tview.NewPages()
app := tview.NewApplication()
2022-01-11 14:55:35 +01:00
// Check if F key is pressed, and switch slide accordingly.
app.SetInputCapture(func(event *tcell.EventKey) *tcell.EventKey {
2022-01-11 14:55:35 +01:00
switch event.Key() {
case tcell.KeyF1:
pages.SwitchToPage("console")
return nil
2022-01-11 14:55:35 +01:00
case tcell.KeyF2:
pages.SwitchToPage("message")
return nil
2022-01-11 14:55:35 +01:00
case tcell.KeyF3:
pages.SwitchToPage("info")
return nil
}
return event
})
info := tview.NewTextView().
SetDynamicColors(true).
SetRegions(true).
SetWrap(false)
// The slides to draw, and their name.
// NB: This slice is being looped over further below, to create the menu
2022-01-09 09:36:48 +01:00
// elements. If adding a new slide, make sure that slides are ordered in
// chronological order, so we can auto generate the info menu with it's
// corresponding F key based on the slice index+1.
slides := []slide{
2022-01-11 14:55:35 +01:00
{name: "console", key: tcell.KeyF1, primitive: console(app)},
{name: "message", key: tcell.KeyF2, primitive: messageSlide(app)},
2022-01-11 14:55:35 +01:00
{name: "info", key: tcell.KeyF3, primitive: infoSlide(app)},
}
2022-01-11 10:49:24 +01:00
// Add a page for each slide.
for i, v := range slides {
if i == 0 {
pages.AddPage(v.name, v.primitive, true, true)
fmt.Fprintf(info, " F%v:%v ", i+1, v.name)
continue
}
pages.AddPage(v.name, v.primitive, true, false)
fmt.Fprintf(info, " F%v:%v ", i+1, v.name)
}
// Create the main layout.
layout := tview.NewFlex()
//layout.SetBorder(true)
layout.SetDirection(tview.FlexRow).
AddItem(pages, 0, 10, true).
AddItem(info, 1, 1, false)
2022-01-09 09:36:48 +01:00
root := app.SetRoot(layout, true)
root.EnableMouse(true)
if err := root.Run(); err != nil {
log.Printf("error: root.Run(): %v\n", err)
os.Exit(1)
}
return nil
}
2022-01-11 14:55:35 +01:00
// ---------------------------------------------------------------------
// Slides
// ---------------------------------------------------------------------
func infoSlide(app *tview.Application) tview.Primitive {
flex := tview.NewFlex()
flex.SetTitle("info")
flex.SetBorder(true)
textView := tview.NewTextView()
flex.AddItem(textView, 0, 1, false)
fmt.Fprintf(textView, "Information page for Stew.\n")
return flex
}
func messageSlide(app *tview.Application) tview.Primitive {
2022-01-11 10:49:24 +01:00
// pageMessage is a struct for holding all the main forms and
// views used in the message slide, so we can easily reference
// them later in the code.
type pageMessage struct {
flex *tview.Flex
msgInputForm *tview.Form
msgOutputForm *tview.TextView
logForm *tview.TextView
2022-01-11 10:49:24 +01:00
saveForm *tview.Form
}
p := pageMessage{}
p.msgInputForm = tview.NewForm()
2022-01-11 10:49:24 +01:00
p.msgInputForm.SetBorder(true).SetTitle("Message input").SetTitleAlign(tview.AlignLeft)
p.msgOutputForm = tview.NewTextView()
p.msgOutputForm.SetBorder(true).SetTitle("Message output").SetTitleAlign(tview.AlignLeft)
p.msgOutputForm.SetChangedFunc(func() {
// Will cause the log window to be redrawn as soon as
// new output are detected.
app.Draw()
})
p.logForm = tview.NewTextView()
p.logForm.SetBorder(true).SetTitle("Log/Status").SetTitleAlign(tview.AlignLeft)
p.logForm.SetChangedFunc(func() {
// Will cause the log window to be redrawn as soon as
// new output are detected.
app.Draw()
})
2022-01-11 10:49:24 +01:00
p.saveForm = tview.NewForm()
p.saveForm.SetBorder(true).SetTitle("Save message").SetTitleAlign(tview.AlignLeft)
// Create a flex layout.
//
2022-01-11 10:49:24 +01:00
// Create the outer flex layout.
p.flex = tview.NewFlex().SetDirection(tview.FlexRow).
2022-01-11 10:49:24 +01:00
// Add a flex for the top windows with columns.
AddItem(tview.NewFlex().SetDirection(tview.FlexColumn).
AddItem(p.msgInputForm, 0, 10, false).
2022-01-11 10:49:24 +01:00
// Add a new flex for splitting output form horizontally.
AddItem(tview.NewFlex().SetDirection(tview.FlexRow).
// Add the message output form.
AddItem(p.msgOutputForm, 0, 10, false).
// Add the save message form.
AddItem(p.saveForm, 0, 2, false),
0, 10, false),
0, 10, false).
2022-01-11 10:49:24 +01:00
// Add a flex for the bottom log window.
AddItem(tview.NewFlex().
2022-01-11 10:49:24 +01:00
// Add the log form.
AddItem(p.logForm, 0, 2, false),
2022-01-11 10:49:24 +01:00
0, 1, false)
2022-01-10 13:21:30 +01:00
m := tuiMessage{}
2022-01-09 09:36:48 +01:00
// Draw all the message input field with values on the screen.
//
// 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.
//
// 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.
mRefVal := reflect.ValueOf(m)
for i := 0; i < mRefVal.NumField(); i++ {
fieldName := mRefVal.Type().Field(i).Name
switch fieldName {
2022-01-09 09:36:48 +01:00
case "_":
case "ToNode":
// Get nodes from file.
2022-01-09 09:36:48 +01:00
values, err := getNodeNames("nodeslist.cfg")
if err != nil {
2022-01-09 09:36:48 +01:00
log.Printf("error: unable to open file: %v\n", err)
}
item := tview.NewDropDown()
item.SetLabelColor(tcell.ColorIndianRed)
item.SetLabel(fieldName).SetOptions(values, nil)
p.msgInputForm.AddFormItem(item)
//c.msgForm.AddDropDown(mRefVal.Type().Field(i).Name, values, 0, nil).SetItemPadding(1)
2022-01-09 09:36:48 +01:00
case "ToNodes":
value := `"ship1","ship2","ship3"`
p.msgInputForm.AddInputField(fieldName, value, 30, nil, nil)
case "Method":
var m Method
ma := m.GetMethodsAvailable()
values := []string{}
for k := range ma.Methodhandlers {
values = append(values, string(k))
}
p.msgInputForm.AddDropDown(fieldName, values, 0, nil).SetItemPadding(1)
2022-01-09 09:36:48 +01:00
case "MethodArgs":
value := ``
p.msgInputForm.AddInputField(fieldName, value, 30, nil, nil)
case "ReplyMethod":
var m Method
rm := m.GetReplyMethods()
values := []string{}
for _, k := range rm {
values = append(values, string(k))
}
p.msgInputForm.AddDropDown(fieldName, values, 0, nil).SetItemPadding(1)
2022-01-09 09:36:48 +01:00
case "ReplyMethodArgs":
value := ``
p.msgInputForm.AddInputField(fieldName, value, 30, nil, nil)
case "ACKTimeout":
value := 30
p.msgInputForm.AddInputField(fieldName, fmt.Sprintf("%d", value), 30, validateInteger, nil)
case "Retries":
value := 1
p.msgInputForm.AddInputField(fieldName, fmt.Sprintf("%d", value), 30, validateInteger, nil)
case "ReplyACKTimeout":
value := 30
p.msgInputForm.AddInputField(fieldName, fmt.Sprintf("%d", value), 30, validateInteger, nil)
case "ReplyRetries":
value := 1
p.msgInputForm.AddInputField(fieldName, fmt.Sprintf("%d", value), 30, validateInteger, nil)
case "MethodTimeout":
value := 120
p.msgInputForm.AddInputField(fieldName, fmt.Sprintf("%d", value), 30, validateInteger, nil)
2022-01-09 09:36:48 +01:00
case "ReplyMethodTimeout":
value := 120
p.msgInputForm.AddInputField(fieldName, fmt.Sprintf("%d", value), 30, validateInteger, nil)
case "Directory":
value := "/some-dir/"
p.msgInputForm.AddInputField(fieldName, value, 30, nil, nil)
case "FileName":
value := ".log"
p.msgInputForm.AddInputField(fieldName, value, 30, nil, nil)
2022-01-09 09:36:48 +01:00
case "RelayViaNode":
// Get nodes from file.
values, err := getNodeNames("nodeslist.cfg")
if err != nil {
log.Printf("error: unable to open file: %v\n", err)
os.Exit(1)
}
item := tview.NewDropDown()
item.SetLabelColor(tcell.ColorIndianRed)
item.SetLabel(fieldName).SetOptions(values, nil)
p.msgInputForm.AddFormItem(item)
//c.msgForm.AddDropDown(mRefVal.Type().Field(i).Name, values, 0, nil).SetItemPadding(1)
case "RelayReplyMethod":
var m Method
rm := m.GetReplyMethods()
values := []string{}
for _, k := range rm {
values = append(values, string(k))
}
p.msgInputForm.AddDropDown(fieldName, values, 0, nil).SetItemPadding(1)
default:
// 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.
2022-01-09 09:36:48 +01:00
p.msgInputForm.AddDropDown("error: no case for: "+fieldName, []string{"1", "2"}, 0, nil).SetItemPadding(1)
}
}
2022-01-11 10:49:24 +01:00
// Variable to hold the last output created when the generate button have
// been pushed.
var lastGeneratedMessage []byte
var saveFileName string
// Add Buttons below the message fields. Like Generate and Exit.
p.msgInputForm.
// 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.
AddButton("generate to console", func() {
p.msgOutputForm.Clear()
2022-01-10 13:21:30 +01:00
m := tuiMessage{}
2022-01-09 09:36:48 +01:00
// Loop trough all the form fields, check the value of each
// form field, and add the value to m.
for i := 0; i < p.msgInputForm.GetFormItemCount(); i++ {
fi := p.msgInputForm.GetFormItem(i)
label, value := getLabelAndValue(fi)
switch label {
case "ToNode":
2022-01-10 13:21:30 +01:00
v := Node(value)
m.ToNode = &v
2022-01-09 09:36:48 +01:00
case "ToNodes":
2022-01-10 13:21:30 +01:00
slice, err := stringToNode(value)
if err != nil {
fmt.Fprintf(p.logForm, "%v : error: ReplyMethodArgs missing or malformed format, should be \"arg0\",\"arg1\",\"arg2\", %v\n", time.Now().Format("Mon Jan _2 15:04:05 2006"), err)
return
}
2022-01-10 13:21:30 +01:00
m.ToNodes = slice
case "Method":
2022-01-10 13:21:30 +01:00
v := Method(value)
m.Method = &v
2022-01-09 09:36:48 +01:00
case "MethodArgs":
2022-01-10 13:21:30 +01:00
slice, err := stringToSlice(value)
if err != nil {
fmt.Fprintf(p.logForm, "%v : error: ReplyMethodArgs missing or malformed format, should be \"arg0\",\"arg1\",\"arg2\", %v\n", time.Now().Format("Mon Jan _2 15:04:05 2006"), err)
return
2022-01-09 09:36:48 +01:00
}
2022-01-10 13:21:30 +01:00
m.MethodArgs = slice
case "ReplyMethod":
2022-01-10 13:21:30 +01:00
v := Method(value)
m.ReplyMethod = &v
2022-01-09 09:36:48 +01:00
case "ReplyMethodArgs":
2022-01-10 13:21:30 +01:00
slice, err := stringToSlice(value)
if err != nil {
fmt.Fprintf(p.logForm, "%v : error: ReplyMethodArgs missing or malformed format, should be \"arg0\",\"arg1\",\"arg2\", %v\n", time.Now().Format("Mon Jan _2 15:04:05 2006"), err)
return
2022-01-09 09:36:48 +01:00
}
2022-01-10 13:21:30 +01:00
m.ReplyMethodArgs = slice
case "ACKTimeout":
v, _ := strconv.Atoi(value)
2022-01-10 13:21:30 +01:00
m.ACKTimeout = &v
case "Retries":
v, _ := strconv.Atoi(value)
2022-01-10 13:21:30 +01:00
m.Retries = &v
case "ReplyACKTimeout":
v, _ := strconv.Atoi(value)
2022-01-10 13:21:30 +01:00
m.ReplyACKTimeout = &v
case "ReplyRetries":
v, _ := strconv.Atoi(value)
2022-01-10 13:21:30 +01:00
m.ReplyRetries = &v
case "MethodTimeout":
v, _ := strconv.Atoi(value)
2022-01-10 13:21:30 +01:00
m.MethodTimeout = &v
2022-01-09 09:36:48 +01:00
case "ReplyMethodTimeout":
v, _ := strconv.Atoi(value)
2022-01-10 13:21:30 +01:00
m.ReplyMethodTimeout = &v
case "Directory":
2022-01-10 13:21:30 +01:00
m.Directory = &value
case "FileName":
2022-01-10 13:21:30 +01:00
m.FileName = &value
2022-01-09 09:36:48 +01:00
case "RelayViaNode":
2022-01-10 13:21:30 +01:00
v := Node(value)
m.RelayViaNode = &v
2022-01-09 09:36:48 +01:00
case "RelayReplyMethod":
2022-01-10 13:21:30 +01:00
v := Method(value)
m.RelayReplyMethod = &v
default:
2022-01-09 09:36:48 +01:00
fmt.Fprintf(p.logForm, "%v : error: did not find case definition for how to handle the \"%v\" within the switch statement\n", time.Now().Format("Mon Jan _2 15:04:05 2006"), label)
2022-01-10 13:21:30 +01:00
return
}
}
2022-01-10 13:21:30 +01:00
msgs := []tuiMessage{}
msgs = append(msgs, m)
msgsIndented, err := json.MarshalIndent(msgs, "", " ")
if err != nil {
fmt.Fprintf(p.logForm, "%v : error: jsonIndent failed: %v\n", time.Now().Format("Mon Jan _2 15:04:05 2006"), err)
}
2022-01-11 10:49:24 +01:00
// Copy the message to a variable outside this scope so we can use
// the content for example if we want to save the message to file.
lastGeneratedMessage = msgsIndented
_, err = p.msgOutputForm.Write(msgsIndented)
if err != nil {
fmt.Fprintf(p.logForm, "%v : error: write to fh failed: %v\n", time.Now().Format("Mon Jan _2 15:04:05 2006"), err)
}
}).
// Add exit button.
AddButton("exit", func() {
app.Stop()
})
app.SetFocus(p.msgInputForm)
2022-01-11 10:49:24 +01:00
p.saveForm.
AddInputField("FileName", "", 40, nil, func(text string) {
saveFileName = text
}).
AddButton("save", func() {
messageFolder := "messages"
if saveFileName == "" {
fmt.Fprintf(p.logForm, "error: missing filename\n")
return
}
if _, err := os.Stat(messageFolder); os.IsNotExist(err) {
err := os.MkdirAll(messageFolder, 0700)
if err != nil {
fmt.Fprintf(p.logForm, "error: failed to create messages folder: %v\n", err)
return
}
}
file := filepath.Join(messageFolder, saveFileName)
fh, err := os.OpenFile(file, os.O_CREATE|os.O_RDWR|os.O_TRUNC, 0755)
if err != nil {
fmt.Fprintf(p.logForm, "error: opening file for writing: %v\n", err)
return
}
defer fh.Close()
_, err = fh.Write([]byte(lastGeneratedMessage))
if err != nil {
fmt.Fprintf(p.logForm, "error: writing message to file: %v\n", err)
return
}
2022-01-11 14:55:35 +01:00
fmt.Fprintf(p.logForm, "info: succesfully wrote message to file: %v\n", file)
2022-01-11 10:49:24 +01:00
})
return p.flex
}
2022-01-11 14:55:35 +01:00
func console(app *tview.Application) tview.Primitive {
// pageMessage is a struct for holding all the main forms and
// views used in the message slide, so we can easily reference
// them later in the code.
type pageMessage struct {
flex *tview.Flex
selectForm *tview.Form
outputForm *tview.TextView
}
p := pageMessage{}
p.selectForm = tview.NewForm()
p.selectForm.SetBorder(true).SetTitle("select").SetTitleAlign(tview.AlignLeft)
p.outputForm = tview.NewTextView()
p.outputForm.SetBorder(true).SetTitle("output").SetTitleAlign(tview.AlignLeft)
p.outputForm.SetChangedFunc(func() {
// Will cause the log window to be redrawn as soon as
// new output are detected.
app.Draw()
})
// Create a flex layout.
//
// Create the outer flex layout.
p.flex = tview.NewFlex().SetDirection(tview.FlexRow).
// Add a flex for the top windows with columns.
AddItem(tview.NewFlex().SetDirection(tview.FlexColumn).
AddItem(p.selectForm, 0, 3, false).
// Add the message output form.
AddItem(p.outputForm, 0, 10, false),
0, 10, false)
// Add a flex for the bottom log window.
// Add items.
// Create nodes dropdown field.
nodesList, err := getNodeNames("nodeslist.cfg")
if err != nil {
fmt.Fprintf(p.outputForm, "error: failed to open nodeslist.cfg file\n")
}
item := tview.NewDropDown()
item.SetLabelColor(tcell.ColorIndianRed)
item.SetLabel("nodes").SetOptions(nodesList, nil)
p.selectForm.AddFormItem(item)
// Create messages dropdown field.
fInfo, err := ioutil.ReadDir("messages")
if err != nil {
fmt.Fprintf(p.outputForm, "error: failed to read files from messages dir\n")
}
values := []string{}
for _, v := range fInfo {
values = append(values, v.Name())
}
p.selectForm.AddDropDown("method", values, 0, nil).SetItemPadding(1)
p.selectForm.AddButton("send message", func() {
// here ........
})
return p.flex
}
// ---------------------------------------------------------------------
// Helper functions
// ---------------------------------------------------------------------
2022-01-10 13:21:30 +01:00
// stringToSlice will Split the comma separated string
// into a and remove the start and end ampersand.
func stringToSlice(s string) (*[]string, error) {
if s == "" {
return nil, nil
}
var stringSlice []string
sp := strings.Split(s, ",")
for _, v := range sp {
// Check if format is correct, return if not.
pre := strings.HasPrefix(v, "\"")
suf := strings.HasSuffix(v, "\"")
if !pre || !suf {
return nil, fmt.Errorf("stringToSlice: missing leading or ending ampersand")
}
// Remove leading and ending ampersand.
v = v[1:]
v = strings.TrimSuffix(v, "\"")
stringSlice = append(stringSlice, v)
}
return &stringSlice, nil
}
// stringToNodes will Split the comma separated slice
// of nodes, and remove the start and end ampersand.
func stringToNode(s string) (*[]Node, error) {
if s == "" {
return nil, nil
}
var nodeSlice []Node
sp := strings.Split(s, ",")
for _, v := range sp {
// Check if format is correct, return if not.
pre := strings.HasPrefix(v, "\"")
suf := strings.HasSuffix(v, "\"")
if !pre || !suf {
return nil, fmt.Errorf("stringToSlice: missing leading or ending ampersand")
}
// Remove leading and ending ampersand.
v = v[1:]
v = strings.TrimSuffix(v, "\"")
nodeSlice = append(nodeSlice, Node(v))
}
return &nodeSlice, nil
}
// Will return the Label And the text Value of an input or dropdown form field.
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
}
// Check if number is int.
func validateInteger(text string, ch rune) bool {
if text == "-" {
return true
}
_, err := strconv.Atoi(text)
return err == nil
}
// getNodes will load all the node names from a file, and return a slice of
// string values, each representing a unique node.
2022-01-09 09:36:48 +01:00
func getNodeNames(fileName string) ([]string, error) {
dirPath, err := os.Getwd()
if err != nil {
return nil, fmt.Errorf("error: tui: unable to get working directory: %v", err)
}
filePath := filepath.Join(dirPath, fileName)
log.Printf(" * filepath : %v\n", filePath)
fh, err := os.Open(filePath)
if err != nil {
2022-01-09 09:36:48 +01:00
return nil, fmt.Errorf("error: tui: you should create a file named nodeslist.cfg with all your nodes : %v", err)
}
defer fh.Close()
nodes := []string{}
2022-01-11 10:49:24 +01:00
// append a blank node at the beginning of the slice, so the dropdown
// can be set to blank
nodes = append(nodes, "")
scanner := bufio.NewScanner(fh)
for scanner.Scan() {
node := scanner.Text()
nodes = append(nodes, node)
}
return nodes, nil
}