diff --git a/.gitignore b/.gitignore index 7cca55a878910921e9e7d8b16f40077e0378c157..5c43daff3ea7556c2ac1204706a6ebaf1f8c958b 100644 --- a/.gitignore +++ b/.gitignore @@ -20,6 +20,12 @@ mock # goland .idea +*/**/.DS_Store # test files app/cmd/test.yaml + +# language +app/i18n/bindata.go +app/i18n/jcli/zh_CN/LC_MESSAGES/jcli.mo +app/i18n/jcli.pot diff --git a/Makefile b/Makefile index dafb5a0dc5dc8be6d4757aeb089fc9131cc4392e..7f04029f147a63cab5f28818b39e18eed61f2b11 100644 --- a/Makefile +++ b/Makefile @@ -3,6 +3,7 @@ CGO_ENABLED = 0 GO := go BUILD_TARGET = build COMMIT := $(shell git rev-parse --short HEAD) +# CHANGE_LOG := $(shell echo -n "$(shell hub release show $(shell hub release --include-drafts -L 1))" | base64) VERSION := dev-$(shell git describe --tags $(shell git rev-list --tags --max-count=1)) BUILDFLAGS = -ldflags "-X github.com/jenkins-zh/jenkins-cli/app.version=$(VERSION) -X github.com/jenkins-zh/jenkins-cli/app.commit=$(COMMIT)" COVERED_MAIN_SRC_FILE=./main @@ -14,15 +15,15 @@ gen-mock: init: gen-mock -darwin: ## Build for OSX +darwin: gen-data GO111MODULE=on CGO_ENABLED=$(CGO_ENABLED) GOOS=darwin GOARCH=amd64 $(GO) $(BUILD_TARGET) $(BUILDFLAGS) -o bin/darwin/$(NAME) $(MAIN_SRC_FILE) chmod +x bin/darwin/$(NAME) -linux: ## Build for linux +linux: gen-data CGO_ENABLED=$(CGO_ENABLED) GOOS=linux GOARCH=amd64 $(GO) $(BUILD_TARGET) $(BUILDFLAGS) -o bin/linux/$(NAME) $(MAIN_SRC_FILE) chmod +x bin/linux/$(NAME) -win: ## Build for windows +win: gen-data go get github.com/inconshreveable/mousetrap go get github.com/mattn/go-isatty CGO_ENABLED=$(CGO_ENABLED) GOOS=windows GOARCH=386 $(GO) $(BUILD_TARGET) $(BUILDFLAGS) -o bin/windows/$(NAME).exe $(MAIN_SRC_FILE) @@ -45,14 +46,28 @@ clean: ## Clean the generated artifacts copy: darwin sudo cp bin/darwin/$(NAME) $(shell which jcli) -tools: +tools: i18n-tools go get -u golang.org/x/lint/golint +i18n-tools: + go get github.com/gosexy/gettext/go-xgettext + go get github.com/go-bindata/go-bindata/... + verify: go vet ./... - golint -set_exit_status ./... + golint -set_exit_status app/cmd/... + golint -set_exit_status app/helper/... + golint -set_exit_status app/i18n/i18n.go + golint -set_exit_status app/. + golint -set_exit_status client/... + golint -set_exit_status util/... + +fmt: + go fmt ./util/... + go fmt ./client/... + go fmt ./app/... -test: verify fmt +test: gen-data verify fmt mkdir -p bin go vet ./... go test ./... -v -coverprofile coverage.out @@ -65,7 +80,9 @@ dep: go get gopkg.in/yaml.v2 go get github.com/Pallinder/go-randomdata -fmt: - go fmt ./util/... - go fmt ./client/... - go fmt ./app/... +JCLI_FILES="app/cmd/*.go" +gettext: + go-xgettext -k=i18n.T "${JCLI_FILES}" > app/i18n/jcli.pot + +gen-data: + cd app/i18n && go-bindata -o bindata.go -pkg i18n jcli/zh_CN/LC_MESSAGES/ diff --git a/app/cmd/center.go b/app/cmd/center.go index b811541dc7ec39ad621f21acec809ab4fb58b2e9..e332e709f50ee4f600b7063d073764cb27a0d868 100644 --- a/app/cmd/center.go +++ b/app/cmd/center.go @@ -4,6 +4,8 @@ import ( "fmt" "net/http" + "github.com/jenkins-zh/jenkins-cli/app/i18n" + "github.com/jenkins-zh/jenkins-cli/app/helper" "github.com/jenkins-zh/jenkins-cli/client" @@ -26,13 +28,14 @@ func init() { var centerCmd = &cobra.Command{ Use: "center", - Short: "Manage your update center", - Long: `Manage your update center`, - Run: func(cmd *cobra.Command, _ []string) { + Short: i18n.T("Manage your update center"), + Long: i18n.T("Manage your update center"), + RunE: func(cmd *cobra.Command, _ []string) (err error) { jenkins := getCurrentJenkinsFromOptionsOrDie() printJenkinsStatus(jenkins, cmd, centerOption.RoundTripper) - printUpdateCenter(jenkins, cmd, centerOption.RoundTripper) + _, err = printUpdateCenter(jenkins, cmd, centerOption.RoundTripper) + return }, } diff --git a/app/cmd/center_download.go b/app/cmd/center_download.go index 3a9f02d4dc8fd3ad402facb7ee550a483fb1e1d1..650d707d8e0b46963828d196b255653409b5a3d7 100644 --- a/app/cmd/center_download.go +++ b/app/cmd/center_download.go @@ -4,6 +4,7 @@ import ( "net/http" "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/client" "github.com/spf13/cobra" @@ -25,18 +26,22 @@ var centerDownloadOption CenterDownloadOption func init() { centerCmd.AddCommand(centerDownloadCmd) - centerDownloadCmd.Flags().BoolVarP(¢erDownloadOption.LTS, "lts", "", true, "If you want to download Jenkins as LTS") + centerDownloadCmd.Flags().BoolVarP(¢erDownloadOption.LTS, "lts", "", true, + i18n.T("If you want to download Jenkins as LTS")) centerDownloadCmd.Flags().StringVarP(¢erDownloadOption.Version, "war-version", "", "", - "Version of the Jenkins which you want to download") - centerDownloadCmd.Flags().StringVarP(¢erDownloadOption.Mirror, "mirror", "m", "default", "The mirror site of Jenkins") - centerDownloadCmd.Flags().BoolVarP(¢erDownloadOption.ShowProgress, "progress", "p", true, "If you want to show the download progress") - centerDownloadCmd.Flags().StringVarP(¢erDownloadOption.Output, "output", "o", "jenkins.war", "The file of output") + i18n.T("Version of the Jenkins which you want to download")) + centerDownloadCmd.Flags().StringVarP(¢erDownloadOption.Mirror, "mirror", "m", "default", + i18n.T("The mirror site of Jenkins")) + centerDownloadCmd.Flags().BoolVarP(¢erDownloadOption.ShowProgress, "progress", "p", true, + i18n.T("If you want to show the download progress")) + centerDownloadCmd.Flags().StringVarP(¢erDownloadOption.Output, "output", "o", "jenkins.war", + i18n.T("The file of output")) } var centerDownloadCmd = &cobra.Command{ Use: "download", - Short: "Download Jenkins", - Long: `Download Jenkins from a mirror site. You can get more mirror sites from https://jenkins-zh.cn/tutorial/management/mirror/`, + Short: i18n.T("Download Jenkins"), + Long: i18n.T(`Download Jenkins from a mirror site. You can get more mirror sites from https://jenkins-zh.cn/tutorial/management/mirror/`), Run: func(cmd *cobra.Command, _ []string) { mirrorSite := getMirror(centerDownloadOption.Mirror) if mirrorSite == "" { diff --git a/app/cmd/center_upgrade.go b/app/cmd/center_upgrade.go index ef72330067652b1db1bf6f1b534c56e9ad8f5cee..9d263d44281708ce0ce4dec092e643b7e76ecd49 100644 --- a/app/cmd/center_upgrade.go +++ b/app/cmd/center_upgrade.go @@ -3,7 +3,7 @@ package cmd import ( "net/http" - "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/client" "github.com/spf13/cobra" @@ -22,9 +22,9 @@ func init() { var centerUpgradeCmd = &cobra.Command{ Use: "upgrade", - Short: "Upgrade your Jenkins", - Long: `Upgrade your Jenkins`, - Run: func(cmd *cobra.Command, _ []string) { + Short: i18n.T("Upgrade your Jenkins"), + Long: i18n.T("Upgrade your Jenkins"), + RunE: func(cmd *cobra.Command, _ []string) (err error) { jenkins := getCurrentJenkinsFromOptionsOrDie() jclient := &client.UpdateCenterManager{ @@ -38,7 +38,7 @@ var centerUpgradeCmd = &cobra.Command{ jclient.Proxy = jenkins.Proxy jclient.ProxyAuth = jenkins.ProxyAuth - err := jclient.Upgrade() - helper.CheckErr(cmd, err) + err = jclient.Upgrade() + return }, } diff --git a/app/cmd/center_watch.go b/app/cmd/center_watch.go index 1979ef0f7d1a7a66b854081976788e0d48e8ae56..0cce90d6af5286dcf7feadd43b5316eabeea464d 100644 --- a/app/cmd/center_watch.go +++ b/app/cmd/center_watch.go @@ -4,6 +4,7 @@ import ( "net/http" "time" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/client" "github.com/spf13/cobra" ) @@ -23,9 +24,9 @@ var centerWatchOption CenterWatchOption func init() { centerCmd.AddCommand(centerWatchCmd) centerWatchCmd.Flags().BoolVarP(¢erWatchOption.UtilNeedRestart, "util-need-restart", "", false, - "The watch will be continue util Jenkins needs restart") + i18n.T("The watch will be continue util Jenkins needs restart")) centerWatchCmd.Flags().BoolVarP(¢erWatchOption.UtilInstallComplete, "util-install-complete", "", false, - "The watch will be continue util all Jenkins plugins installation is completed") + i18n.T("The watch will be continue util all Jenkins plugins installation is completed")) centerWatchOption.SetFlag(centerWatchCmd) } diff --git a/app/cmd/common.go b/app/cmd/common.go index fbe0ee6621038cf6ad79fd74e74c71f10728966e..606b835fd240c3858639e79b0c82814785e1bb94 100644 --- a/app/cmd/common.go +++ b/app/cmd/common.go @@ -5,6 +5,7 @@ import ( "fmt" "github.com/AlecAivazis/survey/v2" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/client" "github.com/spf13/cobra" "gopkg.in/yaml.v2" @@ -96,7 +97,8 @@ type InteractiveOption struct { // SetFlag set the option flag to this cmd func (b *InteractiveOption) SetFlag(cmd *cobra.Command) { - cmd.Flags().BoolVarP(&b.Interactive, "interactive", "i", false, "Interactive mode") + cmd.Flags().BoolVarP(&b.Interactive, "interactive", "i", false, + i18n.T("Interactive mode")) } // HookOption is the option whether skip command hook diff --git a/app/cmd/completion.go b/app/cmd/completion.go index 96156434cb1ad2245a67442abc50d1ed8e4d626e..289ee7a47611644c8f4ba98ed9dd3ecdbd22a74d 100644 --- a/app/cmd/completion.go +++ b/app/cmd/completion.go @@ -1,6 +1,7 @@ package cmd import ( + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/spf13/cobra" ) @@ -10,8 +11,8 @@ func init() { var completionCmd = &cobra.Command{ Use: "completion", - Short: "Genereate bash completion scripts", - Long: `Genereate bash completion scripts`, + Short: i18n.T("Genereate bash completion scripts"), + Long: i18n.T("Genereate bash completion scripts"), Example: `# Installing bash completion on macOS using homebrew ## If running Bash 3.2 included with macOS brew install bash-completion @@ -20,7 +21,7 @@ var completionCmd = &cobra.Command{ ## you may need add the completion to your completion directory jcli completion > $(brew --prefix)/etc/bash_completion.d/jcli ## If you get trouble, please visit https://github.com/jenkins-zh/jenkins-cli/issues/83.`, - Run: func(cmd *cobra.Command, _ []string) { - rootCmd.GenBashCompletion(cmd.OutOrStdout()) + RunE: func(cmd *cobra.Command, _ []string) error { + return rootCmd.GenBashCompletion(cmd.OutOrStdout()) }, } diff --git a/app/cmd/config.go b/app/cmd/config.go index 800a1885570f64e1593a8d8735e7072c6009e08b..a027681861d0ac543f972d79b127cf0d1340c0dd 100644 --- a/app/cmd/config.go +++ b/app/cmd/config.go @@ -2,13 +2,17 @@ package cmd import ( "fmt" + + "github.com/jenkins-zh/jenkins-cli/app/i18n" + + "io/ioutil" + "log" + "os" + "github.com/mitchellh/go-homedir" "github.com/spf13/cobra" "go.uber.org/zap" "gopkg.in/yaml.v2" - "io/ioutil" - "log" - "os" ) // ConfigOptions is the config cmd option @@ -25,8 +29,8 @@ func init() { var configCmd = &cobra.Command{ Use: "config", Aliases: []string{"cfg"}, - Short: "Manage the config of jcli", - Long: `Manage the config of jcli`, + Short: i18n.T("Manage the config of jcli"), + Long: i18n.T("Manage the config of jcli"), Run: func(cmd *cobra.Command, _ []string) { current := getCurrentJenkins() if current.Description != "" { diff --git a/app/cmd/config_add.go b/app/cmd/config_add.go index 0d2d850716ecaae512023fb68caa792e14558df0..2a9b5d1e4a8fb00037e2e65b94f6724fc9570240 100644 --- a/app/cmd/config_add.go +++ b/app/cmd/config_add.go @@ -4,6 +4,8 @@ import ( "fmt" "log" + "github.com/jenkins-zh/jenkins-cli/app/i18n" + "github.com/spf13/cobra" ) @@ -16,13 +18,20 @@ var configAddOptions ConfigAddOptions func init() { configCmd.AddCommand(configAddCmd) - configAddCmd.Flags().StringVarP(&configAddOptions.Name, "name", "n", "", "Name of the Jenkins") - configAddCmd.Flags().StringVarP(&configAddOptions.URL, "url", "", "", "URL of the Jenkins") - configAddCmd.Flags().StringVarP(&configAddOptions.UserName, "username", "u", "", "UserName of the Jenkins") - configAddCmd.Flags().StringVarP(&configAddOptions.Token, "token", "t", "", "Token of the Jenkins") - configAddCmd.Flags().StringVarP(&configAddOptions.Proxy, "proxy", "p", "", "Proxy of the Jenkins") - configAddCmd.Flags().StringVarP(&configAddOptions.ProxyAuth, "proxyAuth", "a", "", "ProxyAuth of the Jenkins") - configAddCmd.Flags().StringVarP(&configAddOptions.Description, "description", "d", "", "Description of the Jenkins") + configAddCmd.Flags().StringVarP(&configAddOptions.Name, "name", "n", "", + i18n.T("Name of the Jenkins")) + configAddCmd.Flags().StringVarP(&configAddOptions.URL, "url", "", "", + i18n.T("URL of the Jenkins")) + configAddCmd.Flags().StringVarP(&configAddOptions.UserName, "username", "u", "", + i18n.T("UserName of the Jenkins")) + configAddCmd.Flags().StringVarP(&configAddOptions.Token, "token", "t", "", + i18n.T("Token of the Jenkins")) + configAddCmd.Flags().StringVarP(&configAddOptions.Proxy, "proxy", "p", "", + i18n.T("Proxy of the Jenkins")) + configAddCmd.Flags().StringVarP(&configAddOptions.ProxyAuth, "proxyAuth", "a", "", + i18n.T("ProxyAuth of the Jenkins")) + configAddCmd.Flags().StringVarP(&configAddOptions.Description, "description", "d", "", + i18n.T("Description of the Jenkins")) } var configAddCmd = &cobra.Command{ diff --git a/app/cmd/config_edit.go b/app/cmd/config_edit.go index a9b22705ef866147e1e2b670b02360f82b37c795..fa16664494e1821b250aaa504b46c5f0eb312c22 100644 --- a/app/cmd/config_edit.go +++ b/app/cmd/config_edit.go @@ -4,9 +4,11 @@ import ( "fmt" "log" + "github.com/AlecAivazis/survey/v2" + "github.com/jenkins-zh/jenkins-cli/app/i18n" + "io/ioutil" - "github.com/AlecAivazis/survey/v2" "github.com/spf13/cobra" ) @@ -16,8 +18,8 @@ func init() { var configEditCmd = &cobra.Command{ Use: "edit", - Short: "Edit a Jenkins config", - Long: `Edit a Jenkins config`, + Short: i18n.T("Edit a Jenkins config"), + Long: i18n.T(`Edit a Jenkins config`), Run: func(_ *cobra.Command, _ []string) { current := getCurrentJenkinsFromOptionsOrDie() configPath := configOptions.ConfigFileLocation diff --git a/app/cmd/config_generate.go b/app/cmd/config_generate.go index 74063839619f84389fc0d6d47aa5c12ff2d55840..94213e9686d8c62a4fe8ac3be14d7992b158a970 100644 --- a/app/cmd/config_generate.go +++ b/app/cmd/config_generate.go @@ -1,13 +1,15 @@ package cmd import ( + "io/ioutil" + "os" + "github.com/AlecAivazis/survey/v2" "github.com/atotto/clipboard" "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/spf13/cobra" "gopkg.in/yaml.v2" - "io/ioutil" - "os" ) // ConfigGenerateOption is the config generate cmd option @@ -20,15 +22,17 @@ var configGenerateOption ConfigGenerateOption func init() { configCmd.AddCommand(configGenerateCmd) - configGenerateCmd.Flags().BoolVarP(&configGenerateOption.Interactive, "interactive", "i", true, "Interactive mode") - configGenerateCmd.Flags().BoolVarP(&configGenerateOption.Copy, "copy", "c", false, "Copy the output into clipboard") + configGenerateCmd.Flags().BoolVarP(&configGenerateOption.Interactive, "interactive", "i", true, + i18n.T("Interactive mode")) + configGenerateCmd.Flags().BoolVarP(&configGenerateOption.Copy, "copy", "c", false, + i18n.T("Copy the output into clipboard")) } var configGenerateCmd = &cobra.Command{ Use: "generate", Aliases: []string{"gen"}, - Short: "Generate a sample config file for you", - Long: `Generate a sample config file for you`, + Short: i18n.T("Generate a sample config file for you"), + Long: i18n.T("Generate a sample config file for you"), Run: func(cmd *cobra.Command, _ []string) { data, err := generateSampleConfig() if err == nil { diff --git a/app/cmd/config_list.go b/app/cmd/config_list.go index 5f7cb590a65ceef972b90813bc5cb3a9ce96f171..9a65dd448d539f42f26424553894b2460676fa45 100644 --- a/app/cmd/config_list.go +++ b/app/cmd/config_list.go @@ -5,6 +5,7 @@ import ( "fmt" "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/util" "github.com/spf13/cobra" @@ -24,8 +25,8 @@ func init() { var configListCmd = &cobra.Command{ Use: "list", - Short: "List all Jenkins config items", - Long: `List all Jenkins config items`, + Short: i18n.T("List all Jenkins config items"), + Long: i18n.T("List all Jenkins config items"), Run: func(cmd *cobra.Command, _ []string) { current := getCurrentJenkins() diff --git a/app/cmd/config_remove.go b/app/cmd/config_remove.go index 51e702103157a6a2789cf896959216dee82f849f..9e5edf460a20b6203380555392e81964ae8bf335 100644 --- a/app/cmd/config_remove.go +++ b/app/cmd/config_remove.go @@ -2,8 +2,8 @@ package cmd import ( "fmt" - "log" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/spf13/cobra" ) @@ -12,18 +12,13 @@ func init() { } var configRemoveCmd = &cobra.Command{ - Use: "remove", - Short: "Remove a Jenkins config", - Long: `Remove a Jenkins config`, - Run: func(_ *cobra.Command, args []string) { - if len(args) == 0 { - log.Fatal("You need to give a name") - } - + Use: "remove ", + Short: i18n.T("Remove a Jenkins config"), + Long: i18n.T("Remove a Jenkins config"), + Args: cobra.MinimumNArgs(1), + RunE: func(_ *cobra.Command, args []string) error { target := args[0] - if err := removeJenkins(target); err != nil { - log.Fatal(err) - } + return removeJenkins(target) }, } diff --git a/app/cmd/doc.go b/app/cmd/doc.go index 7147588b4bcfbf30031c6353b243574fa643de1f..a7646215699182b6efcaa2b4a4ca229a04270270 100644 --- a/app/cmd/doc.go +++ b/app/cmd/doc.go @@ -2,12 +2,16 @@ package cmd import ( "fmt" - "github.com/spf13/cobra" - "github.com/spf13/cobra/doc" "path" "path/filepath" "strings" "time" + + "github.com/jenkins-zh/jenkins-cli/app" + + "github.com/jenkins-zh/jenkins-cli/app/i18n" + "github.com/spf13/cobra" + "github.com/spf13/cobra/doc" ) func init() { @@ -20,22 +24,25 @@ date: %s title: "%s" anchor: %s url: %s +version: %s --- ` ) var docCmd = &cobra.Command{ Use: "doc ", - Short: "Generate document for all jcl commands", - Long: `Generate document for all jcl commands`, + Short: i18n.T("Generate document for all jcl commands"), + Long: i18n.T("Generate document for all jcl commands"), Args: cobra.MinimumNArgs(1), - Run: func(cmd *cobra.Command, args []string) { + RunE: func(cmd *cobra.Command, args []string) (err error) { now := time.Now().Format(time.RFC3339) prepender := func(filename string) string { name := filepath.Base(filename) base := strings.TrimSuffix(name, path.Ext(name)) url := "/commands/" + strings.ToLower(base) + "/" - return fmt.Sprintf(gendocFrontmatterTemplate, now, strings.Replace(base, "_", " ", -1), base, url) + return fmt.Sprintf(gendocFrontmatterTemplate, now, + strings.Replace(base, "_", " ", -1), + base, url, app.GetVersion()) } linkHandler := func(name string) string { @@ -45,9 +52,8 @@ var docCmd = &cobra.Command{ outputDir := args[0] - err := doc.GenMarkdownTreeCustom(rootCmd, outputDir, prepender, linkHandler) - if err != nil { - cmd.PrintErr(err) - } + rootCmd.DisableAutoGenTag = true + err = doc.GenMarkdownTreeCustom(rootCmd, outputDir, prepender, linkHandler) + return }, } diff --git a/app/cmd/job_artifact.go b/app/cmd/job_artifact.go index 899e47fdc495e81df4051c6ba5af2d77bc328931..2565c7451ccb7563cff5cd6d05ce36224947c623 100644 --- a/app/cmd/job_artifact.go +++ b/app/cmd/job_artifact.go @@ -3,6 +3,7 @@ package cmd import ( "bytes" "fmt" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "net/http" "strconv" @@ -29,8 +30,8 @@ func init() { var jobArtifactCmd = &cobra.Command{ Use: "artifact [buildID]", - Short: "Print the artifact list of target job", - Long: `Print the artifact list of target job`, + Short: i18n.T("Print the artifact list of target job"), + Long: i18n.T("Print the artifact list of target job"), Run: func(cmd *cobra.Command, args []string) { argLen := len(args) if argLen == 0 { diff --git a/app/cmd/plugin.go b/app/cmd/plugin.go index ae55debaaa924f17dd13b3548dbb5ed92ddbcce1..20b65d870d99f10f42aa2d7aa5acb7241cfd5478 100644 --- a/app/cmd/plugin.go +++ b/app/cmd/plugin.go @@ -1,6 +1,7 @@ package cmd import ( + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/spf13/cobra" ) @@ -17,8 +18,8 @@ func init() { var pluginCmd = &cobra.Command{ Use: "plugin", - Short: "Manage the plugins of Jenkins", - Long: `Manage the plugins of Jenkins`, + Short: i18n.T("Manage the plugins of Jenkins"), + Long: i18n.T("Manage the plugins of Jenkins"), Example: ` jcli plugin list jcli plugin search github jcli plugin check`, diff --git a/app/cmd/plugin_download.go b/app/cmd/plugin_download.go index 0c8a8ab616f39016c53ca1a30ef69b61179ed8bd..5e664088851e0d49a3941e74a51a1db73f4315ae 100644 --- a/app/cmd/plugin_download.go +++ b/app/cmd/plugin_download.go @@ -1,6 +1,7 @@ package cmd import ( + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/client" "github.com/spf13/cobra" "net/http" @@ -21,19 +22,19 @@ var pluginDownloadOption PluginDownloadOption func init() { pluginCmd.AddCommand(pluginDownloadCmd) pluginDownloadCmd.Flags().BoolVarP(&pluginDownloadOption.SkipDependency, "skip-dependency", "", false, - "If you want to skip download dependency of plugin") + i18n.T("If you want to skip download dependency of plugin")) pluginDownloadCmd.Flags().BoolVarP(&pluginDownloadOption.SkipOptional, "skip-optional", "", true, - "If you want to skip download optional dependency of plugin") + i18n.T("If you want to skip download optional dependency of plugin")) pluginDownloadCmd.Flags().BoolVarP(&pluginDownloadOption.UseMirror, "use-mirror", "", true, - "If you want to download plugin from a mirror site") + i18n.T("If you want to download plugin from a mirror site")) pluginDownloadCmd.Flags().BoolVarP(&pluginDownloadOption.ShowProgress, "show-progress", "", true, - "If you want to show the progress of download a plugin") + i18n.T("If you want to show the progress of download a plugin")) } var pluginDownloadCmd = &cobra.Command{ Use: "download ", - Short: "Download the plugins", - Long: `Download the plugins which contain the target plugin and its dependencies`, + Short: i18n.T("Download the plugins"), + Long: i18n.T(`Download the plugins which contain the target plugin and its dependencies`), Args: cobra.MinimumNArgs(1), Run: func(cmd *cobra.Command, args []string) { jClient := &client.PluginAPI{ diff --git a/app/cmd/plugin_install.go b/app/cmd/plugin_install.go index 5ba5f78c11e659fc6873947329d9d9930f22b203..6f2a85772b5d071e54ca5e534598a1246a26548e 100644 --- a/app/cmd/plugin_install.go +++ b/app/cmd/plugin_install.go @@ -3,6 +3,7 @@ package cmd import ( "fmt" "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "net/http" "github.com/AlecAivazis/survey/v2" @@ -23,15 +24,15 @@ var pluginInstallOption PluginInstallOption func init() { pluginCmd.AddCommand(pluginInstallCmd) pluginInstallCmd.Flags().BoolVarP(&pluginInstallOption.UseMirror, "use-mirror", "", true, - "If you want to download plugin from a mirror site") + i18n.T("If you want to download plugin from a mirror site")) pluginInstallCmd.Flags().BoolVarP(&pluginInstallOption.ShowProgress, "show-progress", "", true, - "If you want to show the progress of download a plugin") + i18n.T("If you want to show the progress of download a plugin")) pluginInstallCmd.Flags().StringVarP(&pluginOpt.Suite, "suite", "", "", "Suite of plugins") } var pluginInstallCmd = &cobra.Command{ Use: "install [pluginName]", - Short: "Install the plugins", + Short: i18n.T("Install the plugins"), Long: `Install the plugins Allow you to install a plugin with or without the version`, Example: `jcli plugin install localization-zh-cn diff --git a/app/cmd/plugin_list.go b/app/cmd/plugin_list.go index 0f2df71adb0ef2c25c5e3f5298dd6b7ce4936a62..48c81470da2d5922d7065a97c1a6bd127af1de34 100644 --- a/app/cmd/plugin_list.go +++ b/app/cmd/plugin_list.go @@ -4,6 +4,7 @@ import ( "bytes" "fmt" "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "net/http" "strings" @@ -31,8 +32,8 @@ func init() { var pluginListCmd = &cobra.Command{ Use: "list", - Short: "Print all the plugins which are installed", - Long: `Print all the plugins which are installed`, + Short: i18n.T("Print all the plugins which are installed"), + Long: i18n.T("Print all the plugins which are installed"), Example: ` jcli plugin list --filter name=github jcli plugin list --filter hasUpdate jcli plugin list --no-headers`, diff --git a/app/cmd/plugin_search.go b/app/cmd/plugin_search.go index 64824a8f09086ffeea3b9cfbb78c2449ac149b7c..c5525f33386ed9ecd3cf5a06dc69afc3e2c7ace4 100644 --- a/app/cmd/plugin_search.go +++ b/app/cmd/plugin_search.go @@ -3,6 +3,7 @@ package cmd import ( "bytes" "fmt" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "net/http" "strings" @@ -29,8 +30,8 @@ func init() { var pluginSearchCmd = &cobra.Command{ Use: "search ", - Short: "Print the plugins of your Jenkins", - Long: `Print the plugins of your Jenkins`, + Short: i18n.T("Print the plugins of your Jenkins"), + Long: i18n.T("Print the plugins of your Jenkins"), Args: cobra.MinimumNArgs(1), Run: func(cmd *cobra.Command, args []string) { keyword := args[0] diff --git a/app/cmd/plugin_uninstall.go b/app/cmd/plugin_uninstall.go index 4dbd8df64dde100ec5d49ae522223199f488221b..889eb14f491f46fefc257b3b798a558315233364 100644 --- a/app/cmd/plugin_uninstall.go +++ b/app/cmd/plugin_uninstall.go @@ -1,6 +1,7 @@ package cmd import ( + "github.com/jenkins-zh/jenkins-cli/app/i18n" "net/http" "github.com/jenkins-zh/jenkins-cli/app/helper" @@ -22,8 +23,8 @@ func init() { var pluginUninstallCmd = &cobra.Command{ Use: "uninstall [pluginName]", - Short: "Uninstall the plugins", - Long: `Uninstall the plugins`, + Short: i18n.T("Uninstall the plugins"), + Long: i18n.T("Uninstall the plugins"), Args: cobra.MinimumNArgs(1), Run: func(cmd *cobra.Command, args []string) { pluginName := args[0] diff --git a/app/cmd/plugin_upgrade.go b/app/cmd/plugin_upgrade.go index fdb6fc701b263e4b8cd5087fb63c3fb9b10d44ac..763290eb8d7a2c0b3bf64c7e7b9621a6c5204739 100644 --- a/app/cmd/plugin_upgrade.go +++ b/app/cmd/plugin_upgrade.go @@ -6,6 +6,8 @@ import ( "strings" "github.com/AlecAivazis/survey/v2" + "github.com/jenkins-zh/jenkins-cli/app/i18n" + "github.com/jenkins-zh/jenkins-cli/app/helper" "github.com/jenkins-zh/jenkins-cli/client" @@ -28,8 +30,8 @@ func init() { var pluginUpgradeCmd = &cobra.Command{ Use: "upgrade [plugin name]", - Short: "Upgrade the specific plugin", - Long: `Upgrade the specific plugin`, + Short: i18n.T("Upgrade the specific plugin"), + Long: i18n.T("Upgrade the specific plugin"), Example: `jcli plugin upgrade [tab][tab]`, Run: func(cmd *cobra.Command, args []string) { jclient := &client.PluginManager{ diff --git a/app/cmd/plugin_upload.go b/app/cmd/plugin_upload.go index 1e8755aed727ccf5d096d1803e968ed47a6f7fa6..c72126db2abfa0352db6e84d55b5b970f3f3e5fe 100644 --- a/app/cmd/plugin_upload.go +++ b/app/cmd/plugin_upload.go @@ -3,6 +3,7 @@ package cmd import ( "fmt" "github.com/jenkins-zh/jenkins-cli/app/helper" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "io/ioutil" "log" "net/http" @@ -34,21 +35,28 @@ var pluginUploadOption PluginUploadOption func init() { pluginCmd.AddCommand(pluginUploadCmd) - pluginUploadCmd.Flags().BoolVarP(&pluginUploadOption.ShowProgress, "show-progress", "", true, "Whether show the upload progress") - pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.Remote, "remote", "r", "", "Remote plugin URL") - pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.RemoteUser, "remote-user", "", "", "User of remote plugin URL") - pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.RemotePassword, "remote-password", "", "", "Password of remote plugin URL") - pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.RemoteJenkins, "remote-jenkins", "", "", "Remote Jenkins which will find from config list") - - pluginUploadCmd.Flags().BoolVarP(&pluginUploadOption.SkipPreHook, "skip-prehook", "", false, "Whether skip the previous command hook") - pluginUploadCmd.Flags().BoolVarP(&pluginUploadOption.SkipPostHook, "skip-posthook", "", false, "Whether skip the post command hook") + pluginUploadCmd.Flags().BoolVarP(&pluginUploadOption.ShowProgress, "show-progress", "", true, + i18n.T("Whether show the upload progress")) + pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.Remote, "remote", "r", "", + i18n.T("Remote plugin URL")) + pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.RemoteUser, "remote-user", "", "", + i18n.T("User of remote plugin URL")) + pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.RemotePassword, "remote-password", "", "", + i18n.T("Password of remote plugin URL")) + pluginUploadCmd.Flags().StringVarP(&pluginUploadOption.RemoteJenkins, "remote-jenkins", "", "", + i18n.T("Remote Jenkins which will find from config list")) + + pluginUploadCmd.Flags().BoolVarP(&pluginUploadOption.SkipPreHook, "skip-prehook", "", false, + i18n.T("Whether skip the previous command hook")) + pluginUploadCmd.Flags().BoolVarP(&pluginUploadOption.SkipPostHook, "skip-posthook", "", false, + i18n.T("Whether skip the post command hook")) } var pluginUploadCmd = &cobra.Command{ Use: "upload", Aliases: []string{"up"}, - Short: "Upload a plugin to your Jenkins", - Long: `Upload a plugin from local filesystem or remote URL to your Jenkins`, + Short: i18n.T("Upload a plugin to your Jenkins"), + Long: i18n.T(`Upload a plugin from local filesystem or remote URL to your Jenkins`), Example: ` jcli plugin upload --remote https://server/sample.hpi jcli plugin upload sample.hpi jcli plugin upload sample.hpi --show-progress=false`, diff --git a/app/cmd/queue.go b/app/cmd/queue.go index ca896745885684fd17000fa2820a2b133da34e77..26516620a0af9f66d6f847f91a00b890ea0c7826 100644 --- a/app/cmd/queue.go +++ b/app/cmd/queue.go @@ -12,7 +12,4 @@ var queueCmd = &cobra.Command{ Use: "queue", Short: "Manage the queue of your Jenkins", Long: `Manage the queue of your Jenkins`, - Run: func(cmd *cobra.Command, _ []string) { - cmd.Help() - }, } diff --git a/app/cmd/restart.go b/app/cmd/restart.go index 3b24847859791a6c0cb2b906d908e1c95e374a53..6cdec0b8b9798ca436004ca4499c991a31307531 100644 --- a/app/cmd/restart.go +++ b/app/cmd/restart.go @@ -5,6 +5,7 @@ import ( "net/http" "github.com/AlecAivazis/survey/v2" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/client" "github.com/spf13/cobra" ) @@ -25,8 +26,8 @@ func init() { var restartCmd = &cobra.Command{ Use: "restart", - Short: "Restart your Jenkins", - Long: `Restart your Jenkins`, + Short: i18n.T("Restart your Jenkins"), + Long: i18n.T("Restart your Jenkins"), Run: func(cmd *cobra.Command, _ []string) { jenkins := getCurrentJenkinsFromOptionsOrDie() if !restartOption.Batch { diff --git a/app/cmd/root.go b/app/cmd/root.go index 11c1720768cf39c31230cfbaa6c8d8eb6358aa58..65b9c3b01a067b079057f05f58db70b532dbe3a1 100644 --- a/app/cmd/root.go +++ b/app/cmd/root.go @@ -8,6 +8,7 @@ import ( "os/exec" "strings" + "github.com/jenkins-zh/jenkins-cli/app/i18n" "github.com/jenkins-zh/jenkins-cli/util" "github.com/jenkins-zh/jenkins-cli/client" @@ -31,21 +32,19 @@ type RootOptions struct { var rootCmd = &cobra.Command{ Use: "jcli", - Short: "jcli is a tool which could help you with your multiple Jenkins", + Short: i18n.T("jcli is a tool which could help you with your multiple Jenkins"), Long: `jcli is Jenkins CLI which could help with your multiple Jenkins, Manage your Jenkins and your pipelines More information could found at https://jenkins-zh.cn`, - BashCompletionFunction: jcliBashCompletionFunc, - PersistentPreRun: func(cmd *cobra.Command, args []string) { - var err error - if logger, err = util.InitLogger(rootOptions.LoggerLevel); err != nil { - cmd.PrintErrln(err) - } else { + PersistentPreRunE: func(cmd *cobra.Command, args []string) (err error) { + if logger, err = util.InitLogger(rootOptions.LoggerLevel); err == nil { client.SetLogger(logger) } + return }, + BashCompletionFunction: jcliBashCompletionFunc, Run: func(cmd *cobra.Command, args []string) { - cmd.Println("Jenkins CLI (jcli) manage your Jenkins") + cmd.Println(i18n.T("Jenkins CLI (jcli) manage your Jenkins")) if rootOptions.Version { cmd.Printf("Version: %s\n", app.GetVersion()) cmd.Printf("Commit: %s\n", app.GetCommit()) @@ -58,7 +57,6 @@ var rootCmd = &cobra.Command{ cmd.Println("Cannot found the configuration") } } - }, } @@ -73,12 +71,15 @@ var rootOptions RootOptions func init() { cobra.OnInitialize(initConfig) - rootCmd.PersistentFlags().StringVarP(&rootOptions.ConfigFile, "configFile", "", "", "An alternative config file") - rootCmd.PersistentFlags().StringVarP(&rootOptions.Jenkins, "jenkins", "j", "", "Select a Jenkins server for this time") - rootCmd.PersistentFlags().BoolVarP(&rootOptions.Version, "version", "v", false, "Print the version of Jenkins CLI") + rootCmd.PersistentFlags().StringVarP(&rootOptions.ConfigFile, "configFile", "", "", + i18n.T("An alternative config file")) + rootCmd.PersistentFlags().StringVarP(&rootOptions.Jenkins, "jenkins", "j", "", + i18n.T("Select a Jenkins server for this time")) rootCmd.PersistentFlags().BoolVarP(&rootOptions.Debug, "debug", "", false, "Print the output into debug.html") rootCmd.PersistentFlags().StringVarP(&rootOptions.LoggerLevel, "logger-level", "", "warn", "Logger level which could be: debug, info, warn, error") + rootCmd.Flags().BoolVarP(&rootOptions.Version, "version", "v", false, + i18n.T("Print the version of Jenkins CLI")) rootCmd.SetOut(os.Stdout) } diff --git a/app/i18n/i18n.go b/app/i18n/i18n.go new file mode 100644 index 0000000000000000000000000000000000000000..4029f5377f29778f173a73741ca3ece82daebcb9 --- /dev/null +++ b/app/i18n/i18n.go @@ -0,0 +1,120 @@ +package i18n + +import ( + "archive/zip" + "bytes" + "fmt" + "io" + "os" + "strings" + + "github.com/chai2010/gettext-go/gettext" +) + +var knownTranslations = map[string][]string{ + "jcli": { + "default", + "en_US", + "zh_CN", + }, + // only used for unit tests. + "test": { + "default", + "en_US", + }, +} + +func loadSystemLanguage() string { + // Implements the following locale priority order: LC_ALL, LC_MESSAGES, LANG + // Similarly to: https://www.gnu.org/software/gettext/manual/html_node/Locale-Environment-Variables.html + langStr := os.Getenv("LC_ALL") + if langStr == "" { + langStr = os.Getenv("LC_MESSAGES") + } + if langStr == "" { + langStr = os.Getenv("LANG") + } + if langStr == "" { + return "default" + } + pieces := strings.Split(langStr, ".") + if len(pieces) != 2 { + return "default" + } + return pieces[0] +} + +func findLanguage(root string, getLanguageFn func() string) string { + langStr := getLanguageFn() + + translations := knownTranslations[root] + for ix := range translations { + if translations[ix] == langStr { + return langStr + } + } + return "default" +} + +// LoadTranslations loads translation files. getLanguageFn should return a language +// string (e.g. 'en-US'). If getLanguageFn is nil, then the loadSystemLanguage function +// is used, which uses the 'LANG' environment variable. +func LoadTranslations(root string, getLanguageFn func() string) (err error) { + if getLanguageFn == nil { + getLanguageFn = loadSystemLanguage + } + + langStr := findLanguage(root, getLanguageFn) + translationFiles := []string{ + //"jcli/zh_CN/LC_MESSAGES/jcli.mo", + "jcli/zh_CN/LC_MESSAGES/jcli.po", + } + + buf := new(bytes.Buffer) + w := zip.NewWriter(buf) + + // Make sure to check the error on Close. + var f io.Writer + var data []byte + for _, file := range translationFiles { + filename := file + if f, err = w.Create(file); err != nil { + return + } + + if data, err = Asset(filename); err != nil { + return + } + + if _, err = f.Write(data); err != nil { + return + } + } + + if err = w.Close(); err == nil { + gettext.BindTextdomain("jcli", root+".zip", buf.Bytes()) + gettext.Textdomain("jcli") + gettext.SetLocale(langStr) + } + return +} + +var i18nLoaded = false + +// T translates a string, possibly substituting arguments into it along +// the way. If len(args) is > 0, args1 is assumed to be the plural value +// and plural translation is used. +func T(defaultValue string, args ...int) string { + if !i18nLoaded { + i18nLoaded = true + if err := LoadTranslations("jcli", nil); err != nil { + fmt.Println(err) + } + } + + if len(args) == 0 { + return gettext.PGettext("", defaultValue) + } + return fmt.Sprintf(gettext.PNGettext("", defaultValue, defaultValue+".plural", args[0]), + args[0]) +} diff --git a/app/i18n/i18n_test.go b/app/i18n/i18n_test.go new file mode 100644 index 0000000000000000000000000000000000000000..28a57e880b3d29fb48332f338c235748a01ee4cb --- /dev/null +++ b/app/i18n/i18n_test.go @@ -0,0 +1,99 @@ +package i18n + +import ( + "fmt" + . "github.com/onsi/ginkgo" + . "github.com/onsi/gomega" + "os" +) + +var _ = Describe("test LoadTranslations", func() { + var ( + root string + getLangFunc func() string + err error + ) + + JustBeforeEach(func() { + err = LoadTranslations(root, getLangFunc) + }) + + AfterEach(func() { + root = "" + getLangFunc = nil + }) + + It("default param", func() { + Expect(err).NotTo(HaveOccurred()) + }) + + Context("unknown language", func() { + BeforeEach(func() { + getLangFunc = func() string { + return "fake" + } + }) + + It("should not have error", func() { + Expect(err).NotTo(HaveOccurred()) + }) + }) + + Context("given invalid environment of language", func() { + var ( + osEnvErr error + ) + + BeforeEach(func() { + osEnvErr = os.Setenv("LC_ALL", "zh_CN") + }) + + It("should not have error", func() { + Expect(osEnvErr).NotTo(HaveOccurred()) + Expect(err).NotTo(HaveOccurred()) + }) + }) + + Context("given valid environment of language", func() { + var ( + osEnvErr error + ) + + BeforeEach(func() { + root = "jcli" + osEnvErr = os.Setenv("LC_ALL", "zh_CN.utf-8") + }) + + It("should not have error", func() { + Expect(osEnvErr).NotTo(HaveOccurred()) + Expect(err).NotTo(HaveOccurred()) + }) + }) +}) + +var _ = Describe("test i18n function T", func() { + var ( + text string + args []int + result string + ) + + JustBeforeEach(func() { + result = T(text, args...) + }) + + It("simple case, without args", func() { + Expect(result).To(Equal(text)) + }) + + Context("with args", func() { + BeforeEach(func() { + text = "fake %d" + args = []int{1} + }) + + It("should success", func() { + Expect(result).To(Equal(fmt.Sprintf(text, 1))) + }) + }) +}) diff --git a/app/i18n/jcli/zh_CN/LC_MESSAGES/jcli.po b/app/i18n/jcli/zh_CN/LC_MESSAGES/jcli.po new file mode 100644 index 0000000000000000000000000000000000000000..e4a441ab6c23d526060aa45e6bdfea222160ab28 --- /dev/null +++ b/app/i18n/jcli/zh_CN/LC_MESSAGES/jcli.po @@ -0,0 +1,239 @@ +# SOME DESCRIPTIVE TITLE. +# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER +# This file is distributed under the same license as the PACKAGE package. +# FIRST AUTHOR , YEAR. +# +msgid "" +msgstr "" +"Project-Id-Version: \n" +"Report-Msgid-Bugs-To: EMAIL\n" +"POT-Creation-Date: 2019-11-21 13:00+0800\n" +"PO-Revision-Date: 2019-11-21 22:17+0800\n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.2.4\n" +"Last-Translator: rick \n" +"Plural-Forms: nplurals=1; plural=0;\n" +"Language: zh_CN\n" + +#: app/cmd/restart.go:29 app/cmd/restart.go:30 +msgid "Restart your Jenkins" +msgstr "" + +#: app/cmd/root.go:79 +msgid "Select a Jenkins server for this time" +msgstr "选择本次执行的 Jenkins" + +#: app/cmd/config_add.go:30 +msgid "Proxy of the Jenkins" +msgstr "" + +#: app/cmd/plugin_upgrade.go:32 app/cmd/plugin_upgrade.go:33 +msgid "Upgrade the specific plugin" +msgstr "升级指定插件" + +#: app/cmd/config.go:31 app/cmd/config.go:32 +msgid "Manage the config of jcli" +msgstr "管理 jcli 的配置" + +#: app/cmd/config_generate.go:28 +msgid "Copy the output into clipboard" +msgstr "拷贝输出到剪贴板" + +#: app/cmd/root.go:77 +msgid "An alternative config file" +msgstr "指定另外一个配置文件" + +#: app/cmd/config_list.go:19 app/cmd/config_list.go:20 +msgid "List all Jenkins config items" +msgstr "列出所有的 Jenkins 配置项" + +#: app/cmd/doc.go:30 app/cmd/doc.go:31 +msgid "Generate document for all jcl commands" +msgstr "为所有的 jcli 命令生成文档" + +#: app/cmd/root.go:37 +msgid "jcli is a tool which could help you with your multiple Jenkins" +msgstr "jcli 是帮助你管理多个 Jenkins 的工具" + +#: app/cmd/root.go:55 +msgid "Jenkins CLI (jcli) manage your Jenkins" +msgstr "Jenkins CLI (jcli) 管理你的 Jenkins" + +#: app/cmd/plugin_upload.go:50 +msgid "Whether skip the previous command hook" +msgstr "是否跳过前置命令钩子" + +#: app/cmd/plugin.go:21 app/cmd/plugin.go:22 +msgid "Manage the plugins of Jenkins" +msgstr "管理 Jenkins 的插件" + +#: app/cmd/root.go:81 +msgid "Print the version of Jenkins CLI" +msgstr "打印 Jenkins CLI 的版本" + +#: app/cmd/completion.go:14 app/cmd/completion.go:15 +msgid "Genereate bash completion scripts" +msgstr "生成 bash 自动补全的脚本" + +#: app/cmd/plugin_download.go:31 app/cmd/plugin_install.go:29 +msgid "If you want to show the progress of download a plugin" +msgstr "你是否希望显示插件下载的进度" + +#: app/cmd/plugin_download.go:37 +msgid "Download the plugins which contain the target plugin and its dependencies" +msgstr "" + +#: app/cmd/plugin_uninstall.go:26 app/cmd/plugin_uninstall.go:27 +msgid "Uninstall the plugins" +msgstr "卸载插件" + +#: app/cmd/plugin_upload.go:41 +msgid "Remote plugin URL" +msgstr "远程插件 URL" + +#: app/cmd/plugin_upload.go:52 +msgid "Whether skip the post command hook" +msgstr "是否跳过后置命令钩子" + +#: app/cmd/center_download.go:32 +msgid "Version of the Jenkins which you want to download" +msgstr "" + +#: app/cmd/config_add.go:22 +msgid "Name of the Jenkins" +msgstr "" + +#: app/cmd/config_list.go:28 app/cmd/config_list.go:29 +msgid "List all Jenkins config items" +msgstr "" + +#: app/cmd/config_generate.go:34 app/cmd/config_generate.go:35 +msgid "Generate a sample config file for you" +msgstr "" + +#: app/cmd/config_add.go:28 +msgid "Token of the Jenkins" +msgstr "" + +#: app/cmd/config_add.go:32 +msgid "ProxyAuth of the Jenkins" +msgstr "" + +#: app/cmd/root.go:36 +msgid "jcli is a tool which could help you with your multiple Jenkins" +msgstr "" + +#: app/cmd/center_watch.go:27 +msgid "The watch will be continue util Jenkins needs restart" +msgstr "" + +#: app/cmd/config_add.go:26 +msgid "UserName of the Jenkins" +msgstr "" + +#: app/cmd/config_edit.go:21 app/cmd/config_edit.go:22 +msgid "Edit a Jenkins config" +msgstr "" + +#: app/cmd/plugin_download.go:25 +msgid "If you want to skip download dependency of plugin" +msgstr "你是否想要跳过下载依赖插件" + +#: app/cmd/plugin_upload.go:45 +msgid "Password of remote plugin URL" +msgstr "" + +#: app/cmd/plugin_upload.go:58 +msgid "Upload a plugin to your Jenkins" +msgstr "上传插件到你的 Jenkins" + +#: app/cmd/center.go:30 +msgid "Manage your update center" +msgstr "管理你的 Jenkins 更新中心" + +#: app/cmd/job_artifact.go:33 app/cmd/job_artifact.go:34 +msgid "Print the artifact list of target job" +msgstr "" + +#: app/cmd/plugin_download.go:27 +msgid "If you want to skip download optional dependency of plugin" +msgstr "" + +#: app/cmd/plugin_download.go:36 +msgid "Download the plugins" +msgstr "下载插件" + +#: app/cmd/plugin_install.go:35 +msgid "Install the plugins" +msgstr "安装插件" + +#: app/cmd/plugin_search.go:33 app/cmd/plugin_search.go:34 +msgid "Print the plugins of your Jenkins" +msgstr "" + +#: app/cmd/plugin_upload.go:39 +msgid "Whether show the upload progress" +msgstr "是否显示上传进度" + +#: app/cmd/plugin_upload.go:47 +msgid "Remote Jenkins which will find from config list" +msgstr "" + +#: app/cmd/plugin_download.go:29 app/cmd/plugin_install.go:27 +msgid "If you want to download plugin from a mirror site" +msgstr "你是否想要从镜像站点下载插件" + +#: app/cmd/plugin_list.go:35 app/cmd/plugin_list.go:36 +msgid "Print all the plugins which are installed" +msgstr "" + +#: app/cmd/center_watch.go:29 +msgid "The watch will be continue util all Jenkins plugins installation is completed" +msgstr "" + +#: app/cmd/common.go:101 app/cmd/config_generate.go:26 +msgid "Interactive mode" +msgstr "" + +#: app/cmd/plugin_download.go:29 app/cmd/plugin_install.go:27 +msgid "If you want to download plugin from a mirror site" +msgstr "" + +#: app/cmd/plugin_upload.go:43 +msgid "User of remote plugin URL" +msgstr "" + +#: app/cmd/plugin_upload.go:47 +msgid "Remote Jenkins which will find from config list" +msgstr "" + +#: app/cmd/plugin_upload.go:58 +msgid "Upload a plugin to your Jenkins" +msgstr "" + +#: app/cmd/root.go:76 +msgid "An alternative config file" +msgstr "" + +#: app/cmd/center.go:31 app/cmd/center.go:32 +msgid "Manage your update center" +msgstr "" + +#: app/cmd/center_download.go:36 +msgid "If you want to show the download progress" +msgstr "" + +#: app/cmd/completion.go:14 app/cmd/completion.go:15 +msgid "Genereate bash completion scripts" +msgstr "" + +#: app/cmd/config.go:32 app/cmd/config.go:33 +msgid "Manage the config of jcli" +msgstr "管理 jcli 的配置信息" + +#: app/cmd/plugin_upload.go:59 +msgid "Upload a plugin from local filesystem or remote URL to your Jenkins" +msgstr "上传来自本地文件系统或者远程 URL 的插件到你的 Jenkins" diff --git a/app/i18n/setup_test.go b/app/i18n/setup_test.go new file mode 100644 index 0000000000000000000000000000000000000000..8df476da3aa3e0363bbcc10ff70b79d9d66f1704 --- /dev/null +++ b/app/i18n/setup_test.go @@ -0,0 +1,16 @@ +package i18n_test + +import ( + "testing" + + "github.com/onsi/ginkgo/reporters" + + . "github.com/onsi/ginkgo" + . "github.com/onsi/gomega" +) + +func TestI18n(t *testing.T) { + RegisterFailHandler(Fail) + junitReporter := reporters.NewJUnitReporter("test-i18n.xml") + RunSpecsWithDefaultAndCustomReporters(t, "app", []Reporter{junitReporter}) +} diff --git a/app/version.go b/app/version.go index f6045379ac864aea719463bf2c8e996732375b09..6f37d8fdd6d8acd59b7d2698a818bf7b169dbfd4 100644 --- a/app/version.go +++ b/app/version.go @@ -1,6 +1,8 @@ package app -import "fmt" +import ( + "fmt" +) var ( version string diff --git a/go.mod b/go.mod index 2351e45db792a5fc00e84bcb8ca47fd16a196b56..99ecf126aad3a1a97fc97ea1dcf68e8d0c61d4c6 100644 --- a/go.mod +++ b/go.mod @@ -7,10 +7,14 @@ require ( github.com/Netflix/go-expect v0.0.0-20180615182759-c93bf25de8e8 github.com/Pallinder/go-randomdata v1.2.0 github.com/atotto/clipboard v0.1.2 + github.com/chai2010/gettext-go v0.0.0-20170215093142-bf70f2a70fb1 + github.com/go-bindata/go-bindata v3.1.2+incompatible // indirect github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b github.com/golang/mock v1.3.1 + github.com/gosexy/gettext v0.0.0-20160830220431-74466a0a0c4a // indirect github.com/gosuri/uilive v0.0.3 // indirect github.com/gosuri/uiprogress v0.0.1 + github.com/jessevdk/go-flags v1.4.0 // indirect github.com/mattn/go-isatty v0.0.10 // indirect github.com/mitchellh/go-homedir v1.1.0 github.com/onsi/ginkgo v1.10.3 diff --git a/go.sum b/go.sum index defba0462cf465d4b220f010801ab38994d44e82..e54b81cb597bac01de4e165c3d17a26e96b878b1 100644 --- a/go.sum +++ b/go.sum @@ -1,51 +1,154 @@ +cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= github.com/AlecAivazis/survey/v2 v2.0.1 h1:fQuEpIq5H/rJMaGqiOtPBgnRZP1oexFYRi6bonYTi0w= github.com/AlecAivazis/survey/v2 v2.0.1/go.mod h1:WYBhg6f0y/fNYUuesWQc0PKbJcEliGcYHB9sNT3Bg74= github.com/AlecAivazis/survey/v2 v2.0.4 h1:qzXnJSzXEvmUllWqMBWpZndvT2YfoAUzAMvZUax3L2M= github.com/AlecAivazis/survey/v2 v2.0.4/go.mod h1:WYBhg6f0y/fNYUuesWQc0PKbJcEliGcYHB9sNT3Bg74= +github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8= +github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI= +github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0= +github.com/Azure/go-autorest/autorest/date v0.1.0/go.mod h1:plvfp3oPSKwf2DNjlBjWF/7vwR+cUD/ELuzDCXwHUVA= +github.com/Azure/go-autorest/autorest/mocks v0.1.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0= +github.com/Azure/go-autorest/autorest/mocks v0.2.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0= +github.com/Azure/go-autorest/logger v0.1.0/go.mod h1:oExouG+K6PryycPJfVSxi/koC6LSNgds39diKLz7Vrc= +github.com/Azure/go-autorest/tracing v0.5.0/go.mod h1:r/s2XiOKccPW3HrqB+W0TQzfbtp2fGCgRFtBroKn4Dk= github.com/BurntSushi/toml v0.3.1 h1:WXkYYl6Yr3qBf1K79EBnL4mak0OimBfB0XUf9Vl28OQ= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= +github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= +github.com/MakeNowJust/heredoc v0.0.0-20170808103936-bb23615498cd/go.mod h1:64YHyfSL2R96J44Nlwm39UHepQbyR5q10x7iYa1ks2E= +github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ= github.com/Netflix/go-expect v0.0.0-20180615182759-c93bf25de8e8 h1:xzYJEypr/85nBpB11F9br+3HUrpgb+fcm5iADzXXYEw= github.com/Netflix/go-expect v0.0.0-20180615182759-c93bf25de8e8/go.mod h1:oX5x61PbNXchhh0oikYAH+4Pcfw5LKv21+Jnpr6r6Pc= github.com/Pallinder/go-randomdata v1.2.0 h1:DZ41wBchNRb/0GfsePLiSwb0PHZmT67XY00lCDlaYPg= github.com/Pallinder/go-randomdata v1.2.0/go.mod h1:yHmJgulpD2Nfrm0cR9tI/+oAgRqCQQixsA8HyRZfV9Y= +github.com/PuerkitoBio/purell v1.0.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= +github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= +github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= +github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= +github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= github.com/atotto/clipboard v0.1.2 h1:YZCtFu5Ie8qX2VmVTBnrqLSiU9XOWwqNRmdT3gIQzbY= github.com/atotto/clipboard v0.1.2/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI= +github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= +github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= +github.com/blang/semver v3.5.0+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk= +github.com/chai2010/gettext-go v0.0.0-20160711120539-c6fed771bfd5/go.mod h1:/iP1qXHoty45bqomnu2LM+VVyAEdWN+vtSHGlQgyxbw= +github.com/chai2010/gettext-go v0.0.0-20170215093142-bf70f2a70fb1 h1:HD4PLRzjuCVW79mQ0/pdsalOLHJ+FaEoqJLxfltpb2U= +github.com/chai2010/gettext-go v0.0.0-20170215093142-bf70f2a70fb1/go.mod h1:/iP1qXHoty45bqomnu2LM+VVyAEdWN+vtSHGlQgyxbw= +github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= github.com/cpuguy83/go-md2man v1.0.10 h1:BSKMNlYxDvnunlTymqtgONjNnaRV1sTpcovwwjF22jk= github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= +github.com/davecgh/go-spew v0.0.0-20151105211317-5215b55f46b2/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/daviddengcn/go-colortext v0.0.0-20160507010035-511bcaf42ccd/go.mod h1:dv4zxwHi5C/8AeI+4gX4dCWOIvNi7I6JCSX0HvlKPgE= +github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= +github.com/docker/distribution v2.7.1+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= +github.com/docker/docker v0.7.3-0.20190327010347-be7ac8be2ae0/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= +github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM= +github.com/elazarl/goproxy v0.0.0-20170405201442-c4fc26588b6e/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc= +github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs= +github.com/emicklei/go-restful v2.9.5+incompatible/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs= +github.com/evanphx/json-patch v4.2.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= +github.com/exponent-io/jsonpath v0.0.0-20151013193312-d6023ce2651d/go.mod h1:ZZMPRZwes7CROmyNKgQzC3XPs6L/G2EJLHddWejkmf4= +github.com/fatih/camelcase v1.0.0/go.mod h1:yN2Sb0lFhZJUdVvtELVWefmrXpuZESvPmqwoZc+/fpc= github.com/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I= github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= +github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/go-bindata/go-bindata v3.1.2+incompatible h1:5vjJMVhowQdPzjE1LdxyFF7YFTXg5IgGVW4gBr5IbvE= +github.com/go-bindata/go-bindata v3.1.2+incompatible/go.mod h1:xK8Dsgwmeed+BBsSy2XTopBn/8uK2HWuGSnA11C3Joo= +github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= +github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= +github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas= +github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0= +github.com/go-openapi/jsonpointer v0.19.2/go.mod h1:3akKfEdA7DF1sugOqz1dVQHBcuDBPKZGEoHC/NkiQRg= +github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= +github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg= +github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwohSTlpa0o73RUL1owJc= +github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc= +github.com/go-openapi/spec v0.19.2/go.mod h1:sCxk3jxKgioEJikev4fgkNmwS+3kuYdJtcsZsD5zxMY= +github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I= +github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= +github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= +github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.3.1 h1:qGJ6qTW+x6xX/my+8YUVl4WNpX9B7+/l2tRsHGZ7f2s= github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= +github.com/golang/protobuf v0.0.0-20161109072736-4bd1920723d7/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golangplus/bytes v0.0.0-20160111154220-45c989fe5450/go.mod h1:Bk6SMAONeMXrxql8uvOKuAZSu8aM5RUGv+1C6IJaEho= +github.com/golangplus/fmt v0.0.0-20150411045040-2a5d6d7d2995/go.mod h1:lJgMEyOkYFkPcDKwRXegd+iM6E7matEszMG5HhwytU8= +github.com/golangplus/testing v0.0.0-20180327235837-af21d9c3145e/go.mod h1:0AA//k/eakGydO4jKRoRL2j92ZKSzTgj9tclaCrvXHk= +github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= +github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= +github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= +github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/gofuzz v0.0.0-20161122191042-44d81051d367/go.mod h1:HP5RmnzzSNb993RKQDq4+1A4ia9nllfqcQFTQJedwGI= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= +github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= +github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY= +github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8= +github.com/gosexy/gettext v0.0.0-20160830220431-74466a0a0c4a h1:N2b2mb4Gki1SlF3WuhR9P1YHOpl7oy/b+xxX4A3iM2E= +github.com/gosexy/gettext v0.0.0-20160830220431-74466a0a0c4a/go.mod h1:IEJaV4/6J0VpoQ33kFCUUP6umRjrcBVEbOva6XCub/Q= github.com/gosuri/uilive v0.0.3 h1:kvo6aB3pez9Wbudij8srWo4iY6SFTTxTKOkb+uRCE8I= github.com/gosuri/uilive v0.0.3/go.mod h1:qkLSc0A5EXSP6B04TrN4oQoxqFI7A8XvoXSlJi8cwk8= github.com/gosuri/uiprogress v0.0.1 h1:0kpv/XY/qTmFWl/SkaJykZXrBBzwwadmW8fRb7RJSxw= github.com/gosuri/uiprogress v0.0.1/go.mod h1:C1RTYn4Sc7iEyf6j8ft5dyoZ4212h8G1ol9QQluh5+0= +github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA= +github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= github.com/hinshun/vt10x v0.0.0-20180616224451-1954e6464174 h1:WlZsjVhE8Af9IcZDGgJGQpNflI3+MJSBhsgT5PCtzBQ= github.com/hinshun/vt10x v0.0.0-20180616224451-1954e6464174/go.mod h1:DqJ97dSdRW1W22yXSB90986pcOyQ7r45iio1KN2ez1A= github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI= github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA= github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM= github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= +github.com/jessevdk/go-flags v1.4.0 h1:4IU2WS7AumrZ/40jfhf4QVDMsQwqA7VEHozFRrGARJA= +github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= +github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= +github.com/json-iterator/go v0.0.0-20180612202835-f2b4162afba3/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= +github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= +github.com/json-iterator/go v1.1.8/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= +github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 h1:Z9n2FFNUXsshfwJMBgNA0RU6/i7WVaAegv3PtuIHPMs= github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51/go.mod h1:CzGEWj7cYgsdH8dAjBGEr58BoE7ScuLd+fwFZ44+/x8= +github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/pty v1.1.4 h1:5Myjjh3JY/NaAi4IsUbHADytDyl1VE1Y9PXDlL+P/VQ= github.com/kr/pty v1.1.4/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/pty v1.1.5/go.mod h1:9r2w37qlBe7rQ6e1fg1S/9xpWHSnaqNdHD3WcMdbPDA= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de/go.mod h1:zAbeS9B/r2mtpb6U+EI2rYA5OAXxsYw6wTamcNW+zcE= +github.com/lithammer/dedent v1.1.0/go.mod h1:jrXYCQtgg0nJiN+StA2KgR7w6CiQNv9Fd/Z9BP0jIOc= github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= +github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= github.com/mattn/go-colorable v0.1.2 h1:/bC9yWikZXAL9uJdulbSfyVNIR3n3trXl+v8+1sx8mU= github.com/mattn/go-colorable v0.1.2/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= github.com/mattn/go-isatty v0.0.8 h1:HLtExJ+uU2HOZ+wI0Tt5DtUDrx8yhUqDcp7fYERX4CE= @@ -54,47 +157,83 @@ github.com/mattn/go-isatty v0.0.9 h1:d5US/mDsogSGW37IV293h//ZFaeajb69h+EHFsv2xGg github.com/mattn/go-isatty v0.0.9/go.mod h1:YNRxwqDuOph6SZLI9vUUz6OYw3QyUt7WiY2yME+cCiQ= github.com/mattn/go-isatty v0.0.10 h1:qxFzApOv4WsAL965uUPIsXzAKCZxN2p9UqdhFS4ZW10= github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= +github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b h1:j7+1HpAFS1zy5+Q4qx1fWh90gTKwiN4QCGoY9TWyyO4= github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b/go.mod h1:01TrycV0kFyexm33Z7vhZRXopbI8J3TDReVlkTgMUxE= github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= +github.com/mitchellh/go-wordwrap v1.0.0/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo= github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v0.0.0-20180320133207-05fbef0ca5da/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= +github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw= +github.com/onsi/ginkgo v0.0.0-20170829012221-11459a886d9c/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.9.0 h1:SZjF721BByVj8QH636/8S2DnX4n0Re3SteMmw3N+tzc= github.com/onsi/ginkgo v1.9.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.10.2 h1:uqH7bpe+ERSiDa34FDOF7RikN6RzXgduUF8yarlZp94= github.com/onsi/ginkgo v1.10.2/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.10.3 h1:OoxbjfXVZyod1fmWYhI7SEyaD8B00ynP3T+D5GiyHOY= github.com/onsi/ginkgo v1.10.3/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA= github.com/onsi/gomega v1.6.0 h1:8XTW0fcJZEq9q+Upcyws4JSGua2MFysCL5xkaSgHc+M= github.com/onsi/gomega v1.6.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= github.com/onsi/gomega v1.7.0 h1:XPnZz8VVBHjVsy1vzJmRwIcSwiUO+JFfrv/xGiigmME= github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= github.com/onsi/gomega v1.7.1 h1:K0jcRCwNQM3vFGh1ppMtDh/+7ApJrjldlX8fA0jDTLQ= github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= +github.com/opencontainers/go-digest v1.0.0-rc1/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s= github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= +github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU= +github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pmezard/go-difflib v0.0.0-20151028094244-d8ed2627bdf0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= +github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= +github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/remyoudompheng/bigfft v0.0.0-20170806203942-52369c62f446/go.mod h1:uYEyJGbgTkfkS4+E/PavXkNJcbFIpEtjt2B0KDQ5+9M= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo= github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= +github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= +github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= +github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk= github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= github.com/spf13/cobra v0.0.5 h1:f0B+LkLX6DtmRH1isoNA9VTtNUK9K8xYd28JNNfOv/s= github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU= github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo= +github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= github.com/spf13/pflag v1.0.3 h1:zPAT6CGy6wXeQ7NtTnaTerfKOsV6V6F8agHXFiazDkg= github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE= +github.com/stretchr/testify v0.0.0-20151208002404-e3a8ff8ce365/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.2.1/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= +github.com/xlab/handysort v0.0.0-20150421192137-fb3537ed64a1/go.mod h1:QcJo0QPSfTONNIgpN5RA8prR7fF8nkF6cTWTcNerRO8= github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= +go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.uber.org/atomic v1.5.0 h1:OI5t8sDa1Or+q8AeE+yKeB/SDYioSHAgcVljj9JIETY= go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= go.uber.org/multierr v1.3.0 h1:sFPn2GLc3poCkfrpIXGhBD2X0CMIo4Q/zSULXrj/+uc= @@ -103,57 +242,120 @@ go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee h1:0mgffUl7nfd+FpvXMVz4IDEa go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA= go.uber.org/zap v1.12.0 h1:dySoUQPFBGj6xwjmBzageVL8jGi8uxc6bEmJQjA06bw= go.uber.org/zap v1.12.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= +golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= go.uber.org/zap v1.13.0 h1:nR6NoDBgAf67s68NhaXbsojM+2gxp3S1hWkHDl27pVU= go.uber.org/zap v1.13.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20190211182817-74369b46fc67/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190530122614-20be4c3c3ed5 h1:8dUaAV7K4uHsF56JQWkprecIQKdPHtR9jCHF5nB8uzc= golang.org/x/crypto v0.0.0-20190530122614-20be4c3c3ed5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20190125153040-c74c464bbbf2/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20190312203227-4b39c73a6495/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= +golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= +golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= +golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= +golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac h1:8R1esu+8QioDxo4E4mX6bFztO+dMTM49DNAaWfO5OeY= golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= golang.org/x/lint v0.0.0-20190930215403-16217165b5de h1:5hukYrvBGR8/eNkX5mdUezrA6JiaEZDtJb9Ei+1LlBs= golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= +golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3 h1:0GoQqolDA55aaLxZyTzK/Y2ePZzZTUrRacwib7cNsYQ= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190620200207-3b0461eec859 h1:R/3boaszxrf1GEUWTVDzSKVwLmSJpwZ1yqXm8j0v2QI= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20191004110552-13f9640d40b9/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20170830134202-bb24a47a89ea/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180606202747-9527bec2660b/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a h1:1n5lsVfiQW3yfsRGu98756EH1YthsFqr/5mxHduZW2A= golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190209173611-3b5209105503/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190530182044-ad28b68e88f1 h1:R4dVlxdmKenVdMRS/tTspEpSTRWINYrHD8ySIU9yCIU= golang.org/x/sys v0.0.0-20190530182044-ad28b68e88f1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190616124812-15dcb6c0061f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a h1:aYOabOQFp6Vj6W1F80affTUvO9UxmJRx8K0gsfABByQ= golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191008105621-543471e840be h1:QAcqgptGM8IQBC9K/RC4o+O9YmqEm0diQn9QmZw/0mU= golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/text v0.0.0-20160726164857-2910a502d2bf/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= +golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20181011042414-1f849cf54d09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190206041539-40960b6deb8e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190425150028-36563e24a262 h1:qsl9y/CJx34tuA7QCPNp86JNJe4spst6Ff8MjvPUdPg= golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190911230505-6bfd74cf029c h1:ZgedNh8bIOBjyY5XEG0kR/41dSN9H+5jFZWuR/TgA1g= golang.org/x/tools v0.0.0-20190911230505-6bfd74cf029c/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20190920225731-5eefd052ad72/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191029041327-9cc4af7d6b2c/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191029190741-b9c20aec41a5 h1:hKsoRgsbwY1NafxrwTs+k64bikrLBkAgPir1TNCj3Zs= golang.org/x/tools v0.0.0-20191029190741-b9c20aec41a5/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gonum.org/v1/gonum v0.0.0-20190331200053-3d26580ed485/go.mod h1:2ltnJ7xHfj0zHS40VVPYEAAMTa3ZGguvHGBSJeRWqE0= +gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0/go.mod h1:wa6Ws7BG/ESfp6dHfk7C6KdzKA7wR7u/rKwOGE66zvw= +gonum.org/v1/netlib v0.0.0-20190331212654-76723241ea4e/go.mod h1:kS+toOQn6AQKjmKJ7gzohV1XkqsFehRA2FbsbkopSuQ= +google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= +google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= +gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= +gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= @@ -161,7 +363,16 @@ gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4 h1:/eiJrUcujPVeJ3xlSWaiNi3uSVmDGBK1pDHUHAnao1I= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw= +honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= gopkg.in/yaml.v2 v2.2.5 h1:ymVxjfMaHvXD8RqPRmzHHsB3VvucivSkIAvJFDI5O3c= gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= honnef.co/go/tools v0.0.1-2019.2.3 h1:3JgtbtFHMiCmsznwGVTUWbgGov+pVqnlf1dEJTNAXeM= honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= +modernc.org/cc v1.0.0/go.mod h1:1Sk4//wdnYJiUIxnW8ddKpaOJCF37yAdqYnkxUpaYxw= +modernc.org/golex v1.0.0/go.mod h1:b/QX9oBD/LhixY6NDh+IdGv17hgB+51fET1i2kPSmvk= +modernc.org/mathutil v1.0.0/go.mod h1:wU0vUrJsVWBZ4P6e7xtFJEhFSNsfRLJ8H458uRjg03k= +modernc.org/strutil v1.0.0/go.mod h1:lstksw84oURvj9y3tn8lGvRxyRC1S2+g5uuIzNfIOBs= +modernc.org/xc v1.0.0/go.mod h1:mRNCo0bvLjGhHO9WsyuKVU4q0ceiDDDoEeWDJHrNx8I= +vbom.ml/util v0.0.0-20160121211510-db5cfe13f5cc/go.mod h1:so/NYdZXCz+E3ZpW0uAoCj6uzU2+8OWDFv/HxUSs7kI= diff --git a/i18n.md b/i18n.md new file mode 100644 index 0000000000000000000000000000000000000000..7378aabff825043a84935ad1c257ba0c78928d8f --- /dev/null +++ b/i18n.md @@ -0,0 +1,5 @@ +## Tools requirement + +[poedit](https://poedit.net/) + +`make i18n-tools`