-
-
Notifications
You must be signed in to change notification settings - Fork 152
/
humacli.go
308 lines (271 loc) · 7.88 KB
/
humacli.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
package humacli
import (
"context"
"fmt"
"os"
"os/signal"
"path/filepath"
"reflect"
"strconv"
"strings"
"syscall"
"time"
"github.com/danielgtaylor/huma/v2/casing"
"github.com/spf13/cobra"
)
func deref(t reflect.Type) reflect.Type {
for t.Kind() == reflect.Ptr {
t = t.Elem()
}
return t
}
// CLI is an optional command-line interface for a Huma service. It is provided
// as a convenience for quickly building a service with configuration from
// the environment and/or command-line options, all tied to a simple type-safe
// Go struct.
type CLI interface {
// Run the CLI. This will parse the command-line arguments and environment
// variables and then run the appropriate command. If no command is given,
// the default command will call the `OnStart` function to start a server.
Run()
// Root returns the root Cobra command. This can be used to add additional
// commands or flags. Customize it however you like.
Root() *cobra.Command
}
// Hooks is an interface for setting up callbacks for the CLI. It is used to
// start and stop the service.
type Hooks interface {
// OnStart sets a function to call when the service should be started. This
// is called by the default command if no command is given. The callback
// should take whatever steps are necessary to start the server, such as
// `httpServer.ListenAndServer(...)`.
OnStart(func())
// OnStop sets a function to call when the service should be stopped. This
// is called by the default command if no command is given. The callback
// should take whatever steps are necessary to stop the server, such as
// `httpServer.Shutdown(...)`.
OnStop(func())
}
type contextKey string
var optionsKey contextKey = "huma/cli/options"
var durationType = reflect.TypeOf((*time.Duration)(nil)).Elem()
// WithOptions is a helper for custom commands that need to access the options.
//
// cli.Root().AddCommand(&cobra.Command{
// Use: "my-custom-command",
// Run: huma.WithOptions(func(cmd *cobra.Command, args []string, opts *Options) {
// fmt.Println("Hello " + opts.Name)
// }),
// })
func WithOptions[Options any](f func(cmd *cobra.Command, args []string, options *Options)) func(*cobra.Command, []string) {
return func(cmd *cobra.Command, s []string) {
var options *Options = cmd.Context().Value(optionsKey).(*Options)
f(cmd, s, options)
}
}
type option struct {
name string
typ reflect.Type
path []int
}
type cli[Options any] struct {
root *cobra.Command
optInfo []option
onParsed func(Hooks, *Options)
start func()
stop func()
}
func (c *cli[Options]) Run() {
var o Options
existing := c.root.PersistentPreRun
c.root.PersistentPreRun = func(cmd *cobra.Command, args []string) {
// Load config from args/env/files
v := reflect.ValueOf(&o).Elem()
flags := c.root.PersistentFlags()
for _, opt := range c.optInfo {
f := v
for _, i := range opt.path {
f = f.Field(i)
}
var fv reflect.Value
switch deref(opt.typ).Kind() {
case reflect.String:
s, _ := flags.GetString(opt.name)
fv = reflect.ValueOf(s)
case reflect.Int, reflect.Int64:
var i any
if opt.typ == durationType {
i, _ = flags.GetDuration(opt.name)
} else {
i, _ = flags.GetInt64(opt.name)
}
fv = reflect.ValueOf(i).Convert(deref(opt.typ))
case reflect.Bool:
b, _ := flags.GetBool(opt.name)
fv = reflect.ValueOf(b)
}
if opt.typ.Kind() == reflect.Ptr {
ptr := reflect.New(fv.Type())
ptr.Elem().Set(fv)
fv = ptr
}
f.Set(fv)
}
// Run the parsed callback.
c.onParsed(c, &o)
if existing != nil {
existing(cmd, args)
}
// Set options in context, so custom commands can access it.
cmd.SetContext(context.WithValue(cmd.Context(), optionsKey, &o))
}
// Run the command!
c.root.Execute()
}
func (c *cli[O]) Root() *cobra.Command {
return c.root
}
func (c *cli[O]) OnStart(fn func()) {
c.start = fn
}
func (c *cli[O]) OnStop(fn func()) {
c.stop = fn
}
func (c *cli[O]) setupOptions(t reflect.Type, path []int) {
var err error
flags := c.root.PersistentFlags()
for i := 0; i < t.NumField(); i++ {
field := t.Field(i)
if !field.IsExported() {
// This isn't a public field, so we cannot use reflect.Value.Set with
// it. This is usually a struct field with a lowercase name.
fmt.Fprintln(os.Stderr, "warning: ignoring unexported options field", field.Name)
continue
}
currentPath := append([]int{}, path...)
currentPath = append(currentPath, i)
fieldType := deref(field.Type)
if field.Anonymous {
// Embedded struct. This enables composition from e.g. company defaults.
c.setupOptions(fieldType, currentPath)
continue
}
name := field.Tag.Get("name")
if name == "" {
name = casing.Kebab(field.Name)
}
envName := "SERVICE_" + casing.Snake(name, strings.ToUpper)
defaultValue := field.Tag.Get("default")
if v, ok := os.LookupEnv(envName); ok {
// Env vars will override the default value, which is used to document
// what the value is if no options are passed.
defaultValue = v
}
c.optInfo = append(c.optInfo, option{name, field.Type, currentPath})
switch fieldType.Kind() {
case reflect.String:
flags.StringP(name, field.Tag.Get("short"), defaultValue, field.Tag.Get("doc"))
case reflect.Int, reflect.Int64:
var def int64
if defaultValue != "" {
if fieldType == durationType {
var t time.Duration
t, err = time.ParseDuration(defaultValue)
def = int64(t)
} else {
def, err = strconv.ParseInt(defaultValue, 10, 64)
}
if err != nil {
panic(err)
}
}
if fieldType == durationType {
flags.DurationP(name, field.Tag.Get("short"), time.Duration(def), field.Tag.Get("doc"))
} else {
flags.Int64P(name, field.Tag.Get("short"), def, field.Tag.Get("doc"))
}
case reflect.Bool:
var def bool
if defaultValue != "" {
def, err = strconv.ParseBool(defaultValue)
if err != nil {
panic(err)
}
}
flags.BoolP(name, field.Tag.Get("short"), def, field.Tag.Get("doc"))
default:
panic("Unsupported option type: " + field.Type.Kind().String())
}
}
}
// New creates a new CLI. The `onParsed` callback is called after the command
// options have been parsed and the options struct has been populated. You
// should set up a `hooks.OnStart` callback to start the server with your
// chosen router.
//
// // First, define your input options.
// type Options struct {
// Debug bool `doc:"Enable debug logging"`
// Host string `doc:"Hostname to listen on."`
// Port int `doc:"Port to listen on." short:"p" default:"8888"`
// }
//
// // Then, create the CLI.
// cli := humacli.CLI(func(hooks humacli.Hooks, opts *Options) {
// fmt.Printf("Options are debug:%v host:%v port%v\n",
// opts.Debug, opts.Host, opts.Port)
//
// // Set up the router & API
// router := chi.NewRouter()
// api := humachi.New(router, huma.DefaultConfig("My API", "1.0.0"))
// srv := &http.Server{
// Addr: fmt.Sprintf("%s:%d", opts.Host, opts.Port),
// Handler: router,
// // TODO: Set up timeouts!
// }
//
// hooks.OnStart(func() {
// if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
// log.Fatalf("listen: %s\n", err)
// }
// })
//
// hooks.OnStop(func() {
// srv.Shutdown(context.Background())
// })
// })
//
// // Run the thing!
// cli.Run()
func New[O any](onParsed func(Hooks, *O)) CLI {
c := &cli[O]{
root: &cobra.Command{
Use: filepath.Base(os.Args[0]),
},
onParsed: onParsed,
}
var o O
c.setupOptions(reflect.TypeOf(o), []int{})
c.root.Run = func(cmd *cobra.Command, args []string) {
done := make(chan struct{}, 1)
if c.start != nil {
go func() {
c.start()
done <- struct{}{}
}()
}
// Handle graceful shutdown.
quit := make(chan os.Signal, 1)
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
select {
case <-done:
// Server is done, just exit.
case <-quit:
if c.stop != nil {
fmt.Fprintln(os.Stderr, "Gracefully shutting down the server...")
c.stop()
}
}
}
return c
}