-
Notifications
You must be signed in to change notification settings - Fork 1.5k
/
Copy pathroot.go
130 lines (106 loc) · 4.68 KB
/
root.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
/*
Copyright 2021 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package cli
import (
"fmt"
"sort"
"strings"
"github.com/spf13/cobra"
)
const (
pluginKeysHeader = "Plugin keys"
projectVersionsHeader = "Supported project versions"
)
func (c CLI) newRootCmd() *cobra.Command {
cmd := &cobra.Command{
Use: c.commandName,
Long: `CLI tool for building Kubernetes extensions and tools.
`,
Example: c.rootExamples(),
RunE: func(cmd *cobra.Command, args []string) error {
return cmd.Help()
},
}
// Global flags for all subcommands
// NOTE: the current plugin resolution doesn't allow to provide values to this flag different to those configured
// for the project, so default values need to be empty and considered when these two sources are compared.
// Another approach would be to allow users to overwrite the project configuration values. In this case, flags
// would take precedence over project configuration, which would take precedence over CLI defaults.
fs := cmd.PersistentFlags()
fs.String(projectVersionFlag, "", "project version")
fs.StringSlice(pluginsFlag, nil, "plugin keys of the plugin to initialize the project with")
// As the root command will be used to shot the help message under some error conditions,
// like during plugin resolving, we need to allow unknown flags to prevent parsing errors.
cmd.FParseErrWhitelist = cobra.FParseErrWhitelist{UnknownFlags: true}
return cmd
}
// rootExamples builds the examples string for the root command
func (c CLI) rootExamples() string {
str := fmt.Sprintf(`The first step is to initialize your project:
%[1]s init --project-version=<PROJECT VERSION> --plugins=<PLUGIN KEYS>
<PLUGIN KEYS> is a comma-separated list of plugin keys from the following table
and <PROJECT VERSION> a supported project version for these plugins.
%[2]s
For more specific help for the init command of a certain plugins and project version
configuration please run:
%[1]s init --help --project-version=<PROJECT VERSION> --plugins=<PLUGIN KEYS>
`,
c.commandName, c.getPluginTable())
str += fmt.Sprintf("\nDefault project version: %s\n", c.defaultProjectVersion)
if defaultPlugins, hasDefaultPlugins := c.defaultPlugins[c.defaultProjectVersion]; hasDefaultPlugins {
str += fmt.Sprintf("Default plugin keys: %q\n", strings.Join(defaultPlugins, ","))
}
str += fmt.Sprintf(`
After the project has been initialized, run
%[1]s --help
to obtain further info about available commands.`,
c.commandName)
return str
}
// getPluginTable returns an ASCII table of the available plugins and their supported project versions.
func (c CLI) getPluginTable() string {
var (
maxPluginKeyLength = len(pluginKeysHeader)
pluginKeys = make([]string, 0, len(c.plugins))
maxProjectVersionLength = len(projectVersionsHeader)
projectVersions = make(map[string]string, len(c.plugins))
)
for pluginKey, plugin := range c.plugins {
if len(pluginKey) > maxPluginKeyLength {
maxPluginKeyLength = len(pluginKey)
}
pluginKeys = append(pluginKeys, pluginKey)
supportedProjectVersions := plugin.SupportedProjectVersions()
supportedProjectVersionStrs := make([]string, 0, len(supportedProjectVersions))
for _, version := range supportedProjectVersions {
supportedProjectVersionStrs = append(supportedProjectVersionStrs, version.String())
}
supportedProjectVersionsStr := strings.Join(supportedProjectVersionStrs, ", ")
if len(supportedProjectVersionsStr) > maxProjectVersionLength {
maxProjectVersionLength = len(supportedProjectVersionsStr)
}
projectVersions[pluginKey] = supportedProjectVersionsStr
}
lines := make([]string, 0, len(c.plugins)+2)
lines = append(lines, fmt.Sprintf(" %[1]*[2]s | %[3]*[4]s",
maxPluginKeyLength, pluginKeysHeader, maxProjectVersionLength, projectVersionsHeader))
lines = append(lines, strings.Repeat("-", maxPluginKeyLength+2)+"+"+
strings.Repeat("-", maxProjectVersionLength+2))
sort.Strings(pluginKeys)
for _, pluginKey := range pluginKeys {
supportedProjectVersions := projectVersions[pluginKey]
lines = append(lines, fmt.Sprintf(" %[1]*[2]s | %[3]*[4]s",
maxPluginKeyLength, pluginKey, maxProjectVersionLength, supportedProjectVersions))
}
return strings.Join(lines, "\n")
}