ssm-browse: added cd command

Also came up with an approach for dealing with commands that will probably work with contexts
This commit is contained in:
Leon Mika 2022-03-29 10:29:25 +11:00
parent 0b745a6dfa
commit f6f06eb22d
13 changed files with 142 additions and 37 deletions

View file

@ -7,6 +7,7 @@ import (
"github.com/aws/aws-sdk-go-v2/service/ssm" "github.com/aws/aws-sdk-go-v2/service/ssm"
tea "github.com/charmbracelet/bubbletea" tea "github.com/charmbracelet/bubbletea"
"github.com/charmbracelet/lipgloss" "github.com/charmbracelet/lipgloss"
"github.com/lmika/awstools/internal/common/ui/commandctrl"
"github.com/lmika/awstools/internal/common/ui/logging" "github.com/lmika/awstools/internal/common/ui/logging"
"github.com/lmika/awstools/internal/ssm-browse/controllers" "github.com/lmika/awstools/internal/ssm-browse/controllers"
"github.com/lmika/awstools/internal/ssm-browse/providers/awsssm" "github.com/lmika/awstools/internal/ssm-browse/providers/awsssm"
@ -33,7 +34,17 @@ func main() {
service := ssmparameters.NewService(provider) service := ssmparameters.NewService(provider)
ctrl := controllers.New(service) ctrl := controllers.New(service)
model := ui.NewModel(ctrl)
cmdController := commandctrl.NewCommandController()
cmdController.AddCommands(&commandctrl.CommandContext{
Commands: map[string]commandctrl.Command{
"cd": func(args []string) tea.Cmd {
return ctrl.ChangePrefix(args[0])
},
},
})
model := ui.NewModel(ctrl, cmdController)
p := tea.NewProgram(model, tea.WithAltScreen()) p := tea.NewProgram(model, tea.WithAltScreen())

View file

@ -9,15 +9,20 @@ import (
) )
type CommandController struct { type CommandController struct {
commands map[string]Command commandList *CommandContext
} }
func NewCommandController(commands map[string]Command) *CommandController { func NewCommandController() *CommandController {
return &CommandController{ return &CommandController{
commands: commands, commandList: nil,
} }
} }
func (c *CommandController) AddCommands(ctx *CommandContext) {
ctx.parent = c.commandList
c.commandList = ctx
}
func (c *CommandController) Prompt() tea.Cmd { func (c *CommandController) Prompt() tea.Cmd {
return func() tea.Msg { return func() tea.Msg {
return events.PromptForInputMsg{ return events.PromptForInputMsg{
@ -36,10 +41,19 @@ func (c *CommandController) Execute(commandInput string) tea.Cmd {
} }
tokens := shellwords.Split(input) tokens := shellwords.Split(input)
command, ok := c.commands[tokens[0]] command := c.lookupCommand(tokens[0])
if !ok { if command == nil {
return events.SetStatus("no such command: " + tokens[0]) return events.SetStatus("no such command: " + tokens[0])
} }
return command(tokens) return command(tokens[1:])
} }
func (c *CommandController) lookupCommand(name string) Command {
for ctx := c.commandList; ctx != nil; ctx = ctx.parent {
if cmd, ok := ctx.Commands[name]; ok {
return cmd
}
}
return nil
}

View file

@ -9,3 +9,9 @@ func NoArgCommand(cmd tea.Cmd) Command {
return cmd return cmd
} }
} }
type CommandContext struct {
Commands map[string]Command
parent *CommandContext
}

View file

@ -19,16 +19,16 @@ type StatusAndPrompt struct {
width int width int
} }
func New(model layout.ResizingModel, initialMsg string) StatusAndPrompt { func New(model layout.ResizingModel, initialMsg string) *StatusAndPrompt {
textInput := textinput.New() textInput := textinput.New()
return StatusAndPrompt{model: model, statusMessage: initialMsg, textInput: textInput} return &StatusAndPrompt{model: model, statusMessage: initialMsg, textInput: textInput}
} }
func (s StatusAndPrompt) Init() tea.Cmd { func (s *StatusAndPrompt) Init() tea.Cmd {
return s.model.Init() return s.model.Init()
} }
func (s StatusAndPrompt) Update(msg tea.Msg) (tea.Model, tea.Cmd) { func (s *StatusAndPrompt) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) { switch msg := msg.(type) {
case events.ErrorMsg: case events.ErrorMsg:
s.statusMessage = "Error: " + msg.Error() s.statusMessage = "Error: " + msg.Error()
@ -80,18 +80,22 @@ func (s StatusAndPrompt) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
return s, cmd return s, cmd
} }
func (s StatusAndPrompt) View() string { func (s *StatusAndPrompt) InPrompt() bool {
return s.pendingInput != nil
}
func (s *StatusAndPrompt) View() string {
return lipgloss.JoinVertical(lipgloss.Top, s.model.View(), s.viewStatus()) return lipgloss.JoinVertical(lipgloss.Top, s.model.View(), s.viewStatus())
} }
func (s StatusAndPrompt) Resize(w, h int) layout.ResizingModel { func (s *StatusAndPrompt) Resize(w, h int) layout.ResizingModel {
s.width = w s.width = w
submodelHeight := h - lipgloss.Height(s.viewStatus()) submodelHeight := h - lipgloss.Height(s.viewStatus())
s.model = s.model.Resize(w, submodelHeight) s.model = s.model.Resize(w, submodelHeight)
return s return s
} }
func (s StatusAndPrompt) viewStatus() string { func (s *StatusAndPrompt) viewStatus() string {
if s.pendingInput != nil { if s.pendingInput != nil {
return s.textInput.View() return s.textInput.View()
} }

View file

@ -3,5 +3,6 @@ package controllers
import "github.com/lmika/awstools/internal/ssm-browse/models" import "github.com/lmika/awstools/internal/ssm-browse/models"
type NewParameterListMsg struct { type NewParameterListMsg struct {
Prefix string
Parameters *models.SSMParameters Parameters *models.SSMParameters
} }

View file

@ -5,27 +5,53 @@ import (
tea "github.com/charmbracelet/bubbletea" tea "github.com/charmbracelet/bubbletea"
"github.com/lmika/awstools/internal/common/ui/events" "github.com/lmika/awstools/internal/common/ui/events"
"github.com/lmika/awstools/internal/ssm-browse/services/ssmparameters" "github.com/lmika/awstools/internal/ssm-browse/services/ssmparameters"
"sync"
) )
type SSMController struct { type SSMController struct {
service *ssmparameters.Service service *ssmparameters.Service
// state
mutex *sync.Mutex
prefix string
} }
func New(service *ssmparameters.Service) *SSMController { func New(service *ssmparameters.Service) *SSMController {
return &SSMController{ return &SSMController{
service: service, service: service,
prefix: "/",
mutex: new(sync.Mutex),
} }
} }
func (c *SSMController) Fetch() tea.Cmd { func (c *SSMController) Fetch() tea.Cmd {
return func() tea.Msg { return func() tea.Msg {
res, err := c.service.List(context.Background()) res, err := c.service.List(context.Background(), c.prefix)
if err != nil { if err != nil {
return events.Error(err) return events.Error(err)
} }
return NewParameterListMsg{ return NewParameterListMsg{
Prefix: c.prefix,
Parameters: res, Parameters: res,
} }
} }
} }
func (c *SSMController) ChangePrefix(newPrefix string) tea.Cmd {
return func() tea.Msg {
res, err := c.service.List(context.Background(), newPrefix)
if err != nil {
return events.Error(err)
}
c.mutex.Lock()
defer c.mutex.Unlock()
c.prefix = newPrefix
return NewParameterListMsg{
Prefix: c.prefix,
Parameters: res,
}
}
}

View file

@ -2,6 +2,7 @@ package models
type SSMParameters struct { type SSMParameters struct {
Items []SSMParameter Items []SSMParameter
NextToken string
} }
type SSMParameter struct { type SSMParameter struct {

View file

@ -6,6 +6,7 @@ import (
"github.com/aws/aws-sdk-go-v2/service/ssm" "github.com/aws/aws-sdk-go-v2/service/ssm"
"github.com/lmika/awstools/internal/ssm-browse/models" "github.com/lmika/awstools/internal/ssm-browse/models"
"github.com/pkg/errors" "github.com/pkg/errors"
"log"
) )
type Provider struct { type Provider struct {
@ -18,9 +19,16 @@ func NewProvider(client *ssm.Client) *Provider {
} }
} }
func (p *Provider) List(ctx context.Context) (*models.SSMParameters, error) { func (p *Provider) List(ctx context.Context, prefix string, nextToken string) (*models.SSMParameters, error) {
log.Printf("new prefix: %v", prefix)
var nextTokenStr *string = nil
if nextToken != "" {
nextTokenStr = aws.String(nextToken)
}
pars, err := p.client.GetParametersByPath(ctx, &ssm.GetParametersByPathInput{ pars, err := p.client.GetParametersByPath(ctx, &ssm.GetParametersByPathInput{
Path: aws.String("/"), Path: aws.String(prefix),
NextToken: nextTokenStr,
MaxResults: 10, MaxResults: 10,
Recursive: true, Recursive: true,
}) })
@ -30,10 +38,11 @@ func (p *Provider) List(ctx context.Context) (*models.SSMParameters, error) {
res := &models.SSMParameters{ res := &models.SSMParameters{
Items: make([]models.SSMParameter, len(pars.Parameters)), Items: make([]models.SSMParameter, len(pars.Parameters)),
NextToken: aws.ToString(pars.NextToken),
} }
for i, p := range pars.Parameters { for i, p := range pars.Parameters {
res.Items[i] = models.SSMParameter{ res.Items[i] = models.SSMParameter{
Name: aws.ToString(p.Name), Name: aws.ToString(p.Name),
Value: aws.ToString(p.Value), Value: aws.ToString(p.Value),
} }
} }

View file

@ -6,5 +6,5 @@ import (
) )
type SSMProvider interface { type SSMProvider interface {
List(ctx context.Context) (*models.SSMParameters, error) List(ctx context.Context, prefix string, nextToken string) (*models.SSMParameters, error)
} }

View file

@ -15,6 +15,22 @@ func NewService(provider SSMProvider) *Service {
} }
} }
func (s *Service) List(ctx context.Context) (*models.SSMParameters, error) { func (s *Service) List(ctx context.Context, prefix string) (*models.SSMParameters, error) {
return s.provider.List(ctx) var items []models.SSMParameter
var nextToken string
for {
page, err := s.provider.List(ctx, prefix, nextToken)
if err != nil {
return nil, err
}
items = append(items, page.Items...)
nextToken = page.NextToken
if len(items) >= 50 || nextToken == "" {
break
}
}
return &models.SSMParameters{Items: items, NextToken: nextToken}, nil
} }

View file

@ -2,6 +2,7 @@ package ui
import ( import (
tea "github.com/charmbracelet/bubbletea" tea "github.com/charmbracelet/bubbletea"
"github.com/lmika/awstools/internal/common/ui/commandctrl"
"github.com/lmika/awstools/internal/dynamo-browse/ui/teamodels/layout" "github.com/lmika/awstools/internal/dynamo-browse/ui/teamodels/layout"
"github.com/lmika/awstools/internal/dynamo-browse/ui/teamodels/statusandprompt" "github.com/lmika/awstools/internal/dynamo-browse/ui/teamodels/statusandprompt"
"github.com/lmika/awstools/internal/ssm-browse/controllers" "github.com/lmika/awstools/internal/ssm-browse/controllers"
@ -9,26 +10,29 @@ import (
) )
type Model struct { type Model struct {
controller *controllers.SSMController cmdController *commandctrl.CommandController
controller *controllers.SSMController
statusAndPrompt *statusandprompt.StatusAndPrompt
root tea.Model root tea.Model
ssmList *ssmlist.Model ssmList *ssmlist.Model
} }
func NewModel(controller *controllers.SSMController) Model { func NewModel(controller *controllers.SSMController, cmdController *commandctrl.CommandController) Model {
ssmList := ssmlist.New() ssmList := ssmlist.New()
root := layout.FullScreen( statusAndPrompt := statusandprompt.New(ssmList, "Hello SSM")
statusandprompt.New(ssmList, "Hello SSM"),
) root := layout.FullScreen(statusAndPrompt)
return Model{ return Model{
controller: controller, controller: controller,
root: root, cmdController: cmdController,
ssmList: ssmList, root: root,
statusAndPrompt: statusAndPrompt,
ssmList: ssmList,
} }
} }
func (m Model) Init() tea.Cmd { func (m Model) Init() tea.Cmd {
return m.controller.Fetch() return m.controller.Fetch()
} }
@ -36,11 +40,19 @@ func (m Model) Init() tea.Cmd {
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) { func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) { switch msg := msg.(type) {
case controllers.NewParameterListMsg: case controllers.NewParameterListMsg:
m.ssmList.SetPrefix(msg.Prefix)
m.ssmList.SetParameters(msg.Parameters) m.ssmList.SetParameters(msg.Parameters)
case tea.KeyMsg: case tea.KeyMsg:
switch msg.String() { if !m.statusAndPrompt.InPrompt() {
case "ctrl+c", "q": switch msg.String() {
return m, tea.Quit // TEMP
case ":":
return m, m.cmdController.Prompt()
// END TEMP
case "ctrl+c", "q":
return m, tea.Quit
}
} }
} }
@ -52,4 +64,3 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
func (m Model) View() string { func (m Model) View() string {
return m.root.View() return m.root.View()
} }

View file

@ -19,7 +19,7 @@ type Model struct {
} }
func New() *Model { func New() *Model {
frameTitle := frame.NewFrameTitle("SSM", true) frameTitle := frame.NewFrameTitle("SSM: /", true)
table := table.New([]string{"name", "type", "value"}, 0, 0) table := table.New([]string{"name", "type", "value"}, 0, 0)
return &Model{ return &Model{
@ -28,6 +28,10 @@ func New() *Model {
} }
} }
func (m *Model) SetPrefix(newPrefix string) {
m.frameTitle.SetTitle("SSM: " + newPrefix)
}
func (m *Model) SetParameters(parameters *models.SSMParameters) { func (m *Model) SetParameters(parameters *models.SSMParameters) {
m.parameters = parameters m.parameters = parameters
cols := []string{"name", "type", "value"} cols := []string{"name", "type", "value"}

View file

@ -5,6 +5,7 @@ import (
table "github.com/calyptia/go-bubble-table" table "github.com/calyptia/go-bubble-table"
"github.com/lmika/awstools/internal/ssm-browse/models" "github.com/lmika/awstools/internal/ssm-browse/models"
"io" "io"
"strings"
) )
type itemTableRow struct { type itemTableRow struct {
@ -12,7 +13,8 @@ type itemTableRow struct {
} }
func (mtr itemTableRow) Render(w io.Writer, model table.Model, index int) { func (mtr itemTableRow) Render(w io.Writer, model table.Model, index int) {
line := fmt.Sprintf("%s\t%s\t%s", mtr.item.Name, "String", mtr.item.Value) firstLine := strings.SplitN(mtr.item.Value, "\n", 2)[0]
line := fmt.Sprintf("%s\t%s\t%s", mtr.item.Name, "String", firstLine)
if index == model.Cursor() { if index == model.Cursor() {
fmt.Fprintln(w, model.Styles.SelectedRow.Render(line)) fmt.Fprintln(w, model.Styles.SelectedRow.Render(line))