-
Notifications
You must be signed in to change notification settings - Fork 53
/
Copy pathdocopts.go
479 lines (428 loc) · 14.2 KB
/
docopts.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
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
// vim: set ts=4 sw=4 sts=4 noet:
//
// docopts.go is a command line wrapper for docopt.go to be used by bash scripts.
//
package main
import (
"fmt"
"github.com/docopt/docopt-go"
"io"
"io/ioutil"
"os"
"reflect"
"regexp"
"sort"
"strings"
)
// vars defined at compile time
// https://github.com/ahmetb/govvv
var (
Version string
BuildDate string
GitCommit string
GoBuildVersion string
)
var copyleft = `
Copyleft (Ɔ) 2022 Sylvain Viart (golang version).
Copyright (C) 2013 Vladimir Keleshev, Lari Rasku.
License MIT <http://opensource.org/licenses/MIT>.
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.
`
// Version will be build by main() function
var Docopts_Version string
var Usage string = `Shell interface for docopt, the CLI description language.
Usage:
docopts [options] -h <msg> : [<argv>...]
docopts [options] [--no-declare] -A <name> -h <msg> : [<argv>...]
docopts [options] -G <prefix> -h <msg> : [<argv>...]
docopts [options] --no-mangle -h <msg> : [<argv>...]
Options:
-h <msg>, --help=<msg> The help message in docopt format.
Without argument outputs this help.
If - is given, read the help message from
standard input.
If no argument is given, print docopts's own
help message and quit.
-V <msg>, --version=<msg> A version message.
If - is given, read the version message from
standard input. If the help message is also
read from standard input, it is read first.
If no argument is given, print docopts's own
version message and quit.
-s <str>, --separator=<str> The string to use to separate the help message
from the version message when both are given
via standard input. [default: ----]
-O, --options-first Disallow interspersing options and positional
arguments: all arguments starting from the
first one that does not begin with a dash will
be treated as positional arguments.
-H, --no-help Don't handle --help and --version specially.
-A <name> Export the arguments as a Bash 4+ associative
array called <name>.
-G <prefix> Don't use associative array but output
Bash 3.2 compatible GLOBAL variables assignment:
<prefix>_{mangled_args}={parsed_value}
Can be used with numeric incompatible options
as well. See also: --no-mangle
--no-mangle Output parsed option not suitable for bash eval.
Full option names are kept. Rvalue is still
shellquoted. Extra parsing is required.
--no-declare Don't output 'declare -A <name>', used only
with -A argument.
--debug Output extra parsing information for debugging.
Output cannot be used in bash eval.
`
// testing trick, out can be mocked to catch stdout and validate
// https://stackoverflow.com/questions/34462355/how-to-deal-with-the-fmt-golang-library-package-for-cli-testing
var out io.Writer = os.Stdout
// debug helper
func print_args(args docopt.Opts, message string) {
fmt.Printf("################## %s ##################\n", message)
for _, key := range Sort_args_keys(args) {
fmt.Printf("%20s : %v\n", key, args[key])
}
}
func Sort_args_keys(args docopt.Opts) []string {
keys_list := make([]string, len(args))
i := 0
for k, _ := range args {
keys_list[i] = k
i++
}
sort.Strings(keys_list)
return keys_list
}
// Store global behavior to avoid passing many optional arguments to methods.
type Docopts struct {
Global_prefix string
Mangle_key bool
Output_declare bool
Exit_function bool
}
// output bash 4+ compatible assoc array, suitable for eval.
func (d *Docopts) Print_bash_args(bash_assoc string, args docopt.Opts) {
// Reuse python's fake nested Bash arrays for repeatable arguments with values.
// The structure is:
// bash_assoc[key,#]=length
// bash_assoc[key,i]=value
// 'i' is an integer from 0 to length-1
// length can be 0, for empty array
if d.Output_declare {
fmt.Fprintf(out, "declare -A %s\n", bash_assoc)
}
for _, key := range Sort_args_keys(args) {
value := args[key]
// some golang tricks here using reflection to loop over the map[]
rt := reflect.TypeOf(value)
if IsArray(rt) {
// all array is outputed even 0 size
val_arr := value.([]string)
for index, v := range val_arr {
fmt.Fprintf(out, "%s['%s,%d']=%s\n", bash_assoc, Shellquote(key), index, To_bash(v))
}
// size of the array
fmt.Fprintf(out, "%s['%s,#']=%d\n", bash_assoc, Shellquote(key), len(val_arr))
} else {
// value is not an array
fmt.Fprintf(out, "%s['%s']=%s\n", bash_assoc, Shellquote(key), To_bash(value))
}
}
}
// Check if a value is an array
func IsArray(rt reflect.Type) bool {
if rt == nil {
return false
}
switch rt.Kind() {
case reflect.Slice:
return true
case reflect.Array:
return true
default:
return false
}
}
func Shellquote(s string) string {
return strings.Replace(s, "'", `'\''`, -1)
}
func IsBashIdentifier(s string) bool {
identifier := regexp.MustCompile(`^([A-Za-z]|[A-Za-z_][0-9A-Za-z_]+)$`)
return identifier.MatchString(s)
}
// Convert a parsed type to a text string suitable for bash eval
// as a right-hand side of an assignment.
// Handles quoting for string, no quote for number or bool.
func To_bash(v interface{}) string {
var s string
switch v.(type) {
case bool:
s = fmt.Sprintf("%v", v.(bool))
case int:
s = fmt.Sprintf("%d", v.(int))
case string:
s = fmt.Sprintf("'%s'", Shellquote(v.(string)))
case []string:
arr := v.([]string)
if len(arr) == 0 {
// bash empty array
s = "()"
} else {
// escape all strings
arr_out := make([]string, len(arr))
for i, e := range arr {
arr_out[i] = Shellquote(e)
}
s = fmt.Sprintf("('%s')", strings.Join(arr_out[:], "' '"))
}
case nil:
s = ""
default:
panic(fmt.Sprintf("To_bash():unsuported type: %v for '%v'", reflect.TypeOf(v), v))
}
return s
}
// Performs output for bash Globals (not bash 4+ assoc) Names are mangled to become
// suitable for bash eval.
// If Docopts.Mangle_key is false: simply print left-hand side assignment verbatim.
// used for --no-mangle
func (d *Docopts) Print_bash_global(args docopt.Opts) error {
var new_name string
var err error
var out_buf string
varmap := make(map[string]string)
// docopt.Opts is of type map[string]interface{}
// so value is an interface{}
for _, key := range Sort_args_keys(args) {
if d.Mangle_key {
if key == "--" && d.Global_prefix == "" {
// skip double-dash that can't be mangled #52
// so double-dash is not printed for bash
// but still parsed by docopts
continue
}
new_name, err = d.Name_mangle(key)
if err != nil {
return err
}
} else {
// --no-mangle option behavior
new_name = key
}
// test if already present in the map
prev_key, seen := varmap[new_name]
if seen {
return fmt.Errorf("%s: two or more elements have identically mangled names", prev_key)
} else {
varmap[new_name] = key
}
out_buf += fmt.Sprintf("%s=%s\n", new_name, To_bash(args[key]))
}
// final output
fmt.Fprintf(out, "%s", out_buf)
return nil
}
// Transform a parsed option or place-holder name into a bash identifier if possible.
// It Docopts.Global_prefix is prepended if given, wrong prefix may produce invalid
// bash identifier and this method will fail too.
func (d *Docopts) Name_mangle(elem string) (string, error) {
var v string
if d.Global_prefix == "" && (elem == "-" || elem == "--") {
return "", fmt.Errorf("Mangling not supported for: '%s'", elem)
}
if Match(`^<.*>$`, elem) {
v = elem[1 : len(elem)-1]
} else if Match(`^-[^-]$`, elem) {
v = fmt.Sprintf("%c", elem[1])
} else if Match(`^--.+$`, elem) {
v = elem[2:]
} else {
// also this case for '-' when d.Global_prefix != ""
v = elem
}
// alter output if we have a prefix
key_fmt := "%s"
if d.Global_prefix != "" {
key_fmt = fmt.Sprintf("%s_%%s", d.Global_prefix)
}
v = fmt.Sprintf(key_fmt, strings.Replace(v, "-", "_", -1))
if !IsBashIdentifier(v) {
return "", fmt.Errorf("cannot transform into a bash identifier: '%s' => '%s'", elem, v)
}
return v, nil
}
// helper for lazy typing
func Match(regex string, source string) bool {
matched, _ := regexp.MatchString(regex, source)
return matched
}
// Experimental: Change bash exit source code based on '--function' parameter
func (d *Docopts) Get_exit_code(exit_code int) (str_code string) {
if d.Exit_function {
str_code = fmt.Sprintf("return %d", exit_code)
} else {
str_code = fmt.Sprintf("exit %d", exit_code)
}
return
}
// Our HelpHandler which outputs bash source code to be evaled as error and stop or
// display program's help or version.
func (d *Docopts) HelpHandler_for_bash_eval(err error, usage string) {
if err != nil {
fmt.Printf("echo 'error: %s\n%s' >&2\n%s\n",
Shellquote(err.Error()),
Shellquote(usage),
d.Get_exit_code(64),
)
os.Exit(1)
} else {
// --help or --version found and --no-help was not given
fmt.Printf("echo '%s'\n%s\n", Shellquote(usage), d.Get_exit_code(0))
os.Exit(0)
}
}
// HelpHandler for go parser which parses docopts options. See: HelpHandler_for_bash_eval for parsing
// bash options. This handler is called when docopts itself detects a parse error on docopts usage.
// If docopts parsing is OK, then HelpHandler_for_bash_eval will be called by a second parser based on the
// help string given with -h <msg> or --help=<msg>. This behavior is a legacy behavior from docopts python
// previous version. This introduce strange hack in option parsing and may be changed after initial docopts go
// version release.
func HelpHandler_golang(err error, usage string) {
if err != nil {
err_str := err.Error()
// we hack for our polymorphic argument -h or -V
// it was the same hack in python version
if len(err_str) >= 9 {
if err_str[0:2] == "-h" || err_str[0:6] == "--help" {
// print full usage message (global var)
fmt.Println(strings.TrimSpace(Usage))
os.Exit(0)
}
if err_str[0:2] == "-V" || err_str[0:9] == "--version" {
fmt.Println(strings.TrimSpace(Docopts_Version))
os.Exit(0)
}
}
// When we have an error with err_str empty, this is a special case:
// we received an usage string which MUST receive an argument and no argument has been
// given by the user. So this is a valid, from golang point of view but not for bash.
if len(err_str) == 0 {
// no arg at all, display small usage, also exits 1
d := &Docopts{Exit_function: false}
d.HelpHandler_for_bash_eval(fmt.Errorf("no argument"), usage)
}
// real error
fmt.Fprintf(os.Stderr, "my error: %v, %v\n", err, usage)
os.Exit(1)
} else {
// no error, never reached?
fmt.Println(usage)
os.Exit(0)
}
}
func docopts_error(msg string, err error) {
if err != nil {
msg = fmt.Sprintf(msg, err)
}
fmt.Fprintf(os.Stderr, "docopts:error: %s\n", msg)
os.Exit(1)
}
func main() {
golang_parser := &docopt.Parser{
OptionsFirst: true,
SkipHelpFlags: true,
HelpHandler: HelpHandler_golang,
}
// build Docopts_Version string
Docopts_Version = fmt.Sprintf("docopts %s commit %s built at %s\nbuilt from: %s\n%s",
Version,
GitCommit,
BuildDate,
GoBuildVersion,
strings.TrimSpace(copyleft))
arguments, err := golang_parser.ParseArgs(Usage, nil, Docopts_Version)
if err != nil {
msg := fmt.Sprintf("mypanic: %v\n", err)
panic(msg)
}
debug := arguments["--debug"].(bool)
if debug {
print_args(arguments, "golang")
}
// create our Docopts struct
d := &Docopts{
Global_prefix: "",
Mangle_key: true,
Output_declare: true,
// Exit_function is experimental
Exit_function: false,
}
// parse docopts's own arguments
argv := arguments["<argv>"].([]string)
doc := arguments["--help"].(string)
bash_version, _ := arguments.String("--version")
options_first := arguments["--options-first"].(bool)
no_help := arguments["--no-help"].(bool)
separator := arguments["--separator"].(string)
d.Mangle_key = !arguments["--no-mangle"].(bool)
d.Output_declare = !arguments["--no-declare"].(bool)
global_prefix, err := arguments.String("-G")
if err == nil {
d.Global_prefix = global_prefix
}
// read from stdin
if doc == "-" && bash_version == "-" {
bytes, _ := ioutil.ReadAll(os.Stdin)
arr := strings.Split(string(bytes), separator)
if len(arr) == 2 {
doc, bash_version = arr[0], arr[1]
} else {
msg := "error: help + version stdin, not found"
if debug {
msg += fmt.Sprintf("\nseparator is: '%s'\n", separator)
msg += fmt.Sprintf("spliting has given %d blocs, exactly 2 are expected\n", len(arr))
}
panic(msg)
}
} else if doc == "-" {
bytes, _ := ioutil.ReadAll(os.Stdin)
doc = string(bytes)
} else if bash_version == "-" {
bytes, _ := ioutil.ReadAll(os.Stdin)
bash_version = string(bytes)
}
doc = strings.TrimSpace(doc)
bash_version = strings.TrimSpace(bash_version)
if debug {
fmt.Printf("%20s : %v\n", "doc", doc)
fmt.Printf("%20s : %v\n", "bash_version", bash_version)
}
// now parses bash program's arguments
parser := &docopt.Parser{
HelpHandler: d.HelpHandler_for_bash_eval,
OptionsFirst: options_first,
SkipHelpFlags: no_help,
}
bash_args, err := parser.ParseArgs(doc, argv, bash_version)
if err == nil {
if debug {
print_args(bash_args, "bash")
fmt.Println("----------------------------------------")
}
name, err := arguments.String("-A")
if err == nil {
if !IsBashIdentifier(name) {
fmt.Printf("-A: not a valid Bash identifier: '%s'", name)
return
}
d.Print_bash_args(name, bash_args)
} else {
err = d.Print_bash_global(bash_args)
if err != nil {
docopts_error("Print_bash_global:%v", err)
}
}
} else {
panic(err)
}
}