pgm/cmd/add.go
2020-10-24 23:12:10 -05:00

386 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{}
})
}