forked from SchwarzIT/go-template
-
Notifications
You must be signed in to change notification settings - Fork 0
/
options.go
391 lines (345 loc) · 12.3 KB
/
options.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
package gotemplate
import (
"bytes"
"fmt"
"os"
"os/exec"
"path"
"regexp"
"strings"
"github.com/schwarzit/go-template/pkg/repos"
)
// nolint: lll // official regex for semver patterns that can't be broken up into multiple lines
const semverRegex = `^(?P<major>0|[1-9]\d*)\.(?P<minor>0|[1-9]\d*)\.(?P<patch>0|[1-9]\d*)(?:-(?P<prerelease>(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+(?P<buildmetadata>[0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$`
type ErrOutOfRange struct {
Value int
Min int
Max int
}
func (e *ErrOutOfRange) Error() string {
return fmt.Sprintf("%d: value out of range (min: %d, max: %d)", e.Value, e.Min, e.Max)
}
// ErrInvalidaPattern indicates that an error occurred while matching
// a value with a pattern.
// The pattern as well as a description for the pattern is included in the error message.
type ErrInvalidPattern struct {
Value string
Pattern string
Description string
}
func (e *ErrInvalidPattern) Error() string {
return fmt.Sprintf("%s: invalid pattern (expected %s (pattern: %s))", e.Value, e.Description, e.Pattern)
}
// Validator is a single method interface that validates that a given value is valid.
// If any error happens during validation or if the value is not valid an error will be returned.
type Validator interface {
Validate(value interface{}) error
}
// ValidatorFunc is a function implementing the Validator interface.
type ValidatorFunc func(value interface{}) error
func (f ValidatorFunc) Validate(value interface{}) error {
return f(value)
}
// Option is a struct containing all needed configuration for options to customize the template.
type Option struct {
// name is the name of the option that will be used to reference it and also that will be shown on the cli.
name string
// description is the description of the option that should be shown.
// It's a StringValuer since it could depend on some earlier input.
description StringValuer
// defaultValue is the default value of the option.
// It's a Valuer since it could be depend on earlier inputs or some http call.
defaultValue Valuer
// validator is used to validate an input value if it can be used as the value for this option.
// If it is not set it will by default by valid.
validator Validator
// shouldDisplay decides whether the option is shown when the values are loaded interactively.
// In most cases this is used to ensure options are only shown if needed values have been supplied earlier.
// If it is not set it will by default be shown.
shouldDisplay BoolValuer
// postHook is some function that will be executed after all options are loaded.
// This can for example be used to remove files from the created project folder or initialize tools based on inputs.
// The passed interface contains the value of the option for convenience (technically also contained in optionValues)
// targetDir indicates the working directory of the postHook
postHook PostHookFunc
}
type PostHookFunc func(value interface{}, optionValues *OptionValues, targetDir string) error
func NewOption(name string, description StringValuer, defaultValue Valuer, opts ...NewOptionOption) Option {
option := Option{
name: name,
description: description,
defaultValue: defaultValue,
}
for _, opt := range opts {
opt(&option)
}
return option
}
type NewOptionOption func(*Option)
func WithValidator(validator Validator) NewOptionOption {
return func(o *Option) {
o.validator = validator
}
}
func WithShouldDisplay(shouldDisplay BoolValuer) NewOptionOption {
return func(o *Option) {
o.shouldDisplay = shouldDisplay
}
}
func WithPosthook(postHook PostHookFunc) NewOptionOption {
return func(o *Option) {
o.postHook = postHook
}
}
func (s *Option) Name() string {
return s.name
}
func (s *Option) Description(currentValues *OptionValues) string {
return s.description.Value(currentValues)
}
// Default either returns the default value (possibly calculated with currentValues).
func (s *Option) Default(currentValues *OptionValues) interface{} {
return s.defaultValue.Value(currentValues)
}
// ShouldDisplay returns a bool value indicating whether the option should be shown or not.
// If shouldDisplay variable is not set on the option true is returned.
func (s *Option) ShouldDisplay(currentValues *OptionValues) bool {
if s.shouldDisplay != nil {
return s.shouldDisplay.Value(currentValues)
}
return true
}
// Validate validates the value if a validator is specified.
func (s *Option) Validate(value interface{}) error {
if s.validator != nil {
return s.validator.Validate(value)
}
return nil
}
// PostHook executes the registered postHook if there is any.
func (s *Option) PostHook(v interface{}, optionValues *OptionValues, targetDir string) error {
if s.postHook != nil {
return s.postHook(v, optionValues, targetDir)
}
return nil
}
// Category is used to wrap multiple extensions into one organizational unit.
// This is to reduce the amount of required user input if certain categories if extensions
// can be skipped as a category instead of needing to skip all one by one.
type Category struct {
Name string
Options []Option
}
// Options is the main struct wrapping the configuration
// for all allowed parameters and extensions.
// Slices are used instead of maps since the iteration order of maps is undefined/random.
// which could lead to confusion.
type Options struct {
Base []Option
Extensions []Category
}
// OptionValues is a struct mirroring the structure of Options but using maps.
// Instead of the whole option only the set value of the option is kept.
// This makes looking up already supplied option values easier than it would
// be in the Options struct.
type OptionValues struct {
Base OptionNameToValue `yaml:"base"`
Extensions map[string]OptionNameToValue `yaml:"extensions"`
}
func NewOptionValues() *OptionValues {
return &OptionValues{
Base: OptionNameToValue{},
Extensions: map[string]OptionNameToValue{},
}
}
type OptionNameToValue map[string]interface{}
// NewOptions returns all of go/template's options.
func NewOptions(githubTagLister repos.GithubTagLister) *Options { // nolint: funlen // Static initialization
return &Options{
Base: []Option{
{
name: "projectName",
defaultValue: StaticValue("Awesome Project"),
description: StringValue("Name of the project"),
},
{
name: "projectSlug",
defaultValue: DynamicValue(func(ov *OptionValues) interface{} {
projectName := ov.Base["projectName"].(string)
return strings.ReplaceAll(strings.ToLower(projectName), " ", "-")
}),
description: StringValue("Technical name of the project for folders and names. This will also be used as output directory."),
validator: RegexValidator(`^[a-z1-9]+(-[a-z1-9]+)*$`, "only lowercase letters and dashes"),
},
{
name: "projectDescription",
defaultValue: StaticValue("The awesome project provides awesome features to awesome people."),
description: StringValue("Description of the project used in the README."),
},
{
name: "appName",
defaultValue: StaticValue("awesomecli"),
description: StringValue(`The name of the binary that you want to create.
Could be the same your "projectSlug" but since Go supports multiple apps in one repo it could also be sth. else.
For example if your project is for some API there could be one app for the server and one CLI client.`),
validator: RegexValidator(`^[a-z]+$`, "only lowercase letters"),
},
{
name: "moduleName",
defaultValue: DynamicValue(func(vals *OptionValues) interface{} {
projectSlug := vals.Base["projectSlug"].(string)
return fmt.Sprintf("github.com/user/%s", projectSlug)
}),
description: StringValue(`The name of the Go module defined in the "go.mod" file.
This is used if you want to "go get" the module.
Please be aware that this depends on your version control system.
The default points to "github.com" but for devops for example it would look sth. like this "dev.azure.com/org/project/repo.git"`),
validator: RegexValidator(`^[\S]+$`, "no whitespaces"),
},
{
name: "golangciVersion",
defaultValue: DynamicValue(func(_ *OptionValues) interface{} {
latestTag, err := repos.LatestGithubReleaseTag(githubTagLister, "golangci", "golangci-lint")
if err != nil {
return "1.42.1"
}
return latestTag.String()
}),
description: StringValue("Golangci-lint version to use."),
validator: RegexValidator(
semverRegex,
"valid semver version string",
),
},
},
Extensions: []Category{
{
Name: "openSource",
Options: []Option{
{
name: "license",
defaultValue: StaticValue(1),
description: StringValue(`Set an OpenSource license.
Unsure which to pick? Checkout Github's https://choosealicense.com/
Options:
1: MIT License
2: Apache License 2.0
3: GNU AGPLv3
4: GNU GPLv3
5: GNU LGPLv3
6: Mozilla Public License 2.0
7: Boost Software License 1.0
8: The Unlicense
9: Add no license`),
postHook: func(v interface{}, _ *OptionValues, targetDir string) error {
if v.(int) == 9 {
return os.RemoveAll(path.Join(targetDir, "LICENSE"))
}
return nil
},
},
{
name: "author",
defaultValue: DynamicValue(func(vals *OptionValues) interface{} {
buffer := &bytes.Buffer{}
gitName := exec.Command("git", "config", "--get", "user.name")
gitName.Stdout = buffer
if err := gitName.Run(); err != nil || len(buffer.Bytes()) == 0 {
return "Marty Mc Fly"
}
return strings.TrimSpace(buffer.String())
}),
description: StringValue(`License author`),
shouldDisplay: DynamicBoolValue(func(vals *OptionValues) bool {
switch vals.Extensions["openSource"]["license"].(int) {
case 1, 2:
return true
}
return false
}),
},
{
name: "codeowner",
defaultValue: DynamicValue(func(vals *OptionValues) interface{} {
buffer := &bytes.Buffer{}
gitMail := exec.Command("git", "config", "--get", "user.email")
gitMail.Stdout = buffer
if err := gitMail.Run(); err != nil || len(buffer.Bytes()) == 0 {
return "Marty.Mc.Fly@future.back"
}
return strings.TrimSpace(buffer.String())
}),
description: StringValue("Set the codeowner of the project"),
},
},
},
{
Name: "grpc",
Options: []Option{
{
name: "base",
defaultValue: StaticValue(false),
description: StringValue("Base configuration for gRPC"),
postHook: func(v interface{}, _ *OptionValues, targetDir string) error {
set := v.(bool)
files := []string{"api/proto", "tools.go", "buf.gen.yaml", "buf.yaml", "api/openapi.v1.yml"}
if set {
return os.RemoveAll(path.Join(targetDir, "api/openapi.v1.yml"))
}
return removeAllBut(targetDir, files, "api/openapi.v1.yml")
},
},
{
name: "grpcGateway",
defaultValue: StaticValue(false),
description: StringValue("Extend gRPC configuration with grpc-gateway"),
shouldDisplay: DynamicBoolValue(func(vals *OptionValues) bool {
return vals.Extensions["grpc"]["base"].(bool)
}),
},
},
},
},
}
}
// removeAllBut removes all files in the toRemove slice except for the exception.
func removeAllBut(targetDir string, toRemove []string, exception string) error {
for _, item := range toRemove {
if item == exception {
continue
}
if err := os.RemoveAll(path.Join(targetDir, item)); err != nil {
return err
}
}
return nil
}
// RangeValidator validates that value is in between or equal to min and max.
func RangeValidator(min, max int) ValidatorFunc {
return func(value interface{}) error {
val := value.(int)
if val < min || val > max {
return &ErrOutOfRange{
Value: val,
Min: min,
Max: max,
}
}
return nil
}
}
// RegexValidator returns a ValidatorFunc to validate a given value against a regex pattern.
// If the pattern doesn't match a ErrInvalidPattern is returned with a description on what the pattern means.
func RegexValidator(pattern, description string) ValidatorFunc {
return func(value interface{}) error {
str := value.(string)
matched, err := regexp.MatchString(pattern, str)
if err != nil {
return err
}
if !matched {
return &ErrInvalidPattern{Value: str, Pattern: pattern, Description: description}
}
return nil
}
}