399 lines
8.2 KiB
Go
399 lines
8.2 KiB
Go
/*
|
|
Copyright © 2020 NAME HERE <EMAIL ADDRESS>
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
package cmd
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
input "github.com/charmbracelet/bubbles/textinput"
|
|
tea "github.com/charmbracelet/bubbletea"
|
|
te "github.com/muesli/termenv"
|
|
"github.com/spf13/cobra"
|
|
"io/ioutil"
|
|
"log"
|
|
"os"
|
|
"pgm/data"
|
|
"pgm/logger"
|
|
"time"
|
|
)
|
|
|
|
// addCmd represents the add command
|
|
var addCmd = &cobra.Command{
|
|
Use: "add",
|
|
Short: "A brief description of your command",
|
|
Long: `A longer description that spans multiple lines and likely contains examples
|
|
and usage of using your command. For example:
|
|
|
|
Cobra is a CLI library for Go that empowers applications.
|
|
This application is a tool to generate the needed files
|
|
to quickly create a Cobra application.`,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
if err := tea.NewProgram(initialModel()).Start(); err != nil {
|
|
fmt.Printf("could not start program: %s\n", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
|
|
},
|
|
}
|
|
|
|
func init() {
|
|
rootCmd.AddCommand(addCmd)
|
|
|
|
// Here you will define your flags and configuration settings.
|
|
|
|
// Cobra supports Persistent Flags which will work for this command
|
|
// and all subcommands, e.g.:
|
|
// addCmd.PersistentFlags().String("foo", "", "A help for foo")
|
|
|
|
// Cobra supports local flags which will only run when this command
|
|
// is called directly, e.g.:
|
|
// addCmd.Flags().BoolP("toggle", "t", false, "Help message for toggle")
|
|
}
|
|
|
|
|
|
const focusedTextColor = "205"
|
|
|
|
var (
|
|
color = te.ColorProfile().Color
|
|
focusedPrompt = te.String("> ").Foreground(color("205")).String()
|
|
blurredPrompt = "> "
|
|
focusedSubmitButton = "[ " + te.String("Submit").Foreground(color("205")).String() + " ]"
|
|
blurredSubmitButton = "[ " + te.String("Submit").Foreground(color("240")).String() + " ]"
|
|
)
|
|
|
|
|
|
// Will ask a user for a unique name until they exit or provide one.
|
|
func cfgRename(c data.HostDetails) {
|
|
p := tea.NewProgram(renameModel(c))
|
|
|
|
if err := p.Start(); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
}
|
|
|
|
// writes json config to distinct file
|
|
func toFile(fp string, c data.HostDetails) {
|
|
var cfg []byte
|
|
lgd := logger.Logger("[WROTE] configuration file:" + c.DatabaseName + "::" + c.Hostname)
|
|
if lgd != true {
|
|
fmt.Println("failed to log")
|
|
}
|
|
f, err := os.Create(fp)
|
|
if err != nil {
|
|
log.Fatal("unable to create file: " + fp)
|
|
}
|
|
defer f.Close()
|
|
if err != nil {
|
|
log.Fatal("error writing to file: " + err.Error())
|
|
}
|
|
cfg, err = json.MarshalIndent(c, "", " ")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
_ = ioutil.WriteFile(fp, cfg, 0644)
|
|
tea.Quit()
|
|
}
|
|
|
|
|
|
type model struct {
|
|
index int
|
|
dbFileName input.Model
|
|
hostName input.Model
|
|
dbName input.Model
|
|
userName input.Model
|
|
secret input.Model
|
|
submitButton string
|
|
}
|
|
|
|
func initialModel() model {
|
|
dbFileName := input.NewModel()
|
|
dbFileName.Placeholder = "database file name (no spaces)"
|
|
dbFileName.Focus()
|
|
dbFileName.Prompt = focusedPrompt
|
|
dbFileName.TextColor = focusedTextColor
|
|
|
|
hostName := input.NewModel()
|
|
hostName.Placeholder = "hostname"
|
|
hostName.Prompt = blurredPrompt
|
|
|
|
dbName := input.NewModel()
|
|
dbName.Placeholder = "database name on host"
|
|
dbName.Prompt = blurredPrompt
|
|
|
|
userName := input.NewModel()
|
|
userName.Placeholder = "username"
|
|
userName.Prompt = blurredPrompt
|
|
|
|
secret := input.NewModel()
|
|
secret.Placeholder = "secret"
|
|
secret.Prompt = blurredPrompt
|
|
|
|
|
|
|
|
return model{0, dbFileName, hostName, dbName, userName, secret, blurredSubmitButton}
|
|
|
|
}
|
|
|
|
|
|
func (m model) Init() tea.Cmd {
|
|
return tea.Batch(
|
|
input.Blink(m.dbFileName),
|
|
input.Blink(m.hostName),
|
|
input.Blink(m.dbName),
|
|
input.Blink(m.userName),
|
|
input.Blink(m.secret),
|
|
)
|
|
}
|
|
|
|
|
|
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|
var cmd tea.Cmd
|
|
|
|
switch msg := msg.(type) {
|
|
case tea.KeyMsg:
|
|
switch msg.String() {
|
|
|
|
case "ctrl+c":
|
|
return m, tea.Quit
|
|
|
|
// Cycle between inputs
|
|
case "tab", "shift+tab", "enter", "up", "down":
|
|
|
|
inputs := []input.Model{
|
|
m.dbFileName,
|
|
m.hostName,
|
|
m.dbName,
|
|
m.userName,
|
|
m.secret,
|
|
}
|
|
|
|
s := msg.String()
|
|
|
|
// Did the user press enter while the submit button was focused?
|
|
// If so, exit.
|
|
if s == "enter" && m.index == len(inputs) {
|
|
configurator(inputs)
|
|
return m, tea.Quit
|
|
}
|
|
|
|
// Cycle indexes
|
|
if s == "up" || s == "shift+tab" {
|
|
m.index--
|
|
} else {
|
|
m.index++
|
|
}
|
|
|
|
if m.index > len(inputs) {
|
|
m.index = 0
|
|
} else if m.index < 0 {
|
|
m.index = len(inputs)
|
|
}
|
|
|
|
for i := 0; i <= len(inputs)-1; i++ {
|
|
if i == m.index {
|
|
// Set focused state
|
|
inputs[i].Focus()
|
|
inputs[i].Prompt = focusedPrompt
|
|
inputs[i].TextColor = focusedTextColor
|
|
continue
|
|
}
|
|
// Remove focused state
|
|
inputs[i].Blur()
|
|
inputs[i].Prompt = blurredPrompt
|
|
inputs[i].TextColor = ""
|
|
}
|
|
|
|
m.dbFileName = inputs[0]
|
|
m.hostName = inputs[1]
|
|
m.dbName = inputs[2]
|
|
m.userName = inputs[3]
|
|
m.secret = inputs[4]
|
|
|
|
|
|
if m.index == len(inputs) {
|
|
m.submitButton = focusedSubmitButton
|
|
} else {
|
|
m.submitButton = blurredSubmitButton
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
}
|
|
|
|
// Handle character input and blinks
|
|
m, cmd = updateInputs(msg, m)
|
|
return m, cmd
|
|
}
|
|
|
|
|
|
func updateInputs(msg tea.Msg, m model) (model, tea.Cmd) {
|
|
var (
|
|
cmd tea.Cmd
|
|
cmds []tea.Cmd
|
|
)
|
|
|
|
m.dbFileName, cmd = input.Update(msg, m.dbFileName)
|
|
cmds = append(cmds, cmd)
|
|
|
|
m.hostName, cmd = input.Update(msg, m.hostName)
|
|
cmds = append(cmds, cmd)
|
|
|
|
m.dbName, cmd = input.Update(msg, m.dbName)
|
|
cmds = append(cmds, cmd)
|
|
|
|
m.userName, cmd = input.Update(msg, m.userName)
|
|
cmds = append(cmds, cmd)
|
|
|
|
m.secret, cmd = input.Update(msg, m.secret)
|
|
|
|
|
|
return m, tea.Batch(cmds...)
|
|
}
|
|
|
|
|
|
func (m model) View() string {
|
|
s := "\n"
|
|
|
|
inputs := []string{
|
|
input.View(m.dbFileName),
|
|
input.View(m.hostName),
|
|
input.View(m.dbName),
|
|
input.View(m.userName),
|
|
input.View(m.secret),
|
|
}
|
|
|
|
for i := 0; i < len(inputs); i++ {
|
|
s += inputs[i]
|
|
if i < len(inputs)-1 {
|
|
s += "\n"
|
|
}
|
|
}
|
|
|
|
s += "\n\n" + m.submitButton + "\n"
|
|
return s
|
|
}
|
|
|
|
// Writes configs to .pgee directory
|
|
func configurator(t []input.Model) {
|
|
var c data.HostDetails
|
|
var cfgFileName string
|
|
var fp string
|
|
var homeDir string
|
|
c = data.HostDetails{t[4].Value(), t[1].Value(), t[2].Value(), t[3].Value()}
|
|
cfgFileName = t[0].Value()
|
|
homeDir, err := os.UserHomeDir()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if _, err := os.Stat(homeDir + "/.pgm/hosts/" + cfgFileName); os.IsNotExist(err) {
|
|
fp = homeDir + "/.pgm/hosts/" + cfgFileName
|
|
} else {
|
|
fmt.Println("duplicate config name")
|
|
cfgRename(c)
|
|
}
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
toFile(fp, c)
|
|
|
|
|
|
}
|
|
|
|
func renameModel(c data.HostDetails) rename {
|
|
inputModel := input.NewModel()
|
|
inputModel.Placeholder = "filename"
|
|
inputModel.Focus()
|
|
inputModel.CharLimit = 20
|
|
|
|
return rename{
|
|
textInput: inputModel,
|
|
conn: c,
|
|
err: nil,
|
|
}
|
|
}
|
|
|
|
type rename struct {
|
|
textInput input.Model
|
|
conn data.HostDetails
|
|
chosen bool
|
|
err error
|
|
}
|
|
|
|
type tickMsg struct{}
|
|
type errMsg error
|
|
|
|
func (m rename) Init() tea.Cmd {
|
|
return input.Blink(m.textInput)
|
|
}
|
|
|
|
func (m rename) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|
var cmd tea.Cmd
|
|
|
|
switch msg := msg.(type) {
|
|
case tea.KeyMsg:
|
|
switch msg.Type {
|
|
case tea.KeyCtrlC:
|
|
fallthrough
|
|
case tea.KeyEsc:
|
|
fallthrough
|
|
case tea.KeyEnter:
|
|
m.chosen = true
|
|
return m, Frame()
|
|
}
|
|
|
|
// We handle errors just like any other message
|
|
case errMsg:
|
|
m.err = msg
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
m.textInput, cmd = input.Update(msg, m.textInput)
|
|
return m, cmd
|
|
}
|
|
|
|
func (m rename) View() string {
|
|
var s string
|
|
if !m.chosen {
|
|
s = fmt.Sprintf(
|
|
"rename your file please\n\n%s\n\n%s",
|
|
input.View(m.textInput),
|
|
"(esc to quit)",
|
|
) + "\n"
|
|
} else {
|
|
renameCfg(m)
|
|
}
|
|
return s
|
|
}
|
|
|
|
func renameCfg(m rename) {
|
|
inputs := []string{
|
|
input.View(m.textInput),
|
|
}
|
|
toFile(inputs[0], m.conn)
|
|
}
|
|
|
|
type FrameMsg struct{}
|
|
|
|
// Frame event
|
|
func Frame() tea.Cmd {
|
|
return tea.Tick(time.Second/60, func(time.Time) tea.Msg {
|
|
return FrameMsg{}
|
|
})
|
|
} |