Skip to content

Commit 409c949

Browse files
feat: create form init
1 parent 951a2e0 commit 409c949

File tree

13 files changed

+359
-16
lines changed

13 files changed

+359
-16
lines changed

Dockerfile

+3-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,9 @@ RUN go mod download
88

99
COPY . .
1010

11-
RUN go build -o ./main ./cmd/api/
11+
ENV TERM=xterm-256color
12+
13+
RUN go build -o ./main .
1214

1315
EXPOSE 22
1416

cmd/create.go

+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
package cmd
2+
3+
import (
4+
"fmt"
5+
"os"
6+
"strconv"
7+
8+
tea "github.com/charmbracelet/bubbletea"
9+
"github.com/charmbracelet/lipgloss"
10+
"github.com/devmegablaster/bashform/internal/create"
11+
"github.com/devmegablaster/bashform/internal/services"
12+
"github.com/spf13/cobra"
13+
)
14+
15+
func (c *CLI) Create() *cobra.Command {
16+
newFormCmd := &cobra.Command{
17+
Use: "create [number of questions] [share code]",
18+
Args: cobra.ExactArgs(2),
19+
Short: "Create a new form with a specific number of questions and shareable code",
20+
Aliases: []string{"c"},
21+
RunE: func(cmd *cobra.Command, args []string) error {
22+
23+
n, err := strconv.Atoi(args[0])
24+
if err != nil {
25+
return err
26+
}
27+
28+
if n < 1 {
29+
return fmt.Errorf("number of questions must be greater than 0")
30+
}
31+
32+
var code string
33+
34+
if len(args) < 2 {
35+
code = ""
36+
} else {
37+
code = args[1]
38+
}
39+
40+
client := services.NewClient(c.cfg.Api.BaseUrl, c.PubKey)
41+
42+
check, err := client.GetForm(code)
43+
44+
if err != nil {
45+
cmd.Println(lipgloss.NewStyle().Foreground(lipgloss.Color("#ef4444")).Render("\nError checking code...\n"))
46+
}
47+
48+
if check.Code != "" {
49+
cmd.Println(lipgloss.NewStyle().Foreground(lipgloss.Color("#ef4444")).Render("\nA Form with that code already exists!\n"))
50+
return nil
51+
}
52+
53+
cr := create.NewModel(c.Session, n, code, client)
54+
55+
p := tea.NewProgram(cr,
56+
tea.WithAltScreen(),
57+
tea.WithInput(c.Session),
58+
tea.WithOutput(c.Session))
59+
60+
if _, err := p.Run(); err != nil {
61+
fmt.Fprintf(os.Stderr, "Error starting program: %v", err)
62+
os.Exit(1)
63+
}
64+
65+
return nil
66+
},
67+
}
68+
69+
return newFormCmd
70+
}

cmd/form.go

+6-3
Original file line numberDiff line numberDiff line change
@@ -17,15 +17,18 @@ func (c *CLI) Form() *cobra.Command {
1717
RunE: func(cmd *cobra.Command, args []string) error {
1818

1919
client := services.NewClient(c.cfg.Api.BaseUrl, c.PubKey)
20-
formData := client.GetForm(args[0])
20+
formData, err := client.GetForm(args[0])
21+
if err != nil {
22+
cmd.Println("Form not found...")
23+
return nil
24+
}
2125

2226
model := form.NewModel(formData, client, c.Session)
2327
p := tea.NewProgram(model,
24-
tea.WithMouseCellMotion(),
2528
tea.WithInput(cmd.InOrStdin()),
2629
tea.WithOutput(cmd.OutOrStdout()),
2730
)
28-
_, err := p.Run()
31+
_, err = p.Run()
2932
return err
3033
},
3134
}

cmd/root.go

+1
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,7 @@ func (c *CLI) Init() {
5252

5353
// Add Commands
5454
c.AddCommand(c.Form())
55+
c.AddCommand(c.Create())
5556
}
5657

5758
func (c *CLI) Run() error {

config.yaml

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
ssh:
22
host: 0.0.0.0
3-
port: 22
3+
port: 23
44

55
api:
6-
base_url: http://bashform-api:8000
6+
base_url: http://localhost:8000

docker-compose.yaml

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
services:
22
ssh:
3+
tty: true
34
build:
45
context: .
56
dockerfile: Dockerfile

go.mod

+3-3
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,10 @@ module github.com/devmegablaster/bashform
33
go 1.23.2
44

55
require (
6+
github.com/charmbracelet/bubbles v0.20.0
67
github.com/charmbracelet/bubbletea v1.2.4
78
github.com/charmbracelet/huh v0.6.0
8-
github.com/charmbracelet/huh/spinner v0.0.0-20241211235322-ceae3bbcfbb4
9+
github.com/charmbracelet/lipgloss v1.0.0
910
github.com/charmbracelet/log v0.4.0
1011
github.com/charmbracelet/ssh v0.0.0-20241211182756-4fe22b0f1b7c
1112
github.com/charmbracelet/wish v1.4.4
@@ -19,9 +20,7 @@ require (
1920
github.com/atotto/clipboard v0.1.4 // indirect
2021
github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect
2122
github.com/catppuccin/go v0.2.0 // indirect
22-
github.com/charmbracelet/bubbles v0.20.0 // indirect
2323
github.com/charmbracelet/keygen v0.5.1 // indirect
24-
github.com/charmbracelet/lipgloss v1.0.0 // indirect
2524
github.com/charmbracelet/x/ansi v0.4.5 // indirect
2625
github.com/charmbracelet/x/conpty v0.1.0 // indirect
2726
github.com/charmbracelet/x/errors v0.0.0-20240508181413-e8d8b6e2de86 // indirect
@@ -49,6 +48,7 @@ require (
4948
github.com/rivo/uniseg v0.4.7 // indirect
5049
github.com/sagikazarmark/locafero v0.4.0 // indirect
5150
github.com/sagikazarmark/slog-shim v0.1.0 // indirect
51+
github.com/sahilm/fuzzy v0.1.1 // indirect
5252
github.com/sourcegraph/conc v0.3.0 // indirect
5353
github.com/spf13/afero v1.11.0 // indirect
5454
github.com/spf13/cast v1.6.0 // indirect

go.sum

+4-2
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,6 @@ github.com/charmbracelet/bubbletea v1.2.4 h1:KN8aCViA0eps9SCOThb2/XPIlea3ANJLUkv
1414
github.com/charmbracelet/bubbletea v1.2.4/go.mod h1:Qr6fVQw+wX7JkWWkVyXYk/ZUQ92a6XNekLXa3rR18MM=
1515
github.com/charmbracelet/huh v0.6.0 h1:mZM8VvZGuE0hoDXq6XLxRtgfWyTI3b2jZNKh0xWmax8=
1616
github.com/charmbracelet/huh v0.6.0/go.mod h1:GGNKeWCeNzKpEOh/OJD8WBwTQjV3prFAtQPpLv+AVwU=
17-
github.com/charmbracelet/huh/spinner v0.0.0-20241211235322-ceae3bbcfbb4 h1:FTJ/03WaUpEiZ5oK4/n22eqyAtj8Pi0Uu64oo9ZzBU8=
18-
github.com/charmbracelet/huh/spinner v0.0.0-20241211235322-ceae3bbcfbb4/go.mod h1:3/xTBdgqRzAb+eUKRAGi9ix/K6QxsS0nGtd4zp+/tJs=
1917
github.com/charmbracelet/keygen v0.5.1 h1:zBkkYPtmKDVTw+cwUyY6ZwGDhRxXkEp0Oxs9sqMLqxI=
2018
github.com/charmbracelet/keygen v0.5.1/go.mod h1:zznJVmK/GWB6dAtjluqn2qsttiCBhA5MZSiwb80fcHw=
2119
github.com/charmbracelet/lipgloss v1.0.0 h1:O7VkGDvqEdGi93X+DeqsQ7PKHDgtQfF8j8/O2qFMQNg=
@@ -67,6 +65,8 @@ github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
6765
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
6866
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
6967
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
68+
github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc=
69+
github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw=
7070
github.com/lucasb-eyer/go-colorful v1.2.0 h1:1nnpGOrhyZZuNyfu1QjKiUICQ74+3FNCN69Aj6K7nkY=
7171
github.com/lucasb-eyer/go-colorful v1.2.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
7272
github.com/magiconair/properties v1.8.7 h1:IeQXZAiQcpL9mgcAe1Nu6cX9LLw6ExEHKjN0VQdvPDY=
@@ -102,6 +102,8 @@ github.com/sagikazarmark/locafero v0.4.0 h1:HApY1R9zGo4DBgr7dqsTH/JJxLTTsOt7u6ke
102102
github.com/sagikazarmark/locafero v0.4.0/go.mod h1:Pe1W6UlPYUk/+wc/6KFhbORCfqzgYEpgQ3O5fPuL3H4=
103103
github.com/sagikazarmark/slog-shim v0.1.0 h1:diDBnUNK9N/354PgrxMywXnAwEr1QZcOr6gto+ugjYE=
104104
github.com/sagikazarmark/slog-shim v0.1.0/go.mod h1:SrcSrq8aKtyuqEI1uvTDTK1arOWRIczQRv+GVI1AkeQ=
105+
github.com/sahilm/fuzzy v0.1.1 h1:ceu5RHF8DGgoi+/dR5PsECjCDH1BE3Fnmpo7aVXOdRA=
106+
github.com/sahilm/fuzzy v0.1.1/go.mod h1:VFvziUEIMCrT6A6tw2RFIXPXXmzXbOsSHF0DOI8ZK9Y=
105107
github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo=
106108
github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0=
107109
github.com/spf13/afero v1.11.0 h1:WJQKhtpdm3v2IzqG8VMqrr6Rf3UYpEF239Jy9wNepM8=

internal/create/create.go

+165
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,165 @@
1+
package create
2+
3+
import (
4+
"fmt"
5+
"strings"
6+
"time"
7+
8+
tea "github.com/charmbracelet/bubbletea"
9+
"github.com/charmbracelet/huh"
10+
"github.com/charmbracelet/lipgloss"
11+
"github.com/charmbracelet/ssh"
12+
"github.com/devmegablaster/bashform/internal/constants"
13+
"github.com/devmegablaster/bashform/internal/models"
14+
"github.com/devmegablaster/bashform/internal/services"
15+
)
16+
17+
type Model struct {
18+
width int
19+
height int
20+
questionsForm *huh.Form
21+
code string
22+
n int
23+
client *services.Client
24+
isCreating bool
25+
isCreated bool
26+
err error
27+
init bool
28+
initTime time.Time
29+
formResp *models.FormResponse
30+
}
31+
32+
func NewModel(session ssh.Session, n int, code string, client *services.Client) *Model {
33+
pty, _, _ := session.Pty()
34+
35+
return &Model{
36+
width: pty.Window.Width,
37+
height: pty.Window.Height,
38+
questionsForm: starterForm(n),
39+
code: code,
40+
n: n,
41+
client: client,
42+
init: true,
43+
initTime: time.Now(),
44+
}
45+
}
46+
47+
func (m *Model) Init() tea.Cmd {
48+
return m.questionsForm.Init()
49+
}
50+
51+
func (m *Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
52+
var cmd tea.Cmd
53+
54+
form, cmd := m.questionsForm.Update(msg)
55+
if f, ok := form.(*huh.Form); ok {
56+
m.questionsForm = f
57+
}
58+
59+
if m.questionsForm.State == huh.StateCompleted && !m.isCreating {
60+
m.CreateRequest()
61+
}
62+
63+
if m.init && time.Since(m.initTime) > 2*time.Second {
64+
m.init = false
65+
}
66+
67+
switch msg := msg.(type) {
68+
case tea.KeyMsg:
69+
switch msg.String() {
70+
case "ctrl+c":
71+
return m, tea.Quit
72+
case "q":
73+
if m.questionsForm.State == huh.StateCompleted {
74+
return m, tea.Quit
75+
}
76+
}
77+
}
78+
79+
return m, cmd
80+
}
81+
82+
func (m *Model) View() string {
83+
84+
var content string
85+
86+
content = m.questionsForm.View()
87+
88+
if m.isCreating {
89+
content = "Creating your form..."
90+
}
91+
92+
if m.isCreated {
93+
successMsg := lipgloss.NewStyle().Foreground(lipgloss.Color("#22c55e")).Bold(true).Render("Form created successfully!")
94+
linkMsg := lipgloss.NewStyle().Foreground(lipgloss.Color("#3b82f6")).Render("Your Access Command:")
95+
accessMsg := lipgloss.NewStyle().Foreground(lipgloss.Color("#3b82f6")).Render(fmt.Sprintf("ssh -t bashform.me f %s", m.formResp.Data.Code))
96+
helpMsg := lipgloss.NewStyle().Foreground(lipgloss.Color("#626262")).Render("q or ctrl+c to exit")
97+
98+
content = successMsg + "\n\n" + linkMsg + "\n" + accessMsg + "\n\n" + helpMsg
99+
}
100+
101+
if m.err != nil {
102+
content = fmt.Sprintf("Error creating form: %s", m.err)
103+
}
104+
105+
if m.init {
106+
return lipgloss.Place(m.width, m.height, lipgloss.Center, lipgloss.Center, constants.Logo)
107+
}
108+
109+
box := lipgloss.NewStyle().
110+
Border(lipgloss.RoundedBorder()).
111+
BorderForeground(lipgloss.Color("#94a3b8")).
112+
Align(lipgloss.Center).
113+
Padding(1, 2).
114+
Render(content)
115+
116+
return lipgloss.Place(
117+
m.width,
118+
m.height,
119+
lipgloss.Center,
120+
lipgloss.Center,
121+
lipgloss.JoinVertical(
122+
lipgloss.Center,
123+
lipgloss.NewStyle().Foreground(lipgloss.Color("#3b82f6")).MarginBottom(1).Bold(true).Render("New Form"),
124+
box,
125+
))
126+
}
127+
128+
func (m *Model) CreateRequest() {
129+
m.isCreating = true
130+
131+
questions := []models.QuestionRequest{}
132+
133+
for i := 0; i < m.n; i++ {
134+
question := models.QuestionRequest{
135+
Text: m.questionsForm.GetString(fmt.Sprintf("question%d", i)),
136+
Type: m.questionsForm.GetString(fmt.Sprintf("type%d", i)),
137+
}
138+
139+
options := strings.Split(m.questionsForm.GetString(fmt.Sprintf("options%d", i)), ",")
140+
optionRequests := []models.OptionRequest{}
141+
for _, option := range options {
142+
optionRequests = append(optionRequests, models.OptionRequest{Text: strings.TrimSpace(option)})
143+
}
144+
145+
question.Options = optionRequests
146+
questions = append(questions, question)
147+
}
148+
149+
formRequest := models.FormRequest{
150+
Name: m.questionsForm.GetString("name"),
151+
Description: m.questionsForm.GetString("description"),
152+
Code: m.code,
153+
Questions: questions,
154+
}
155+
156+
form, err := m.client.CreateForm(formRequest)
157+
if err != nil {
158+
m.err = err
159+
return
160+
}
161+
162+
m.formResp = form
163+
164+
m.isCreated = true
165+
}

internal/create/form.go

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
package create
2+
3+
import (
4+
"fmt"
5+
6+
"github.com/charmbracelet/huh"
7+
)
8+
9+
func questionForm(index int) *huh.Group {
10+
return huh.NewGroup(
11+
huh.NewSelect[string]().
12+
Title("Question Type").
13+
Options(
14+
huh.NewOption("Text", "text"),
15+
huh.NewOption("Textarea", "textarea"),
16+
huh.NewOption("Select", "select"),
17+
).Key(fmt.Sprintf("type%d", index)).Validate(huh.ValidateNotEmpty()),
18+
huh.NewInput().Title("Question").Key(fmt.Sprintf("question%d", index)).Validate(huh.ValidateNotEmpty()),
19+
huh.NewInput().Title("Options (comma separated) [For select questions only]").Key(fmt.Sprintf("options%d", index)),
20+
huh.NewConfirm().Title("Required?").Key(fmt.Sprintf("required%d", index)),
21+
)
22+
}
23+
24+
func starterForm(n int) *huh.Form {
25+
questions := make([]*huh.Group, n)
26+
for i := 0; i < n; i++ {
27+
questions[i] = questionForm(i)
28+
}
29+
30+
questions = append(questions, huh.NewGroup(
31+
huh.NewConfirm().Title("Allow multiple submissions by a user?").Key("allowMultiple"),
32+
))
33+
34+
allGroups := append([]*huh.Group{
35+
huh.NewGroup(
36+
huh.NewInput().Title("Form Name").Key("name").Validate(huh.ValidateNotEmpty()),
37+
huh.NewInput().Title("Form Description").Key("description").Validate(huh.ValidateNotEmpty()),
38+
),
39+
}, questions...)
40+
41+
return huh.NewForm(
42+
allGroups...,
43+
)
44+
}

0 commit comments

Comments
 (0)