Skip to content

Commit 6d1badd

Browse files
committed
dagger: function completion
1 parent 3ee6a55 commit 6d1badd

File tree

4 files changed

+126
-13
lines changed

4 files changed

+126
-13
lines changed

completers/dagger_completer/cmd/call.go

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ var callCmd = &cobra.Command{
1515

1616
func init() {
1717
carapace.Gen(callCmd).Standalone()
18+
callCmd.Flags().SetInterspersed(false)
1819

1920
callCmd.PersistentFlags().Bool("focus", false, "Only show output for focused commands")
2021
callCmd.PersistentFlags().Bool("json", false, "Present result as JSON")

completers/dagger_completer/cmd/functions.go

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ func init() {
1717
carapace.Gen(functionsCmd).Standalone()
1818

1919
functionsCmd.PersistentFlags().Bool("focus", false, "Only show output for focused commands")
20+
functionsCmd.PersistentFlags().Bool("json", false, "Output as json")
2021
functionsCmd.PersistentFlags().StringP("mod", "m", "", "Path to dagger.json config file for the module or a directory containing that file. Either local path (e.g. \"/path/to/some/dir\") or a github repo (e.g. \"github.com/dagger/dagger/path/to/some/subdir\")")
2122
rootCmd.AddCommand(functionsCmd)
2223

completers/dagger_completer/cmd/run.go

+6
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ package cmd
22

33
import (
44
"github.com/rsteube/carapace"
5+
"github.com/rsteube/carapace-bridge/pkg/actions/bridge"
56
"github.com/spf13/cobra"
67
)
78

@@ -15,8 +16,13 @@ var runCmd = &cobra.Command{
1516

1617
func init() {
1718
carapace.Gen(runCmd).Standalone()
19+
runCmd.Flags().SetInterspersed(false)
1820

1921
runCmd.Flags().String("cleanup-timeout", "", "max duration to wait between SIGTERM and SIGKILL on interrupt")
2022
runCmd.Flags().Bool("focus", false, "Only show output for focused commands.")
2123
rootCmd.AddCommand(runCmd)
24+
25+
carapace.Gen(runCmd).PositionalAnyCompletion(
26+
bridge.ActionCarapaceBin(),
27+
)
2228
}

pkg/actions/tools/dagger/function.go

+118-13
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,132 @@
11
package dagger
22

33
import (
4-
"regexp"
5-
"strings"
4+
"encoding/json"
5+
"os"
6+
"time"
7+
"unicode"
68

79
"github.com/rsteube/carapace"
10+
"github.com/rsteube/carapace/pkg/cache"
11+
"github.com/rsteube/carapace/pkg/cache/key"
12+
"github.com/spf13/cobra"
813
)
914

10-
// ActionFunctions completes functions
15+
type function struct {
16+
Name string `json:"Name"`
17+
Description string `json:"Description"`
18+
Args []struct {
19+
Name string `json:"Name"`
20+
Description string `json:"Description"`
21+
TypeDef struct {
22+
Kind string `json:"Kind"`
23+
Optional bool `json:"Optional"`
24+
AsObject *struct {
25+
Name string `json:"Name"`
26+
Functions any `json:"Functions"`
27+
Fields any `json:"Fields"`
28+
Constructor any `json:"Constructor"`
29+
SourceModuleName string `json:"SourceModuleName"`
30+
} `json:"AsObject"`
31+
AsInterface any `json:"AsInterface"`
32+
AsInput any `json:"AsInput"`
33+
AsList any `json:"AsList"`
34+
} `json:"TypeDef"`
35+
DefaultValue string `json:"DefaultValue"`
36+
} `json:"Args"`
37+
}
38+
39+
type daggerFile struct {
40+
Name string
41+
Sdk string
42+
Source string
43+
}
44+
45+
// ActionFunctions completes functions and their arguments
1146
//
12-
// container-echo (dagger call container-echo --string-arg yo stdout)
1347
// grep-dir (dagger call grep-dir --directory-arg . --pattern GrepDir)
48+
// --directory-arg
1449
func ActionFunctions() carapace.Action {
15-
return carapace.ActionExecCommand("dagger", "functions", "--silent")(func(output []byte) carapace.Action {
16-
lines := strings.Split(string(output), "\n")
17-
r := regexp.MustCompile(`^(?P<name>[^ ]+) +example usage: "(?P<example>.*)"$`)
18-
19-
vals := make([]string, 0)
20-
for _, line := range lines[1:] {
21-
if matches := r.FindStringSubmatch(line); matches != nil {
22-
vals = append(vals, matches[1], matches[2])
50+
return carapace.ActionCallback(func(c carapace.Context) carapace.Action {
51+
content, err := os.ReadFile("dagger.json")
52+
if err != nil {
53+
return carapace.ActionMessage(err.Error())
54+
}
55+
56+
var d daggerFile
57+
if err := json.Unmarshal(content, &d); err != nil {
58+
return carapace.ActionMessage(err.Error())
59+
}
60+
61+
output, err := cache.Cache(24*time.Hour, key.FolderStats(d.Source))(func() ([]byte, error) {
62+
return c.Command("dagger", "functions", "--json").Output()
63+
})
64+
if err != nil {
65+
return carapace.ActionMessage(err.Error())
66+
}
67+
68+
var functions []function
69+
if err := json.Unmarshal(output, &functions); err != nil {
70+
return carapace.ActionMessage(err.Error())
71+
}
72+
73+
cmd := &cobra.Command{}
74+
carapace.Gen(cmd).Standalone()
75+
for _, f := range functions {
76+
f.Name = toKebab(f.Name)
77+
subCmd := &cobra.Command{
78+
Use: f.Name,
79+
Short: f.Description,
80+
Run: func(cmd *cobra.Command, args []string) {},
2381
}
82+
carapace.Gen(subCmd).Standalone()
83+
84+
for _, arg := range f.Args {
85+
arg.Name = toKebab(arg.Name)
86+
87+
// TODO transform camelcase to kebab
88+
switch arg.TypeDef.Kind { // TODO more types
89+
case "STRING_KIND", "OBJECT_KIND":
90+
subCmd.Flags().String(arg.Name, arg.DefaultValue, arg.Description)
91+
default:
92+
return carapace.ActionMessage("unknown kind %s", arg.TypeDef.Kind)
93+
}
94+
95+
if arg.TypeDef.Optional {
96+
subCmd.Flag(arg.Name).NoOptDefVal = " "
97+
}
98+
99+
localArg := arg
100+
carapace.Gen(subCmd).FlagCompletion(carapace.ActionMap{
101+
arg.Name: carapace.ActionCallback(func(c carapace.Context) carapace.Action {
102+
if localArg.TypeDef.AsObject == nil {
103+
return carapace.ActionValues()
104+
}
105+
106+
switch localArg.TypeDef.AsObject.Name { // TODO more
107+
case "Directory":
108+
return carapace.ActionDirectories()
109+
default:
110+
return carapace.ActionValues()
111+
}
112+
}),
113+
})
114+
}
115+
116+
cmd.AddCommand(subCmd)
24117
}
25-
return carapace.ActionValuesDescribed(vals...)
118+
119+
return carapace.ActionExecute(cmd)
26120
})
27121
}
122+
123+
func toKebab(s string) string {
124+
runes := make([]rune, 0)
125+
for _, r := range []rune(s) {
126+
if unicode.IsUpper(r) {
127+
runes = append(runes, '-')
128+
}
129+
runes = append(runes, unicode.ToLower(r))
130+
}
131+
return string(runes)
132+
}

0 commit comments

Comments
 (0)