/
helper.go
107 lines (102 loc) · 2.77 KB
/
helper.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
package flags
import (
"fmt"
"sort"
"strings"
wrap "gopkg.in/wrap.v1"
)
func formatHelp(name, desc string) string {
desc = wrap.Space(desc, 55)
desc = strings.Replace(desc, "\n", "\n ", -1)
if len(name) < 22 {
return " " + name + strings.Repeat(" ", 22-len(name)) + desc
}
return " " + name + "\n " + desc
}
// Usage creates a usage string for the given argument definitions.
func Usage(pos *Positional, opt *Optional) string {
builder := strings.Builder{}
builder.WriteString("[-h | --help]")
if opt != nil && len(opt.Args) > 0 {
builder.WriteString(" [<args>]")
}
if pos != nil {
for _, name := range pos.Order {
builder.WriteString(fmt.Sprintf(" <%s>", name))
}
if pos.In != nil {
builder.WriteString(" [<infile>]")
}
if pos.Out != nil {
builder.WriteString(" [<outfile>]")
}
}
return builder.String()
}
// Help creaes a help string for the given argument definitions.
func Help(pos *Positional, opt *Optional) string {
parts := []string{}
if pos != nil {
parts = append(parts, "\npositional arguments")
for _, name := range pos.Order {
usage := pos.Args[name].Usage
name = fmt.Sprintf("<%s>", name)
parts = append(parts, formatHelp(name, usage))
}
if pos.In != nil {
usage := wrap.Space(pos.In.Usage, 55)
parts = append(parts, formatHelp("[<infile>]", usage))
}
if pos.Out != nil {
usage := wrap.Space(pos.Out.Usage, 55)
parts = append(parts, formatHelp("[<outfile>]", usage))
}
}
if opt != nil {
parts = append(parts, "\noptional arguments:")
names := []optionalName{}
for long := range opt.Args {
name := optionalName{0, long}
for short := range opt.Alias {
if opt.Alias[short] == long {
name.Short = short
}
}
names = append(names, name)
}
sort.Sort(byShort(names))
for _, name := range names {
long, short := name.Long, name.Short
arg := opt.Args[long]
usage := fmt.Sprintf("%s (value: %s)", arg.Usage, arg.Value)
flag := ""
switch arg.Value.(type) {
case *BoolValue:
flag = "--" + long
if short != 0 {
flag = fmt.Sprintf("-%c, %s", short, flag)
}
case SliceValue:
flags := []string{}
if short != 0 {
flags = append(flags,
fmt.Sprintf("-%[1]c <%[2]s> [<%[2]s> ...]", short, long),
fmt.Sprintf("-%[1]c <%[2]s> [-%[1]c <%[2]s> ...]", short, long),
)
}
flags = append(flags,
fmt.Sprintf("--%[1]s <%[1]s> [<%[1]s> ...]", long),
fmt.Sprintf("--%[1]s <%[1]s> [--%[1]s <%[1]s> ...]", long),
)
flag = strings.Join(flags, ",\n ")
default:
flag = fmt.Sprintf("--%[1]s <%[1]s>", long)
if short != 0 {
flag = fmt.Sprintf("-%c <%s>, %s", short, long, flag)
}
}
parts = append(parts, formatHelp(flag, usage))
}
}
return strings.Join(parts, "\n")
}