mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 08:30:25 +08:00 
			
		
		
		
	Rename scripts to build and add revive command as a new build tool command (#10942)
Co-authored-by: techknowlogick <techknowlogick@gitea.io>
This commit is contained in:
		@@ -339,7 +339,7 @@ steps:
 | 
				
			|||||||
    pull: default
 | 
					    pull: default
 | 
				
			||||||
    image: alpine:3.11
 | 
					    image: alpine:3.11
 | 
				
			||||||
    commands:
 | 
					    commands:
 | 
				
			||||||
      - ./scripts/update-locales.sh
 | 
					      - ./build/update-locales.sh
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  - name: push
 | 
					  - name: push
 | 
				
			||||||
    pull: always
 | 
					    pull: always
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										2
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							@@ -6,5 +6,5 @@ conf/* linguist-vendored
 | 
				
			|||||||
docker/* linguist-vendored
 | 
					docker/* linguist-vendored
 | 
				
			||||||
options/* linguist-vendored
 | 
					options/* linguist-vendored
 | 
				
			||||||
public/* linguist-vendored
 | 
					public/* linguist-vendored
 | 
				
			||||||
scripts/* linguist-vendored
 | 
					build/* linguist-vendored
 | 
				
			||||||
templates/* linguist-vendored
 | 
					templates/* linguist-vendored
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										7
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										7
									
								
								Makefile
									
									
									
									
									
								
							@@ -80,7 +80,7 @@ TAGS ?=
 | 
				
			|||||||
TAGS_SPLIT := $(subst $(COMMA), ,$(TAGS))
 | 
					TAGS_SPLIT := $(subst $(COMMA), ,$(TAGS))
 | 
				
			||||||
TAGS_EVIDENCE := $(MAKE_EVIDENCE_DIR)/tags
 | 
					TAGS_EVIDENCE := $(MAKE_EVIDENCE_DIR)/tags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
GO_DIRS := cmd integrations models modules routers scripts services vendor
 | 
					GO_DIRS := cmd integrations models modules routers build services vendor
 | 
				
			||||||
GO_SOURCES := $(wildcard *.go)
 | 
					GO_SOURCES := $(wildcard *.go)
 | 
				
			||||||
GO_SOURCES += $(shell find $(GO_DIRS) -type f -name "*.go" -not -path modules/options/bindata.go -not -path modules/public/bindata.go -not -path modules/templates/bindata.go)
 | 
					GO_SOURCES += $(shell find $(GO_DIRS) -type f -name "*.go" -not -path modules/options/bindata.go -not -path modules/public/bindata.go -not -path modules/templates/bindata.go)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -234,10 +234,7 @@ errcheck:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
.PHONY: revive
 | 
					.PHONY: revive
 | 
				
			||||||
revive:
 | 
					revive:
 | 
				
			||||||
	@hash revive > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | 
						GO111MODULE=on $(GO) run -mod=vendor build/lint.go -config .revive.toml -exclude=./vendor/... ./... || exit 1
 | 
				
			||||||
		$(GO) get -u github.com/mgechev/revive; \
 | 
					 | 
				
			||||||
	fi
 | 
					 | 
				
			||||||
	revive -config .revive.toml -exclude=./vendor/... ./... || exit 1
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
.PHONY: misspell-check
 | 
					.PHONY: misspell-check
 | 
				
			||||||
misspell-check:
 | 
					misspell-check:
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										325
									
								
								build/lint.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										325
									
								
								build/lint.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,325 @@
 | 
				
			|||||||
 | 
					// Copyright 2020 The Gitea Authors. All rights reserved.
 | 
				
			||||||
 | 
					// Copyright (c) 2018 Minko Gechev. All rights reserved.
 | 
				
			||||||
 | 
					// Use of this source code is governed by a MIT-style
 | 
				
			||||||
 | 
					// license that can be found in the LICENSE file.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// +build ignore
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package main
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"flag"
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"io/ioutil"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
						"path/filepath"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/BurntSushi/toml"
 | 
				
			||||||
 | 
						"github.com/mgechev/dots"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/formatter"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/rule"
 | 
				
			||||||
 | 
						"github.com/mitchellh/go-homedir"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func fail(err string) {
 | 
				
			||||||
 | 
						fmt.Fprintln(os.Stderr, err)
 | 
				
			||||||
 | 
						os.Exit(1)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var defaultRules = []lint.Rule{
 | 
				
			||||||
 | 
						&rule.VarDeclarationsRule{},
 | 
				
			||||||
 | 
						&rule.PackageCommentsRule{},
 | 
				
			||||||
 | 
						&rule.DotImportsRule{},
 | 
				
			||||||
 | 
						&rule.BlankImportsRule{},
 | 
				
			||||||
 | 
						&rule.ExportedRule{},
 | 
				
			||||||
 | 
						&rule.VarNamingRule{},
 | 
				
			||||||
 | 
						&rule.IndentErrorFlowRule{},
 | 
				
			||||||
 | 
						&rule.IfReturnRule{},
 | 
				
			||||||
 | 
						&rule.RangeRule{},
 | 
				
			||||||
 | 
						&rule.ErrorfRule{},
 | 
				
			||||||
 | 
						&rule.ErrorNamingRule{},
 | 
				
			||||||
 | 
						&rule.ErrorStringsRule{},
 | 
				
			||||||
 | 
						&rule.ReceiverNamingRule{},
 | 
				
			||||||
 | 
						&rule.IncrementDecrementRule{},
 | 
				
			||||||
 | 
						&rule.ErrorReturnRule{},
 | 
				
			||||||
 | 
						&rule.UnexportedReturnRule{},
 | 
				
			||||||
 | 
						&rule.TimeNamingRule{},
 | 
				
			||||||
 | 
						&rule.ContextKeysType{},
 | 
				
			||||||
 | 
						&rule.ContextAsArgumentRule{},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var allRules = append([]lint.Rule{
 | 
				
			||||||
 | 
						&rule.ArgumentsLimitRule{},
 | 
				
			||||||
 | 
						&rule.CyclomaticRule{},
 | 
				
			||||||
 | 
						&rule.FileHeaderRule{},
 | 
				
			||||||
 | 
						&rule.EmptyBlockRule{},
 | 
				
			||||||
 | 
						&rule.SuperfluousElseRule{},
 | 
				
			||||||
 | 
						&rule.ConfusingNamingRule{},
 | 
				
			||||||
 | 
						&rule.GetReturnRule{},
 | 
				
			||||||
 | 
						&rule.ModifiesParamRule{},
 | 
				
			||||||
 | 
						&rule.ConfusingResultsRule{},
 | 
				
			||||||
 | 
						&rule.DeepExitRule{},
 | 
				
			||||||
 | 
						&rule.UnusedParamRule{},
 | 
				
			||||||
 | 
						&rule.UnreachableCodeRule{},
 | 
				
			||||||
 | 
						&rule.AddConstantRule{},
 | 
				
			||||||
 | 
						&rule.FlagParamRule{},
 | 
				
			||||||
 | 
						&rule.UnnecessaryStmtRule{},
 | 
				
			||||||
 | 
						&rule.StructTagRule{},
 | 
				
			||||||
 | 
						&rule.ModifiesValRecRule{},
 | 
				
			||||||
 | 
						&rule.ConstantLogicalExprRule{},
 | 
				
			||||||
 | 
						&rule.BoolLiteralRule{},
 | 
				
			||||||
 | 
						&rule.RedefinesBuiltinIDRule{},
 | 
				
			||||||
 | 
						&rule.ImportsBlacklistRule{},
 | 
				
			||||||
 | 
						&rule.FunctionResultsLimitRule{},
 | 
				
			||||||
 | 
						&rule.MaxPublicStructsRule{},
 | 
				
			||||||
 | 
						&rule.RangeValInClosureRule{},
 | 
				
			||||||
 | 
						&rule.RangeValAddress{},
 | 
				
			||||||
 | 
						&rule.WaitGroupByValueRule{},
 | 
				
			||||||
 | 
						&rule.AtomicRule{},
 | 
				
			||||||
 | 
						&rule.EmptyLinesRule{},
 | 
				
			||||||
 | 
						&rule.LineLengthLimitRule{},
 | 
				
			||||||
 | 
						&rule.CallToGCRule{},
 | 
				
			||||||
 | 
						&rule.DuplicatedImportsRule{},
 | 
				
			||||||
 | 
						&rule.ImportShadowingRule{},
 | 
				
			||||||
 | 
						&rule.BareReturnRule{},
 | 
				
			||||||
 | 
						&rule.UnusedReceiverRule{},
 | 
				
			||||||
 | 
						&rule.UnhandledErrorRule{},
 | 
				
			||||||
 | 
						&rule.CognitiveComplexityRule{},
 | 
				
			||||||
 | 
						&rule.StringOfIntRule{},
 | 
				
			||||||
 | 
					}, defaultRules...)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var allFormatters = []lint.Formatter{
 | 
				
			||||||
 | 
						&formatter.Stylish{},
 | 
				
			||||||
 | 
						&formatter.Friendly{},
 | 
				
			||||||
 | 
						&formatter.JSON{},
 | 
				
			||||||
 | 
						&formatter.NDJSON{},
 | 
				
			||||||
 | 
						&formatter.Default{},
 | 
				
			||||||
 | 
						&formatter.Unix{},
 | 
				
			||||||
 | 
						&formatter.Checkstyle{},
 | 
				
			||||||
 | 
						&formatter.Plain{},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getFormatters() map[string]lint.Formatter {
 | 
				
			||||||
 | 
						result := map[string]lint.Formatter{}
 | 
				
			||||||
 | 
						for _, f := range allFormatters {
 | 
				
			||||||
 | 
							result[f.Name()] = f
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return result
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getLintingRules(config *lint.Config) []lint.Rule {
 | 
				
			||||||
 | 
						rulesMap := map[string]lint.Rule{}
 | 
				
			||||||
 | 
						for _, r := range allRules {
 | 
				
			||||||
 | 
							rulesMap[r.Name()] = r
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						lintingRules := []lint.Rule{}
 | 
				
			||||||
 | 
						for name := range config.Rules {
 | 
				
			||||||
 | 
							rule, ok := rulesMap[name]
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								fail("cannot find rule: " + name)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							lintingRules = append(lintingRules, rule)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return lintingRules
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func parseConfig(path string) *lint.Config {
 | 
				
			||||||
 | 
						config := &lint.Config{}
 | 
				
			||||||
 | 
						file, err := ioutil.ReadFile(path)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							fail("cannot read the config file")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						_, err = toml.Decode(string(file), config)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							fail("cannot parse the config file: " + err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return config
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func normalizeConfig(config *lint.Config) {
 | 
				
			||||||
 | 
						if config.Confidence == 0 {
 | 
				
			||||||
 | 
							config.Confidence = 0.8
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						severity := config.Severity
 | 
				
			||||||
 | 
						if severity != "" {
 | 
				
			||||||
 | 
							for k, v := range config.Rules {
 | 
				
			||||||
 | 
								if v.Severity == "" {
 | 
				
			||||||
 | 
									v.Severity = severity
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								config.Rules[k] = v
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							for k, v := range config.Directives {
 | 
				
			||||||
 | 
								if v.Severity == "" {
 | 
				
			||||||
 | 
									v.Severity = severity
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								config.Directives[k] = v
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getConfig() *lint.Config {
 | 
				
			||||||
 | 
						config := defaultConfig()
 | 
				
			||||||
 | 
						if configPath != "" {
 | 
				
			||||||
 | 
							config = parseConfig(configPath)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						normalizeConfig(config)
 | 
				
			||||||
 | 
						return config
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getFormatter() lint.Formatter {
 | 
				
			||||||
 | 
						formatters := getFormatters()
 | 
				
			||||||
 | 
						formatter := formatters["default"]
 | 
				
			||||||
 | 
						if formatterName != "" {
 | 
				
			||||||
 | 
							f, ok := formatters[formatterName]
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								fail("unknown formatter " + formatterName)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							formatter = f
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return formatter
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func buildDefaultConfigPath() string {
 | 
				
			||||||
 | 
						var result string
 | 
				
			||||||
 | 
						if homeDir, err := homedir.Dir(); err == nil {
 | 
				
			||||||
 | 
							result = filepath.Join(homeDir, "revive.toml")
 | 
				
			||||||
 | 
							if _, err := os.Stat(result); err != nil {
 | 
				
			||||||
 | 
								result = ""
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return result
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func defaultConfig() *lint.Config {
 | 
				
			||||||
 | 
						defaultConfig := lint.Config{
 | 
				
			||||||
 | 
							Confidence: 0.0,
 | 
				
			||||||
 | 
							Severity:   lint.SeverityWarning,
 | 
				
			||||||
 | 
							Rules:      map[string]lint.RuleConfig{},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, r := range defaultRules {
 | 
				
			||||||
 | 
							defaultConfig.Rules[r.Name()] = lint.RuleConfig{}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return &defaultConfig
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func normalizeSplit(strs []string) []string {
 | 
				
			||||||
 | 
						res := []string{}
 | 
				
			||||||
 | 
						for _, s := range strs {
 | 
				
			||||||
 | 
							t := strings.Trim(s, " \t")
 | 
				
			||||||
 | 
							if len(t) > 0 {
 | 
				
			||||||
 | 
								res = append(res, t)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return res
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getPackages() [][]string {
 | 
				
			||||||
 | 
						globs := normalizeSplit(flag.Args())
 | 
				
			||||||
 | 
						if len(globs) == 0 {
 | 
				
			||||||
 | 
							globs = append(globs, ".")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						packages, err := dots.ResolvePackages(globs, normalizeSplit(excludePaths))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							fail(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return packages
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type arrayFlags []string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (i *arrayFlags) String() string {
 | 
				
			||||||
 | 
						return strings.Join([]string(*i), " ")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (i *arrayFlags) Set(value string) error {
 | 
				
			||||||
 | 
						*i = append(*i, value)
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var configPath string
 | 
				
			||||||
 | 
					var excludePaths arrayFlags
 | 
				
			||||||
 | 
					var formatterName string
 | 
				
			||||||
 | 
					var help bool
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var originalUsage = flag.Usage
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func init() {
 | 
				
			||||||
 | 
						flag.Usage = func() {
 | 
				
			||||||
 | 
							originalUsage()
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// command line help strings
 | 
				
			||||||
 | 
						const (
 | 
				
			||||||
 | 
							configUsage    = "path to the configuration TOML file, defaults to $HOME/revive.toml, if present (i.e. -config myconf.toml)"
 | 
				
			||||||
 | 
							excludeUsage   = "list of globs which specify files to be excluded (i.e. -exclude foo/...)"
 | 
				
			||||||
 | 
							formatterUsage = "formatter to be used for the output (i.e. -formatter stylish)"
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						defaultConfigPath := buildDefaultConfigPath()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						flag.StringVar(&configPath, "config", defaultConfigPath, configUsage)
 | 
				
			||||||
 | 
						flag.Var(&excludePaths, "exclude", excludeUsage)
 | 
				
			||||||
 | 
						flag.StringVar(&formatterName, "formatter", "", formatterUsage)
 | 
				
			||||||
 | 
						flag.Parse()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func main() {
 | 
				
			||||||
 | 
						config := getConfig()
 | 
				
			||||||
 | 
						formatter := getFormatter()
 | 
				
			||||||
 | 
						packages := getPackages()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						revive := lint.New(func(file string) ([]byte, error) {
 | 
				
			||||||
 | 
							return ioutil.ReadFile(file)
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						lintingRules := getLintingRules(config)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						failures, err := revive.Lint(packages, lintingRules, *config)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							fail(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						formatChan := make(chan lint.Failure)
 | 
				
			||||||
 | 
						exitChan := make(chan bool)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var output string
 | 
				
			||||||
 | 
						go (func() {
 | 
				
			||||||
 | 
							output, err = formatter.Format(formatChan, *config)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								fail(err.Error())
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							exitChan <- true
 | 
				
			||||||
 | 
						})()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						exitCode := 0
 | 
				
			||||||
 | 
						for f := range failures {
 | 
				
			||||||
 | 
							if f.Confidence < config.Confidence {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if exitCode == 0 {
 | 
				
			||||||
 | 
								exitCode = config.WarningCode
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if c, ok := config.Rules[f.RuleName]; ok && c.Severity == lint.SeverityError {
 | 
				
			||||||
 | 
								exitCode = config.ErrorCode
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if c, ok := config.Directives[f.RuleName]; ok && c.Severity == lint.SeverityError {
 | 
				
			||||||
 | 
								exitCode = config.ErrorCode
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							formatChan <- f
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						close(formatChan)
 | 
				
			||||||
 | 
						<-exitChan
 | 
				
			||||||
 | 
						if output != "" {
 | 
				
			||||||
 | 
							fmt.Println(output)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						os.Exit(exitCode)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										18
									
								
								build/vendor.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								build/vendor.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
				
			|||||||
 | 
					// Copyright 2020 The Gitea Authors. All rights reserved.
 | 
				
			||||||
 | 
					// Use of this source code is governed by a MIT-style
 | 
				
			||||||
 | 
					// license that can be found in the LICENSE file.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package build
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						// for lint
 | 
				
			||||||
 | 
						_ "github.com/BurntSushi/toml"
 | 
				
			||||||
 | 
						_ "github.com/mgechev/dots"
 | 
				
			||||||
 | 
						_ "github.com/mgechev/revive/formatter"
 | 
				
			||||||
 | 
						_ "github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						_ "github.com/mgechev/revive/rule"
 | 
				
			||||||
 | 
						_ "github.com/mitchellh/go-homedir"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// for embed
 | 
				
			||||||
 | 
						_ "github.com/shurcooL/vfsgen"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
							
								
								
									
										9
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								go.mod
									
									
									
									
									
								
							@@ -16,6 +16,7 @@ require (
 | 
				
			|||||||
	gitea.com/macaron/macaron v1.4.0
 | 
						gitea.com/macaron/macaron v1.4.0
 | 
				
			||||||
	gitea.com/macaron/session v0.0.0-20191207215012-613cebf0674d
 | 
						gitea.com/macaron/session v0.0.0-20191207215012-613cebf0674d
 | 
				
			||||||
	gitea.com/macaron/toolbox v0.0.0-20190822013122-05ff0fc766b7
 | 
						gitea.com/macaron/toolbox v0.0.0-20190822013122-05ff0fc766b7
 | 
				
			||||||
 | 
						github.com/BurntSushi/toml v0.3.1
 | 
				
			||||||
	github.com/PuerkitoBio/goquery v1.5.0
 | 
						github.com/PuerkitoBio/goquery v1.5.0
 | 
				
			||||||
	github.com/RoaringBitmap/roaring v0.4.21 // indirect
 | 
						github.com/RoaringBitmap/roaring v0.4.21 // indirect
 | 
				
			||||||
	github.com/bgentry/speakeasy v0.1.0 // indirect
 | 
						github.com/bgentry/speakeasy v0.1.0 // indirect
 | 
				
			||||||
@@ -67,17 +68,20 @@ require (
 | 
				
			|||||||
	github.com/lunny/dingtalk_webhook v0.0.0-20171025031554-e3534c89ef96
 | 
						github.com/lunny/dingtalk_webhook v0.0.0-20171025031554-e3534c89ef96
 | 
				
			||||||
	github.com/mailru/easyjson v0.7.0 // indirect
 | 
						github.com/mailru/easyjson v0.7.0 // indirect
 | 
				
			||||||
	github.com/markbates/goth v1.61.2
 | 
						github.com/markbates/goth v1.61.2
 | 
				
			||||||
	github.com/mattn/go-isatty v0.0.7
 | 
						github.com/mattn/go-isatty v0.0.11
 | 
				
			||||||
	github.com/mattn/go-oci8 v0.0.0-20190320171441-14ba190cf52d // indirect
 | 
						github.com/mattn/go-oci8 v0.0.0-20190320171441-14ba190cf52d // indirect
 | 
				
			||||||
	github.com/mattn/go-sqlite3 v1.11.0
 | 
						github.com/mattn/go-sqlite3 v1.11.0
 | 
				
			||||||
	github.com/mcuadros/go-version v0.0.0-20190308113854-92cdf37c5b75
 | 
						github.com/mcuadros/go-version v0.0.0-20190308113854-92cdf37c5b75
 | 
				
			||||||
 | 
						github.com/mgechev/dots v0.0.0-20190921121421-c36f7dcfbb81
 | 
				
			||||||
 | 
						github.com/mgechev/revive v1.0.2
 | 
				
			||||||
	github.com/microcosm-cc/bluemonday v0.0.0-20161012083705-f77f16ffc87a
 | 
						github.com/microcosm-cc/bluemonday v0.0.0-20161012083705-f77f16ffc87a
 | 
				
			||||||
 | 
						github.com/mitchellh/go-homedir v1.1.0
 | 
				
			||||||
	github.com/msteinert/pam v0.0.0-20151204160544-02ccfbfaf0cc
 | 
						github.com/msteinert/pam v0.0.0-20151204160544-02ccfbfaf0cc
 | 
				
			||||||
	github.com/nfnt/resize v0.0.0-20160724205520-891127d8d1b5
 | 
						github.com/nfnt/resize v0.0.0-20160724205520-891127d8d1b5
 | 
				
			||||||
	github.com/niklasfasching/go-org v0.1.9
 | 
						github.com/niklasfasching/go-org v0.1.9
 | 
				
			||||||
	github.com/oliamb/cutter v0.2.2
 | 
						github.com/oliamb/cutter v0.2.2
 | 
				
			||||||
	github.com/olivere/elastic/v7 v7.0.9
 | 
						github.com/olivere/elastic/v7 v7.0.9
 | 
				
			||||||
	github.com/pkg/errors v0.8.1
 | 
						github.com/pkg/errors v0.9.1
 | 
				
			||||||
	github.com/pquerna/otp v0.0.0-20160912161815-54653902c20e
 | 
						github.com/pquerna/otp v0.0.0-20160912161815-54653902c20e
 | 
				
			||||||
	github.com/prometheus/client_golang v1.1.0
 | 
						github.com/prometheus/client_golang v1.1.0
 | 
				
			||||||
	github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4 // indirect
 | 
						github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4 // indirect
 | 
				
			||||||
@@ -107,7 +111,6 @@ require (
 | 
				
			|||||||
	golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45
 | 
						golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45
 | 
				
			||||||
	golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527
 | 
						golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527
 | 
				
			||||||
	golang.org/x/text v0.3.2
 | 
						golang.org/x/text v0.3.2
 | 
				
			||||||
	golang.org/x/tools v0.0.0-20191213221258-04c2e8eff935 // indirect
 | 
					 | 
				
			||||||
	gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc // indirect
 | 
						gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc // indirect
 | 
				
			||||||
	gopkg.in/asn1-ber.v1 v1.0.0-20150924051756-4e86f4367175 // indirect
 | 
						gopkg.in/asn1-ber.v1 v1.0.0-20150924051756-4e86f4367175 // indirect
 | 
				
			||||||
	gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df
 | 
						gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										30
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								go.sum
									
									
									
									
									
								
							@@ -157,6 +157,10 @@ github.com/facebookgo/stack v0.0.0-20160209184415-751773369052 h1:JWuenKqqX8nojt
 | 
				
			|||||||
github.com/facebookgo/stack v0.0.0-20160209184415-751773369052/go.mod h1:UbMTZqLaRiH3MsBH8va0n7s1pQYcu3uTb8G4tygF4Zg=
 | 
					github.com/facebookgo/stack v0.0.0-20160209184415-751773369052/go.mod h1:UbMTZqLaRiH3MsBH8va0n7s1pQYcu3uTb8G4tygF4Zg=
 | 
				
			||||||
github.com/facebookgo/subset v0.0.0-20150612182917-8dac2c3c4870 h1:E2s37DuLxFhQDg5gKsWoLBOB0n+ZW8s599zru8FJ2/Y=
 | 
					github.com/facebookgo/subset v0.0.0-20150612182917-8dac2c3c4870 h1:E2s37DuLxFhQDg5gKsWoLBOB0n+ZW8s599zru8FJ2/Y=
 | 
				
			||||||
github.com/facebookgo/subset v0.0.0-20150612182917-8dac2c3c4870/go.mod h1:5tD+neXqOorC30/tWg0LCSkrqj/AR6gu8yY8/fpw1q0=
 | 
					github.com/facebookgo/subset v0.0.0-20150612182917-8dac2c3c4870/go.mod h1:5tD+neXqOorC30/tWg0LCSkrqj/AR6gu8yY8/fpw1q0=
 | 
				
			||||||
 | 
					github.com/fatih/color v1.9.0 h1:8xPHl4/q1VyqGIPif1F+1V3Y3lSmrq01EabUW3CoW5s=
 | 
				
			||||||
 | 
					github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU=
 | 
				
			||||||
 | 
					github.com/fatih/structtag v1.2.0 h1:/OdNE99OxoI/PqaW/SuSK9uxxT3f/tcSZgon/ssNSx4=
 | 
				
			||||||
 | 
					github.com/fatih/structtag v1.2.0/go.mod h1:mBJUNpUnHmRKrKlQQlmCrh5PuhftFbNv8Ys4/aAZl94=
 | 
				
			||||||
github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568 h1:BHsljHzVlRcyQhjrss6TZTdY2VfCqZPbv5k3iBFa2ZQ=
 | 
					github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568 h1:BHsljHzVlRcyQhjrss6TZTdY2VfCqZPbv5k3iBFa2ZQ=
 | 
				
			||||||
github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc=
 | 
					github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc=
 | 
				
			||||||
github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw=
 | 
					github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw=
 | 
				
			||||||
@@ -184,6 +188,7 @@ github.com/go-git/go-git/v5 v5.0.0/go.mod h1:oYD8y9kWsGINPFJoLdaScGCN6dlKg23blmC
 | 
				
			|||||||
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
 | 
					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-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
 | 
				
			||||||
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
 | 
					github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
 | 
				
			||||||
 | 
					github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas=
 | 
				
			||||||
github.com/go-openapi/analysis v0.0.0-20180825180245-b006789cd277/go.mod h1:k70tL6pCuVxPJOHXQ+wIac1FUrvNkHolPie/cLEU6hI=
 | 
					github.com/go-openapi/analysis v0.0.0-20180825180245-b006789cd277/go.mod h1:k70tL6pCuVxPJOHXQ+wIac1FUrvNkHolPie/cLEU6hI=
 | 
				
			||||||
github.com/go-openapi/analysis v0.17.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
 | 
					github.com/go-openapi/analysis v0.17.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
 | 
				
			||||||
github.com/go-openapi/analysis v0.18.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
 | 
					github.com/go-openapi/analysis v0.18.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
 | 
				
			||||||
@@ -399,10 +404,15 @@ github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7
 | 
				
			|||||||
github.com/markbates/going v1.0.0/go.mod h1:I6mnB4BPnEeqo85ynXIx1ZFLLbtiLHNXVgWeFO9OGOA=
 | 
					github.com/markbates/going v1.0.0/go.mod h1:I6mnB4BPnEeqo85ynXIx1ZFLLbtiLHNXVgWeFO9OGOA=
 | 
				
			||||||
github.com/markbates/goth v1.61.2 h1:jDowrUH5qw8KGuQdKwFhLzkXkTYCIPfz3LHADJsiPIs=
 | 
					github.com/markbates/goth v1.61.2 h1:jDowrUH5qw8KGuQdKwFhLzkXkTYCIPfz3LHADJsiPIs=
 | 
				
			||||||
github.com/markbates/goth v1.61.2/go.mod h1:qh2QfwZoWRucQ+DR5KVKC6dUGkNCToWh4vS45GIzFsY=
 | 
					github.com/markbates/goth v1.61.2/go.mod h1:qh2QfwZoWRucQ+DR5KVKC6dUGkNCToWh4vS45GIzFsY=
 | 
				
			||||||
github.com/mattn/go-isatty v0.0.7 h1:UvyT9uN+3r7yLEYSlJsbQGdsaB/a0DlgWP3pql6iwOc=
 | 
					github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
 | 
				
			||||||
github.com/mattn/go-isatty v0.0.7/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
 | 
					github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.11 h1:FxPOTFNqGkuDUGi3H/qkUbQO4ZiBa2brKq5r0l8TGeM=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE=
 | 
				
			||||||
github.com/mattn/go-oci8 v0.0.0-20190320171441-14ba190cf52d h1:m+dSK37rFf2fqppZhg15yI2IwC9BtucBiRwSDm9VL8g=
 | 
					github.com/mattn/go-oci8 v0.0.0-20190320171441-14ba190cf52d h1:m+dSK37rFf2fqppZhg15yI2IwC9BtucBiRwSDm9VL8g=
 | 
				
			||||||
github.com/mattn/go-oci8 v0.0.0-20190320171441-14ba190cf52d/go.mod h1:/M9VLO+lUPmxvoOK2PfWRZ8mTtB4q1Hy9lEGijv9Nr8=
 | 
					github.com/mattn/go-oci8 v0.0.0-20190320171441-14ba190cf52d/go.mod h1:/M9VLO+lUPmxvoOK2PfWRZ8mTtB4q1Hy9lEGijv9Nr8=
 | 
				
			||||||
 | 
					github.com/mattn/go-runewidth v0.0.7 h1:Ei8KR0497xHyKJPAv59M1dkC+rOZCMBJ+t3fZ+twI54=
 | 
				
			||||||
 | 
					github.com/mattn/go-runewidth v0.0.7/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI=
 | 
				
			||||||
github.com/mattn/go-sqlite3 v1.10.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
 | 
					github.com/mattn/go-sqlite3 v1.10.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
 | 
				
			||||||
github.com/mattn/go-sqlite3 v1.11.0 h1:LDdKkqtYlom37fkvqs8rMPFKAMe8+SgjbwZ6ex1/A/Q=
 | 
					github.com/mattn/go-sqlite3 v1.11.0 h1:LDdKkqtYlom37fkvqs8rMPFKAMe8+SgjbwZ6ex1/A/Q=
 | 
				
			||||||
github.com/mattn/go-sqlite3 v1.11.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
 | 
					github.com/mattn/go-sqlite3 v1.11.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
 | 
				
			||||||
@@ -410,6 +420,10 @@ github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0j
 | 
				
			|||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
 | 
					github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
 | 
				
			||||||
github.com/mcuadros/go-version v0.0.0-20190308113854-92cdf37c5b75 h1:Pijfgr7ZuvX7QIQiEwLdRVr3RoMG+i0SbBO1Qu+7yVk=
 | 
					github.com/mcuadros/go-version v0.0.0-20190308113854-92cdf37c5b75 h1:Pijfgr7ZuvX7QIQiEwLdRVr3RoMG+i0SbBO1Qu+7yVk=
 | 
				
			||||||
github.com/mcuadros/go-version v0.0.0-20190308113854-92cdf37c5b75/go.mod h1:76rfSfYPWj01Z85hUf/ituArm797mNKcvINh1OlsZKo=
 | 
					github.com/mcuadros/go-version v0.0.0-20190308113854-92cdf37c5b75/go.mod h1:76rfSfYPWj01Z85hUf/ituArm797mNKcvINh1OlsZKo=
 | 
				
			||||||
 | 
					github.com/mgechev/dots v0.0.0-20190921121421-c36f7dcfbb81 h1:QASJXOGm2RZ5Ardbc86qNFvby9AqkLDibfChMtAg5QM=
 | 
				
			||||||
 | 
					github.com/mgechev/dots v0.0.0-20190921121421-c36f7dcfbb81/go.mod h1:KQ7+USdGKfpPjXk4Ga+5XxQM4Lm4e3gAogrreFAYpOg=
 | 
				
			||||||
 | 
					github.com/mgechev/revive v1.0.2 h1:v0NxxQ7fSFz/u1NQydPo6EGdq7va0J1BtsZmae6kzUg=
 | 
				
			||||||
 | 
					github.com/mgechev/revive v1.0.2/go.mod h1:rb0dQy1LVAxW9SWy5R3LPUjevzUbUS316U5MFySA2lo=
 | 
				
			||||||
github.com/microcosm-cc/bluemonday v0.0.0-20161012083705-f77f16ffc87a h1:d18LCO3ctH2kugUqt0pEyKKP8L+IYrocaPqGFilhTKk=
 | 
					github.com/microcosm-cc/bluemonday v0.0.0-20161012083705-f77f16ffc87a h1:d18LCO3ctH2kugUqt0pEyKKP8L+IYrocaPqGFilhTKk=
 | 
				
			||||||
github.com/microcosm-cc/bluemonday v0.0.0-20161012083705-f77f16ffc87a/go.mod h1:hsXNsILzKxV+sX77C5b8FSuKF00vh2OMYv+xgHpAMF4=
 | 
					github.com/microcosm-cc/bluemonday v0.0.0-20161012083705-f77f16ffc87a/go.mod h1:hsXNsILzKxV+sX77C5b8FSuKF00vh2OMYv+xgHpAMF4=
 | 
				
			||||||
github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
 | 
					github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
 | 
				
			||||||
@@ -434,6 +448,8 @@ github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLA
 | 
				
			|||||||
github.com/niklasfasching/go-org v0.1.9 h1:Toz8WMIt+qJb52uYEk1YD/muLuOOmRt1CfkV+bKVMkI=
 | 
					github.com/niklasfasching/go-org v0.1.9 h1:Toz8WMIt+qJb52uYEk1YD/muLuOOmRt1CfkV+bKVMkI=
 | 
				
			||||||
github.com/niklasfasching/go-org v0.1.9/go.mod h1:AsLD6X7djzRIz4/RFZu8vwRL0VGjUvGZCCH1Nz0VdrU=
 | 
					github.com/niklasfasching/go-org v0.1.9/go.mod h1:AsLD6X7djzRIz4/RFZu8vwRL0VGjUvGZCCH1Nz0VdrU=
 | 
				
			||||||
github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U=
 | 
					github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U=
 | 
				
			||||||
 | 
					github.com/olekukonko/tablewriter v0.0.4 h1:vHD/YYe1Wolo78koG299f7V/VAS08c6IpCLn+Ejf/w8=
 | 
				
			||||||
 | 
					github.com/olekukonko/tablewriter v0.0.4/go.mod h1:zq6QwlOf5SlnkVbMSr5EoBv3636FWnp+qbPhuoO21uA=
 | 
				
			||||||
github.com/oliamb/cutter v0.2.2 h1:Lfwkya0HHNU1YLnGv2hTkzHfasrSMkgv4Dn+5rmlk3k=
 | 
					github.com/oliamb/cutter v0.2.2 h1:Lfwkya0HHNU1YLnGv2hTkzHfasrSMkgv4Dn+5rmlk3k=
 | 
				
			||||||
github.com/oliamb/cutter v0.2.2/go.mod h1:4BenG2/4GuRBDbVm/OPahDVqbrOemzpPiG5mi1iryBU=
 | 
					github.com/oliamb/cutter v0.2.2/go.mod h1:4BenG2/4GuRBDbVm/OPahDVqbrOemzpPiG5mi1iryBU=
 | 
				
			||||||
github.com/olivere/elastic/v7 v7.0.9 h1:+bTR1xJbfLYD8WnTBt9672mFlKxjfWRJpEQ1y8BMS3g=
 | 
					github.com/olivere/elastic/v7 v7.0.9 h1:+bTR1xJbfLYD8WnTBt9672mFlKxjfWRJpEQ1y8BMS3g=
 | 
				
			||||||
@@ -457,6 +473,8 @@ github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi
 | 
				
			|||||||
github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
					github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
				
			||||||
github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I=
 | 
					github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I=
 | 
				
			||||||
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
					github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
				
			||||||
 | 
					github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
 | 
				
			||||||
 | 
					github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
				
			||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
 | 
					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/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
				
			||||||
github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35/go.mod h1:prYjPmNq4d1NPVmpShWobRqXY3q7Vp+80DqgxxUrUIA=
 | 
					github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35/go.mod h1:prYjPmNq4d1NPVmpShWobRqXY3q7Vp+80DqgxxUrUIA=
 | 
				
			||||||
@@ -631,6 +649,7 @@ golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTk
 | 
				
			|||||||
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
 | 
					golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
 | 
				
			||||||
golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
 | 
					golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
 | 
				
			||||||
golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE=
 | 
					golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE=
 | 
				
			||||||
 | 
					golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee h1:WG0RUwxtNT4qqaXX3DPA8zHFNm/D9xaBpxzHt1WcA/E=
 | 
				
			||||||
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
 | 
					golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
 | 
				
			||||||
golang.org/x/net v0.0.0-20180218175443-cbe0f9307d01/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
 | 
					golang.org/x/net v0.0.0-20180218175443-cbe0f9307d01/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-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
 | 
				
			||||||
@@ -690,6 +709,7 @@ golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7w
 | 
				
			|||||||
golang.org/x/sys v0.0.0-20190907184412-d223b2b6db03/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
					golang.org/x/sys v0.0.0-20190907184412-d223b2b6db03/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
				
			||||||
golang.org/x/sys v0.0.0-20191010194322-b09406accb47 h1:/XfQ9z7ib8eEJX2hdgFTZJ/ntt0swNk5oYBziWeTCvY=
 | 
					golang.org/x/sys v0.0.0-20191010194322-b09406accb47 h1:/XfQ9z7ib8eEJX2hdgFTZJ/ntt0swNk5oYBziWeTCvY=
 | 
				
			||||||
golang.org/x/sys v0.0.0-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
					golang.org/x/sys v0.0.0-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
				
			||||||
 | 
					golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
				
			||||||
golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527 h1:uYVVQ9WP/Ds2ROhcaGPeIdVq0RIXVLwsHlnvJ+cT1So=
 | 
					golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527 h1:uYVVQ9WP/Ds2ROhcaGPeIdVq0RIXVLwsHlnvJ+cT1So=
 | 
				
			||||||
golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
					golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
				
			||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
 | 
					golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
 | 
				
			||||||
@@ -714,9 +734,10 @@ golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgw
 | 
				
			|||||||
golang.org/x/tools v0.0.0-20190617190820-da514acc4774/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
 | 
					golang.org/x/tools v0.0.0-20190617190820-da514acc4774/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
 | 
				
			||||||
golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
 | 
					golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
 | 
				
			||||||
golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
 | 
					golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
 | 
				
			||||||
golang.org/x/tools v0.0.0-20191213221258-04c2e8eff935 h1:kJQZhwFzSwJS2BxboKjdZzWczQOZx8VuH7Y8hhuGUtM=
 | 
					golang.org/x/tools v0.0.0-20200225230052-807dcd883420 h1:4RJNOV+2rLxMEfr6QIpC7GEv9MjD6ApGXTCLrNF9+eA=
 | 
				
			||||||
golang.org/x/tools v0.0.0-20191213221258-04c2e8eff935/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
 | 
					golang.org/x/tools v0.0.0-20200225230052-807dcd883420/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
 | 
				
			||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
					golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
				
			||||||
 | 
					golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898 h1:/atklqdjdhuosWIl6AIbOeHJjicWYPqR9bpxqxYG2pA=
 | 
				
			||||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
					golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
				
			||||||
google.golang.org/api v0.3.1/go.mod h1:6wY9I6uQWHQ8EM57III9mq/AjF+i8G65rmVagqKMtkk=
 | 
					google.golang.org/api v0.3.1/go.mod h1:6wY9I6uQWHQ8EM57III9mq/AjF+i8G65rmVagqKMtkk=
 | 
				
			||||||
google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
 | 
					google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
 | 
				
			||||||
@@ -788,6 +809,7 @@ honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWh
 | 
				
			|||||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
 | 
					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=
 | 
					honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
 | 
				
			||||||
honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
 | 
					honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
 | 
				
			||||||
 | 
					k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I=
 | 
				
			||||||
mvdan.cc/xurls/v2 v2.1.0 h1:KaMb5GLhlcSX+e+qhbRJODnUUBvlw01jt4yrjFIHAuA=
 | 
					mvdan.cc/xurls/v2 v2.1.0 h1:KaMb5GLhlcSX+e+qhbRJODnUUBvlw01jt4yrjFIHAuA=
 | 
				
			||||||
mvdan.cc/xurls/v2 v2.1.0/go.mod h1:5GrSd9rOnKOpZaji1OZLYL/yeAAtGDlo/cFe+8K5n8E=
 | 
					mvdan.cc/xurls/v2 v2.1.0/go.mod h1:5GrSd9rOnKOpZaji1OZLYL/yeAAtGDlo/cFe+8K5n8E=
 | 
				
			||||||
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
 | 
					rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										4
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								main.go
									
									
									
									
									
								
							@@ -21,8 +21,8 @@ import (
 | 
				
			|||||||
	_ "code.gitea.io/gitea/modules/markup/markdown"
 | 
						_ "code.gitea.io/gitea/modules/markup/markdown"
 | 
				
			||||||
	_ "code.gitea.io/gitea/modules/markup/orgmode"
 | 
						_ "code.gitea.io/gitea/modules/markup/orgmode"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// for embed
 | 
						// for build
 | 
				
			||||||
	_ "github.com/shurcooL/vfsgen"
 | 
						_ "code.gitea.io/gitea/build"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	"github.com/urfave/cli"
 | 
						"github.com/urfave/cli"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,4 +6,4 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
package options
 | 
					package options
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//go:generate go run -mod=vendor ../../scripts/generate-bindata.go ../../options options bindata.go
 | 
					//go:generate go run -mod=vendor ../../build/generate-bindata.go ../../options options bindata.go
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,4 +6,4 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
package public
 | 
					package public
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//go:generate go run -mod=vendor  ../../scripts/generate-bindata.go ../../public public bindata.go
 | 
					//go:generate go run -mod=vendor  ../../build/generate-bindata.go ../../public public bindata.go
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,4 +6,4 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
package templates
 | 
					package templates
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//go:generate go run -mod=vendor ../../scripts/generate-bindata.go ../../templates templates bindata.go
 | 
					//go:generate go run -mod=vendor ../../build/generate-bindata.go ../../templates templates bindata.go
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										20
									
								
								vendor/github.com/fatih/color/LICENSE.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								vendor/github.com/fatih/color/LICENSE.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,20 @@
 | 
				
			|||||||
 | 
					The MIT License (MIT)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright (c) 2013 Fatih Arslan
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Permission is hereby granted, free of charge, to any person obtaining a copy of
 | 
				
			||||||
 | 
					this software and associated documentation files (the "Software"), to deal in
 | 
				
			||||||
 | 
					the Software without restriction, including without limitation the rights to
 | 
				
			||||||
 | 
					use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
 | 
				
			||||||
 | 
					the Software, and to permit persons to whom the Software is furnished to do so,
 | 
				
			||||||
 | 
					subject to the following conditions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The above copyright notice and this permission notice shall be included in all
 | 
				
			||||||
 | 
					copies or substantial portions of the Software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
				
			||||||
 | 
					IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
 | 
				
			||||||
 | 
					FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
 | 
				
			||||||
 | 
					COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
 | 
				
			||||||
 | 
					IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 | 
				
			||||||
 | 
					CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | 
				
			||||||
							
								
								
									
										182
									
								
								vendor/github.com/fatih/color/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										182
									
								
								vendor/github.com/fatih/color/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,182 @@
 | 
				
			|||||||
 | 
					# Archived project. No maintenance. 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					This project is not maintained anymore and is archived. Feel free to fork and
 | 
				
			||||||
 | 
					make your own changes if needed. For more detail read my blog post: [Taking an indefinite sabbatical from my projects](https://arslan.io/2018/10/09/taking-an-indefinite-sabbatical-from-my-projects/)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Thanks to everyone for their valuable feedback and contributions.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Color [](https://godoc.org/github.com/fatih/color) 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Color lets you use colorized outputs in terms of [ANSI Escape
 | 
				
			||||||
 | 
					Codes](http://en.wikipedia.org/wiki/ANSI_escape_code#Colors) in Go (Golang). It
 | 
				
			||||||
 | 
					has support for Windows too! The API can be used in several ways, pick one that
 | 
				
			||||||
 | 
					suits you.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Install
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```bash
 | 
				
			||||||
 | 
					go get github.com/fatih/color
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Examples
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Standard colors
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					// Print with default helper functions
 | 
				
			||||||
 | 
					color.Cyan("Prints text in cyan.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// A newline will be appended automatically
 | 
				
			||||||
 | 
					color.Blue("Prints %s in blue.", "text")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// These are using the default foreground colors
 | 
				
			||||||
 | 
					color.Red("We have red")
 | 
				
			||||||
 | 
					color.Magenta("And many others ..")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Mix and reuse colors
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					// Create a new color object
 | 
				
			||||||
 | 
					c := color.New(color.FgCyan).Add(color.Underline)
 | 
				
			||||||
 | 
					c.Println("Prints cyan text with an underline.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Or just add them to New()
 | 
				
			||||||
 | 
					d := color.New(color.FgCyan, color.Bold)
 | 
				
			||||||
 | 
					d.Printf("This prints bold cyan %s\n", "too!.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Mix up foreground and background colors, create new mixes!
 | 
				
			||||||
 | 
					red := color.New(color.FgRed)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					boldRed := red.Add(color.Bold)
 | 
				
			||||||
 | 
					boldRed.Println("This will print text in bold red.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					whiteBackground := red.Add(color.BgWhite)
 | 
				
			||||||
 | 
					whiteBackground.Println("Red text with white background.")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Use your own output (io.Writer)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					// Use your own io.Writer output
 | 
				
			||||||
 | 
					color.New(color.FgBlue).Fprintln(myWriter, "blue color!")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					blue := color.New(color.FgBlue)
 | 
				
			||||||
 | 
					blue.Fprint(writer, "This will print text in blue.")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Custom print functions (PrintFunc)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					// Create a custom print function for convenience
 | 
				
			||||||
 | 
					red := color.New(color.FgRed).PrintfFunc()
 | 
				
			||||||
 | 
					red("Warning")
 | 
				
			||||||
 | 
					red("Error: %s", err)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Mix up multiple attributes
 | 
				
			||||||
 | 
					notice := color.New(color.Bold, color.FgGreen).PrintlnFunc()
 | 
				
			||||||
 | 
					notice("Don't forget this...")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Custom fprint functions (FprintFunc)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					blue := color.New(FgBlue).FprintfFunc()
 | 
				
			||||||
 | 
					blue(myWriter, "important notice: %s", stars)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Mix up with multiple attributes
 | 
				
			||||||
 | 
					success := color.New(color.Bold, color.FgGreen).FprintlnFunc()
 | 
				
			||||||
 | 
					success(myWriter, "Don't forget this...")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Insert into noncolor strings (SprintFunc)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					// Create SprintXxx functions to mix strings with other non-colorized strings:
 | 
				
			||||||
 | 
					yellow := color.New(color.FgYellow).SprintFunc()
 | 
				
			||||||
 | 
					red := color.New(color.FgRed).SprintFunc()
 | 
				
			||||||
 | 
					fmt.Printf("This is a %s and this is %s.\n", yellow("warning"), red("error"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					info := color.New(color.FgWhite, color.BgGreen).SprintFunc()
 | 
				
			||||||
 | 
					fmt.Printf("This %s rocks!\n", info("package"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Use helper functions
 | 
				
			||||||
 | 
					fmt.Println("This", color.RedString("warning"), "should be not neglected.")
 | 
				
			||||||
 | 
					fmt.Printf("%v %v\n", color.GreenString("Info:"), "an important message.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Windows supported too! Just don't forget to change the output to color.Output
 | 
				
			||||||
 | 
					fmt.Fprintf(color.Output, "Windows support: %s", color.GreenString("PASS"))
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Plug into existing code
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					// Use handy standard colors
 | 
				
			||||||
 | 
					color.Set(color.FgYellow)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					fmt.Println("Existing text will now be in yellow")
 | 
				
			||||||
 | 
					fmt.Printf("This one %s\n", "too")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					color.Unset() // Don't forget to unset
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// You can mix up parameters
 | 
				
			||||||
 | 
					color.Set(color.FgMagenta, color.Bold)
 | 
				
			||||||
 | 
					defer color.Unset() // Use it in your function
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					fmt.Println("All text will now be bold magenta.")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Disable/Enable color
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					There might be a case where you want to explicitly disable/enable color output. the 
 | 
				
			||||||
 | 
					`go-isatty` package will automatically disable color output for non-tty output streams 
 | 
				
			||||||
 | 
					(for example if the output were piped directly to `less`)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					`Color` has support to disable/enable colors both globally and for single color 
 | 
				
			||||||
 | 
					definitions. For example suppose you have a CLI app and a `--no-color` bool flag. You 
 | 
				
			||||||
 | 
					can easily disable the color output with:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var flagNoColor = flag.Bool("no-color", false, "Disable color output")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if *flagNoColor {
 | 
				
			||||||
 | 
						color.NoColor = true // disables colorized output
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					It also has support for single color definitions (local). You can
 | 
				
			||||||
 | 
					disable/enable color output on the fly:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					c := color.New(color.FgCyan)
 | 
				
			||||||
 | 
					c.Println("Prints cyan text")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					c.DisableColor()
 | 
				
			||||||
 | 
					c.Println("This is printed without any color")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					c.EnableColor()
 | 
				
			||||||
 | 
					c.Println("This prints again cyan...")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Todo
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* Save/Return previous values
 | 
				
			||||||
 | 
					* Evaluate fmt.Formatter interface
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Credits
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 * [Fatih Arslan](https://github.com/fatih)
 | 
				
			||||||
 | 
					 * Windows support via @mattn: [colorable](https://github.com/mattn/go-colorable)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## License
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The MIT License (MIT) - see [`LICENSE.md`](https://github.com/fatih/color/blob/master/LICENSE.md) for more details
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										603
									
								
								vendor/github.com/fatih/color/color.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										603
									
								
								vendor/github.com/fatih/color/color.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,603 @@
 | 
				
			|||||||
 | 
					package color
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"io"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
						"sync"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mattn/go-colorable"
 | 
				
			||||||
 | 
						"github.com/mattn/go-isatty"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						// NoColor defines if the output is colorized or not. It's dynamically set to
 | 
				
			||||||
 | 
						// false or true based on the stdout's file descriptor referring to a terminal
 | 
				
			||||||
 | 
						// or not. This is a global option and affects all colors. For more control
 | 
				
			||||||
 | 
						// over each color block use the methods DisableColor() individually.
 | 
				
			||||||
 | 
						NoColor = os.Getenv("TERM") == "dumb" ||
 | 
				
			||||||
 | 
							(!isatty.IsTerminal(os.Stdout.Fd()) && !isatty.IsCygwinTerminal(os.Stdout.Fd()))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Output defines the standard output of the print functions. By default
 | 
				
			||||||
 | 
						// os.Stdout is used.
 | 
				
			||||||
 | 
						Output = colorable.NewColorableStdout()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Error defines a color supporting writer for os.Stderr.
 | 
				
			||||||
 | 
						Error = colorable.NewColorableStderr()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// colorsCache is used to reduce the count of created Color objects and
 | 
				
			||||||
 | 
						// allows to reuse already created objects with required Attribute.
 | 
				
			||||||
 | 
						colorsCache   = make(map[Attribute]*Color)
 | 
				
			||||||
 | 
						colorsCacheMu sync.Mutex // protects colorsCache
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Color defines a custom color object which is defined by SGR parameters.
 | 
				
			||||||
 | 
					type Color struct {
 | 
				
			||||||
 | 
						params  []Attribute
 | 
				
			||||||
 | 
						noColor *bool
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Attribute defines a single SGR Code
 | 
				
			||||||
 | 
					type Attribute int
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const escape = "\x1b"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Base attributes
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						Reset Attribute = iota
 | 
				
			||||||
 | 
						Bold
 | 
				
			||||||
 | 
						Faint
 | 
				
			||||||
 | 
						Italic
 | 
				
			||||||
 | 
						Underline
 | 
				
			||||||
 | 
						BlinkSlow
 | 
				
			||||||
 | 
						BlinkRapid
 | 
				
			||||||
 | 
						ReverseVideo
 | 
				
			||||||
 | 
						Concealed
 | 
				
			||||||
 | 
						CrossedOut
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Foreground text colors
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						FgBlack Attribute = iota + 30
 | 
				
			||||||
 | 
						FgRed
 | 
				
			||||||
 | 
						FgGreen
 | 
				
			||||||
 | 
						FgYellow
 | 
				
			||||||
 | 
						FgBlue
 | 
				
			||||||
 | 
						FgMagenta
 | 
				
			||||||
 | 
						FgCyan
 | 
				
			||||||
 | 
						FgWhite
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Foreground Hi-Intensity text colors
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						FgHiBlack Attribute = iota + 90
 | 
				
			||||||
 | 
						FgHiRed
 | 
				
			||||||
 | 
						FgHiGreen
 | 
				
			||||||
 | 
						FgHiYellow
 | 
				
			||||||
 | 
						FgHiBlue
 | 
				
			||||||
 | 
						FgHiMagenta
 | 
				
			||||||
 | 
						FgHiCyan
 | 
				
			||||||
 | 
						FgHiWhite
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Background text colors
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						BgBlack Attribute = iota + 40
 | 
				
			||||||
 | 
						BgRed
 | 
				
			||||||
 | 
						BgGreen
 | 
				
			||||||
 | 
						BgYellow
 | 
				
			||||||
 | 
						BgBlue
 | 
				
			||||||
 | 
						BgMagenta
 | 
				
			||||||
 | 
						BgCyan
 | 
				
			||||||
 | 
						BgWhite
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Background Hi-Intensity text colors
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						BgHiBlack Attribute = iota + 100
 | 
				
			||||||
 | 
						BgHiRed
 | 
				
			||||||
 | 
						BgHiGreen
 | 
				
			||||||
 | 
						BgHiYellow
 | 
				
			||||||
 | 
						BgHiBlue
 | 
				
			||||||
 | 
						BgHiMagenta
 | 
				
			||||||
 | 
						BgHiCyan
 | 
				
			||||||
 | 
						BgHiWhite
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// New returns a newly created color object.
 | 
				
			||||||
 | 
					func New(value ...Attribute) *Color {
 | 
				
			||||||
 | 
						c := &Color{params: make([]Attribute, 0)}
 | 
				
			||||||
 | 
						c.Add(value...)
 | 
				
			||||||
 | 
						return c
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Set sets the given parameters immediately. It will change the color of
 | 
				
			||||||
 | 
					// output with the given SGR parameters until color.Unset() is called.
 | 
				
			||||||
 | 
					func Set(p ...Attribute) *Color {
 | 
				
			||||||
 | 
						c := New(p...)
 | 
				
			||||||
 | 
						c.Set()
 | 
				
			||||||
 | 
						return c
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Unset resets all escape attributes and clears the output. Usually should
 | 
				
			||||||
 | 
					// be called after Set().
 | 
				
			||||||
 | 
					func Unset() {
 | 
				
			||||||
 | 
						if NoColor {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fmt.Fprintf(Output, "%s[%dm", escape, Reset)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Set sets the SGR sequence.
 | 
				
			||||||
 | 
					func (c *Color) Set() *Color {
 | 
				
			||||||
 | 
						if c.isNoColorSet() {
 | 
				
			||||||
 | 
							return c
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fmt.Fprintf(Output, c.format())
 | 
				
			||||||
 | 
						return c
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) unset() {
 | 
				
			||||||
 | 
						if c.isNoColorSet() {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						Unset()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) setWriter(w io.Writer) *Color {
 | 
				
			||||||
 | 
						if c.isNoColorSet() {
 | 
				
			||||||
 | 
							return c
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fmt.Fprintf(w, c.format())
 | 
				
			||||||
 | 
						return c
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) unsetWriter(w io.Writer) {
 | 
				
			||||||
 | 
						if c.isNoColorSet() {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if NoColor {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fmt.Fprintf(w, "%s[%dm", escape, Reset)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Add is used to chain SGR parameters. Use as many as parameters to combine
 | 
				
			||||||
 | 
					// and create custom color objects. Example: Add(color.FgRed, color.Underline).
 | 
				
			||||||
 | 
					func (c *Color) Add(value ...Attribute) *Color {
 | 
				
			||||||
 | 
						c.params = append(c.params, value...)
 | 
				
			||||||
 | 
						return c
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) prepend(value Attribute) {
 | 
				
			||||||
 | 
						c.params = append(c.params, 0)
 | 
				
			||||||
 | 
						copy(c.params[1:], c.params[0:])
 | 
				
			||||||
 | 
						c.params[0] = value
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Fprint formats using the default formats for its operands and writes to w.
 | 
				
			||||||
 | 
					// Spaces are added between operands when neither is a string.
 | 
				
			||||||
 | 
					// It returns the number of bytes written and any write error encountered.
 | 
				
			||||||
 | 
					// On Windows, users should wrap w with colorable.NewColorable() if w is of
 | 
				
			||||||
 | 
					// type *os.File.
 | 
				
			||||||
 | 
					func (c *Color) Fprint(w io.Writer, a ...interface{}) (n int, err error) {
 | 
				
			||||||
 | 
						c.setWriter(w)
 | 
				
			||||||
 | 
						defer c.unsetWriter(w)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Fprint(w, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Print formats using the default formats for its operands and writes to
 | 
				
			||||||
 | 
					// standard output. Spaces are added between operands when neither is a
 | 
				
			||||||
 | 
					// string. It returns the number of bytes written and any write error
 | 
				
			||||||
 | 
					// encountered. This is the standard fmt.Print() method wrapped with the given
 | 
				
			||||||
 | 
					// color.
 | 
				
			||||||
 | 
					func (c *Color) Print(a ...interface{}) (n int, err error) {
 | 
				
			||||||
 | 
						c.Set()
 | 
				
			||||||
 | 
						defer c.unset()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Fprint(Output, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Fprintf formats according to a format specifier and writes to w.
 | 
				
			||||||
 | 
					// It returns the number of bytes written and any write error encountered.
 | 
				
			||||||
 | 
					// On Windows, users should wrap w with colorable.NewColorable() if w is of
 | 
				
			||||||
 | 
					// type *os.File.
 | 
				
			||||||
 | 
					func (c *Color) Fprintf(w io.Writer, format string, a ...interface{}) (n int, err error) {
 | 
				
			||||||
 | 
						c.setWriter(w)
 | 
				
			||||||
 | 
						defer c.unsetWriter(w)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Fprintf(w, format, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Printf formats according to a format specifier and writes to standard output.
 | 
				
			||||||
 | 
					// It returns the number of bytes written and any write error encountered.
 | 
				
			||||||
 | 
					// This is the standard fmt.Printf() method wrapped with the given color.
 | 
				
			||||||
 | 
					func (c *Color) Printf(format string, a ...interface{}) (n int, err error) {
 | 
				
			||||||
 | 
						c.Set()
 | 
				
			||||||
 | 
						defer c.unset()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Fprintf(Output, format, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Fprintln formats using the default formats for its operands and writes to w.
 | 
				
			||||||
 | 
					// Spaces are always added between operands and a newline is appended.
 | 
				
			||||||
 | 
					// On Windows, users should wrap w with colorable.NewColorable() if w is of
 | 
				
			||||||
 | 
					// type *os.File.
 | 
				
			||||||
 | 
					func (c *Color) Fprintln(w io.Writer, a ...interface{}) (n int, err error) {
 | 
				
			||||||
 | 
						c.setWriter(w)
 | 
				
			||||||
 | 
						defer c.unsetWriter(w)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Fprintln(w, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Println formats using the default formats for its operands and writes to
 | 
				
			||||||
 | 
					// standard output. Spaces are always added between operands and a newline is
 | 
				
			||||||
 | 
					// appended. It returns the number of bytes written and any write error
 | 
				
			||||||
 | 
					// encountered. This is the standard fmt.Print() method wrapped with the given
 | 
				
			||||||
 | 
					// color.
 | 
				
			||||||
 | 
					func (c *Color) Println(a ...interface{}) (n int, err error) {
 | 
				
			||||||
 | 
						c.Set()
 | 
				
			||||||
 | 
						defer c.unset()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Fprintln(Output, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Sprint is just like Print, but returns a string instead of printing it.
 | 
				
			||||||
 | 
					func (c *Color) Sprint(a ...interface{}) string {
 | 
				
			||||||
 | 
						return c.wrap(fmt.Sprint(a...))
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Sprintln is just like Println, but returns a string instead of printing it.
 | 
				
			||||||
 | 
					func (c *Color) Sprintln(a ...interface{}) string {
 | 
				
			||||||
 | 
						return c.wrap(fmt.Sprintln(a...))
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Sprintf is just like Printf, but returns a string instead of printing it.
 | 
				
			||||||
 | 
					func (c *Color) Sprintf(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return c.wrap(fmt.Sprintf(format, a...))
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FprintFunc returns a new function that prints the passed arguments as
 | 
				
			||||||
 | 
					// colorized with color.Fprint().
 | 
				
			||||||
 | 
					func (c *Color) FprintFunc() func(w io.Writer, a ...interface{}) {
 | 
				
			||||||
 | 
						return func(w io.Writer, a ...interface{}) {
 | 
				
			||||||
 | 
							c.Fprint(w, a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PrintFunc returns a new function that prints the passed arguments as
 | 
				
			||||||
 | 
					// colorized with color.Print().
 | 
				
			||||||
 | 
					func (c *Color) PrintFunc() func(a ...interface{}) {
 | 
				
			||||||
 | 
						return func(a ...interface{}) {
 | 
				
			||||||
 | 
							c.Print(a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FprintfFunc returns a new function that prints the passed arguments as
 | 
				
			||||||
 | 
					// colorized with color.Fprintf().
 | 
				
			||||||
 | 
					func (c *Color) FprintfFunc() func(w io.Writer, format string, a ...interface{}) {
 | 
				
			||||||
 | 
						return func(w io.Writer, format string, a ...interface{}) {
 | 
				
			||||||
 | 
							c.Fprintf(w, format, a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PrintfFunc returns a new function that prints the passed arguments as
 | 
				
			||||||
 | 
					// colorized with color.Printf().
 | 
				
			||||||
 | 
					func (c *Color) PrintfFunc() func(format string, a ...interface{}) {
 | 
				
			||||||
 | 
						return func(format string, a ...interface{}) {
 | 
				
			||||||
 | 
							c.Printf(format, a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FprintlnFunc returns a new function that prints the passed arguments as
 | 
				
			||||||
 | 
					// colorized with color.Fprintln().
 | 
				
			||||||
 | 
					func (c *Color) FprintlnFunc() func(w io.Writer, a ...interface{}) {
 | 
				
			||||||
 | 
						return func(w io.Writer, a ...interface{}) {
 | 
				
			||||||
 | 
							c.Fprintln(w, a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PrintlnFunc returns a new function that prints the passed arguments as
 | 
				
			||||||
 | 
					// colorized with color.Println().
 | 
				
			||||||
 | 
					func (c *Color) PrintlnFunc() func(a ...interface{}) {
 | 
				
			||||||
 | 
						return func(a ...interface{}) {
 | 
				
			||||||
 | 
							c.Println(a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// SprintFunc returns a new function that returns colorized strings for the
 | 
				
			||||||
 | 
					// given arguments with fmt.Sprint(). Useful to put into or mix into other
 | 
				
			||||||
 | 
					// string. Windows users should use this in conjunction with color.Output, example:
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					//	put := New(FgYellow).SprintFunc()
 | 
				
			||||||
 | 
					//	fmt.Fprintf(color.Output, "This is a %s", put("warning"))
 | 
				
			||||||
 | 
					func (c *Color) SprintFunc() func(a ...interface{}) string {
 | 
				
			||||||
 | 
						return func(a ...interface{}) string {
 | 
				
			||||||
 | 
							return c.wrap(fmt.Sprint(a...))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// SprintfFunc returns a new function that returns colorized strings for the
 | 
				
			||||||
 | 
					// given arguments with fmt.Sprintf(). Useful to put into or mix into other
 | 
				
			||||||
 | 
					// string. Windows users should use this in conjunction with color.Output.
 | 
				
			||||||
 | 
					func (c *Color) SprintfFunc() func(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return func(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
							return c.wrap(fmt.Sprintf(format, a...))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// SprintlnFunc returns a new function that returns colorized strings for the
 | 
				
			||||||
 | 
					// given arguments with fmt.Sprintln(). Useful to put into or mix into other
 | 
				
			||||||
 | 
					// string. Windows users should use this in conjunction with color.Output.
 | 
				
			||||||
 | 
					func (c *Color) SprintlnFunc() func(a ...interface{}) string {
 | 
				
			||||||
 | 
						return func(a ...interface{}) string {
 | 
				
			||||||
 | 
							return c.wrap(fmt.Sprintln(a...))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// sequence returns a formatted SGR sequence to be plugged into a "\x1b[...m"
 | 
				
			||||||
 | 
					// an example output might be: "1;36" -> bold cyan
 | 
				
			||||||
 | 
					func (c *Color) sequence() string {
 | 
				
			||||||
 | 
						format := make([]string, len(c.params))
 | 
				
			||||||
 | 
						for i, v := range c.params {
 | 
				
			||||||
 | 
							format[i] = strconv.Itoa(int(v))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return strings.Join(format, ";")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// wrap wraps the s string with the colors attributes. The string is ready to
 | 
				
			||||||
 | 
					// be printed.
 | 
				
			||||||
 | 
					func (c *Color) wrap(s string) string {
 | 
				
			||||||
 | 
						if c.isNoColorSet() {
 | 
				
			||||||
 | 
							return s
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return c.format() + s + c.unformat()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) format() string {
 | 
				
			||||||
 | 
						return fmt.Sprintf("%s[%sm", escape, c.sequence())
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) unformat() string {
 | 
				
			||||||
 | 
						return fmt.Sprintf("%s[%dm", escape, Reset)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DisableColor disables the color output. Useful to not change any existing
 | 
				
			||||||
 | 
					// code and still being able to output. Can be used for flags like
 | 
				
			||||||
 | 
					// "--no-color". To enable back use EnableColor() method.
 | 
				
			||||||
 | 
					func (c *Color) DisableColor() {
 | 
				
			||||||
 | 
						c.noColor = boolPtr(true)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// EnableColor enables the color output. Use it in conjunction with
 | 
				
			||||||
 | 
					// DisableColor(). Otherwise this method has no side effects.
 | 
				
			||||||
 | 
					func (c *Color) EnableColor() {
 | 
				
			||||||
 | 
						c.noColor = boolPtr(false)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) isNoColorSet() bool {
 | 
				
			||||||
 | 
						// check first if we have user setted action
 | 
				
			||||||
 | 
						if c.noColor != nil {
 | 
				
			||||||
 | 
							return *c.noColor
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// if not return the global option, which is disabled by default
 | 
				
			||||||
 | 
						return NoColor
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Equals returns a boolean value indicating whether two colors are equal.
 | 
				
			||||||
 | 
					func (c *Color) Equals(c2 *Color) bool {
 | 
				
			||||||
 | 
						if len(c.params) != len(c2.params) {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, attr := range c.params {
 | 
				
			||||||
 | 
							if !c2.attrExists(attr) {
 | 
				
			||||||
 | 
								return false
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return true
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Color) attrExists(a Attribute) bool {
 | 
				
			||||||
 | 
						for _, attr := range c.params {
 | 
				
			||||||
 | 
							if attr == a {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func boolPtr(v bool) *bool {
 | 
				
			||||||
 | 
						return &v
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getCachedColor(p Attribute) *Color {
 | 
				
			||||||
 | 
						colorsCacheMu.Lock()
 | 
				
			||||||
 | 
						defer colorsCacheMu.Unlock()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						c, ok := colorsCache[p]
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							c = New(p)
 | 
				
			||||||
 | 
							colorsCache[p] = c
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return c
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func colorPrint(format string, p Attribute, a ...interface{}) {
 | 
				
			||||||
 | 
						c := getCachedColor(p)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if !strings.HasSuffix(format, "\n") {
 | 
				
			||||||
 | 
							format += "\n"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(a) == 0 {
 | 
				
			||||||
 | 
							c.Print(format)
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							c.Printf(format, a...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func colorString(format string, p Attribute, a ...interface{}) string {
 | 
				
			||||||
 | 
						c := getCachedColor(p)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(a) == 0 {
 | 
				
			||||||
 | 
							return c.SprintFunc()(format)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return c.SprintfFunc()(format, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Black is a convenient helper function to print with black foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func Black(format string, a ...interface{}) { colorPrint(format, FgBlack, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Red is a convenient helper function to print with red foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func Red(format string, a ...interface{}) { colorPrint(format, FgRed, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Green is a convenient helper function to print with green foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func Green(format string, a ...interface{}) { colorPrint(format, FgGreen, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Yellow is a convenient helper function to print with yellow foreground.
 | 
				
			||||||
 | 
					// A newline is appended to format by default.
 | 
				
			||||||
 | 
					func Yellow(format string, a ...interface{}) { colorPrint(format, FgYellow, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Blue is a convenient helper function to print with blue foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func Blue(format string, a ...interface{}) { colorPrint(format, FgBlue, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Magenta is a convenient helper function to print with magenta foreground.
 | 
				
			||||||
 | 
					// A newline is appended to format by default.
 | 
				
			||||||
 | 
					func Magenta(format string, a ...interface{}) { colorPrint(format, FgMagenta, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Cyan is a convenient helper function to print with cyan foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func Cyan(format string, a ...interface{}) { colorPrint(format, FgCyan, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// White is a convenient helper function to print with white foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func White(format string, a ...interface{}) { colorPrint(format, FgWhite, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// BlackString is a convenient helper function to return a string with black
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func BlackString(format string, a ...interface{}) string { return colorString(format, FgBlack, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// RedString is a convenient helper function to return a string with red
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func RedString(format string, a ...interface{}) string { return colorString(format, FgRed, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GreenString is a convenient helper function to return a string with green
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func GreenString(format string, a ...interface{}) string { return colorString(format, FgGreen, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// YellowString is a convenient helper function to return a string with yellow
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func YellowString(format string, a ...interface{}) string { return colorString(format, FgYellow, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// BlueString is a convenient helper function to return a string with blue
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func BlueString(format string, a ...interface{}) string { return colorString(format, FgBlue, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// MagentaString is a convenient helper function to return a string with magenta
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func MagentaString(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return colorString(format, FgMagenta, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CyanString is a convenient helper function to return a string with cyan
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func CyanString(format string, a ...interface{}) string { return colorString(format, FgCyan, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// WhiteString is a convenient helper function to return a string with white
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func WhiteString(format string, a ...interface{}) string { return colorString(format, FgWhite, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiBlack is a convenient helper function to print with hi-intensity black foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiBlack(format string, a ...interface{}) { colorPrint(format, FgHiBlack, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiRed is a convenient helper function to print with hi-intensity red foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiRed(format string, a ...interface{}) { colorPrint(format, FgHiRed, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiGreen is a convenient helper function to print with hi-intensity green foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiGreen(format string, a ...interface{}) { colorPrint(format, FgHiGreen, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiYellow is a convenient helper function to print with hi-intensity yellow foreground.
 | 
				
			||||||
 | 
					// A newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiYellow(format string, a ...interface{}) { colorPrint(format, FgHiYellow, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiBlue is a convenient helper function to print with hi-intensity blue foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiBlue(format string, a ...interface{}) { colorPrint(format, FgHiBlue, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiMagenta is a convenient helper function to print with hi-intensity magenta foreground.
 | 
				
			||||||
 | 
					// A newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiMagenta(format string, a ...interface{}) { colorPrint(format, FgHiMagenta, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiCyan is a convenient helper function to print with hi-intensity cyan foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiCyan(format string, a ...interface{}) { colorPrint(format, FgHiCyan, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiWhite is a convenient helper function to print with hi-intensity white foreground. A
 | 
				
			||||||
 | 
					// newline is appended to format by default.
 | 
				
			||||||
 | 
					func HiWhite(format string, a ...interface{}) { colorPrint(format, FgHiWhite, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiBlackString is a convenient helper function to return a string with hi-intensity black
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiBlackString(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return colorString(format, FgHiBlack, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiRedString is a convenient helper function to return a string with hi-intensity red
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiRedString(format string, a ...interface{}) string { return colorString(format, FgHiRed, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiGreenString is a convenient helper function to return a string with hi-intensity green
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiGreenString(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return colorString(format, FgHiGreen, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiYellowString is a convenient helper function to return a string with hi-intensity yellow
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiYellowString(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return colorString(format, FgHiYellow, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiBlueString is a convenient helper function to return a string with hi-intensity blue
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiBlueString(format string, a ...interface{}) string { return colorString(format, FgHiBlue, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiMagentaString is a convenient helper function to return a string with hi-intensity magenta
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiMagentaString(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return colorString(format, FgHiMagenta, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiCyanString is a convenient helper function to return a string with hi-intensity cyan
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiCyanString(format string, a ...interface{}) string { return colorString(format, FgHiCyan, a...) }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HiWhiteString is a convenient helper function to return a string with hi-intensity white
 | 
				
			||||||
 | 
					// foreground.
 | 
				
			||||||
 | 
					func HiWhiteString(format string, a ...interface{}) string {
 | 
				
			||||||
 | 
						return colorString(format, FgHiWhite, a...)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										133
									
								
								vendor/github.com/fatih/color/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								vendor/github.com/fatih/color/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,133 @@
 | 
				
			|||||||
 | 
					/*
 | 
				
			||||||
 | 
					Package color is an ANSI color package to output colorized or SGR defined
 | 
				
			||||||
 | 
					output to the standard output. The API can be used in several way, pick one
 | 
				
			||||||
 | 
					that suits you.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Use simple and default helper functions with predefined foreground colors:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    color.Cyan("Prints text in cyan.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // a newline will be appended automatically
 | 
				
			||||||
 | 
					    color.Blue("Prints %s in blue.", "text")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // More default foreground colors..
 | 
				
			||||||
 | 
					    color.Red("We have red")
 | 
				
			||||||
 | 
					    color.Yellow("Yellow color too!")
 | 
				
			||||||
 | 
					    color.Magenta("And many others ..")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Hi-intensity colors
 | 
				
			||||||
 | 
					    color.HiGreen("Bright green color.")
 | 
				
			||||||
 | 
					    color.HiBlack("Bright black means gray..")
 | 
				
			||||||
 | 
					    color.HiWhite("Shiny white color!")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					However there are times where custom color mixes are required. Below are some
 | 
				
			||||||
 | 
					examples to create custom color objects and use the print functions of each
 | 
				
			||||||
 | 
					separate color object.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Create a new color object
 | 
				
			||||||
 | 
					    c := color.New(color.FgCyan).Add(color.Underline)
 | 
				
			||||||
 | 
					    c.Println("Prints cyan text with an underline.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Or just add them to New()
 | 
				
			||||||
 | 
					    d := color.New(color.FgCyan, color.Bold)
 | 
				
			||||||
 | 
					    d.Printf("This prints bold cyan %s\n", "too!.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Mix up foreground and background colors, create new mixes!
 | 
				
			||||||
 | 
					    red := color.New(color.FgRed)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    boldRed := red.Add(color.Bold)
 | 
				
			||||||
 | 
					    boldRed.Println("This will print text in bold red.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    whiteBackground := red.Add(color.BgWhite)
 | 
				
			||||||
 | 
					    whiteBackground.Println("Red text with White background.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Use your own io.Writer output
 | 
				
			||||||
 | 
					    color.New(color.FgBlue).Fprintln(myWriter, "blue color!")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    blue := color.New(color.FgBlue)
 | 
				
			||||||
 | 
					    blue.Fprint(myWriter, "This will print text in blue.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You can create PrintXxx functions to simplify even more:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Create a custom print function for convenient
 | 
				
			||||||
 | 
					    red := color.New(color.FgRed).PrintfFunc()
 | 
				
			||||||
 | 
					    red("warning")
 | 
				
			||||||
 | 
					    red("error: %s", err)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Mix up multiple attributes
 | 
				
			||||||
 | 
					    notice := color.New(color.Bold, color.FgGreen).PrintlnFunc()
 | 
				
			||||||
 | 
					    notice("don't forget this...")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You can also FprintXxx functions to pass your own io.Writer:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    blue := color.New(FgBlue).FprintfFunc()
 | 
				
			||||||
 | 
					    blue(myWriter, "important notice: %s", stars)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Mix up with multiple attributes
 | 
				
			||||||
 | 
					    success := color.New(color.Bold, color.FgGreen).FprintlnFunc()
 | 
				
			||||||
 | 
					    success(myWriter, don't forget this...")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Or create SprintXxx functions to mix strings with other non-colorized strings:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    yellow := New(FgYellow).SprintFunc()
 | 
				
			||||||
 | 
					    red := New(FgRed).SprintFunc()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fmt.Printf("this is a %s and this is %s.\n", yellow("warning"), red("error"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    info := New(FgWhite, BgGreen).SprintFunc()
 | 
				
			||||||
 | 
					    fmt.Printf("this %s rocks!\n", info("package"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Windows support is enabled by default. All Print functions work as intended.
 | 
				
			||||||
 | 
					However only for color.SprintXXX functions, user should use fmt.FprintXXX and
 | 
				
			||||||
 | 
					set the output to color.Output:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fmt.Fprintf(color.Output, "Windows support: %s", color.GreenString("PASS"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    info := New(FgWhite, BgGreen).SprintFunc()
 | 
				
			||||||
 | 
					    fmt.Fprintf(color.Output, "this %s rocks!\n", info("package"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Using with existing code is possible. Just use the Set() method to set the
 | 
				
			||||||
 | 
					standard output to the given parameters. That way a rewrite of an existing
 | 
				
			||||||
 | 
					code is not required.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Use handy standard colors.
 | 
				
			||||||
 | 
					    color.Set(color.FgYellow)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fmt.Println("Existing text will be now in Yellow")
 | 
				
			||||||
 | 
					    fmt.Printf("This one %s\n", "too")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    color.Unset() // don't forget to unset
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // You can mix up parameters
 | 
				
			||||||
 | 
					    color.Set(color.FgMagenta, color.Bold)
 | 
				
			||||||
 | 
					    defer color.Unset() // use it in your function
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fmt.Println("All text will be now bold magenta.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					There might be a case where you want to disable color output (for example to
 | 
				
			||||||
 | 
					pipe the standard output of your app to somewhere else). `Color` has support to
 | 
				
			||||||
 | 
					disable colors both globally and for single color definition. For example
 | 
				
			||||||
 | 
					suppose you have a CLI app and a `--no-color` bool flag. You can easily disable
 | 
				
			||||||
 | 
					the color output with:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    var flagNoColor = flag.Bool("no-color", false, "Disable color output")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if *flagNoColor {
 | 
				
			||||||
 | 
					    	color.NoColor = true // disables colorized output
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					It also has support for single color definitions (local). You can
 | 
				
			||||||
 | 
					disable/enable color output on the fly:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					     c := color.New(color.FgCyan)
 | 
				
			||||||
 | 
					     c.Println("Prints cyan text")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					     c.DisableColor()
 | 
				
			||||||
 | 
					     c.Println("This is printed without any color")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					     c.EnableColor()
 | 
				
			||||||
 | 
					     c.Println("This prints again cyan...")
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					package color
 | 
				
			||||||
							
								
								
									
										8
									
								
								vendor/github.com/fatih/color/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								vendor/github.com/fatih/color/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					module github.com/fatih/color
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					go 1.13
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					require (
 | 
				
			||||||
 | 
						github.com/mattn/go-colorable v0.1.4
 | 
				
			||||||
 | 
						github.com/mattn/go-isatty v0.0.11
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
							
								
								
									
										8
									
								
								vendor/github.com/fatih/color/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								vendor/github.com/fatih/color/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
 | 
				
			||||||
 | 
					github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.11 h1:FxPOTFNqGkuDUGi3H/qkUbQO4ZiBa2brKq5r0l8TGeM=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE=
 | 
				
			||||||
 | 
					golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
 | 
				
			||||||
 | 
					golang.org/x/sys v0.0.0-20191026070338-33540a1f6037 h1:YyJpGZS1sBuBCzLAR1VEpK193GlqGZbnPFnPV/5Rsb4=
 | 
				
			||||||
 | 
					golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
				
			||||||
							
								
								
									
										60
									
								
								vendor/github.com/fatih/structtag/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								vendor/github.com/fatih/structtag/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,60 @@
 | 
				
			|||||||
 | 
					Copyright (c) 2017, Fatih Arslan
 | 
				
			||||||
 | 
					All rights reserved.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Redistribution and use in source and binary forms, with or without
 | 
				
			||||||
 | 
					modification, are permitted provided that the following conditions are met:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* Redistributions of source code must retain the above copyright notice, this
 | 
				
			||||||
 | 
					  list of conditions and the following disclaimer.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* Redistributions in binary form must reproduce the above copyright notice,
 | 
				
			||||||
 | 
					  this list of conditions and the following disclaimer in the documentation
 | 
				
			||||||
 | 
					  and/or other materials provided with the distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					* Neither the name of structtag nor the names of its
 | 
				
			||||||
 | 
					  contributors may be used to endorse or promote products derived from
 | 
				
			||||||
 | 
					  this software without specific prior written permission.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | 
				
			||||||
 | 
					AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | 
				
			||||||
 | 
					IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 | 
				
			||||||
 | 
					DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
 | 
				
			||||||
 | 
					FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | 
				
			||||||
 | 
					DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 | 
				
			||||||
 | 
					SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
 | 
				
			||||||
 | 
					CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 | 
				
			||||||
 | 
					OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | 
				
			||||||
 | 
					OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					This software includes some portions from Go. Go is used under the terms of the
 | 
				
			||||||
 | 
					BSD like license.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright (c) 2012 The Go Authors. All rights reserved.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Redistribution and use in source and binary forms, with or without
 | 
				
			||||||
 | 
					modification, are permitted provided that the following conditions are
 | 
				
			||||||
 | 
					met:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   * Redistributions of source code must retain the above copyright
 | 
				
			||||||
 | 
					notice, this list of conditions and the following disclaimer.
 | 
				
			||||||
 | 
					   * Redistributions in binary form must reproduce the above
 | 
				
			||||||
 | 
					copyright notice, this list of conditions and the following disclaimer
 | 
				
			||||||
 | 
					in the documentation and/or other materials provided with the
 | 
				
			||||||
 | 
					distribution.
 | 
				
			||||||
 | 
					   * Neither the name of Google Inc. nor the names of its
 | 
				
			||||||
 | 
					contributors may be used to endorse or promote products derived from
 | 
				
			||||||
 | 
					this software without specific prior written permission.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 | 
				
			||||||
 | 
					"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 | 
				
			||||||
 | 
					LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 | 
				
			||||||
 | 
					A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 | 
				
			||||||
 | 
					OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | 
				
			||||||
 | 
					SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 | 
				
			||||||
 | 
					LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 | 
				
			||||||
 | 
					DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 | 
				
			||||||
 | 
					THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | 
				
			||||||
 | 
					(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | 
				
			||||||
 | 
					OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The Go gopher was designed by Renee French. http://reneefrench.blogspot.com/ The design is licensed under the Creative Commons 3.0 Attributions license. Read this article for more details: https://blog.golang.org/gopher
 | 
				
			||||||
							
								
								
									
										73
									
								
								vendor/github.com/fatih/structtag/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								vendor/github.com/fatih/structtag/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,73 @@
 | 
				
			|||||||
 | 
					# structtag [](http://godoc.org/github.com/fatih/structtag) 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					structtag provides an easy way of parsing and manipulating struct tag fields.
 | 
				
			||||||
 | 
					Please vendor the library as it might change in future versions.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Install
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```bash
 | 
				
			||||||
 | 
					go get github.com/fatih/structtag
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Example
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					package main
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"reflect"
 | 
				
			||||||
 | 
						"sort"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/fatih/structtag"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func main() {
 | 
				
			||||||
 | 
						type t struct {
 | 
				
			||||||
 | 
							t string `json:"foo,omitempty,string" xml:"foo"`
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// get field tag
 | 
				
			||||||
 | 
						tag := reflect.TypeOf(t{}).Field(0).Tag
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// ... and start using structtag by parsing the tag
 | 
				
			||||||
 | 
						tags, err := structtag.Parse(string(tag))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							panic(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// iterate over all tags
 | 
				
			||||||
 | 
						for _, t := range tags.Tags() {
 | 
				
			||||||
 | 
							fmt.Printf("tag: %+v\n", t)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// get a single tag
 | 
				
			||||||
 | 
						jsonTag, err := tags.Get("json")
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							panic(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						fmt.Println(jsonTag)         // Output: json:"foo,omitempty,string"
 | 
				
			||||||
 | 
						fmt.Println(jsonTag.Key)     // Output: json
 | 
				
			||||||
 | 
						fmt.Println(jsonTag.Name)    // Output: foo
 | 
				
			||||||
 | 
						fmt.Println(jsonTag.Options) // Output: [omitempty string]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// change existing tag
 | 
				
			||||||
 | 
						jsonTag.Name = "foo_bar"
 | 
				
			||||||
 | 
						jsonTag.Options = nil
 | 
				
			||||||
 | 
						tags.Set(jsonTag)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// add new tag
 | 
				
			||||||
 | 
						tags.Set(&structtag.Tag{
 | 
				
			||||||
 | 
							Key:     "hcl",
 | 
				
			||||||
 | 
							Name:    "foo",
 | 
				
			||||||
 | 
							Options: []string{"squash"},
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// print the tags
 | 
				
			||||||
 | 
						fmt.Println(tags) // Output: json:"foo_bar" xml:"foo" hcl:"foo,squash"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// sort tags according to keys
 | 
				
			||||||
 | 
						sort.Sort(tags)
 | 
				
			||||||
 | 
						fmt.Println(tags) // Output: hcl:"foo,squash" json:"foo_bar" xml:"foo"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
							
								
								
									
										3
									
								
								vendor/github.com/fatih/structtag/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								vendor/github.com/fatih/structtag/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
				
			|||||||
 | 
					module github.com/fatih/structtag
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					go 1.12
 | 
				
			||||||
							
								
								
									
										315
									
								
								vendor/github.com/fatih/structtag/tags.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										315
									
								
								vendor/github.com/fatih/structtag/tags.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,315 @@
 | 
				
			|||||||
 | 
					package structtag
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"errors"
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						errTagSyntax      = errors.New("bad syntax for struct tag pair")
 | 
				
			||||||
 | 
						errTagKeySyntax   = errors.New("bad syntax for struct tag key")
 | 
				
			||||||
 | 
						errTagValueSyntax = errors.New("bad syntax for struct tag value")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						errKeyNotSet      = errors.New("tag key does not exist")
 | 
				
			||||||
 | 
						errTagNotExist    = errors.New("tag does not exist")
 | 
				
			||||||
 | 
						errTagKeyMismatch = errors.New("mismatch between key and tag.key")
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Tags represent a set of tags from a single struct field
 | 
				
			||||||
 | 
					type Tags struct {
 | 
				
			||||||
 | 
						tags []*Tag
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Tag defines a single struct's string literal tag
 | 
				
			||||||
 | 
					type Tag struct {
 | 
				
			||||||
 | 
						// Key is the tag key, such as json, xml, etc..
 | 
				
			||||||
 | 
						// i.e: `json:"foo,omitempty". Here key is: "json"
 | 
				
			||||||
 | 
						Key string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Name is a part of the value
 | 
				
			||||||
 | 
						// i.e: `json:"foo,omitempty". Here name is: "foo"
 | 
				
			||||||
 | 
						Name string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Options is a part of the value. It contains a slice of tag options i.e:
 | 
				
			||||||
 | 
						// `json:"foo,omitempty". Here options is: ["omitempty"]
 | 
				
			||||||
 | 
						Options []string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Parse parses a single struct field tag and returns the set of tags.
 | 
				
			||||||
 | 
					func Parse(tag string) (*Tags, error) {
 | 
				
			||||||
 | 
						var tags []*Tag
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						hasTag := tag != ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// NOTE(arslan) following code is from reflect and vet package with some
 | 
				
			||||||
 | 
						// modifications to collect all necessary information and extend it with
 | 
				
			||||||
 | 
						// usable methods
 | 
				
			||||||
 | 
						for tag != "" {
 | 
				
			||||||
 | 
							// Skip leading space.
 | 
				
			||||||
 | 
							i := 0
 | 
				
			||||||
 | 
							for i < len(tag) && tag[i] == ' ' {
 | 
				
			||||||
 | 
								i++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							tag = tag[i:]
 | 
				
			||||||
 | 
							if tag == "" {
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Scan to colon. A space, a quote or a control character is a syntax
 | 
				
			||||||
 | 
							// error. Strictly speaking, control chars include the range [0x7f,
 | 
				
			||||||
 | 
							// 0x9f], not just [0x00, 0x1f], but in practice, we ignore the
 | 
				
			||||||
 | 
							// multi-byte control characters as it is simpler to inspect the tag's
 | 
				
			||||||
 | 
							// bytes than the tag's runes.
 | 
				
			||||||
 | 
							i = 0
 | 
				
			||||||
 | 
							for i < len(tag) && tag[i] > ' ' && tag[i] != ':' && tag[i] != '"' && tag[i] != 0x7f {
 | 
				
			||||||
 | 
								i++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if i == 0 {
 | 
				
			||||||
 | 
								return nil, errTagKeySyntax
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if i+1 >= len(tag) || tag[i] != ':' {
 | 
				
			||||||
 | 
								return nil, errTagSyntax
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if tag[i+1] != '"' {
 | 
				
			||||||
 | 
								return nil, errTagValueSyntax
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							key := string(tag[:i])
 | 
				
			||||||
 | 
							tag = tag[i+1:]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Scan quoted string to find value.
 | 
				
			||||||
 | 
							i = 1
 | 
				
			||||||
 | 
							for i < len(tag) && tag[i] != '"' {
 | 
				
			||||||
 | 
								if tag[i] == '\\' {
 | 
				
			||||||
 | 
									i++
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								i++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if i >= len(tag) {
 | 
				
			||||||
 | 
								return nil, errTagValueSyntax
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							qvalue := string(tag[:i+1])
 | 
				
			||||||
 | 
							tag = tag[i+1:]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							value, err := strconv.Unquote(qvalue)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return nil, errTagValueSyntax
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							res := strings.Split(value, ",")
 | 
				
			||||||
 | 
							name := res[0]
 | 
				
			||||||
 | 
							options := res[1:]
 | 
				
			||||||
 | 
							if len(options) == 0 {
 | 
				
			||||||
 | 
								options = nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							tags = append(tags, &Tag{
 | 
				
			||||||
 | 
								Key:     key,
 | 
				
			||||||
 | 
								Name:    name,
 | 
				
			||||||
 | 
								Options: options,
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if hasTag && len(tags) == 0 {
 | 
				
			||||||
 | 
							return nil, nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return &Tags{
 | 
				
			||||||
 | 
							tags: tags,
 | 
				
			||||||
 | 
						}, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Get returns the tag associated with the given key. If the key is present
 | 
				
			||||||
 | 
					// in the tag the value (which may be empty) is returned. Otherwise the
 | 
				
			||||||
 | 
					// returned value will be the empty string. The ok return value reports whether
 | 
				
			||||||
 | 
					// the tag exists or not (which the return value is nil).
 | 
				
			||||||
 | 
					func (t *Tags) Get(key string) (*Tag, error) {
 | 
				
			||||||
 | 
						for _, tag := range t.tags {
 | 
				
			||||||
 | 
							if tag.Key == key {
 | 
				
			||||||
 | 
								return tag, nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return nil, errTagNotExist
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Set sets the given tag. If the tag key already exists it'll override it
 | 
				
			||||||
 | 
					func (t *Tags) Set(tag *Tag) error {
 | 
				
			||||||
 | 
						if tag.Key == "" {
 | 
				
			||||||
 | 
							return errKeyNotSet
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						added := false
 | 
				
			||||||
 | 
						for i, tg := range t.tags {
 | 
				
			||||||
 | 
							if tg.Key == tag.Key {
 | 
				
			||||||
 | 
								added = true
 | 
				
			||||||
 | 
								t.tags[i] = tag
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if !added {
 | 
				
			||||||
 | 
							// this means this is a new tag, add it
 | 
				
			||||||
 | 
							t.tags = append(t.tags, tag)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// AddOptions adds the given option for the given key. If the option already
 | 
				
			||||||
 | 
					// exists it doesn't add it again.
 | 
				
			||||||
 | 
					func (t *Tags) AddOptions(key string, options ...string) {
 | 
				
			||||||
 | 
						for i, tag := range t.tags {
 | 
				
			||||||
 | 
							if tag.Key != key {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							for _, opt := range options {
 | 
				
			||||||
 | 
								if !tag.HasOption(opt) {
 | 
				
			||||||
 | 
									tag.Options = append(tag.Options, opt)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							t.tags[i] = tag
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DeleteOptions deletes the given options for the given key
 | 
				
			||||||
 | 
					func (t *Tags) DeleteOptions(key string, options ...string) {
 | 
				
			||||||
 | 
						hasOption := func(option string) bool {
 | 
				
			||||||
 | 
							for _, opt := range options {
 | 
				
			||||||
 | 
								if opt == option {
 | 
				
			||||||
 | 
									return true
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for i, tag := range t.tags {
 | 
				
			||||||
 | 
							if tag.Key != key {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							var updated []string
 | 
				
			||||||
 | 
							for _, opt := range tag.Options {
 | 
				
			||||||
 | 
								if !hasOption(opt) {
 | 
				
			||||||
 | 
									updated = append(updated, opt)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							tag.Options = updated
 | 
				
			||||||
 | 
							t.tags[i] = tag
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Delete deletes the tag for the given keys
 | 
				
			||||||
 | 
					func (t *Tags) Delete(keys ...string) {
 | 
				
			||||||
 | 
						hasKey := func(key string) bool {
 | 
				
			||||||
 | 
							for _, k := range keys {
 | 
				
			||||||
 | 
								if k == key {
 | 
				
			||||||
 | 
									return true
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var updated []*Tag
 | 
				
			||||||
 | 
						for _, tag := range t.tags {
 | 
				
			||||||
 | 
							if !hasKey(tag.Key) {
 | 
				
			||||||
 | 
								updated = append(updated, tag)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						t.tags = updated
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Tags returns a slice of tags. The order is the original tag order unless it
 | 
				
			||||||
 | 
					// was changed.
 | 
				
			||||||
 | 
					func (t *Tags) Tags() []*Tag {
 | 
				
			||||||
 | 
						return t.tags
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Tags returns a slice of tags. The order is the original tag order unless it
 | 
				
			||||||
 | 
					// was changed.
 | 
				
			||||||
 | 
					func (t *Tags) Keys() []string {
 | 
				
			||||||
 | 
						var keys []string
 | 
				
			||||||
 | 
						for _, tag := range t.tags {
 | 
				
			||||||
 | 
							keys = append(keys, tag.Key)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return keys
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// String reassembles the tags into a valid literal tag field representation
 | 
				
			||||||
 | 
					func (t *Tags) String() string {
 | 
				
			||||||
 | 
						tags := t.Tags()
 | 
				
			||||||
 | 
						if len(tags) == 0 {
 | 
				
			||||||
 | 
							return ""
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var buf bytes.Buffer
 | 
				
			||||||
 | 
						for i, tag := range t.Tags() {
 | 
				
			||||||
 | 
							buf.WriteString(tag.String())
 | 
				
			||||||
 | 
							if i != len(tags)-1 {
 | 
				
			||||||
 | 
								buf.WriteString(" ")
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return buf.String()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HasOption returns true if the given option is available in options
 | 
				
			||||||
 | 
					func (t *Tag) HasOption(opt string) bool {
 | 
				
			||||||
 | 
						for _, tagOpt := range t.Options {
 | 
				
			||||||
 | 
							if tagOpt == opt {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Value returns the raw value of the tag, i.e. if the tag is
 | 
				
			||||||
 | 
					// `json:"foo,omitempty", the Value is "foo,omitempty"
 | 
				
			||||||
 | 
					func (t *Tag) Value() string {
 | 
				
			||||||
 | 
						options := strings.Join(t.Options, ",")
 | 
				
			||||||
 | 
						if options != "" {
 | 
				
			||||||
 | 
							return fmt.Sprintf(`%s,%s`, t.Name, options)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return t.Name
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// String reassembles the tag into a valid tag field representation
 | 
				
			||||||
 | 
					func (t *Tag) String() string {
 | 
				
			||||||
 | 
						return fmt.Sprintf(`%s:%q`, t.Key, t.Value())
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GoString implements the fmt.GoStringer interface
 | 
				
			||||||
 | 
					func (t *Tag) GoString() string {
 | 
				
			||||||
 | 
						template := `{
 | 
				
			||||||
 | 
							Key:    '%s',
 | 
				
			||||||
 | 
							Name:   '%s',
 | 
				
			||||||
 | 
							Option: '%s',
 | 
				
			||||||
 | 
						}`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if t.Options == nil {
 | 
				
			||||||
 | 
							return fmt.Sprintf(template, t.Key, t.Name, "nil")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						options := strings.Join(t.Options, ",")
 | 
				
			||||||
 | 
						return fmt.Sprintf(template, t.Key, t.Name, options)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (t *Tags) Len() int {
 | 
				
			||||||
 | 
						return len(t.tags)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (t *Tags) Less(i int, j int) bool {
 | 
				
			||||||
 | 
						return t.tags[i].Key < t.tags[j].Key
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (t *Tags) Swap(i int, j int) {
 | 
				
			||||||
 | 
						t.tags[i], t.tags[j] = t.tags[j], t.tags[i]
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										9
									
								
								vendor/github.com/mattn/go-colorable/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								vendor/github.com/mattn/go-colorable/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					language: go
 | 
				
			||||||
 | 
					go:
 | 
				
			||||||
 | 
					  - tip
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					before_install:
 | 
				
			||||||
 | 
					  - go get github.com/mattn/goveralls
 | 
				
			||||||
 | 
					  - go get golang.org/x/tools/cmd/cover
 | 
				
			||||||
 | 
					script:
 | 
				
			||||||
 | 
					  - $HOME/gopath/bin/goveralls -repotoken xnXqRGwgW3SXIguzxf90ZSK1GPYZPaGrw
 | 
				
			||||||
							
								
								
									
										21
									
								
								vendor/github.com/mattn/go-colorable/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/github.com/mattn/go-colorable/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					The MIT License (MIT)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright (c) 2016 Yasuhiro Matsumoto
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
				
			||||||
 | 
					of this software and associated documentation files (the "Software"), to deal
 | 
				
			||||||
 | 
					in the Software without restriction, including without limitation the rights
 | 
				
			||||||
 | 
					to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
				
			||||||
 | 
					copies of the Software, and to permit persons to whom the Software is
 | 
				
			||||||
 | 
					furnished to do so, subject to the following conditions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The above copyright notice and this permission notice shall be included in all
 | 
				
			||||||
 | 
					copies or substantial portions of the Software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
				
			||||||
 | 
					IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
				
			||||||
 | 
					FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
				
			||||||
 | 
					AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
				
			||||||
 | 
					LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
				
			||||||
 | 
					OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | 
				
			||||||
 | 
					SOFTWARE.
 | 
				
			||||||
							
								
								
									
										48
									
								
								vendor/github.com/mattn/go-colorable/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								vendor/github.com/mattn/go-colorable/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,48 @@
 | 
				
			|||||||
 | 
					# go-colorable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					[](http://godoc.org/github.com/mattn/go-colorable)
 | 
				
			||||||
 | 
					[](https://travis-ci.org/mattn/go-colorable)
 | 
				
			||||||
 | 
					[](https://coveralls.io/github/mattn/go-colorable?branch=master)
 | 
				
			||||||
 | 
					[](https://goreportcard.com/report/mattn/go-colorable)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Colorable writer for windows.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					For example, most of logger packages doesn't show colors on windows. (I know we can do it with ansicon. But I don't want.)
 | 
				
			||||||
 | 
					This package is possible to handle escape sequence for ansi color on windows.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Too Bad!
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## So Good!
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Usage
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					logrus.SetFormatter(&logrus.TextFormatter{ForceColors: true})
 | 
				
			||||||
 | 
					logrus.SetOutput(colorable.NewColorableStdout())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					logrus.Info("succeeded")
 | 
				
			||||||
 | 
					logrus.Warn("not correct")
 | 
				
			||||||
 | 
					logrus.Error("something error")
 | 
				
			||||||
 | 
					logrus.Fatal("panic")
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You can compile above code on non-windows OSs.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Installation
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					$ go get github.com/mattn/go-colorable
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# License
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					MIT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Author
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Yasuhiro Matsumoto (a.k.a mattn)
 | 
				
			||||||
							
								
								
									
										29
									
								
								vendor/github.com/mattn/go-colorable/colorable_appengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/github.com/mattn/go-colorable/colorable_appengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
				
			|||||||
 | 
					// +build appengine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package colorable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"io"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						_ "github.com/mattn/go-isatty"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewColorable returns new instance of Writer which handles escape sequence.
 | 
				
			||||||
 | 
					func NewColorable(file *os.File) io.Writer {
 | 
				
			||||||
 | 
						if file == nil {
 | 
				
			||||||
 | 
							panic("nil passed instead of *os.File to NewColorable()")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return file
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewColorableStdout returns new instance of Writer which handles escape sequence for stdout.
 | 
				
			||||||
 | 
					func NewColorableStdout() io.Writer {
 | 
				
			||||||
 | 
						return os.Stdout
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewColorableStderr returns new instance of Writer which handles escape sequence for stderr.
 | 
				
			||||||
 | 
					func NewColorableStderr() io.Writer {
 | 
				
			||||||
 | 
						return os.Stderr
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										30
									
								
								vendor/github.com/mattn/go-colorable/colorable_others.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								vendor/github.com/mattn/go-colorable/colorable_others.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,30 @@
 | 
				
			|||||||
 | 
					// +build !windows
 | 
				
			||||||
 | 
					// +build !appengine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package colorable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"io"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						_ "github.com/mattn/go-isatty"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewColorable returns new instance of Writer which handles escape sequence.
 | 
				
			||||||
 | 
					func NewColorable(file *os.File) io.Writer {
 | 
				
			||||||
 | 
						if file == nil {
 | 
				
			||||||
 | 
							panic("nil passed instead of *os.File to NewColorable()")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return file
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewColorableStdout returns new instance of Writer which handles escape sequence for stdout.
 | 
				
			||||||
 | 
					func NewColorableStdout() io.Writer {
 | 
				
			||||||
 | 
						return os.Stdout
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewColorableStderr returns new instance of Writer which handles escape sequence for stderr.
 | 
				
			||||||
 | 
					func NewColorableStderr() io.Writer {
 | 
				
			||||||
 | 
						return os.Stderr
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										1005
									
								
								vendor/github.com/mattn/go-colorable/colorable_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1005
									
								
								vendor/github.com/mattn/go-colorable/colorable_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										3
									
								
								vendor/github.com/mattn/go-colorable/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								vendor/github.com/mattn/go-colorable/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
				
			|||||||
 | 
					module github.com/mattn/go-colorable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					require github.com/mattn/go-isatty v0.0.8
 | 
				
			||||||
							
								
								
									
										4
									
								
								vendor/github.com/mattn/go-colorable/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								vendor/github.com/mattn/go-colorable/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
				
			|||||||
 | 
					github.com/mattn/go-isatty v0.0.5 h1:tHXDdz1cpzGaovsTB+TVB8q90WEokoVmfMqoVcrLUgw=
 | 
				
			||||||
 | 
					github.com/mattn/go-isatty v0.0.5/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
 | 
				
			||||||
 | 
					golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223 h1:DH4skfRX4EBpamg7iV4ZlCpblAHI6s6TDM39bFZumv8=
 | 
				
			||||||
 | 
					golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
 | 
				
			||||||
							
								
								
									
										55
									
								
								vendor/github.com/mattn/go-colorable/noncolorable.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								vendor/github.com/mattn/go-colorable/noncolorable.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,55 @@
 | 
				
			|||||||
 | 
					package colorable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"io"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NonColorable holds writer but removes escape sequence.
 | 
				
			||||||
 | 
					type NonColorable struct {
 | 
				
			||||||
 | 
						out io.Writer
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewNonColorable returns new instance of Writer which removes escape sequence from Writer.
 | 
				
			||||||
 | 
					func NewNonColorable(w io.Writer) io.Writer {
 | 
				
			||||||
 | 
						return &NonColorable{out: w}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Write writes data on console
 | 
				
			||||||
 | 
					func (w *NonColorable) Write(data []byte) (n int, err error) {
 | 
				
			||||||
 | 
						er := bytes.NewReader(data)
 | 
				
			||||||
 | 
						var bw [1]byte
 | 
				
			||||||
 | 
					loop:
 | 
				
			||||||
 | 
						for {
 | 
				
			||||||
 | 
							c1, err := er.ReadByte()
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								break loop
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if c1 != 0x1b {
 | 
				
			||||||
 | 
								bw[0] = c1
 | 
				
			||||||
 | 
								w.out.Write(bw[:])
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							c2, err := er.ReadByte()
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								break loop
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if c2 != 0x5b {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							var buf bytes.Buffer
 | 
				
			||||||
 | 
							for {
 | 
				
			||||||
 | 
								c, err := er.ReadByte()
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									break loop
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || c == '@' {
 | 
				
			||||||
 | 
									break
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								buf.Write([]byte(string(c)))
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return len(data), nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										4
									
								
								vendor/github.com/mattn/go-isatty/go.mod
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/github.com/mattn/go-isatty/go.mod
									
									
									
										generated
									
									
										vendored
									
									
								
							@@ -1,3 +1,5 @@
 | 
				
			|||||||
module github.com/mattn/go-isatty
 | 
					module github.com/mattn/go-isatty
 | 
				
			||||||
 | 
					
 | 
				
			||||||
require golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223
 | 
					go 1.12
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					require golang.org/x/sys v0.0.0-20191026070338-33540a1f6037
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										4
									
								
								vendor/github.com/mattn/go-isatty/go.sum
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/github.com/mattn/go-isatty/go.sum
									
									
									
										generated
									
									
										vendored
									
									
								
							@@ -1,2 +1,2 @@
 | 
				
			|||||||
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223 h1:DH4skfRX4EBpamg7iV4ZlCpblAHI6s6TDM39bFZumv8=
 | 
					golang.org/x/sys v0.0.0-20191026070338-33540a1f6037 h1:YyJpGZS1sBuBCzLAR1VEpK193GlqGZbnPFnPV/5Rsb4=
 | 
				
			||||||
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
 | 
					golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										2
									
								
								vendor/github.com/mattn/go-isatty/isatty_others.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								vendor/github.com/mattn/go-isatty/isatty_others.go
									
									
									
										generated
									
									
										vendored
									
									
								
							@@ -1,4 +1,4 @@
 | 
				
			|||||||
// +build appengine js
 | 
					// +build appengine js nacl
 | 
				
			||||||
 | 
					
 | 
				
			||||||
package isatty
 | 
					package isatty
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										22
									
								
								vendor/github.com/mattn/go-isatty/isatty_plan9.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								vendor/github.com/mattn/go-isatty/isatty_plan9.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,22 @@
 | 
				
			|||||||
 | 
					// +build plan9
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package isatty
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"syscall"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsTerminal returns true if the given file descriptor is a terminal.
 | 
				
			||||||
 | 
					func IsTerminal(fd uintptr) bool {
 | 
				
			||||||
 | 
						path, err := syscall.Fd2path(int(fd))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return path == "/dev/cons" || path == "/mnt/term/dev/cons"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsCygwinTerminal return true if the file descriptor is a cygwin or msys2
 | 
				
			||||||
 | 
					// terminal. This is also always false on this environment.
 | 
				
			||||||
 | 
					func IsCygwinTerminal(fd uintptr) bool {
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -1,4 +1,4 @@
 | 
				
			|||||||
// +build linux
 | 
					// +build linux aix
 | 
				
			||||||
// +build !appengine
 | 
					// +build !appengine
 | 
				
			||||||
// +build !android
 | 
					// +build !android
 | 
				
			||||||
 | 
					
 | 
				
			||||||
							
								
								
									
										35
									
								
								vendor/github.com/mattn/go-isatty/isatty_windows.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										35
									
								
								vendor/github.com/mattn/go-isatty/isatty_windows.go
									
									
									
										generated
									
									
										vendored
									
									
								
							@@ -4,6 +4,7 @@
 | 
				
			|||||||
package isatty
 | 
					package isatty
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import (
 | 
					import (
 | 
				
			||||||
 | 
						"errors"
 | 
				
			||||||
	"strings"
 | 
						"strings"
 | 
				
			||||||
	"syscall"
 | 
						"syscall"
 | 
				
			||||||
	"unicode/utf16"
 | 
						"unicode/utf16"
 | 
				
			||||||
@@ -11,15 +12,18 @@ import (
 | 
				
			|||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const (
 | 
					const (
 | 
				
			||||||
	fileNameInfo uintptr = 2
 | 
						objectNameInfo uintptr = 1
 | 
				
			||||||
 | 
						fileNameInfo           = 2
 | 
				
			||||||
	fileTypePipe           = 3
 | 
						fileTypePipe           = 3
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
var (
 | 
					var (
 | 
				
			||||||
	kernel32                         = syscall.NewLazyDLL("kernel32.dll")
 | 
						kernel32                         = syscall.NewLazyDLL("kernel32.dll")
 | 
				
			||||||
 | 
						ntdll                            = syscall.NewLazyDLL("ntdll.dll")
 | 
				
			||||||
	procGetConsoleMode               = kernel32.NewProc("GetConsoleMode")
 | 
						procGetConsoleMode               = kernel32.NewProc("GetConsoleMode")
 | 
				
			||||||
	procGetFileInformationByHandleEx = kernel32.NewProc("GetFileInformationByHandleEx")
 | 
						procGetFileInformationByHandleEx = kernel32.NewProc("GetFileInformationByHandleEx")
 | 
				
			||||||
	procGetFileType                  = kernel32.NewProc("GetFileType")
 | 
						procGetFileType                  = kernel32.NewProc("GetFileType")
 | 
				
			||||||
 | 
						procNtQueryObject                = ntdll.NewProc("NtQueryObject")
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func init() {
 | 
					func init() {
 | 
				
			||||||
@@ -45,7 +49,10 @@ func isCygwinPipeName(name string) bool {
 | 
				
			|||||||
		return false
 | 
							return false
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if token[0] != `\msys` && token[0] != `\cygwin` {
 | 
						if token[0] != `\msys` &&
 | 
				
			||||||
 | 
							token[0] != `\cygwin` &&
 | 
				
			||||||
 | 
							token[0] != `\Device\NamedPipe\msys` &&
 | 
				
			||||||
 | 
							token[0] != `\Device\NamedPipe\cygwin` {
 | 
				
			||||||
		return false
 | 
							return false
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -68,12 +75,36 @@ func isCygwinPipeName(name string) bool {
 | 
				
			|||||||
	return true
 | 
						return true
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// getFileNameByHandle use the undocomented ntdll NtQueryObject to get file full name from file handler
 | 
				
			||||||
 | 
					// since GetFileInformationByHandleEx is not avilable under windows Vista and still some old fashion
 | 
				
			||||||
 | 
					// guys are using Windows XP, this is a workaround for those guys, it will also work on system from
 | 
				
			||||||
 | 
					// Windows vista to 10
 | 
				
			||||||
 | 
					// see https://stackoverflow.com/a/18792477 for details
 | 
				
			||||||
 | 
					func getFileNameByHandle(fd uintptr) (string, error) {
 | 
				
			||||||
 | 
						if procNtQueryObject == nil {
 | 
				
			||||||
 | 
							return "", errors.New("ntdll.dll: NtQueryObject not supported")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var buf [4 + syscall.MAX_PATH]uint16
 | 
				
			||||||
 | 
						var result int
 | 
				
			||||||
 | 
						r, _, e := syscall.Syscall6(procNtQueryObject.Addr(), 5,
 | 
				
			||||||
 | 
							fd, objectNameInfo, uintptr(unsafe.Pointer(&buf)), uintptr(2*len(buf)), uintptr(unsafe.Pointer(&result)), 0)
 | 
				
			||||||
 | 
						if r != 0 {
 | 
				
			||||||
 | 
							return "", e
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return string(utf16.Decode(buf[4 : 4+buf[0]/2])), nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// IsCygwinTerminal() return true if the file descriptor is a cygwin or msys2
 | 
					// IsCygwinTerminal() return true if the file descriptor is a cygwin or msys2
 | 
				
			||||||
// terminal.
 | 
					// terminal.
 | 
				
			||||||
func IsCygwinTerminal(fd uintptr) bool {
 | 
					func IsCygwinTerminal(fd uintptr) bool {
 | 
				
			||||||
	if procGetFileInformationByHandleEx == nil {
 | 
						if procGetFileInformationByHandleEx == nil {
 | 
				
			||||||
 | 
							name, err := getFileNameByHandle(fd)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
			return false
 | 
								return false
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
							return isCygwinPipeName(name)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Cygwin/msys's pty is a pipe.
 | 
						// Cygwin/msys's pty is a pipe.
 | 
				
			||||||
	ft, _, e := syscall.Syscall(procGetFileType.Addr(), 1, fd, 0, 0)
 | 
						ft, _, e := syscall.Syscall(procGetFileType.Addr(), 1, fd, 0, 0)
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										8
									
								
								vendor/github.com/mattn/go-runewidth/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								vendor/github.com/mattn/go-runewidth/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					language: go
 | 
				
			||||||
 | 
					go:
 | 
				
			||||||
 | 
					  - tip
 | 
				
			||||||
 | 
					before_install:
 | 
				
			||||||
 | 
					  - go get github.com/mattn/goveralls
 | 
				
			||||||
 | 
					  - go get golang.org/x/tools/cmd/cover
 | 
				
			||||||
 | 
					script:
 | 
				
			||||||
 | 
					    - $HOME/gopath/bin/goveralls -repotoken lAKAWPzcGsD3A8yBX3BGGtRUdJ6CaGERL
 | 
				
			||||||
							
								
								
									
										21
									
								
								vendor/github.com/mattn/go-runewidth/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/github.com/mattn/go-runewidth/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					The MIT License (MIT)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright (c) 2016 Yasuhiro Matsumoto
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
				
			||||||
 | 
					of this software and associated documentation files (the "Software"), to deal
 | 
				
			||||||
 | 
					in the Software without restriction, including without limitation the rights
 | 
				
			||||||
 | 
					to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
				
			||||||
 | 
					copies of the Software, and to permit persons to whom the Software is
 | 
				
			||||||
 | 
					furnished to do so, subject to the following conditions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The above copyright notice and this permission notice shall be included in all
 | 
				
			||||||
 | 
					copies or substantial portions of the Software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
				
			||||||
 | 
					IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
				
			||||||
 | 
					FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
				
			||||||
 | 
					AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
				
			||||||
 | 
					LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
				
			||||||
 | 
					OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | 
				
			||||||
 | 
					SOFTWARE.
 | 
				
			||||||
							
								
								
									
										27
									
								
								vendor/github.com/mattn/go-runewidth/README.mkd
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								vendor/github.com/mattn/go-runewidth/README.mkd
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
				
			|||||||
 | 
					go-runewidth
 | 
				
			||||||
 | 
					============
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					[](https://travis-ci.org/mattn/go-runewidth)
 | 
				
			||||||
 | 
					[](https://coveralls.io/r/mattn/go-runewidth?branch=HEAD)
 | 
				
			||||||
 | 
					[](http://godoc.org/github.com/mattn/go-runewidth)
 | 
				
			||||||
 | 
					[](https://goreportcard.com/report/github.com/mattn/go-runewidth)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Provides functions to get fixed width of the character or string.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Usage
 | 
				
			||||||
 | 
					-----
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					runewidth.StringWidth("つのだ☆HIRO") == 12
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Author
 | 
				
			||||||
 | 
					------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Yasuhiro Matsumoto
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					License
 | 
				
			||||||
 | 
					-------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					under the MIT License: http://mattn.mit-license.org/2013
 | 
				
			||||||
							
								
								
									
										3
									
								
								vendor/github.com/mattn/go-runewidth/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								vendor/github.com/mattn/go-runewidth/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
				
			|||||||
 | 
					module github.com/mattn/go-runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					go 1.9
 | 
				
			||||||
							
								
								
									
										258
									
								
								vendor/github.com/mattn/go-runewidth/runewidth.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										258
									
								
								vendor/github.com/mattn/go-runewidth/runewidth.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,258 @@
 | 
				
			|||||||
 | 
					package runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//go:generate go run script/generate.go
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						// EastAsianWidth will be set true if the current locale is CJK
 | 
				
			||||||
 | 
						EastAsianWidth bool
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// ZeroWidthJoiner is flag to set to use UTR#51 ZWJ
 | 
				
			||||||
 | 
						ZeroWidthJoiner bool
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// DefaultCondition is a condition in current locale
 | 
				
			||||||
 | 
						DefaultCondition = &Condition{}
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func init() {
 | 
				
			||||||
 | 
						handleEnv()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func handleEnv() {
 | 
				
			||||||
 | 
						env := os.Getenv("RUNEWIDTH_EASTASIAN")
 | 
				
			||||||
 | 
						if env == "" {
 | 
				
			||||||
 | 
							EastAsianWidth = IsEastAsian()
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							EastAsianWidth = env == "1"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// update DefaultCondition
 | 
				
			||||||
 | 
						DefaultCondition.EastAsianWidth = EastAsianWidth
 | 
				
			||||||
 | 
						DefaultCondition.ZeroWidthJoiner = ZeroWidthJoiner
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type interval struct {
 | 
				
			||||||
 | 
						first rune
 | 
				
			||||||
 | 
						last  rune
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type table []interval
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func inTables(r rune, ts ...table) bool {
 | 
				
			||||||
 | 
						for _, t := range ts {
 | 
				
			||||||
 | 
							if inTable(r, t) {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func inTable(r rune, t table) bool {
 | 
				
			||||||
 | 
						// func (t table) IncludesRune(r rune) bool {
 | 
				
			||||||
 | 
						if r < t[0].first {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						bot := 0
 | 
				
			||||||
 | 
						top := len(t) - 1
 | 
				
			||||||
 | 
						for top >= bot {
 | 
				
			||||||
 | 
							mid := (bot + top) >> 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							switch {
 | 
				
			||||||
 | 
							case t[mid].last < r:
 | 
				
			||||||
 | 
								bot = mid + 1
 | 
				
			||||||
 | 
							case t[mid].first > r:
 | 
				
			||||||
 | 
								top = mid - 1
 | 
				
			||||||
 | 
							default:
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var private = table{
 | 
				
			||||||
 | 
						{0x00E000, 0x00F8FF}, {0x0F0000, 0x0FFFFD}, {0x100000, 0x10FFFD},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var nonprint = table{
 | 
				
			||||||
 | 
						{0x0000, 0x001F}, {0x007F, 0x009F}, {0x00AD, 0x00AD},
 | 
				
			||||||
 | 
						{0x070F, 0x070F}, {0x180B, 0x180E}, {0x200B, 0x200F},
 | 
				
			||||||
 | 
						{0x2028, 0x202E}, {0x206A, 0x206F}, {0xD800, 0xDFFF},
 | 
				
			||||||
 | 
						{0xFEFF, 0xFEFF}, {0xFFF9, 0xFFFB}, {0xFFFE, 0xFFFF},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Condition have flag EastAsianWidth whether the current locale is CJK or not.
 | 
				
			||||||
 | 
					type Condition struct {
 | 
				
			||||||
 | 
						EastAsianWidth  bool
 | 
				
			||||||
 | 
						ZeroWidthJoiner bool
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewCondition return new instance of Condition which is current locale.
 | 
				
			||||||
 | 
					func NewCondition() *Condition {
 | 
				
			||||||
 | 
						return &Condition{
 | 
				
			||||||
 | 
							EastAsianWidth:  EastAsianWidth,
 | 
				
			||||||
 | 
							ZeroWidthJoiner: ZeroWidthJoiner,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// RuneWidth returns the number of cells in r.
 | 
				
			||||||
 | 
					// See http://www.unicode.org/reports/tr11/
 | 
				
			||||||
 | 
					func (c *Condition) RuneWidth(r rune) int {
 | 
				
			||||||
 | 
						switch {
 | 
				
			||||||
 | 
						case r < 0 || r > 0x10FFFF || inTables(r, nonprint, combining, notassigned):
 | 
				
			||||||
 | 
							return 0
 | 
				
			||||||
 | 
						case (c.EastAsianWidth && IsAmbiguousWidth(r)) || inTables(r, doublewidth):
 | 
				
			||||||
 | 
							return 2
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							return 1
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Condition) stringWidth(s string) (width int) {
 | 
				
			||||||
 | 
						for _, r := range []rune(s) {
 | 
				
			||||||
 | 
							width += c.RuneWidth(r)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return width
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (c *Condition) stringWidthZeroJoiner(s string) (width int) {
 | 
				
			||||||
 | 
						r1, r2 := rune(0), rune(0)
 | 
				
			||||||
 | 
						for _, r := range []rune(s) {
 | 
				
			||||||
 | 
							if r == 0xFE0E || r == 0xFE0F {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							w := c.RuneWidth(r)
 | 
				
			||||||
 | 
							if r2 == 0x200D && inTables(r, emoji) && inTables(r1, emoji) {
 | 
				
			||||||
 | 
								if width < w {
 | 
				
			||||||
 | 
									width = w
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								width += w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							r1, r2 = r2, r
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return width
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// StringWidth return width as you can see
 | 
				
			||||||
 | 
					func (c *Condition) StringWidth(s string) (width int) {
 | 
				
			||||||
 | 
						if c.ZeroWidthJoiner {
 | 
				
			||||||
 | 
							return c.stringWidthZeroJoiner(s)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return c.stringWidth(s)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Truncate return string truncated with w cells
 | 
				
			||||||
 | 
					func (c *Condition) Truncate(s string, w int, tail string) string {
 | 
				
			||||||
 | 
						if c.StringWidth(s) <= w {
 | 
				
			||||||
 | 
							return s
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						r := []rune(s)
 | 
				
			||||||
 | 
						tw := c.StringWidth(tail)
 | 
				
			||||||
 | 
						w -= tw
 | 
				
			||||||
 | 
						width := 0
 | 
				
			||||||
 | 
						i := 0
 | 
				
			||||||
 | 
						for ; i < len(r); i++ {
 | 
				
			||||||
 | 
							cw := c.RuneWidth(r[i])
 | 
				
			||||||
 | 
							if width+cw > w {
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							width += cw
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return string(r[0:i]) + tail
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Wrap return string wrapped with w cells
 | 
				
			||||||
 | 
					func (c *Condition) Wrap(s string, w int) string {
 | 
				
			||||||
 | 
						width := 0
 | 
				
			||||||
 | 
						out := ""
 | 
				
			||||||
 | 
						for _, r := range []rune(s) {
 | 
				
			||||||
 | 
							cw := RuneWidth(r)
 | 
				
			||||||
 | 
							if r == '\n' {
 | 
				
			||||||
 | 
								out += string(r)
 | 
				
			||||||
 | 
								width = 0
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							} else if width+cw > w {
 | 
				
			||||||
 | 
								out += "\n"
 | 
				
			||||||
 | 
								width = 0
 | 
				
			||||||
 | 
								out += string(r)
 | 
				
			||||||
 | 
								width += cw
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							out += string(r)
 | 
				
			||||||
 | 
							width += cw
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return out
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FillLeft return string filled in left by spaces in w cells
 | 
				
			||||||
 | 
					func (c *Condition) FillLeft(s string, w int) string {
 | 
				
			||||||
 | 
						width := c.StringWidth(s)
 | 
				
			||||||
 | 
						count := w - width
 | 
				
			||||||
 | 
						if count > 0 {
 | 
				
			||||||
 | 
							b := make([]byte, count)
 | 
				
			||||||
 | 
							for i := range b {
 | 
				
			||||||
 | 
								b[i] = ' '
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return string(b) + s
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return s
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FillRight return string filled in left by spaces in w cells
 | 
				
			||||||
 | 
					func (c *Condition) FillRight(s string, w int) string {
 | 
				
			||||||
 | 
						width := c.StringWidth(s)
 | 
				
			||||||
 | 
						count := w - width
 | 
				
			||||||
 | 
						if count > 0 {
 | 
				
			||||||
 | 
							b := make([]byte, count)
 | 
				
			||||||
 | 
							for i := range b {
 | 
				
			||||||
 | 
								b[i] = ' '
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return s + string(b)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return s
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// RuneWidth returns the number of cells in r.
 | 
				
			||||||
 | 
					// See http://www.unicode.org/reports/tr11/
 | 
				
			||||||
 | 
					func RuneWidth(r rune) int {
 | 
				
			||||||
 | 
						return DefaultCondition.RuneWidth(r)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsAmbiguousWidth returns whether is ambiguous width or not.
 | 
				
			||||||
 | 
					func IsAmbiguousWidth(r rune) bool {
 | 
				
			||||||
 | 
						return inTables(r, private, ambiguous)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsNeutralWidth returns whether is neutral width or not.
 | 
				
			||||||
 | 
					func IsNeutralWidth(r rune) bool {
 | 
				
			||||||
 | 
						return inTable(r, neutral)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// StringWidth return width as you can see
 | 
				
			||||||
 | 
					func StringWidth(s string) (width int) {
 | 
				
			||||||
 | 
						return DefaultCondition.StringWidth(s)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Truncate return string truncated with w cells
 | 
				
			||||||
 | 
					func Truncate(s string, w int, tail string) string {
 | 
				
			||||||
 | 
						return DefaultCondition.Truncate(s, w, tail)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Wrap return string wrapped with w cells
 | 
				
			||||||
 | 
					func Wrap(s string, w int) string {
 | 
				
			||||||
 | 
						return DefaultCondition.Wrap(s, w)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FillLeft return string filled in left by spaces in w cells
 | 
				
			||||||
 | 
					func FillLeft(s string, w int) string {
 | 
				
			||||||
 | 
						return DefaultCondition.FillLeft(s, w)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FillRight return string filled in left by spaces in w cells
 | 
				
			||||||
 | 
					func FillRight(s string, w int) string {
 | 
				
			||||||
 | 
						return DefaultCondition.FillRight(s, w)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										8
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_appengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_appengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
				
			|||||||
 | 
					// +build appengine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsEastAsian return true if the current locale is CJK
 | 
				
			||||||
 | 
					func IsEastAsian() bool {
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										9
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_js.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_js.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					// +build js
 | 
				
			||||||
 | 
					// +build !appengine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func IsEastAsian() bool {
 | 
				
			||||||
 | 
						// TODO: Implement this for the web. Detect east asian in a compatible way, and return true.
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										79
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_posix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_posix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,79 @@
 | 
				
			|||||||
 | 
					// +build !windows
 | 
				
			||||||
 | 
					// +build !js
 | 
				
			||||||
 | 
					// +build !appengine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
						"regexp"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var reLoc = regexp.MustCompile(`^[a-z][a-z][a-z]?(?:_[A-Z][A-Z])?\.(.+)`)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var mblenTable = map[string]int{
 | 
				
			||||||
 | 
						"utf-8":   6,
 | 
				
			||||||
 | 
						"utf8":    6,
 | 
				
			||||||
 | 
						"jis":     8,
 | 
				
			||||||
 | 
						"eucjp":   3,
 | 
				
			||||||
 | 
						"euckr":   2,
 | 
				
			||||||
 | 
						"euccn":   2,
 | 
				
			||||||
 | 
						"sjis":    2,
 | 
				
			||||||
 | 
						"cp932":   2,
 | 
				
			||||||
 | 
						"cp51932": 2,
 | 
				
			||||||
 | 
						"cp936":   2,
 | 
				
			||||||
 | 
						"cp949":   2,
 | 
				
			||||||
 | 
						"cp950":   2,
 | 
				
			||||||
 | 
						"big5":    2,
 | 
				
			||||||
 | 
						"gbk":     2,
 | 
				
			||||||
 | 
						"gb2312":  2,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func isEastAsian(locale string) bool {
 | 
				
			||||||
 | 
						charset := strings.ToLower(locale)
 | 
				
			||||||
 | 
						r := reLoc.FindStringSubmatch(locale)
 | 
				
			||||||
 | 
						if len(r) == 2 {
 | 
				
			||||||
 | 
							charset = strings.ToLower(r[1])
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if strings.HasSuffix(charset, "@cjk_narrow") {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for pos, b := range []byte(charset) {
 | 
				
			||||||
 | 
							if b == '@' {
 | 
				
			||||||
 | 
								charset = charset[:pos]
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						max := 1
 | 
				
			||||||
 | 
						if m, ok := mblenTable[charset]; ok {
 | 
				
			||||||
 | 
							max = m
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if max > 1 && (charset[0] != 'u' ||
 | 
				
			||||||
 | 
							strings.HasPrefix(locale, "ja") ||
 | 
				
			||||||
 | 
							strings.HasPrefix(locale, "ko") ||
 | 
				
			||||||
 | 
							strings.HasPrefix(locale, "zh")) {
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsEastAsian return true if the current locale is CJK
 | 
				
			||||||
 | 
					func IsEastAsian() bool {
 | 
				
			||||||
 | 
						locale := os.Getenv("LC_CTYPE")
 | 
				
			||||||
 | 
						if locale == "" {
 | 
				
			||||||
 | 
							locale = os.Getenv("LANG")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// ignore C locale
 | 
				
			||||||
 | 
						if locale == "POSIX" || locale == "C" {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if len(locale) > 1 && locale[0] == 'C' && (locale[1] == '.' || locale[1] == '-') {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return isEastAsian(locale)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										427
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_table.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										427
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_table.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,427 @@
 | 
				
			|||||||
 | 
					package runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var combining = table{
 | 
				
			||||||
 | 
						{0x0300, 0x036F}, {0x0483, 0x0489}, {0x07EB, 0x07F3},
 | 
				
			||||||
 | 
						{0x0C00, 0x0C00}, {0x0C04, 0x0C04}, {0x0D00, 0x0D01},
 | 
				
			||||||
 | 
						{0x135D, 0x135F}, {0x1A7F, 0x1A7F}, {0x1AB0, 0x1ABE},
 | 
				
			||||||
 | 
						{0x1B6B, 0x1B73}, {0x1DC0, 0x1DF9}, {0x1DFB, 0x1DFF},
 | 
				
			||||||
 | 
						{0x20D0, 0x20F0}, {0x2CEF, 0x2CF1}, {0x2DE0, 0x2DFF},
 | 
				
			||||||
 | 
						{0x3099, 0x309A}, {0xA66F, 0xA672}, {0xA674, 0xA67D},
 | 
				
			||||||
 | 
						{0xA69E, 0xA69F}, {0xA6F0, 0xA6F1}, {0xA8E0, 0xA8F1},
 | 
				
			||||||
 | 
						{0xFE20, 0xFE2F}, {0x101FD, 0x101FD}, {0x10376, 0x1037A},
 | 
				
			||||||
 | 
						{0x10F46, 0x10F50}, {0x11300, 0x11301}, {0x1133B, 0x1133C},
 | 
				
			||||||
 | 
						{0x11366, 0x1136C}, {0x11370, 0x11374}, {0x16AF0, 0x16AF4},
 | 
				
			||||||
 | 
						{0x1D165, 0x1D169}, {0x1D16D, 0x1D172}, {0x1D17B, 0x1D182},
 | 
				
			||||||
 | 
						{0x1D185, 0x1D18B}, {0x1D1AA, 0x1D1AD}, {0x1D242, 0x1D244},
 | 
				
			||||||
 | 
						{0x1E000, 0x1E006}, {0x1E008, 0x1E018}, {0x1E01B, 0x1E021},
 | 
				
			||||||
 | 
						{0x1E023, 0x1E024}, {0x1E026, 0x1E02A}, {0x1E8D0, 0x1E8D6},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var doublewidth = table{
 | 
				
			||||||
 | 
						{0x1100, 0x115F}, {0x231A, 0x231B}, {0x2329, 0x232A},
 | 
				
			||||||
 | 
						{0x23E9, 0x23EC}, {0x23F0, 0x23F0}, {0x23F3, 0x23F3},
 | 
				
			||||||
 | 
						{0x25FD, 0x25FE}, {0x2614, 0x2615}, {0x2648, 0x2653},
 | 
				
			||||||
 | 
						{0x267F, 0x267F}, {0x2693, 0x2693}, {0x26A1, 0x26A1},
 | 
				
			||||||
 | 
						{0x26AA, 0x26AB}, {0x26BD, 0x26BE}, {0x26C4, 0x26C5},
 | 
				
			||||||
 | 
						{0x26CE, 0x26CE}, {0x26D4, 0x26D4}, {0x26EA, 0x26EA},
 | 
				
			||||||
 | 
						{0x26F2, 0x26F3}, {0x26F5, 0x26F5}, {0x26FA, 0x26FA},
 | 
				
			||||||
 | 
						{0x26FD, 0x26FD}, {0x2705, 0x2705}, {0x270A, 0x270B},
 | 
				
			||||||
 | 
						{0x2728, 0x2728}, {0x274C, 0x274C}, {0x274E, 0x274E},
 | 
				
			||||||
 | 
						{0x2753, 0x2755}, {0x2757, 0x2757}, {0x2795, 0x2797},
 | 
				
			||||||
 | 
						{0x27B0, 0x27B0}, {0x27BF, 0x27BF}, {0x2B1B, 0x2B1C},
 | 
				
			||||||
 | 
						{0x2B50, 0x2B50}, {0x2B55, 0x2B55}, {0x2E80, 0x2E99},
 | 
				
			||||||
 | 
						{0x2E9B, 0x2EF3}, {0x2F00, 0x2FD5}, {0x2FF0, 0x2FFB},
 | 
				
			||||||
 | 
						{0x3000, 0x303E}, {0x3041, 0x3096}, {0x3099, 0x30FF},
 | 
				
			||||||
 | 
						{0x3105, 0x312F}, {0x3131, 0x318E}, {0x3190, 0x31BA},
 | 
				
			||||||
 | 
						{0x31C0, 0x31E3}, {0x31F0, 0x321E}, {0x3220, 0x3247},
 | 
				
			||||||
 | 
						{0x3250, 0x4DBF}, {0x4E00, 0xA48C}, {0xA490, 0xA4C6},
 | 
				
			||||||
 | 
						{0xA960, 0xA97C}, {0xAC00, 0xD7A3}, {0xF900, 0xFAFF},
 | 
				
			||||||
 | 
						{0xFE10, 0xFE19}, {0xFE30, 0xFE52}, {0xFE54, 0xFE66},
 | 
				
			||||||
 | 
						{0xFE68, 0xFE6B}, {0xFF01, 0xFF60}, {0xFFE0, 0xFFE6},
 | 
				
			||||||
 | 
						{0x16FE0, 0x16FE3}, {0x17000, 0x187F7}, {0x18800, 0x18AF2},
 | 
				
			||||||
 | 
						{0x1B000, 0x1B11E}, {0x1B150, 0x1B152}, {0x1B164, 0x1B167},
 | 
				
			||||||
 | 
						{0x1B170, 0x1B2FB}, {0x1F004, 0x1F004}, {0x1F0CF, 0x1F0CF},
 | 
				
			||||||
 | 
						{0x1F18E, 0x1F18E}, {0x1F191, 0x1F19A}, {0x1F200, 0x1F202},
 | 
				
			||||||
 | 
						{0x1F210, 0x1F23B}, {0x1F240, 0x1F248}, {0x1F250, 0x1F251},
 | 
				
			||||||
 | 
						{0x1F260, 0x1F265}, {0x1F300, 0x1F320}, {0x1F32D, 0x1F335},
 | 
				
			||||||
 | 
						{0x1F337, 0x1F37C}, {0x1F37E, 0x1F393}, {0x1F3A0, 0x1F3CA},
 | 
				
			||||||
 | 
						{0x1F3CF, 0x1F3D3}, {0x1F3E0, 0x1F3F0}, {0x1F3F4, 0x1F3F4},
 | 
				
			||||||
 | 
						{0x1F3F8, 0x1F43E}, {0x1F440, 0x1F440}, {0x1F442, 0x1F4FC},
 | 
				
			||||||
 | 
						{0x1F4FF, 0x1F53D}, {0x1F54B, 0x1F54E}, {0x1F550, 0x1F567},
 | 
				
			||||||
 | 
						{0x1F57A, 0x1F57A}, {0x1F595, 0x1F596}, {0x1F5A4, 0x1F5A4},
 | 
				
			||||||
 | 
						{0x1F5FB, 0x1F64F}, {0x1F680, 0x1F6C5}, {0x1F6CC, 0x1F6CC},
 | 
				
			||||||
 | 
						{0x1F6D0, 0x1F6D2}, {0x1F6D5, 0x1F6D5}, {0x1F6EB, 0x1F6EC},
 | 
				
			||||||
 | 
						{0x1F6F4, 0x1F6FA}, {0x1F7E0, 0x1F7EB}, {0x1F90D, 0x1F971},
 | 
				
			||||||
 | 
						{0x1F973, 0x1F976}, {0x1F97A, 0x1F9A2}, {0x1F9A5, 0x1F9AA},
 | 
				
			||||||
 | 
						{0x1F9AE, 0x1F9CA}, {0x1F9CD, 0x1F9FF}, {0x1FA70, 0x1FA73},
 | 
				
			||||||
 | 
						{0x1FA78, 0x1FA7A}, {0x1FA80, 0x1FA82}, {0x1FA90, 0x1FA95},
 | 
				
			||||||
 | 
						{0x20000, 0x2FFFD}, {0x30000, 0x3FFFD},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var ambiguous = table{
 | 
				
			||||||
 | 
						{0x00A1, 0x00A1}, {0x00A4, 0x00A4}, {0x00A7, 0x00A8},
 | 
				
			||||||
 | 
						{0x00AA, 0x00AA}, {0x00AD, 0x00AE}, {0x00B0, 0x00B4},
 | 
				
			||||||
 | 
						{0x00B6, 0x00BA}, {0x00BC, 0x00BF}, {0x00C6, 0x00C6},
 | 
				
			||||||
 | 
						{0x00D0, 0x00D0}, {0x00D7, 0x00D8}, {0x00DE, 0x00E1},
 | 
				
			||||||
 | 
						{0x00E6, 0x00E6}, {0x00E8, 0x00EA}, {0x00EC, 0x00ED},
 | 
				
			||||||
 | 
						{0x00F0, 0x00F0}, {0x00F2, 0x00F3}, {0x00F7, 0x00FA},
 | 
				
			||||||
 | 
						{0x00FC, 0x00FC}, {0x00FE, 0x00FE}, {0x0101, 0x0101},
 | 
				
			||||||
 | 
						{0x0111, 0x0111}, {0x0113, 0x0113}, {0x011B, 0x011B},
 | 
				
			||||||
 | 
						{0x0126, 0x0127}, {0x012B, 0x012B}, {0x0131, 0x0133},
 | 
				
			||||||
 | 
						{0x0138, 0x0138}, {0x013F, 0x0142}, {0x0144, 0x0144},
 | 
				
			||||||
 | 
						{0x0148, 0x014B}, {0x014D, 0x014D}, {0x0152, 0x0153},
 | 
				
			||||||
 | 
						{0x0166, 0x0167}, {0x016B, 0x016B}, {0x01CE, 0x01CE},
 | 
				
			||||||
 | 
						{0x01D0, 0x01D0}, {0x01D2, 0x01D2}, {0x01D4, 0x01D4},
 | 
				
			||||||
 | 
						{0x01D6, 0x01D6}, {0x01D8, 0x01D8}, {0x01DA, 0x01DA},
 | 
				
			||||||
 | 
						{0x01DC, 0x01DC}, {0x0251, 0x0251}, {0x0261, 0x0261},
 | 
				
			||||||
 | 
						{0x02C4, 0x02C4}, {0x02C7, 0x02C7}, {0x02C9, 0x02CB},
 | 
				
			||||||
 | 
						{0x02CD, 0x02CD}, {0x02D0, 0x02D0}, {0x02D8, 0x02DB},
 | 
				
			||||||
 | 
						{0x02DD, 0x02DD}, {0x02DF, 0x02DF}, {0x0300, 0x036F},
 | 
				
			||||||
 | 
						{0x0391, 0x03A1}, {0x03A3, 0x03A9}, {0x03B1, 0x03C1},
 | 
				
			||||||
 | 
						{0x03C3, 0x03C9}, {0x0401, 0x0401}, {0x0410, 0x044F},
 | 
				
			||||||
 | 
						{0x0451, 0x0451}, {0x2010, 0x2010}, {0x2013, 0x2016},
 | 
				
			||||||
 | 
						{0x2018, 0x2019}, {0x201C, 0x201D}, {0x2020, 0x2022},
 | 
				
			||||||
 | 
						{0x2024, 0x2027}, {0x2030, 0x2030}, {0x2032, 0x2033},
 | 
				
			||||||
 | 
						{0x2035, 0x2035}, {0x203B, 0x203B}, {0x203E, 0x203E},
 | 
				
			||||||
 | 
						{0x2074, 0x2074}, {0x207F, 0x207F}, {0x2081, 0x2084},
 | 
				
			||||||
 | 
						{0x20AC, 0x20AC}, {0x2103, 0x2103}, {0x2105, 0x2105},
 | 
				
			||||||
 | 
						{0x2109, 0x2109}, {0x2113, 0x2113}, {0x2116, 0x2116},
 | 
				
			||||||
 | 
						{0x2121, 0x2122}, {0x2126, 0x2126}, {0x212B, 0x212B},
 | 
				
			||||||
 | 
						{0x2153, 0x2154}, {0x215B, 0x215E}, {0x2160, 0x216B},
 | 
				
			||||||
 | 
						{0x2170, 0x2179}, {0x2189, 0x2189}, {0x2190, 0x2199},
 | 
				
			||||||
 | 
						{0x21B8, 0x21B9}, {0x21D2, 0x21D2}, {0x21D4, 0x21D4},
 | 
				
			||||||
 | 
						{0x21E7, 0x21E7}, {0x2200, 0x2200}, {0x2202, 0x2203},
 | 
				
			||||||
 | 
						{0x2207, 0x2208}, {0x220B, 0x220B}, {0x220F, 0x220F},
 | 
				
			||||||
 | 
						{0x2211, 0x2211}, {0x2215, 0x2215}, {0x221A, 0x221A},
 | 
				
			||||||
 | 
						{0x221D, 0x2220}, {0x2223, 0x2223}, {0x2225, 0x2225},
 | 
				
			||||||
 | 
						{0x2227, 0x222C}, {0x222E, 0x222E}, {0x2234, 0x2237},
 | 
				
			||||||
 | 
						{0x223C, 0x223D}, {0x2248, 0x2248}, {0x224C, 0x224C},
 | 
				
			||||||
 | 
						{0x2252, 0x2252}, {0x2260, 0x2261}, {0x2264, 0x2267},
 | 
				
			||||||
 | 
						{0x226A, 0x226B}, {0x226E, 0x226F}, {0x2282, 0x2283},
 | 
				
			||||||
 | 
						{0x2286, 0x2287}, {0x2295, 0x2295}, {0x2299, 0x2299},
 | 
				
			||||||
 | 
						{0x22A5, 0x22A5}, {0x22BF, 0x22BF}, {0x2312, 0x2312},
 | 
				
			||||||
 | 
						{0x2460, 0x24E9}, {0x24EB, 0x254B}, {0x2550, 0x2573},
 | 
				
			||||||
 | 
						{0x2580, 0x258F}, {0x2592, 0x2595}, {0x25A0, 0x25A1},
 | 
				
			||||||
 | 
						{0x25A3, 0x25A9}, {0x25B2, 0x25B3}, {0x25B6, 0x25B7},
 | 
				
			||||||
 | 
						{0x25BC, 0x25BD}, {0x25C0, 0x25C1}, {0x25C6, 0x25C8},
 | 
				
			||||||
 | 
						{0x25CB, 0x25CB}, {0x25CE, 0x25D1}, {0x25E2, 0x25E5},
 | 
				
			||||||
 | 
						{0x25EF, 0x25EF}, {0x2605, 0x2606}, {0x2609, 0x2609},
 | 
				
			||||||
 | 
						{0x260E, 0x260F}, {0x261C, 0x261C}, {0x261E, 0x261E},
 | 
				
			||||||
 | 
						{0x2640, 0x2640}, {0x2642, 0x2642}, {0x2660, 0x2661},
 | 
				
			||||||
 | 
						{0x2663, 0x2665}, {0x2667, 0x266A}, {0x266C, 0x266D},
 | 
				
			||||||
 | 
						{0x266F, 0x266F}, {0x269E, 0x269F}, {0x26BF, 0x26BF},
 | 
				
			||||||
 | 
						{0x26C6, 0x26CD}, {0x26CF, 0x26D3}, {0x26D5, 0x26E1},
 | 
				
			||||||
 | 
						{0x26E3, 0x26E3}, {0x26E8, 0x26E9}, {0x26EB, 0x26F1},
 | 
				
			||||||
 | 
						{0x26F4, 0x26F4}, {0x26F6, 0x26F9}, {0x26FB, 0x26FC},
 | 
				
			||||||
 | 
						{0x26FE, 0x26FF}, {0x273D, 0x273D}, {0x2776, 0x277F},
 | 
				
			||||||
 | 
						{0x2B56, 0x2B59}, {0x3248, 0x324F}, {0xE000, 0xF8FF},
 | 
				
			||||||
 | 
						{0xFE00, 0xFE0F}, {0xFFFD, 0xFFFD}, {0x1F100, 0x1F10A},
 | 
				
			||||||
 | 
						{0x1F110, 0x1F12D}, {0x1F130, 0x1F169}, {0x1F170, 0x1F18D},
 | 
				
			||||||
 | 
						{0x1F18F, 0x1F190}, {0x1F19B, 0x1F1AC}, {0xE0100, 0xE01EF},
 | 
				
			||||||
 | 
						{0xF0000, 0xFFFFD}, {0x100000, 0x10FFFD},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					var notassigned = table{
 | 
				
			||||||
 | 
						{0x27E6, 0x27ED}, {0x2985, 0x2986},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var neutral = table{
 | 
				
			||||||
 | 
						{0x0000, 0x001F}, {0x007F, 0x00A0}, {0x00A9, 0x00A9},
 | 
				
			||||||
 | 
						{0x00AB, 0x00AB}, {0x00B5, 0x00B5}, {0x00BB, 0x00BB},
 | 
				
			||||||
 | 
						{0x00C0, 0x00C5}, {0x00C7, 0x00CF}, {0x00D1, 0x00D6},
 | 
				
			||||||
 | 
						{0x00D9, 0x00DD}, {0x00E2, 0x00E5}, {0x00E7, 0x00E7},
 | 
				
			||||||
 | 
						{0x00EB, 0x00EB}, {0x00EE, 0x00EF}, {0x00F1, 0x00F1},
 | 
				
			||||||
 | 
						{0x00F4, 0x00F6}, {0x00FB, 0x00FB}, {0x00FD, 0x00FD},
 | 
				
			||||||
 | 
						{0x00FF, 0x0100}, {0x0102, 0x0110}, {0x0112, 0x0112},
 | 
				
			||||||
 | 
						{0x0114, 0x011A}, {0x011C, 0x0125}, {0x0128, 0x012A},
 | 
				
			||||||
 | 
						{0x012C, 0x0130}, {0x0134, 0x0137}, {0x0139, 0x013E},
 | 
				
			||||||
 | 
						{0x0143, 0x0143}, {0x0145, 0x0147}, {0x014C, 0x014C},
 | 
				
			||||||
 | 
						{0x014E, 0x0151}, {0x0154, 0x0165}, {0x0168, 0x016A},
 | 
				
			||||||
 | 
						{0x016C, 0x01CD}, {0x01CF, 0x01CF}, {0x01D1, 0x01D1},
 | 
				
			||||||
 | 
						{0x01D3, 0x01D3}, {0x01D5, 0x01D5}, {0x01D7, 0x01D7},
 | 
				
			||||||
 | 
						{0x01D9, 0x01D9}, {0x01DB, 0x01DB}, {0x01DD, 0x0250},
 | 
				
			||||||
 | 
						{0x0252, 0x0260}, {0x0262, 0x02C3}, {0x02C5, 0x02C6},
 | 
				
			||||||
 | 
						{0x02C8, 0x02C8}, {0x02CC, 0x02CC}, {0x02CE, 0x02CF},
 | 
				
			||||||
 | 
						{0x02D1, 0x02D7}, {0x02DC, 0x02DC}, {0x02DE, 0x02DE},
 | 
				
			||||||
 | 
						{0x02E0, 0x02FF}, {0x0370, 0x0377}, {0x037A, 0x037F},
 | 
				
			||||||
 | 
						{0x0384, 0x038A}, {0x038C, 0x038C}, {0x038E, 0x0390},
 | 
				
			||||||
 | 
						{0x03AA, 0x03B0}, {0x03C2, 0x03C2}, {0x03CA, 0x0400},
 | 
				
			||||||
 | 
						{0x0402, 0x040F}, {0x0450, 0x0450}, {0x0452, 0x052F},
 | 
				
			||||||
 | 
						{0x0531, 0x0556}, {0x0559, 0x058A}, {0x058D, 0x058F},
 | 
				
			||||||
 | 
						{0x0591, 0x05C7}, {0x05D0, 0x05EA}, {0x05EF, 0x05F4},
 | 
				
			||||||
 | 
						{0x0600, 0x061C}, {0x061E, 0x070D}, {0x070F, 0x074A},
 | 
				
			||||||
 | 
						{0x074D, 0x07B1}, {0x07C0, 0x07FA}, {0x07FD, 0x082D},
 | 
				
			||||||
 | 
						{0x0830, 0x083E}, {0x0840, 0x085B}, {0x085E, 0x085E},
 | 
				
			||||||
 | 
						{0x0860, 0x086A}, {0x08A0, 0x08B4}, {0x08B6, 0x08BD},
 | 
				
			||||||
 | 
						{0x08D3, 0x0983}, {0x0985, 0x098C}, {0x098F, 0x0990},
 | 
				
			||||||
 | 
						{0x0993, 0x09A8}, {0x09AA, 0x09B0}, {0x09B2, 0x09B2},
 | 
				
			||||||
 | 
						{0x09B6, 0x09B9}, {0x09BC, 0x09C4}, {0x09C7, 0x09C8},
 | 
				
			||||||
 | 
						{0x09CB, 0x09CE}, {0x09D7, 0x09D7}, {0x09DC, 0x09DD},
 | 
				
			||||||
 | 
						{0x09DF, 0x09E3}, {0x09E6, 0x09FE}, {0x0A01, 0x0A03},
 | 
				
			||||||
 | 
						{0x0A05, 0x0A0A}, {0x0A0F, 0x0A10}, {0x0A13, 0x0A28},
 | 
				
			||||||
 | 
						{0x0A2A, 0x0A30}, {0x0A32, 0x0A33}, {0x0A35, 0x0A36},
 | 
				
			||||||
 | 
						{0x0A38, 0x0A39}, {0x0A3C, 0x0A3C}, {0x0A3E, 0x0A42},
 | 
				
			||||||
 | 
						{0x0A47, 0x0A48}, {0x0A4B, 0x0A4D}, {0x0A51, 0x0A51},
 | 
				
			||||||
 | 
						{0x0A59, 0x0A5C}, {0x0A5E, 0x0A5E}, {0x0A66, 0x0A76},
 | 
				
			||||||
 | 
						{0x0A81, 0x0A83}, {0x0A85, 0x0A8D}, {0x0A8F, 0x0A91},
 | 
				
			||||||
 | 
						{0x0A93, 0x0AA8}, {0x0AAA, 0x0AB0}, {0x0AB2, 0x0AB3},
 | 
				
			||||||
 | 
						{0x0AB5, 0x0AB9}, {0x0ABC, 0x0AC5}, {0x0AC7, 0x0AC9},
 | 
				
			||||||
 | 
						{0x0ACB, 0x0ACD}, {0x0AD0, 0x0AD0}, {0x0AE0, 0x0AE3},
 | 
				
			||||||
 | 
						{0x0AE6, 0x0AF1}, {0x0AF9, 0x0AFF}, {0x0B01, 0x0B03},
 | 
				
			||||||
 | 
						{0x0B05, 0x0B0C}, {0x0B0F, 0x0B10}, {0x0B13, 0x0B28},
 | 
				
			||||||
 | 
						{0x0B2A, 0x0B30}, {0x0B32, 0x0B33}, {0x0B35, 0x0B39},
 | 
				
			||||||
 | 
						{0x0B3C, 0x0B44}, {0x0B47, 0x0B48}, {0x0B4B, 0x0B4D},
 | 
				
			||||||
 | 
						{0x0B56, 0x0B57}, {0x0B5C, 0x0B5D}, {0x0B5F, 0x0B63},
 | 
				
			||||||
 | 
						{0x0B66, 0x0B77}, {0x0B82, 0x0B83}, {0x0B85, 0x0B8A},
 | 
				
			||||||
 | 
						{0x0B8E, 0x0B90}, {0x0B92, 0x0B95}, {0x0B99, 0x0B9A},
 | 
				
			||||||
 | 
						{0x0B9C, 0x0B9C}, {0x0B9E, 0x0B9F}, {0x0BA3, 0x0BA4},
 | 
				
			||||||
 | 
						{0x0BA8, 0x0BAA}, {0x0BAE, 0x0BB9}, {0x0BBE, 0x0BC2},
 | 
				
			||||||
 | 
						{0x0BC6, 0x0BC8}, {0x0BCA, 0x0BCD}, {0x0BD0, 0x0BD0},
 | 
				
			||||||
 | 
						{0x0BD7, 0x0BD7}, {0x0BE6, 0x0BFA}, {0x0C00, 0x0C0C},
 | 
				
			||||||
 | 
						{0x0C0E, 0x0C10}, {0x0C12, 0x0C28}, {0x0C2A, 0x0C39},
 | 
				
			||||||
 | 
						{0x0C3D, 0x0C44}, {0x0C46, 0x0C48}, {0x0C4A, 0x0C4D},
 | 
				
			||||||
 | 
						{0x0C55, 0x0C56}, {0x0C58, 0x0C5A}, {0x0C60, 0x0C63},
 | 
				
			||||||
 | 
						{0x0C66, 0x0C6F}, {0x0C77, 0x0C8C}, {0x0C8E, 0x0C90},
 | 
				
			||||||
 | 
						{0x0C92, 0x0CA8}, {0x0CAA, 0x0CB3}, {0x0CB5, 0x0CB9},
 | 
				
			||||||
 | 
						{0x0CBC, 0x0CC4}, {0x0CC6, 0x0CC8}, {0x0CCA, 0x0CCD},
 | 
				
			||||||
 | 
						{0x0CD5, 0x0CD6}, {0x0CDE, 0x0CDE}, {0x0CE0, 0x0CE3},
 | 
				
			||||||
 | 
						{0x0CE6, 0x0CEF}, {0x0CF1, 0x0CF2}, {0x0D00, 0x0D03},
 | 
				
			||||||
 | 
						{0x0D05, 0x0D0C}, {0x0D0E, 0x0D10}, {0x0D12, 0x0D44},
 | 
				
			||||||
 | 
						{0x0D46, 0x0D48}, {0x0D4A, 0x0D4F}, {0x0D54, 0x0D63},
 | 
				
			||||||
 | 
						{0x0D66, 0x0D7F}, {0x0D82, 0x0D83}, {0x0D85, 0x0D96},
 | 
				
			||||||
 | 
						{0x0D9A, 0x0DB1}, {0x0DB3, 0x0DBB}, {0x0DBD, 0x0DBD},
 | 
				
			||||||
 | 
						{0x0DC0, 0x0DC6}, {0x0DCA, 0x0DCA}, {0x0DCF, 0x0DD4},
 | 
				
			||||||
 | 
						{0x0DD6, 0x0DD6}, {0x0DD8, 0x0DDF}, {0x0DE6, 0x0DEF},
 | 
				
			||||||
 | 
						{0x0DF2, 0x0DF4}, {0x0E01, 0x0E3A}, {0x0E3F, 0x0E5B},
 | 
				
			||||||
 | 
						{0x0E81, 0x0E82}, {0x0E84, 0x0E84}, {0x0E86, 0x0E8A},
 | 
				
			||||||
 | 
						{0x0E8C, 0x0EA3}, {0x0EA5, 0x0EA5}, {0x0EA7, 0x0EBD},
 | 
				
			||||||
 | 
						{0x0EC0, 0x0EC4}, {0x0EC6, 0x0EC6}, {0x0EC8, 0x0ECD},
 | 
				
			||||||
 | 
						{0x0ED0, 0x0ED9}, {0x0EDC, 0x0EDF}, {0x0F00, 0x0F47},
 | 
				
			||||||
 | 
						{0x0F49, 0x0F6C}, {0x0F71, 0x0F97}, {0x0F99, 0x0FBC},
 | 
				
			||||||
 | 
						{0x0FBE, 0x0FCC}, {0x0FCE, 0x0FDA}, {0x1000, 0x10C5},
 | 
				
			||||||
 | 
						{0x10C7, 0x10C7}, {0x10CD, 0x10CD}, {0x10D0, 0x10FF},
 | 
				
			||||||
 | 
						{0x1160, 0x1248}, {0x124A, 0x124D}, {0x1250, 0x1256},
 | 
				
			||||||
 | 
						{0x1258, 0x1258}, {0x125A, 0x125D}, {0x1260, 0x1288},
 | 
				
			||||||
 | 
						{0x128A, 0x128D}, {0x1290, 0x12B0}, {0x12B2, 0x12B5},
 | 
				
			||||||
 | 
						{0x12B8, 0x12BE}, {0x12C0, 0x12C0}, {0x12C2, 0x12C5},
 | 
				
			||||||
 | 
						{0x12C8, 0x12D6}, {0x12D8, 0x1310}, {0x1312, 0x1315},
 | 
				
			||||||
 | 
						{0x1318, 0x135A}, {0x135D, 0x137C}, {0x1380, 0x1399},
 | 
				
			||||||
 | 
						{0x13A0, 0x13F5}, {0x13F8, 0x13FD}, {0x1400, 0x169C},
 | 
				
			||||||
 | 
						{0x16A0, 0x16F8}, {0x1700, 0x170C}, {0x170E, 0x1714},
 | 
				
			||||||
 | 
						{0x1720, 0x1736}, {0x1740, 0x1753}, {0x1760, 0x176C},
 | 
				
			||||||
 | 
						{0x176E, 0x1770}, {0x1772, 0x1773}, {0x1780, 0x17DD},
 | 
				
			||||||
 | 
						{0x17E0, 0x17E9}, {0x17F0, 0x17F9}, {0x1800, 0x180E},
 | 
				
			||||||
 | 
						{0x1810, 0x1819}, {0x1820, 0x1878}, {0x1880, 0x18AA},
 | 
				
			||||||
 | 
						{0x18B0, 0x18F5}, {0x1900, 0x191E}, {0x1920, 0x192B},
 | 
				
			||||||
 | 
						{0x1930, 0x193B}, {0x1940, 0x1940}, {0x1944, 0x196D},
 | 
				
			||||||
 | 
						{0x1970, 0x1974}, {0x1980, 0x19AB}, {0x19B0, 0x19C9},
 | 
				
			||||||
 | 
						{0x19D0, 0x19DA}, {0x19DE, 0x1A1B}, {0x1A1E, 0x1A5E},
 | 
				
			||||||
 | 
						{0x1A60, 0x1A7C}, {0x1A7F, 0x1A89}, {0x1A90, 0x1A99},
 | 
				
			||||||
 | 
						{0x1AA0, 0x1AAD}, {0x1AB0, 0x1ABE}, {0x1B00, 0x1B4B},
 | 
				
			||||||
 | 
						{0x1B50, 0x1B7C}, {0x1B80, 0x1BF3}, {0x1BFC, 0x1C37},
 | 
				
			||||||
 | 
						{0x1C3B, 0x1C49}, {0x1C4D, 0x1C88}, {0x1C90, 0x1CBA},
 | 
				
			||||||
 | 
						{0x1CBD, 0x1CC7}, {0x1CD0, 0x1CFA}, {0x1D00, 0x1DF9},
 | 
				
			||||||
 | 
						{0x1DFB, 0x1F15}, {0x1F18, 0x1F1D}, {0x1F20, 0x1F45},
 | 
				
			||||||
 | 
						{0x1F48, 0x1F4D}, {0x1F50, 0x1F57}, {0x1F59, 0x1F59},
 | 
				
			||||||
 | 
						{0x1F5B, 0x1F5B}, {0x1F5D, 0x1F5D}, {0x1F5F, 0x1F7D},
 | 
				
			||||||
 | 
						{0x1F80, 0x1FB4}, {0x1FB6, 0x1FC4}, {0x1FC6, 0x1FD3},
 | 
				
			||||||
 | 
						{0x1FD6, 0x1FDB}, {0x1FDD, 0x1FEF}, {0x1FF2, 0x1FF4},
 | 
				
			||||||
 | 
						{0x1FF6, 0x1FFE}, {0x2000, 0x200F}, {0x2011, 0x2012},
 | 
				
			||||||
 | 
						{0x2017, 0x2017}, {0x201A, 0x201B}, {0x201E, 0x201F},
 | 
				
			||||||
 | 
						{0x2023, 0x2023}, {0x2028, 0x202F}, {0x2031, 0x2031},
 | 
				
			||||||
 | 
						{0x2034, 0x2034}, {0x2036, 0x203A}, {0x203C, 0x203D},
 | 
				
			||||||
 | 
						{0x203F, 0x2064}, {0x2066, 0x2071}, {0x2075, 0x207E},
 | 
				
			||||||
 | 
						{0x2080, 0x2080}, {0x2085, 0x208E}, {0x2090, 0x209C},
 | 
				
			||||||
 | 
						{0x20A0, 0x20A8}, {0x20AA, 0x20AB}, {0x20AD, 0x20BF},
 | 
				
			||||||
 | 
						{0x20D0, 0x20F0}, {0x2100, 0x2102}, {0x2104, 0x2104},
 | 
				
			||||||
 | 
						{0x2106, 0x2108}, {0x210A, 0x2112}, {0x2114, 0x2115},
 | 
				
			||||||
 | 
						{0x2117, 0x2120}, {0x2123, 0x2125}, {0x2127, 0x212A},
 | 
				
			||||||
 | 
						{0x212C, 0x2152}, {0x2155, 0x215A}, {0x215F, 0x215F},
 | 
				
			||||||
 | 
						{0x216C, 0x216F}, {0x217A, 0x2188}, {0x218A, 0x218B},
 | 
				
			||||||
 | 
						{0x219A, 0x21B7}, {0x21BA, 0x21D1}, {0x21D3, 0x21D3},
 | 
				
			||||||
 | 
						{0x21D5, 0x21E6}, {0x21E8, 0x21FF}, {0x2201, 0x2201},
 | 
				
			||||||
 | 
						{0x2204, 0x2206}, {0x2209, 0x220A}, {0x220C, 0x220E},
 | 
				
			||||||
 | 
						{0x2210, 0x2210}, {0x2212, 0x2214}, {0x2216, 0x2219},
 | 
				
			||||||
 | 
						{0x221B, 0x221C}, {0x2221, 0x2222}, {0x2224, 0x2224},
 | 
				
			||||||
 | 
						{0x2226, 0x2226}, {0x222D, 0x222D}, {0x222F, 0x2233},
 | 
				
			||||||
 | 
						{0x2238, 0x223B}, {0x223E, 0x2247}, {0x2249, 0x224B},
 | 
				
			||||||
 | 
						{0x224D, 0x2251}, {0x2253, 0x225F}, {0x2262, 0x2263},
 | 
				
			||||||
 | 
						{0x2268, 0x2269}, {0x226C, 0x226D}, {0x2270, 0x2281},
 | 
				
			||||||
 | 
						{0x2284, 0x2285}, {0x2288, 0x2294}, {0x2296, 0x2298},
 | 
				
			||||||
 | 
						{0x229A, 0x22A4}, {0x22A6, 0x22BE}, {0x22C0, 0x2311},
 | 
				
			||||||
 | 
						{0x2313, 0x2319}, {0x231C, 0x2328}, {0x232B, 0x23E8},
 | 
				
			||||||
 | 
						{0x23ED, 0x23EF}, {0x23F1, 0x23F2}, {0x23F4, 0x2426},
 | 
				
			||||||
 | 
						{0x2440, 0x244A}, {0x24EA, 0x24EA}, {0x254C, 0x254F},
 | 
				
			||||||
 | 
						{0x2574, 0x257F}, {0x2590, 0x2591}, {0x2596, 0x259F},
 | 
				
			||||||
 | 
						{0x25A2, 0x25A2}, {0x25AA, 0x25B1}, {0x25B4, 0x25B5},
 | 
				
			||||||
 | 
						{0x25B8, 0x25BB}, {0x25BE, 0x25BF}, {0x25C2, 0x25C5},
 | 
				
			||||||
 | 
						{0x25C9, 0x25CA}, {0x25CC, 0x25CD}, {0x25D2, 0x25E1},
 | 
				
			||||||
 | 
						{0x25E6, 0x25EE}, {0x25F0, 0x25FC}, {0x25FF, 0x2604},
 | 
				
			||||||
 | 
						{0x2607, 0x2608}, {0x260A, 0x260D}, {0x2610, 0x2613},
 | 
				
			||||||
 | 
						{0x2616, 0x261B}, {0x261D, 0x261D}, {0x261F, 0x263F},
 | 
				
			||||||
 | 
						{0x2641, 0x2641}, {0x2643, 0x2647}, {0x2654, 0x265F},
 | 
				
			||||||
 | 
						{0x2662, 0x2662}, {0x2666, 0x2666}, {0x266B, 0x266B},
 | 
				
			||||||
 | 
						{0x266E, 0x266E}, {0x2670, 0x267E}, {0x2680, 0x2692},
 | 
				
			||||||
 | 
						{0x2694, 0x269D}, {0x26A0, 0x26A0}, {0x26A2, 0x26A9},
 | 
				
			||||||
 | 
						{0x26AC, 0x26BC}, {0x26C0, 0x26C3}, {0x26E2, 0x26E2},
 | 
				
			||||||
 | 
						{0x26E4, 0x26E7}, {0x2700, 0x2704}, {0x2706, 0x2709},
 | 
				
			||||||
 | 
						{0x270C, 0x2727}, {0x2729, 0x273C}, {0x273E, 0x274B},
 | 
				
			||||||
 | 
						{0x274D, 0x274D}, {0x274F, 0x2752}, {0x2756, 0x2756},
 | 
				
			||||||
 | 
						{0x2758, 0x2775}, {0x2780, 0x2794}, {0x2798, 0x27AF},
 | 
				
			||||||
 | 
						{0x27B1, 0x27BE}, {0x27C0, 0x27E5}, {0x27EE, 0x2984},
 | 
				
			||||||
 | 
						{0x2987, 0x2B1A}, {0x2B1D, 0x2B4F}, {0x2B51, 0x2B54},
 | 
				
			||||||
 | 
						{0x2B5A, 0x2B73}, {0x2B76, 0x2B95}, {0x2B98, 0x2C2E},
 | 
				
			||||||
 | 
						{0x2C30, 0x2C5E}, {0x2C60, 0x2CF3}, {0x2CF9, 0x2D25},
 | 
				
			||||||
 | 
						{0x2D27, 0x2D27}, {0x2D2D, 0x2D2D}, {0x2D30, 0x2D67},
 | 
				
			||||||
 | 
						{0x2D6F, 0x2D70}, {0x2D7F, 0x2D96}, {0x2DA0, 0x2DA6},
 | 
				
			||||||
 | 
						{0x2DA8, 0x2DAE}, {0x2DB0, 0x2DB6}, {0x2DB8, 0x2DBE},
 | 
				
			||||||
 | 
						{0x2DC0, 0x2DC6}, {0x2DC8, 0x2DCE}, {0x2DD0, 0x2DD6},
 | 
				
			||||||
 | 
						{0x2DD8, 0x2DDE}, {0x2DE0, 0x2E4F}, {0x303F, 0x303F},
 | 
				
			||||||
 | 
						{0x4DC0, 0x4DFF}, {0xA4D0, 0xA62B}, {0xA640, 0xA6F7},
 | 
				
			||||||
 | 
						{0xA700, 0xA7BF}, {0xA7C2, 0xA7C6}, {0xA7F7, 0xA82B},
 | 
				
			||||||
 | 
						{0xA830, 0xA839}, {0xA840, 0xA877}, {0xA880, 0xA8C5},
 | 
				
			||||||
 | 
						{0xA8CE, 0xA8D9}, {0xA8E0, 0xA953}, {0xA95F, 0xA95F},
 | 
				
			||||||
 | 
						{0xA980, 0xA9CD}, {0xA9CF, 0xA9D9}, {0xA9DE, 0xA9FE},
 | 
				
			||||||
 | 
						{0xAA00, 0xAA36}, {0xAA40, 0xAA4D}, {0xAA50, 0xAA59},
 | 
				
			||||||
 | 
						{0xAA5C, 0xAAC2}, {0xAADB, 0xAAF6}, {0xAB01, 0xAB06},
 | 
				
			||||||
 | 
						{0xAB09, 0xAB0E}, {0xAB11, 0xAB16}, {0xAB20, 0xAB26},
 | 
				
			||||||
 | 
						{0xAB28, 0xAB2E}, {0xAB30, 0xAB67}, {0xAB70, 0xABED},
 | 
				
			||||||
 | 
						{0xABF0, 0xABF9}, {0xD7B0, 0xD7C6}, {0xD7CB, 0xD7FB},
 | 
				
			||||||
 | 
						{0xD800, 0xDFFF}, {0xFB00, 0xFB06}, {0xFB13, 0xFB17},
 | 
				
			||||||
 | 
						{0xFB1D, 0xFB36}, {0xFB38, 0xFB3C}, {0xFB3E, 0xFB3E},
 | 
				
			||||||
 | 
						{0xFB40, 0xFB41}, {0xFB43, 0xFB44}, {0xFB46, 0xFBC1},
 | 
				
			||||||
 | 
						{0xFBD3, 0xFD3F}, {0xFD50, 0xFD8F}, {0xFD92, 0xFDC7},
 | 
				
			||||||
 | 
						{0xFDF0, 0xFDFD}, {0xFE20, 0xFE2F}, {0xFE70, 0xFE74},
 | 
				
			||||||
 | 
						{0xFE76, 0xFEFC}, {0xFEFF, 0xFEFF}, {0xFFF9, 0xFFFC},
 | 
				
			||||||
 | 
						{0x10000, 0x1000B}, {0x1000D, 0x10026}, {0x10028, 0x1003A},
 | 
				
			||||||
 | 
						{0x1003C, 0x1003D}, {0x1003F, 0x1004D}, {0x10050, 0x1005D},
 | 
				
			||||||
 | 
						{0x10080, 0x100FA}, {0x10100, 0x10102}, {0x10107, 0x10133},
 | 
				
			||||||
 | 
						{0x10137, 0x1018E}, {0x10190, 0x1019B}, {0x101A0, 0x101A0},
 | 
				
			||||||
 | 
						{0x101D0, 0x101FD}, {0x10280, 0x1029C}, {0x102A0, 0x102D0},
 | 
				
			||||||
 | 
						{0x102E0, 0x102FB}, {0x10300, 0x10323}, {0x1032D, 0x1034A},
 | 
				
			||||||
 | 
						{0x10350, 0x1037A}, {0x10380, 0x1039D}, {0x1039F, 0x103C3},
 | 
				
			||||||
 | 
						{0x103C8, 0x103D5}, {0x10400, 0x1049D}, {0x104A0, 0x104A9},
 | 
				
			||||||
 | 
						{0x104B0, 0x104D3}, {0x104D8, 0x104FB}, {0x10500, 0x10527},
 | 
				
			||||||
 | 
						{0x10530, 0x10563}, {0x1056F, 0x1056F}, {0x10600, 0x10736},
 | 
				
			||||||
 | 
						{0x10740, 0x10755}, {0x10760, 0x10767}, {0x10800, 0x10805},
 | 
				
			||||||
 | 
						{0x10808, 0x10808}, {0x1080A, 0x10835}, {0x10837, 0x10838},
 | 
				
			||||||
 | 
						{0x1083C, 0x1083C}, {0x1083F, 0x10855}, {0x10857, 0x1089E},
 | 
				
			||||||
 | 
						{0x108A7, 0x108AF}, {0x108E0, 0x108F2}, {0x108F4, 0x108F5},
 | 
				
			||||||
 | 
						{0x108FB, 0x1091B}, {0x1091F, 0x10939}, {0x1093F, 0x1093F},
 | 
				
			||||||
 | 
						{0x10980, 0x109B7}, {0x109BC, 0x109CF}, {0x109D2, 0x10A03},
 | 
				
			||||||
 | 
						{0x10A05, 0x10A06}, {0x10A0C, 0x10A13}, {0x10A15, 0x10A17},
 | 
				
			||||||
 | 
						{0x10A19, 0x10A35}, {0x10A38, 0x10A3A}, {0x10A3F, 0x10A48},
 | 
				
			||||||
 | 
						{0x10A50, 0x10A58}, {0x10A60, 0x10A9F}, {0x10AC0, 0x10AE6},
 | 
				
			||||||
 | 
						{0x10AEB, 0x10AF6}, {0x10B00, 0x10B35}, {0x10B39, 0x10B55},
 | 
				
			||||||
 | 
						{0x10B58, 0x10B72}, {0x10B78, 0x10B91}, {0x10B99, 0x10B9C},
 | 
				
			||||||
 | 
						{0x10BA9, 0x10BAF}, {0x10C00, 0x10C48}, {0x10C80, 0x10CB2},
 | 
				
			||||||
 | 
						{0x10CC0, 0x10CF2}, {0x10CFA, 0x10D27}, {0x10D30, 0x10D39},
 | 
				
			||||||
 | 
						{0x10E60, 0x10E7E}, {0x10F00, 0x10F27}, {0x10F30, 0x10F59},
 | 
				
			||||||
 | 
						{0x10FE0, 0x10FF6}, {0x11000, 0x1104D}, {0x11052, 0x1106F},
 | 
				
			||||||
 | 
						{0x1107F, 0x110C1}, {0x110CD, 0x110CD}, {0x110D0, 0x110E8},
 | 
				
			||||||
 | 
						{0x110F0, 0x110F9}, {0x11100, 0x11134}, {0x11136, 0x11146},
 | 
				
			||||||
 | 
						{0x11150, 0x11176}, {0x11180, 0x111CD}, {0x111D0, 0x111DF},
 | 
				
			||||||
 | 
						{0x111E1, 0x111F4}, {0x11200, 0x11211}, {0x11213, 0x1123E},
 | 
				
			||||||
 | 
						{0x11280, 0x11286}, {0x11288, 0x11288}, {0x1128A, 0x1128D},
 | 
				
			||||||
 | 
						{0x1128F, 0x1129D}, {0x1129F, 0x112A9}, {0x112B0, 0x112EA},
 | 
				
			||||||
 | 
						{0x112F0, 0x112F9}, {0x11300, 0x11303}, {0x11305, 0x1130C},
 | 
				
			||||||
 | 
						{0x1130F, 0x11310}, {0x11313, 0x11328}, {0x1132A, 0x11330},
 | 
				
			||||||
 | 
						{0x11332, 0x11333}, {0x11335, 0x11339}, {0x1133B, 0x11344},
 | 
				
			||||||
 | 
						{0x11347, 0x11348}, {0x1134B, 0x1134D}, {0x11350, 0x11350},
 | 
				
			||||||
 | 
						{0x11357, 0x11357}, {0x1135D, 0x11363}, {0x11366, 0x1136C},
 | 
				
			||||||
 | 
						{0x11370, 0x11374}, {0x11400, 0x11459}, {0x1145B, 0x1145B},
 | 
				
			||||||
 | 
						{0x1145D, 0x1145F}, {0x11480, 0x114C7}, {0x114D0, 0x114D9},
 | 
				
			||||||
 | 
						{0x11580, 0x115B5}, {0x115B8, 0x115DD}, {0x11600, 0x11644},
 | 
				
			||||||
 | 
						{0x11650, 0x11659}, {0x11660, 0x1166C}, {0x11680, 0x116B8},
 | 
				
			||||||
 | 
						{0x116C0, 0x116C9}, {0x11700, 0x1171A}, {0x1171D, 0x1172B},
 | 
				
			||||||
 | 
						{0x11730, 0x1173F}, {0x11800, 0x1183B}, {0x118A0, 0x118F2},
 | 
				
			||||||
 | 
						{0x118FF, 0x118FF}, {0x119A0, 0x119A7}, {0x119AA, 0x119D7},
 | 
				
			||||||
 | 
						{0x119DA, 0x119E4}, {0x11A00, 0x11A47}, {0x11A50, 0x11AA2},
 | 
				
			||||||
 | 
						{0x11AC0, 0x11AF8}, {0x11C00, 0x11C08}, {0x11C0A, 0x11C36},
 | 
				
			||||||
 | 
						{0x11C38, 0x11C45}, {0x11C50, 0x11C6C}, {0x11C70, 0x11C8F},
 | 
				
			||||||
 | 
						{0x11C92, 0x11CA7}, {0x11CA9, 0x11CB6}, {0x11D00, 0x11D06},
 | 
				
			||||||
 | 
						{0x11D08, 0x11D09}, {0x11D0B, 0x11D36}, {0x11D3A, 0x11D3A},
 | 
				
			||||||
 | 
						{0x11D3C, 0x11D3D}, {0x11D3F, 0x11D47}, {0x11D50, 0x11D59},
 | 
				
			||||||
 | 
						{0x11D60, 0x11D65}, {0x11D67, 0x11D68}, {0x11D6A, 0x11D8E},
 | 
				
			||||||
 | 
						{0x11D90, 0x11D91}, {0x11D93, 0x11D98}, {0x11DA0, 0x11DA9},
 | 
				
			||||||
 | 
						{0x11EE0, 0x11EF8}, {0x11FC0, 0x11FF1}, {0x11FFF, 0x12399},
 | 
				
			||||||
 | 
						{0x12400, 0x1246E}, {0x12470, 0x12474}, {0x12480, 0x12543},
 | 
				
			||||||
 | 
						{0x13000, 0x1342E}, {0x13430, 0x13438}, {0x14400, 0x14646},
 | 
				
			||||||
 | 
						{0x16800, 0x16A38}, {0x16A40, 0x16A5E}, {0x16A60, 0x16A69},
 | 
				
			||||||
 | 
						{0x16A6E, 0x16A6F}, {0x16AD0, 0x16AED}, {0x16AF0, 0x16AF5},
 | 
				
			||||||
 | 
						{0x16B00, 0x16B45}, {0x16B50, 0x16B59}, {0x16B5B, 0x16B61},
 | 
				
			||||||
 | 
						{0x16B63, 0x16B77}, {0x16B7D, 0x16B8F}, {0x16E40, 0x16E9A},
 | 
				
			||||||
 | 
						{0x16F00, 0x16F4A}, {0x16F4F, 0x16F87}, {0x16F8F, 0x16F9F},
 | 
				
			||||||
 | 
						{0x1BC00, 0x1BC6A}, {0x1BC70, 0x1BC7C}, {0x1BC80, 0x1BC88},
 | 
				
			||||||
 | 
						{0x1BC90, 0x1BC99}, {0x1BC9C, 0x1BCA3}, {0x1D000, 0x1D0F5},
 | 
				
			||||||
 | 
						{0x1D100, 0x1D126}, {0x1D129, 0x1D1E8}, {0x1D200, 0x1D245},
 | 
				
			||||||
 | 
						{0x1D2E0, 0x1D2F3}, {0x1D300, 0x1D356}, {0x1D360, 0x1D378},
 | 
				
			||||||
 | 
						{0x1D400, 0x1D454}, {0x1D456, 0x1D49C}, {0x1D49E, 0x1D49F},
 | 
				
			||||||
 | 
						{0x1D4A2, 0x1D4A2}, {0x1D4A5, 0x1D4A6}, {0x1D4A9, 0x1D4AC},
 | 
				
			||||||
 | 
						{0x1D4AE, 0x1D4B9}, {0x1D4BB, 0x1D4BB}, {0x1D4BD, 0x1D4C3},
 | 
				
			||||||
 | 
						{0x1D4C5, 0x1D505}, {0x1D507, 0x1D50A}, {0x1D50D, 0x1D514},
 | 
				
			||||||
 | 
						{0x1D516, 0x1D51C}, {0x1D51E, 0x1D539}, {0x1D53B, 0x1D53E},
 | 
				
			||||||
 | 
						{0x1D540, 0x1D544}, {0x1D546, 0x1D546}, {0x1D54A, 0x1D550},
 | 
				
			||||||
 | 
						{0x1D552, 0x1D6A5}, {0x1D6A8, 0x1D7CB}, {0x1D7CE, 0x1DA8B},
 | 
				
			||||||
 | 
						{0x1DA9B, 0x1DA9F}, {0x1DAA1, 0x1DAAF}, {0x1E000, 0x1E006},
 | 
				
			||||||
 | 
						{0x1E008, 0x1E018}, {0x1E01B, 0x1E021}, {0x1E023, 0x1E024},
 | 
				
			||||||
 | 
						{0x1E026, 0x1E02A}, {0x1E100, 0x1E12C}, {0x1E130, 0x1E13D},
 | 
				
			||||||
 | 
						{0x1E140, 0x1E149}, {0x1E14E, 0x1E14F}, {0x1E2C0, 0x1E2F9},
 | 
				
			||||||
 | 
						{0x1E2FF, 0x1E2FF}, {0x1E800, 0x1E8C4}, {0x1E8C7, 0x1E8D6},
 | 
				
			||||||
 | 
						{0x1E900, 0x1E94B}, {0x1E950, 0x1E959}, {0x1E95E, 0x1E95F},
 | 
				
			||||||
 | 
						{0x1EC71, 0x1ECB4}, {0x1ED01, 0x1ED3D}, {0x1EE00, 0x1EE03},
 | 
				
			||||||
 | 
						{0x1EE05, 0x1EE1F}, {0x1EE21, 0x1EE22}, {0x1EE24, 0x1EE24},
 | 
				
			||||||
 | 
						{0x1EE27, 0x1EE27}, {0x1EE29, 0x1EE32}, {0x1EE34, 0x1EE37},
 | 
				
			||||||
 | 
						{0x1EE39, 0x1EE39}, {0x1EE3B, 0x1EE3B}, {0x1EE42, 0x1EE42},
 | 
				
			||||||
 | 
						{0x1EE47, 0x1EE47}, {0x1EE49, 0x1EE49}, {0x1EE4B, 0x1EE4B},
 | 
				
			||||||
 | 
						{0x1EE4D, 0x1EE4F}, {0x1EE51, 0x1EE52}, {0x1EE54, 0x1EE54},
 | 
				
			||||||
 | 
						{0x1EE57, 0x1EE57}, {0x1EE59, 0x1EE59}, {0x1EE5B, 0x1EE5B},
 | 
				
			||||||
 | 
						{0x1EE5D, 0x1EE5D}, {0x1EE5F, 0x1EE5F}, {0x1EE61, 0x1EE62},
 | 
				
			||||||
 | 
						{0x1EE64, 0x1EE64}, {0x1EE67, 0x1EE6A}, {0x1EE6C, 0x1EE72},
 | 
				
			||||||
 | 
						{0x1EE74, 0x1EE77}, {0x1EE79, 0x1EE7C}, {0x1EE7E, 0x1EE7E},
 | 
				
			||||||
 | 
						{0x1EE80, 0x1EE89}, {0x1EE8B, 0x1EE9B}, {0x1EEA1, 0x1EEA3},
 | 
				
			||||||
 | 
						{0x1EEA5, 0x1EEA9}, {0x1EEAB, 0x1EEBB}, {0x1EEF0, 0x1EEF1},
 | 
				
			||||||
 | 
						{0x1F000, 0x1F003}, {0x1F005, 0x1F02B}, {0x1F030, 0x1F093},
 | 
				
			||||||
 | 
						{0x1F0A0, 0x1F0AE}, {0x1F0B1, 0x1F0BF}, {0x1F0C1, 0x1F0CE},
 | 
				
			||||||
 | 
						{0x1F0D1, 0x1F0F5}, {0x1F10B, 0x1F10C}, {0x1F12E, 0x1F12F},
 | 
				
			||||||
 | 
						{0x1F16A, 0x1F16C}, {0x1F1E6, 0x1F1FF}, {0x1F321, 0x1F32C},
 | 
				
			||||||
 | 
						{0x1F336, 0x1F336}, {0x1F37D, 0x1F37D}, {0x1F394, 0x1F39F},
 | 
				
			||||||
 | 
						{0x1F3CB, 0x1F3CE}, {0x1F3D4, 0x1F3DF}, {0x1F3F1, 0x1F3F3},
 | 
				
			||||||
 | 
						{0x1F3F5, 0x1F3F7}, {0x1F43F, 0x1F43F}, {0x1F441, 0x1F441},
 | 
				
			||||||
 | 
						{0x1F4FD, 0x1F4FE}, {0x1F53E, 0x1F54A}, {0x1F54F, 0x1F54F},
 | 
				
			||||||
 | 
						{0x1F568, 0x1F579}, {0x1F57B, 0x1F594}, {0x1F597, 0x1F5A3},
 | 
				
			||||||
 | 
						{0x1F5A5, 0x1F5FA}, {0x1F650, 0x1F67F}, {0x1F6C6, 0x1F6CB},
 | 
				
			||||||
 | 
						{0x1F6CD, 0x1F6CF}, {0x1F6D3, 0x1F6D4}, {0x1F6E0, 0x1F6EA},
 | 
				
			||||||
 | 
						{0x1F6F0, 0x1F6F3}, {0x1F700, 0x1F773}, {0x1F780, 0x1F7D8},
 | 
				
			||||||
 | 
						{0x1F800, 0x1F80B}, {0x1F810, 0x1F847}, {0x1F850, 0x1F859},
 | 
				
			||||||
 | 
						{0x1F860, 0x1F887}, {0x1F890, 0x1F8AD}, {0x1F900, 0x1F90B},
 | 
				
			||||||
 | 
						{0x1FA00, 0x1FA53}, {0x1FA60, 0x1FA6D}, {0xE0001, 0xE0001},
 | 
				
			||||||
 | 
						{0xE0020, 0xE007F},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var emoji = table{
 | 
				
			||||||
 | 
						{0x203C, 0x203C}, {0x2049, 0x2049}, {0x2122, 0x2122},
 | 
				
			||||||
 | 
						{0x2139, 0x2139}, {0x2194, 0x2199}, {0x21A9, 0x21AA},
 | 
				
			||||||
 | 
						{0x231A, 0x231B}, {0x2328, 0x2328}, {0x2388, 0x2388},
 | 
				
			||||||
 | 
						{0x23CF, 0x23CF}, {0x23E9, 0x23F3}, {0x23F8, 0x23FA},
 | 
				
			||||||
 | 
						{0x24C2, 0x24C2}, {0x25AA, 0x25AB}, {0x25B6, 0x25B6},
 | 
				
			||||||
 | 
						{0x25C0, 0x25C0}, {0x25FB, 0x25FE}, {0x2600, 0x2605},
 | 
				
			||||||
 | 
						{0x2607, 0x2612}, {0x2614, 0x2685}, {0x2690, 0x2705},
 | 
				
			||||||
 | 
						{0x2708, 0x2712}, {0x2714, 0x2714}, {0x2716, 0x2716},
 | 
				
			||||||
 | 
						{0x271D, 0x271D}, {0x2721, 0x2721}, {0x2728, 0x2728},
 | 
				
			||||||
 | 
						{0x2733, 0x2734}, {0x2744, 0x2744}, {0x2747, 0x2747},
 | 
				
			||||||
 | 
						{0x274C, 0x274C}, {0x274E, 0x274E}, {0x2753, 0x2755},
 | 
				
			||||||
 | 
						{0x2757, 0x2757}, {0x2763, 0x2767}, {0x2795, 0x2797},
 | 
				
			||||||
 | 
						{0x27A1, 0x27A1}, {0x27B0, 0x27B0}, {0x27BF, 0x27BF},
 | 
				
			||||||
 | 
						{0x2934, 0x2935}, {0x2B05, 0x2B07}, {0x2B1B, 0x2B1C},
 | 
				
			||||||
 | 
						{0x2B50, 0x2B50}, {0x2B55, 0x2B55}, {0x3030, 0x3030},
 | 
				
			||||||
 | 
						{0x303D, 0x303D}, {0x3297, 0x3297}, {0x3299, 0x3299},
 | 
				
			||||||
 | 
						{0x1F000, 0x1F0FF}, {0x1F10D, 0x1F10F}, {0x1F12F, 0x1F12F},
 | 
				
			||||||
 | 
						{0x1F16C, 0x1F171}, {0x1F17E, 0x1F17F}, {0x1F18E, 0x1F18E},
 | 
				
			||||||
 | 
						{0x1F191, 0x1F19A}, {0x1F1AD, 0x1F1E5}, {0x1F201, 0x1F20F},
 | 
				
			||||||
 | 
						{0x1F21A, 0x1F21A}, {0x1F22F, 0x1F22F}, {0x1F232, 0x1F23A},
 | 
				
			||||||
 | 
						{0x1F23C, 0x1F23F}, {0x1F249, 0x1F3FA}, {0x1F400, 0x1F53D},
 | 
				
			||||||
 | 
						{0x1F546, 0x1F64F}, {0x1F680, 0x1F6FF}, {0x1F774, 0x1F77F},
 | 
				
			||||||
 | 
						{0x1F7D5, 0x1F7FF}, {0x1F80C, 0x1F80F}, {0x1F848, 0x1F84F},
 | 
				
			||||||
 | 
						{0x1F85A, 0x1F85F}, {0x1F888, 0x1F88F}, {0x1F8AE, 0x1F8FF},
 | 
				
			||||||
 | 
						{0x1F90C, 0x1F93A}, {0x1F93C, 0x1F945}, {0x1F947, 0x1FFFD},
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										28
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								vendor/github.com/mattn/go-runewidth/runewidth_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
				
			|||||||
 | 
					// +build windows
 | 
				
			||||||
 | 
					// +build !appengine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package runewidth
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"syscall"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						kernel32               = syscall.NewLazyDLL("kernel32")
 | 
				
			||||||
 | 
						procGetConsoleOutputCP = kernel32.NewProc("GetConsoleOutputCP")
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsEastAsian return true if the current locale is CJK
 | 
				
			||||||
 | 
					func IsEastAsian() bool {
 | 
				
			||||||
 | 
						r1, _, _ := procGetConsoleOutputCP.Call()
 | 
				
			||||||
 | 
						if r1 == 0 {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						switch int(r1) {
 | 
				
			||||||
 | 
						case 932, 51932, 936, 949, 950:
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										2
									
								
								vendor/github.com/mgechev/dots/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								vendor/github.com/mgechev/dots/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					language: go
 | 
				
			||||||
 | 
					go: master
 | 
				
			||||||
							
								
								
									
										21
									
								
								vendor/github.com/mgechev/dots/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/github.com/mgechev/dots/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					MIT License
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright (c) 2018 Minko Gechev
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
				
			||||||
 | 
					of this software and associated documentation files (the "Software"), to deal
 | 
				
			||||||
 | 
					in the Software without restriction, including without limitation the rights
 | 
				
			||||||
 | 
					to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
				
			||||||
 | 
					copies of the Software, and to permit persons to whom the Software is
 | 
				
			||||||
 | 
					furnished to do so, subject to the following conditions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The above copyright notice and this permission notice shall be included in all
 | 
				
			||||||
 | 
					copies or substantial portions of the Software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
				
			||||||
 | 
					IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
				
			||||||
 | 
					FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
				
			||||||
 | 
					AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
				
			||||||
 | 
					LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
				
			||||||
 | 
					OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | 
				
			||||||
 | 
					SOFTWARE.
 | 
				
			||||||
							
								
								
									
										100
									
								
								vendor/github.com/mgechev/dots/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								vendor/github.com/mgechev/dots/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,100 @@
 | 
				
			|||||||
 | 
					[](https://travis-ci.org/mgechev/dots)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Dots
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Implements the wildcard file matching in Go used by golint, go test etc.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Usage
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					import "github.com/mgechev/dots"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func main() {
 | 
				
			||||||
 | 
					  result, err := dots.Resolve([]string{"./fixtures/..."}, []string{"./fixtures/foo"})
 | 
				
			||||||
 | 
					  for _, f := range result {
 | 
				
			||||||
 | 
					    fmt.Println(f);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					If we suppose that we have the following directory structure:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```text
 | 
				
			||||||
 | 
					├── README.md
 | 
				
			||||||
 | 
					├── fixtures
 | 
				
			||||||
 | 
					│   ├── bar
 | 
				
			||||||
 | 
					│   │   ├── bar1.go
 | 
				
			||||||
 | 
					│   │   └── bar2.go
 | 
				
			||||||
 | 
					│   ├── baz
 | 
				
			||||||
 | 
					│   │   ├── baz1.go
 | 
				
			||||||
 | 
					│   │   ├── baz2.go
 | 
				
			||||||
 | 
					│   │   └── baz3.go
 | 
				
			||||||
 | 
					│   └── foo
 | 
				
			||||||
 | 
					│       ├── foo1.go
 | 
				
			||||||
 | 
					│       ├── foo2.go
 | 
				
			||||||
 | 
					│       └── foo3.go
 | 
				
			||||||
 | 
					└── main.go
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The result will be:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```text
 | 
				
			||||||
 | 
					fixtures/bar/bar1.go
 | 
				
			||||||
 | 
					fixtures/bar/bar2.go
 | 
				
			||||||
 | 
					fixtures/baz/baz1.go
 | 
				
			||||||
 | 
					fixtures/baz/baz2.go
 | 
				
			||||||
 | 
					fixtures/baz/baz3.go
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					`dots` supports wildcard in both - the first and the last argument of `Resolve`, which means that you can ignore files based on a wildcard:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					dots.Resolve([]string{"github.com/mgechev/dots"}, []string{"./..."}) // empty list
 | 
				
			||||||
 | 
					dots.Resolve([]string{"./fixtures/bar/..."}, []string{"./fixture/foo/...", "./fixtures/baz/..."}) // bar1.go, bar2.go
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Preserve package structure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					`dots` allow you to receive a slice of slices where each nested slice represents an individual package:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```go
 | 
				
			||||||
 | 
					dots.ResolvePackages([]string{"github.com/mgechev/dots/..."}, []string{})
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					So we will get the result:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```text
 | 
				
			||||||
 | 
					[
 | 
				
			||||||
 | 
					  [
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/bar/bar1.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/bar/bar2.go"
 | 
				
			||||||
 | 
					  ],
 | 
				
			||||||
 | 
					  [
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/baz/baz1.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/baz/baz2.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/baz/baz3.go"
 | 
				
			||||||
 | 
					  ],
 | 
				
			||||||
 | 
					  [
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/foo/foo1.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/foo/foo2.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/dummy/foo/foo3.go"
 | 
				
			||||||
 | 
					  ],
 | 
				
			||||||
 | 
					  [
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/pkg/baz/baz1.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/pkg/baz/baz2.go"
 | 
				
			||||||
 | 
					  ],
 | 
				
			||||||
 | 
					  [
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/pkg/foo/foo1.go",
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/pkg/foo/foo2.go"
 | 
				
			||||||
 | 
					  ],
 | 
				
			||||||
 | 
					  [
 | 
				
			||||||
 | 
					    "$GOROOT/src/github.com/mgechev/dots/fixtures/pkg/foo/bar/bar1.go"
 | 
				
			||||||
 | 
					  ]
 | 
				
			||||||
 | 
					]
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					This method is especially useful, when you want to perform type checking over given package from the result.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## License
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					MIT
 | 
				
			||||||
							
								
								
									
										456
									
								
								vendor/github.com/mgechev/dots/resolve.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										456
									
								
								vendor/github.com/mgechev/dots/resolve.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,456 @@
 | 
				
			|||||||
 | 
					package dots
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/build"
 | 
				
			||||||
 | 
						"log"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
						"path"
 | 
				
			||||||
 | 
						"path/filepath"
 | 
				
			||||||
 | 
						"regexp"
 | 
				
			||||||
 | 
						"runtime"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						buildContext = build.Default
 | 
				
			||||||
 | 
						goroot       = filepath.Clean(runtime.GOROOT())
 | 
				
			||||||
 | 
						gorootSrc    = filepath.Join(goroot, "src")
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func flatten(arr [][]string) []string {
 | 
				
			||||||
 | 
						var res []string
 | 
				
			||||||
 | 
						for _, e := range arr {
 | 
				
			||||||
 | 
							res = append(res, e...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return res
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Resolve accepts a slice of paths with optional "..." placeholder and a slice with paths to be skipped.
 | 
				
			||||||
 | 
					// The final result is the set of all files from the selected directories subtracted with
 | 
				
			||||||
 | 
					// the files in the skip slice.
 | 
				
			||||||
 | 
					func Resolve(includePatterns, skipPatterns []string) ([]string, error) {
 | 
				
			||||||
 | 
						skip, err := resolvePatterns(skipPatterns)
 | 
				
			||||||
 | 
						filter := newPathFilter(flatten(skip))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pathSet := map[string]bool{}
 | 
				
			||||||
 | 
						includePackages, err := resolvePatterns(includePatterns)
 | 
				
			||||||
 | 
						include := flatten(includePackages)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var result []string
 | 
				
			||||||
 | 
						for _, i := range include {
 | 
				
			||||||
 | 
							if _, ok := pathSet[i]; !ok && !filter(i) {
 | 
				
			||||||
 | 
								pathSet[i] = true
 | 
				
			||||||
 | 
								result = append(result, i)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return result, err
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ResolvePackages accepts a slice of paths with optional "..." placeholder and a slice with paths to be skipped.
 | 
				
			||||||
 | 
					// The final result is the set of all files from the selected directories subtracted with
 | 
				
			||||||
 | 
					// the files in the skip slice. The difference between `Resolve` and `ResolvePackages`
 | 
				
			||||||
 | 
					// is that `ResolvePackages` preserves the package structure in the nested slices.
 | 
				
			||||||
 | 
					func ResolvePackages(includePatterns, skipPatterns []string) ([][]string, error) {
 | 
				
			||||||
 | 
						skip, err := resolvePatterns(skipPatterns)
 | 
				
			||||||
 | 
						filter := newPathFilter(flatten(skip))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pathSet := map[string]bool{}
 | 
				
			||||||
 | 
						include, err := resolvePatterns(includePatterns)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var result [][]string
 | 
				
			||||||
 | 
						for _, p := range include {
 | 
				
			||||||
 | 
							var packageFiles []string
 | 
				
			||||||
 | 
							for _, f := range p {
 | 
				
			||||||
 | 
								if _, ok := pathSet[f]; !ok && !filter(f) {
 | 
				
			||||||
 | 
									pathSet[f] = true
 | 
				
			||||||
 | 
									packageFiles = append(packageFiles, f)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							result = append(result, packageFiles)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return result, err
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func isDir(filename string) bool {
 | 
				
			||||||
 | 
						fi, err := os.Stat(filename)
 | 
				
			||||||
 | 
						return err == nil && fi.IsDir()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func exists(filename string) bool {
 | 
				
			||||||
 | 
						_, err := os.Stat(filename)
 | 
				
			||||||
 | 
						return err == nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func resolveDir(dirname string) ([]string, error) {
 | 
				
			||||||
 | 
						pkg, err := build.ImportDir(dirname, 0)
 | 
				
			||||||
 | 
						return resolveImportedPackage(pkg, err)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func resolvePackage(pkgname string) ([]string, error) {
 | 
				
			||||||
 | 
						pkg, err := build.Import(pkgname, ".", 0)
 | 
				
			||||||
 | 
						return resolveImportedPackage(pkg, err)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func resolveImportedPackage(pkg *build.Package, err error) ([]string, error) {
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if _, nogo := err.(*build.NoGoError); nogo {
 | 
				
			||||||
 | 
								// Don't complain if the failure is due to no Go source files.
 | 
				
			||||||
 | 
								return nil, nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var files []string
 | 
				
			||||||
 | 
						files = append(files, pkg.GoFiles...)
 | 
				
			||||||
 | 
						files = append(files, pkg.CgoFiles...)
 | 
				
			||||||
 | 
						files = append(files, pkg.TestGoFiles...)
 | 
				
			||||||
 | 
						if pkg.Dir != "." {
 | 
				
			||||||
 | 
							for i, f := range files {
 | 
				
			||||||
 | 
								files[i] = filepath.Join(pkg.Dir, f)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return files, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func resolvePatterns(patterns []string) ([][]string, error) {
 | 
				
			||||||
 | 
						var files [][]string
 | 
				
			||||||
 | 
						for _, pattern := range patterns {
 | 
				
			||||||
 | 
							f, err := resolvePattern(pattern)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return nil, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							files = append(files, f...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return files, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func resolvePattern(pattern string) ([][]string, error) {
 | 
				
			||||||
 | 
						// dirsRun, filesRun, and pkgsRun indicate whether golint is applied to
 | 
				
			||||||
 | 
						// directory, file or package targets. The distinction affects which
 | 
				
			||||||
 | 
						// checks are run. It is no valid to mix target types.
 | 
				
			||||||
 | 
						var dirsRun, filesRun, pkgsRun int
 | 
				
			||||||
 | 
						var matches []string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if strings.HasSuffix(pattern, "/...") && isDir(pattern[:len(pattern)-len("/...")]) {
 | 
				
			||||||
 | 
							dirsRun = 1
 | 
				
			||||||
 | 
							for _, dirname := range matchPackagesInFS(pattern) {
 | 
				
			||||||
 | 
								matches = append(matches, dirname)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else if isDir(pattern) {
 | 
				
			||||||
 | 
							dirsRun = 1
 | 
				
			||||||
 | 
							matches = append(matches, pattern)
 | 
				
			||||||
 | 
						} else if exists(pattern) {
 | 
				
			||||||
 | 
							filesRun = 1
 | 
				
			||||||
 | 
							matches = append(matches, pattern)
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							pkgsRun = 1
 | 
				
			||||||
 | 
							matches = append(matches, pattern)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						result := [][]string{}
 | 
				
			||||||
 | 
						switch {
 | 
				
			||||||
 | 
						case dirsRun == 1:
 | 
				
			||||||
 | 
							for _, dir := range matches {
 | 
				
			||||||
 | 
								res, err := resolveDir(dir)
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									return nil, err
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								result = append(result, res)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						case filesRun == 1:
 | 
				
			||||||
 | 
							return [][]string{matches}, nil
 | 
				
			||||||
 | 
						case pkgsRun == 1:
 | 
				
			||||||
 | 
							for _, pkg := range importPaths(matches) {
 | 
				
			||||||
 | 
								res, err := resolvePackage(pkg)
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									return nil, err
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								result = append(result, res)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return result, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func newPathFilter(skip []string) func(string) bool {
 | 
				
			||||||
 | 
						filter := map[string]bool{}
 | 
				
			||||||
 | 
						for _, name := range skip {
 | 
				
			||||||
 | 
							filter[name] = true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return func(path string) bool {
 | 
				
			||||||
 | 
							base := filepath.Base(path)
 | 
				
			||||||
 | 
							if filter[base] || filter[path] {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return base != "." && base != ".." && strings.ContainsAny(base[0:1], "_.")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// importPathsNoDotExpansion returns the import paths to use for the given
 | 
				
			||||||
 | 
					// command line, but it does no ... expansion.
 | 
				
			||||||
 | 
					func importPathsNoDotExpansion(args []string) []string {
 | 
				
			||||||
 | 
						if len(args) == 0 {
 | 
				
			||||||
 | 
							return []string{"."}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						var out []string
 | 
				
			||||||
 | 
						for _, a := range args {
 | 
				
			||||||
 | 
							// Arguments are supposed to be import paths, but
 | 
				
			||||||
 | 
							// as a courtesy to Windows developers, rewrite \ to /
 | 
				
			||||||
 | 
							// in command-line arguments.  Handles .\... and so on.
 | 
				
			||||||
 | 
							if filepath.Separator == '\\' {
 | 
				
			||||||
 | 
								a = strings.Replace(a, `\`, `/`, -1)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Put argument in canonical form, but preserve leading ./.
 | 
				
			||||||
 | 
							if strings.HasPrefix(a, "./") {
 | 
				
			||||||
 | 
								a = "./" + path.Clean(a)
 | 
				
			||||||
 | 
								if a == "./." {
 | 
				
			||||||
 | 
									a = "."
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								a = path.Clean(a)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if a == "all" || a == "std" {
 | 
				
			||||||
 | 
								out = append(out, matchPackages(a)...)
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							out = append(out, a)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return out
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// importPaths returns the import paths to use for the given command line.
 | 
				
			||||||
 | 
					func importPaths(args []string) []string {
 | 
				
			||||||
 | 
						args = importPathsNoDotExpansion(args)
 | 
				
			||||||
 | 
						var out []string
 | 
				
			||||||
 | 
						for _, a := range args {
 | 
				
			||||||
 | 
							if strings.Contains(a, "...") {
 | 
				
			||||||
 | 
								if build.IsLocalImport(a) {
 | 
				
			||||||
 | 
									out = append(out, matchPackagesInFS(a)...)
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									out = append(out, matchPackages(a)...)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							out = append(out, a)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return out
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// matchPattern(pattern)(name) reports whether
 | 
				
			||||||
 | 
					// name matches pattern.  Pattern is a limited glob
 | 
				
			||||||
 | 
					// pattern in which '...' means 'any string' and there
 | 
				
			||||||
 | 
					// is no other special syntax.
 | 
				
			||||||
 | 
					func matchPattern(pattern string) func(name string) bool {
 | 
				
			||||||
 | 
						re := regexp.QuoteMeta(pattern)
 | 
				
			||||||
 | 
						re = strings.Replace(re, `\.\.\.`, `.*`, -1)
 | 
				
			||||||
 | 
						// Special case: foo/... matches foo too.
 | 
				
			||||||
 | 
						if strings.HasSuffix(re, `/.*`) {
 | 
				
			||||||
 | 
							re = re[:len(re)-len(`/.*`)] + `(/.*)?`
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						reg := regexp.MustCompile(`^` + re + `$`)
 | 
				
			||||||
 | 
						return func(name string) bool {
 | 
				
			||||||
 | 
							return reg.MatchString(name)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// hasPathPrefix reports whether the path s begins with the
 | 
				
			||||||
 | 
					// elements in prefix.
 | 
				
			||||||
 | 
					func hasPathPrefix(s, prefix string) bool {
 | 
				
			||||||
 | 
						switch {
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						case len(s) == len(prefix):
 | 
				
			||||||
 | 
							return s == prefix
 | 
				
			||||||
 | 
						case len(s) > len(prefix):
 | 
				
			||||||
 | 
							if prefix != "" && prefix[len(prefix)-1] == '/' {
 | 
				
			||||||
 | 
								return strings.HasPrefix(s, prefix)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return s[len(prefix)] == '/' && s[:len(prefix)] == prefix
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// treeCanMatchPattern(pattern)(name) reports whether
 | 
				
			||||||
 | 
					// name or children of name can possibly match pattern.
 | 
				
			||||||
 | 
					// Pattern is the same limited glob accepted by matchPattern.
 | 
				
			||||||
 | 
					func treeCanMatchPattern(pattern string) func(name string) bool {
 | 
				
			||||||
 | 
						wildCard := false
 | 
				
			||||||
 | 
						if i := strings.Index(pattern, "..."); i >= 0 {
 | 
				
			||||||
 | 
							wildCard = true
 | 
				
			||||||
 | 
							pattern = pattern[:i]
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return func(name string) bool {
 | 
				
			||||||
 | 
							return len(name) <= len(pattern) && hasPathPrefix(pattern, name) ||
 | 
				
			||||||
 | 
								wildCard && strings.HasPrefix(name, pattern)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func matchPackages(pattern string) []string {
 | 
				
			||||||
 | 
						match := func(string) bool { return true }
 | 
				
			||||||
 | 
						treeCanMatch := func(string) bool { return true }
 | 
				
			||||||
 | 
						if pattern != "all" && pattern != "std" {
 | 
				
			||||||
 | 
							match = matchPattern(pattern)
 | 
				
			||||||
 | 
							treeCanMatch = treeCanMatchPattern(pattern)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						have := map[string]bool{
 | 
				
			||||||
 | 
							"builtin": true, // ignore pseudo-package that exists only for documentation
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !buildContext.CgoEnabled {
 | 
				
			||||||
 | 
							have["runtime/cgo"] = true // ignore during walk
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						var pkgs []string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Commands
 | 
				
			||||||
 | 
						cmd := filepath.Join(goroot, "src/cmd") + string(filepath.Separator)
 | 
				
			||||||
 | 
						filepath.Walk(cmd, func(path string, fi os.FileInfo, err error) error {
 | 
				
			||||||
 | 
							if err != nil || !fi.IsDir() || path == cmd {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							name := path[len(cmd):]
 | 
				
			||||||
 | 
							if !treeCanMatch(name) {
 | 
				
			||||||
 | 
								return filepath.SkipDir
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							// Commands are all in cmd/, not in subdirectories.
 | 
				
			||||||
 | 
							if strings.Contains(name, string(filepath.Separator)) {
 | 
				
			||||||
 | 
								return filepath.SkipDir
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// We use, e.g., cmd/gofmt as the pseudo import path for gofmt.
 | 
				
			||||||
 | 
							name = "cmd/" + name
 | 
				
			||||||
 | 
							if have[name] {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							have[name] = true
 | 
				
			||||||
 | 
							if !match(name) {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							_, err = buildContext.ImportDir(path, 0)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								if _, noGo := err.(*build.NoGoError); !noGo {
 | 
				
			||||||
 | 
									log.Print(err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							pkgs = append(pkgs, name)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, src := range buildContext.SrcDirs() {
 | 
				
			||||||
 | 
							if (pattern == "std" || pattern == "cmd") && src != gorootSrc {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							src = filepath.Clean(src) + string(filepath.Separator)
 | 
				
			||||||
 | 
							root := src
 | 
				
			||||||
 | 
							if pattern == "cmd" {
 | 
				
			||||||
 | 
								root += "cmd" + string(filepath.Separator)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							filepath.Walk(root, func(path string, fi os.FileInfo, err error) error {
 | 
				
			||||||
 | 
								if err != nil || !fi.IsDir() || path == src {
 | 
				
			||||||
 | 
									return nil
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								// Avoid .foo, _foo, and testdata directory trees.
 | 
				
			||||||
 | 
								_, elem := filepath.Split(path)
 | 
				
			||||||
 | 
								if strings.HasPrefix(elem, ".") || strings.HasPrefix(elem, "_") || elem == "testdata" {
 | 
				
			||||||
 | 
									return filepath.SkipDir
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								name := filepath.ToSlash(path[len(src):])
 | 
				
			||||||
 | 
								if pattern == "std" && (strings.Contains(name, ".") || name == "cmd") {
 | 
				
			||||||
 | 
									// The name "std" is only the standard library.
 | 
				
			||||||
 | 
									// If the name is cmd, it's the root of the command tree.
 | 
				
			||||||
 | 
									return filepath.SkipDir
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if !treeCanMatch(name) {
 | 
				
			||||||
 | 
									return filepath.SkipDir
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if have[name] {
 | 
				
			||||||
 | 
									return nil
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								have[name] = true
 | 
				
			||||||
 | 
								if !match(name) {
 | 
				
			||||||
 | 
									return nil
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								_, err = buildContext.ImportDir(path, 0)
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									if _, noGo := err.(*build.NoGoError); noGo {
 | 
				
			||||||
 | 
										return nil
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								pkgs = append(pkgs, name)
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return pkgs
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func matchPackagesInFS(pattern string) []string {
 | 
				
			||||||
 | 
						// Find directory to begin the scan.
 | 
				
			||||||
 | 
						// Could be smarter but this one optimization
 | 
				
			||||||
 | 
						// is enough for now, since ... is usually at the
 | 
				
			||||||
 | 
						// end of a path.
 | 
				
			||||||
 | 
						i := strings.Index(pattern, "...")
 | 
				
			||||||
 | 
						dir, _ := path.Split(pattern[:i])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// pattern begins with ./ or ../.
 | 
				
			||||||
 | 
						// path.Clean will discard the ./ but not the ../.
 | 
				
			||||||
 | 
						// We need to preserve the ./ for pattern matching
 | 
				
			||||||
 | 
						// and in the returned import paths.
 | 
				
			||||||
 | 
						prefix := ""
 | 
				
			||||||
 | 
						if strings.HasPrefix(pattern, "./") {
 | 
				
			||||||
 | 
							prefix = "./"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						match := matchPattern(pattern)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var pkgs []string
 | 
				
			||||||
 | 
						filepath.Walk(dir, func(path string, fi os.FileInfo, err error) error {
 | 
				
			||||||
 | 
							if err != nil || !fi.IsDir() {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if path == dir {
 | 
				
			||||||
 | 
								// filepath.Walk starts at dir and recurses. For the recursive case,
 | 
				
			||||||
 | 
								// the path is the result of filepath.Join, which calls filepath.Clean.
 | 
				
			||||||
 | 
								// The initial case is not Cleaned, though, so we do this explicitly.
 | 
				
			||||||
 | 
								//
 | 
				
			||||||
 | 
								// This converts a path like "./io/" to "io". Without this step, running
 | 
				
			||||||
 | 
								// "cd $GOROOT/src/pkg; go list ./io/..." would incorrectly skip the io
 | 
				
			||||||
 | 
								// package, because prepending the prefix "./" to the unclean path would
 | 
				
			||||||
 | 
								// result in "././io", and match("././io") returns false.
 | 
				
			||||||
 | 
								path = filepath.Clean(path)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Avoid .foo, _foo, and testdata directory trees, but do not avoid "." or "..".
 | 
				
			||||||
 | 
							_, elem := filepath.Split(path)
 | 
				
			||||||
 | 
							dot := strings.HasPrefix(elem, ".") && elem != "." && elem != ".."
 | 
				
			||||||
 | 
							if dot || strings.HasPrefix(elem, "_") || elem == "testdata" {
 | 
				
			||||||
 | 
								return filepath.SkipDir
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							name := prefix + filepath.ToSlash(path)
 | 
				
			||||||
 | 
							if !match(name) {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if _, err = build.ImportDir(path, 0); err != nil {
 | 
				
			||||||
 | 
								if _, noGo := err.(*build.NoGoError); !noGo {
 | 
				
			||||||
 | 
									log.Print(err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							pkgs = append(pkgs, name)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						return pkgs
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										21
									
								
								vendor/github.com/mgechev/revive/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/github.com/mgechev/revive/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					MIT License
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright (c) 2018 Minko Gechev
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
				
			||||||
 | 
					of this software and associated documentation files (the "Software"), to deal
 | 
				
			||||||
 | 
					in the Software without restriction, including without limitation the rights
 | 
				
			||||||
 | 
					to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
				
			||||||
 | 
					copies of the Software, and to permit persons to whom the Software is
 | 
				
			||||||
 | 
					furnished to do so, subject to the following conditions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The above copyright notice and this permission notice shall be included in all
 | 
				
			||||||
 | 
					copies or substantial portions of the Software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
				
			||||||
 | 
					IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
				
			||||||
 | 
					FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
				
			||||||
 | 
					AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
				
			||||||
 | 
					LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
				
			||||||
 | 
					OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | 
				
			||||||
 | 
					SOFTWARE.
 | 
				
			||||||
							
								
								
									
										76
									
								
								vendor/github.com/mgechev/revive/formatter/checkstyle.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								vendor/github.com/mgechev/revive/formatter/checkstyle.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,76 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"encoding/xml"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						plainTemplate "text/template"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Checkstyle is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to Checkstyle-like format.
 | 
				
			||||||
 | 
					type Checkstyle struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *Checkstyle) Name() string {
 | 
				
			||||||
 | 
						return "checkstyle"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type issue struct {
 | 
				
			||||||
 | 
						Line       int
 | 
				
			||||||
 | 
						Col        int
 | 
				
			||||||
 | 
						What       string
 | 
				
			||||||
 | 
						Confidence float64
 | 
				
			||||||
 | 
						Severity   lint.Severity
 | 
				
			||||||
 | 
						RuleName   string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *Checkstyle) Format(failures <-chan lint.Failure, config lint.Config) (string, error) {
 | 
				
			||||||
 | 
						var issues = map[string][]issue{}
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							buf := new(bytes.Buffer)
 | 
				
			||||||
 | 
							xml.Escape(buf, []byte(failure.Failure))
 | 
				
			||||||
 | 
							what := buf.String()
 | 
				
			||||||
 | 
							iss := issue{
 | 
				
			||||||
 | 
								Line:       failure.Position.Start.Line,
 | 
				
			||||||
 | 
								Col:        failure.Position.Start.Column,
 | 
				
			||||||
 | 
								What:       what,
 | 
				
			||||||
 | 
								Confidence: failure.Confidence,
 | 
				
			||||||
 | 
								Severity:   severity(config, failure),
 | 
				
			||||||
 | 
								RuleName:   failure.RuleName,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							fn := failure.GetFilename()
 | 
				
			||||||
 | 
							if issues[fn] == nil {
 | 
				
			||||||
 | 
								issues[fn] = make([]issue, 0)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							issues[fn] = append(issues[fn], iss)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						t, err := plainTemplate.New("revive").Parse(checkstyleTemplate)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return "", err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						buf := new(bytes.Buffer)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						err = t.Execute(buf, issues)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return "", err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return buf.String(), nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const checkstyleTemplate = `<?xml version='1.0' encoding='UTF-8'?>
 | 
				
			||||||
 | 
					<checkstyle version="5.0">
 | 
				
			||||||
 | 
					{{- range $k, $v := . }}
 | 
				
			||||||
 | 
					    <file name="{{ $k }}">
 | 
				
			||||||
 | 
					      {{- range $i, $issue := $v }}
 | 
				
			||||||
 | 
					      <error line="{{ $issue.Line }}" column="{{ $issue.Col }}" message="{{ $issue.What }} (confidence {{ $issue.Confidence}})" severity="{{ $issue.Severity }}" source="revive/{{ $issue.RuleName }}"/>
 | 
				
			||||||
 | 
					      {{- end }}
 | 
				
			||||||
 | 
					    </file>
 | 
				
			||||||
 | 
					{{- end }}
 | 
				
			||||||
 | 
					</checkstyle>`
 | 
				
			||||||
							
								
								
									
										26
									
								
								vendor/github.com/mgechev/revive/formatter/default.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								vendor/github.com/mgechev/revive/formatter/default.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Default is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to text.
 | 
				
			||||||
 | 
					type Default struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *Default) Name() string {
 | 
				
			||||||
 | 
						return "default"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *Default) Format(failures <-chan lint.Failure, _ lint.Config) (string, error) {
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							fmt.Printf("%v: %s\n", failure.Position.Start, failure.Failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return "", nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										146
									
								
								vendor/github.com/mgechev/revive/formatter/friendly.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								vendor/github.com/mgechev/revive/formatter/friendly.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,146 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"sort"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/fatih/color"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"github.com/olekukonko/tablewriter"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						errorEmoji   = color.RedString("✘")
 | 
				
			||||||
 | 
						warningEmoji = color.YellowString("⚠")
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var newLines = map[rune]bool{
 | 
				
			||||||
 | 
						0x000A: true,
 | 
				
			||||||
 | 
						0x000B: true,
 | 
				
			||||||
 | 
						0x000C: true,
 | 
				
			||||||
 | 
						0x000D: true,
 | 
				
			||||||
 | 
						0x0085: true,
 | 
				
			||||||
 | 
						0x2028: true,
 | 
				
			||||||
 | 
						0x2029: true,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Friendly is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to JSON.
 | 
				
			||||||
 | 
					type Friendly struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *Friendly) Name() string {
 | 
				
			||||||
 | 
						return "friendly"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *Friendly) Format(failures <-chan lint.Failure, config lint.Config) (string, error) {
 | 
				
			||||||
 | 
						errorMap := map[string]int{}
 | 
				
			||||||
 | 
						warningMap := map[string]int{}
 | 
				
			||||||
 | 
						totalErrors := 0
 | 
				
			||||||
 | 
						totalWarnings := 0
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							sev := severity(config, failure)
 | 
				
			||||||
 | 
							f.printFriendlyFailure(failure, sev)
 | 
				
			||||||
 | 
							if sev == lint.SeverityWarning {
 | 
				
			||||||
 | 
								warningMap[failure.RuleName] = warningMap[failure.RuleName] + 1
 | 
				
			||||||
 | 
								totalWarnings++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if sev == lint.SeverityError {
 | 
				
			||||||
 | 
								errorMap[failure.RuleName] = errorMap[failure.RuleName] + 1
 | 
				
			||||||
 | 
								totalErrors++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						f.printSummary(totalErrors, totalWarnings)
 | 
				
			||||||
 | 
						f.printStatistics(color.RedString("Errors:"), errorMap)
 | 
				
			||||||
 | 
						f.printStatistics(color.YellowString("Warnings:"), warningMap)
 | 
				
			||||||
 | 
						return "", nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *Friendly) printFriendlyFailure(failure lint.Failure, severity lint.Severity) {
 | 
				
			||||||
 | 
						f.printHeaderRow(failure, severity)
 | 
				
			||||||
 | 
						f.printFilePosition(failure)
 | 
				
			||||||
 | 
						fmt.Println()
 | 
				
			||||||
 | 
						fmt.Println()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *Friendly) printHeaderRow(failure lint.Failure, severity lint.Severity) {
 | 
				
			||||||
 | 
						emoji := warningEmoji
 | 
				
			||||||
 | 
						if severity == lint.SeverityError {
 | 
				
			||||||
 | 
							emoji = errorEmoji
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						fmt.Print(f.table([][]string{{emoji, "https://revive.run/r#" + failure.RuleName, color.GreenString(failure.Failure)}}))
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *Friendly) printFilePosition(failure lint.Failure) {
 | 
				
			||||||
 | 
						fmt.Printf("  %s:%d:%d", failure.GetFilename(), failure.Position.Start.Line, failure.Position.Start.Column)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type statEntry struct {
 | 
				
			||||||
 | 
						name     string
 | 
				
			||||||
 | 
						failures int
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *Friendly) printSummary(errors, warnings int) {
 | 
				
			||||||
 | 
						emoji := warningEmoji
 | 
				
			||||||
 | 
						if errors > 0 {
 | 
				
			||||||
 | 
							emoji = errorEmoji
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						problemsLabel := "problems"
 | 
				
			||||||
 | 
						if errors+warnings == 1 {
 | 
				
			||||||
 | 
							problemsLabel = "problem"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						warningsLabel := "warnings"
 | 
				
			||||||
 | 
						if warnings == 1 {
 | 
				
			||||||
 | 
							warningsLabel = "warning"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						errorsLabel := "errors"
 | 
				
			||||||
 | 
						if errors == 1 {
 | 
				
			||||||
 | 
							errorsLabel = "error"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						str := fmt.Sprintf("%d %s (%d %s, %d %s)", errors+warnings, problemsLabel, errors, errorsLabel, warnings, warningsLabel)
 | 
				
			||||||
 | 
						if errors > 0 {
 | 
				
			||||||
 | 
							fmt.Printf("%s %s\n", emoji, color.RedString(str))
 | 
				
			||||||
 | 
							fmt.Println()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if warnings > 0 {
 | 
				
			||||||
 | 
							fmt.Printf("%s %s\n", emoji, color.YellowString(str))
 | 
				
			||||||
 | 
							fmt.Println()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *Friendly) printStatistics(header string, stats map[string]int) {
 | 
				
			||||||
 | 
						if len(stats) == 0 {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						var data []statEntry
 | 
				
			||||||
 | 
						for name, total := range stats {
 | 
				
			||||||
 | 
							data = append(data, statEntry{name, total})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						sort.Slice(data, func(i, j int) bool {
 | 
				
			||||||
 | 
							return data[i].failures > data[j].failures
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						formatted := [][]string{}
 | 
				
			||||||
 | 
						for _, entry := range data {
 | 
				
			||||||
 | 
							formatted = append(formatted, []string{color.GreenString(fmt.Sprintf("%d", entry.failures)), entry.name})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						fmt.Println(header)
 | 
				
			||||||
 | 
						fmt.Println(f.table(formatted))
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *Friendly) table(rows [][]string) string {
 | 
				
			||||||
 | 
						buf := new(bytes.Buffer)
 | 
				
			||||||
 | 
						table := tablewriter.NewWriter(buf)
 | 
				
			||||||
 | 
						table.SetBorder(false)
 | 
				
			||||||
 | 
						table.SetColumnSeparator("")
 | 
				
			||||||
 | 
						table.SetRowSeparator("")
 | 
				
			||||||
 | 
						table.SetAutoWrapText(false)
 | 
				
			||||||
 | 
						table.AppendBulk(rows)
 | 
				
			||||||
 | 
						table.Render()
 | 
				
			||||||
 | 
						return buf.String()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										40
									
								
								vendor/github.com/mgechev/revive/formatter/json.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								vendor/github.com/mgechev/revive/formatter/json.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,40 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"encoding/json"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// JSON is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to JSON.
 | 
				
			||||||
 | 
					type JSON struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *JSON) Name() string {
 | 
				
			||||||
 | 
						return "json"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// jsonObject defines a JSON object of an failure
 | 
				
			||||||
 | 
					type jsonObject struct {
 | 
				
			||||||
 | 
						Severity     lint.Severity
 | 
				
			||||||
 | 
						lint.Failure `json:",inline"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *JSON) Format(failures <-chan lint.Failure, config lint.Config) (string, error) {
 | 
				
			||||||
 | 
						var slice []jsonObject
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							obj := jsonObject{}
 | 
				
			||||||
 | 
							obj.Severity = severity(config, failure)
 | 
				
			||||||
 | 
							obj.Failure = failure
 | 
				
			||||||
 | 
							slice = append(slice, obj)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						result, err := json.Marshal(slice)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return "", err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return string(result), err
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										34
									
								
								vendor/github.com/mgechev/revive/formatter/ndjson.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								vendor/github.com/mgechev/revive/formatter/ndjson.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,34 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"encoding/json"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NDJSON is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to NDJSON stream.
 | 
				
			||||||
 | 
					type NDJSON struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *NDJSON) Name() string {
 | 
				
			||||||
 | 
						return "ndjson"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *NDJSON) Format(failures <-chan lint.Failure, config lint.Config) (string, error) {
 | 
				
			||||||
 | 
						enc := json.NewEncoder(os.Stdout)
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							obj := jsonObject{}
 | 
				
			||||||
 | 
							obj.Severity = severity(config, failure)
 | 
				
			||||||
 | 
							obj.Failure = failure
 | 
				
			||||||
 | 
							err := enc.Encode(obj)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return "", err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return "", nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										26
									
								
								vendor/github.com/mgechev/revive/formatter/plain.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								vendor/github.com/mgechev/revive/formatter/plain.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Plain is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to JSON.
 | 
				
			||||||
 | 
					type Plain struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *Plain) Name() string {
 | 
				
			||||||
 | 
						return "plain"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *Plain) Format(failures <-chan lint.Failure, _ lint.Config) (string, error) {
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							fmt.Printf("%v: %s %s\n", failure.Position.Start, failure.Failure, "https://revive.run/r#"+failure.RuleName)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return "", nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										13
									
								
								vendor/github.com/mgechev/revive/formatter/severity.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								vendor/github.com/mgechev/revive/formatter/severity.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,13 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import "github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func severity(config lint.Config, failure lint.Failure) lint.Severity {
 | 
				
			||||||
 | 
						if config, ok := config.Rules[failure.RuleName]; ok && config.Severity == lint.SeverityError {
 | 
				
			||||||
 | 
							return lint.SeverityError
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if config, ok := config.Directives[failure.RuleName]; ok && config.Severity == lint.SeverityError {
 | 
				
			||||||
 | 
							return lint.SeverityError
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return lint.SeverityWarning
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										89
									
								
								vendor/github.com/mgechev/revive/formatter/stylish.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								vendor/github.com/mgechev/revive/formatter/stylish.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,89 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/fatih/color"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"github.com/olekukonko/tablewriter"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Stylish is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to JSON.
 | 
				
			||||||
 | 
					type Stylish struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *Stylish) Name() string {
 | 
				
			||||||
 | 
						return "stylish"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func formatFailure(failure lint.Failure, severity lint.Severity) []string {
 | 
				
			||||||
 | 
						fString := color.CyanString(failure.Failure)
 | 
				
			||||||
 | 
						fName := color.RedString("https://revive.run/r#" + failure.RuleName)
 | 
				
			||||||
 | 
						lineColumn := failure.Position
 | 
				
			||||||
 | 
						pos := fmt.Sprintf("(%d, %d)", lineColumn.Start.Line, lineColumn.Start.Column)
 | 
				
			||||||
 | 
						if severity == lint.SeverityWarning {
 | 
				
			||||||
 | 
							fName = color.YellowString("https://revive.run/r#" + failure.RuleName)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return []string{failure.GetFilename(), pos, fName, fString}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *Stylish) Format(failures <-chan lint.Failure, config lint.Config) (string, error) {
 | 
				
			||||||
 | 
						var result [][]string
 | 
				
			||||||
 | 
						var totalErrors = 0
 | 
				
			||||||
 | 
						var total = 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for f := range failures {
 | 
				
			||||||
 | 
							total++
 | 
				
			||||||
 | 
							currentType := severity(config, f)
 | 
				
			||||||
 | 
							if currentType == lint.SeverityError {
 | 
				
			||||||
 | 
								totalErrors++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							result = append(result, formatFailure(f, lint.Severity(currentType)))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						ps := "problems"
 | 
				
			||||||
 | 
						if total == 1 {
 | 
				
			||||||
 | 
							ps = "problem"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileReport := make(map[string][][]string)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, row := range result {
 | 
				
			||||||
 | 
							if _, ok := fileReport[row[0]]; !ok {
 | 
				
			||||||
 | 
								fileReport[row[0]] = [][]string{}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							fileReport[row[0]] = append(fileReport[row[0]], []string{row[1], row[2], row[3]})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						output := ""
 | 
				
			||||||
 | 
						for filename, val := range fileReport {
 | 
				
			||||||
 | 
							buf := new(bytes.Buffer)
 | 
				
			||||||
 | 
							table := tablewriter.NewWriter(buf)
 | 
				
			||||||
 | 
							table.SetBorder(false)
 | 
				
			||||||
 | 
							table.SetColumnSeparator("")
 | 
				
			||||||
 | 
							table.SetRowSeparator("")
 | 
				
			||||||
 | 
							table.SetAutoWrapText(false)
 | 
				
			||||||
 | 
							table.AppendBulk(val)
 | 
				
			||||||
 | 
							table.Render()
 | 
				
			||||||
 | 
							c := color.New(color.Underline)
 | 
				
			||||||
 | 
							output += c.SprintfFunc()(filename + "\n")
 | 
				
			||||||
 | 
							output += buf.String() + "\n"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						suffix := fmt.Sprintf(" %d %s (%d errors) (%d warnings)", total, ps, totalErrors, total-totalErrors)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if total > 0 && totalErrors > 0 {
 | 
				
			||||||
 | 
							suffix = color.RedString("\n ✖" + suffix)
 | 
				
			||||||
 | 
						} else if total > 0 && totalErrors == 0 {
 | 
				
			||||||
 | 
							suffix = color.YellowString("\n ✖" + suffix)
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							suffix, output = "", ""
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return output + suffix, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										27
									
								
								vendor/github.com/mgechev/revive/formatter/unix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								vendor/github.com/mgechev/revive/formatter/unix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,27 @@
 | 
				
			|||||||
 | 
					package formatter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Unix is an implementation of the Formatter interface
 | 
				
			||||||
 | 
					// which formats the errors to a simple line based error format
 | 
				
			||||||
 | 
					//  main.go:24:9: [errorf] should replace errors.New(fmt.Sprintf(...)) with fmt.Errorf(...)
 | 
				
			||||||
 | 
					type Unix struct {
 | 
				
			||||||
 | 
						Metadata lint.FormatterMetadata
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the name of the formatter
 | 
				
			||||||
 | 
					func (f *Unix) Name() string {
 | 
				
			||||||
 | 
						return "unix"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Format formats the failures gotten from the lint.
 | 
				
			||||||
 | 
					func (f *Unix) Format(failures <-chan lint.Failure, _ lint.Config) (string, error) {
 | 
				
			||||||
 | 
						for failure := range failures {
 | 
				
			||||||
 | 
							fmt.Printf("%v: [%s] %s\n", failure.Position.Start, failure.RuleName, failure.Failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return "", nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										32
									
								
								vendor/github.com/mgechev/revive/lint/config.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								vendor/github.com/mgechev/revive/lint/config.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,32 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Arguments is type used for the arguments of a rule.
 | 
				
			||||||
 | 
					type Arguments = []interface{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// RuleConfig is type used for the rule configuration.
 | 
				
			||||||
 | 
					type RuleConfig struct {
 | 
				
			||||||
 | 
						Arguments Arguments
 | 
				
			||||||
 | 
						Severity  Severity
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// RulesConfig defines the config for all rules.
 | 
				
			||||||
 | 
					type RulesConfig = map[string]RuleConfig
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DirectiveConfig is type used for the linter directive configuration.
 | 
				
			||||||
 | 
					type DirectiveConfig struct {
 | 
				
			||||||
 | 
						Severity Severity
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DirectivesConfig defines the config for all directives.
 | 
				
			||||||
 | 
					type DirectivesConfig = map[string]DirectiveConfig
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Config defines the config of the linter.
 | 
				
			||||||
 | 
					type Config struct {
 | 
				
			||||||
 | 
						IgnoreGeneratedHeader bool `toml:"ignoreGeneratedHeader"`
 | 
				
			||||||
 | 
						Confidence            float64
 | 
				
			||||||
 | 
						Severity              Severity
 | 
				
			||||||
 | 
						Rules                 RulesConfig      `toml:"rule"`
 | 
				
			||||||
 | 
						ErrorCode             int              `toml:"errorCode"`
 | 
				
			||||||
 | 
						WarningCode           int              `toml:"warningCode"`
 | 
				
			||||||
 | 
						Directives            DirectivesConfig `toml:"directive"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										39
									
								
								vendor/github.com/mgechev/revive/lint/failure.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								vendor/github.com/mgechev/revive/lint/failure.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						// SeverityWarning declares failures of type warning
 | 
				
			||||||
 | 
						SeverityWarning = "warning"
 | 
				
			||||||
 | 
						// SeverityError declares failures of type error.
 | 
				
			||||||
 | 
						SeverityError = "error"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Severity is the type for the failure types.
 | 
				
			||||||
 | 
					type Severity string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FailurePosition returns the failure position
 | 
				
			||||||
 | 
					type FailurePosition struct {
 | 
				
			||||||
 | 
						Start token.Position
 | 
				
			||||||
 | 
						End   token.Position
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Failure defines a struct for a linting failure.
 | 
				
			||||||
 | 
					type Failure struct {
 | 
				
			||||||
 | 
						Failure    string
 | 
				
			||||||
 | 
						RuleName   string
 | 
				
			||||||
 | 
						Category   string
 | 
				
			||||||
 | 
						Position   FailurePosition
 | 
				
			||||||
 | 
						Node       ast.Node `json:"-"`
 | 
				
			||||||
 | 
						Confidence float64
 | 
				
			||||||
 | 
						// For future use
 | 
				
			||||||
 | 
						ReplacementLine string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GetFilename returns the filename.
 | 
				
			||||||
 | 
					func (f *Failure) GetFilename() string {
 | 
				
			||||||
 | 
						return f.Position.Start.Filename
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										278
									
								
								vendor/github.com/mgechev/revive/lint/file.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										278
									
								
								vendor/github.com/mgechev/revive/lint/file.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,278 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/parser"
 | 
				
			||||||
 | 
						"go/printer"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"go/types"
 | 
				
			||||||
 | 
						"math"
 | 
				
			||||||
 | 
						"regexp"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// File abstraction used for representing files.
 | 
				
			||||||
 | 
					type File struct {
 | 
				
			||||||
 | 
						Name    string
 | 
				
			||||||
 | 
						Pkg     *Package
 | 
				
			||||||
 | 
						content []byte
 | 
				
			||||||
 | 
						AST     *ast.File
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsTest returns if the file contains tests.
 | 
				
			||||||
 | 
					func (f *File) IsTest() bool { return strings.HasSuffix(f.Name, "_test.go") }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Content returns the file's content.
 | 
				
			||||||
 | 
					func (f *File) Content() []byte {
 | 
				
			||||||
 | 
						return f.content
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// NewFile creates a new file
 | 
				
			||||||
 | 
					func NewFile(name string, content []byte, pkg *Package) (*File, error) {
 | 
				
			||||||
 | 
						f, err := parser.ParseFile(pkg.fset, name, content, parser.ParseComments)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return &File{
 | 
				
			||||||
 | 
							Name:    name,
 | 
				
			||||||
 | 
							content: content,
 | 
				
			||||||
 | 
							Pkg:     pkg,
 | 
				
			||||||
 | 
							AST:     f,
 | 
				
			||||||
 | 
						}, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ToPosition returns line and column for given position.
 | 
				
			||||||
 | 
					func (f *File) ToPosition(pos token.Pos) token.Position {
 | 
				
			||||||
 | 
						return f.Pkg.fset.Position(pos)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Render renters a node.
 | 
				
			||||||
 | 
					func (f *File) Render(x interface{}) string {
 | 
				
			||||||
 | 
						var buf bytes.Buffer
 | 
				
			||||||
 | 
						if err := printer.Fprint(&buf, f.Pkg.fset, x); err != nil {
 | 
				
			||||||
 | 
							panic(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return buf.String()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CommentMap builds a comment map for the file.
 | 
				
			||||||
 | 
					func (f *File) CommentMap() ast.CommentMap {
 | 
				
			||||||
 | 
						return ast.NewCommentMap(f.Pkg.fset, f.AST, f.AST.Comments)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var basicTypeKinds = map[types.BasicKind]string{
 | 
				
			||||||
 | 
						types.UntypedBool:    "bool",
 | 
				
			||||||
 | 
						types.UntypedInt:     "int",
 | 
				
			||||||
 | 
						types.UntypedRune:    "rune",
 | 
				
			||||||
 | 
						types.UntypedFloat:   "float64",
 | 
				
			||||||
 | 
						types.UntypedComplex: "complex128",
 | 
				
			||||||
 | 
						types.UntypedString:  "string",
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsUntypedConst reports whether expr is an untyped constant,
 | 
				
			||||||
 | 
					// and indicates what its default type is.
 | 
				
			||||||
 | 
					// scope may be nil.
 | 
				
			||||||
 | 
					func (f *File) IsUntypedConst(expr ast.Expr) (defType string, ok bool) {
 | 
				
			||||||
 | 
						// Re-evaluate expr outside of its context to see if it's untyped.
 | 
				
			||||||
 | 
						// (An expr evaluated within, for example, an assignment context will get the type of the LHS.)
 | 
				
			||||||
 | 
						exprStr := f.Render(expr)
 | 
				
			||||||
 | 
						tv, err := types.Eval(f.Pkg.fset, f.Pkg.TypesPkg, expr.Pos(), exprStr)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return "", false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if b, ok := tv.Type.(*types.Basic); ok {
 | 
				
			||||||
 | 
							if dt, ok := basicTypeKinds[b.Kind()]; ok {
 | 
				
			||||||
 | 
								return dt, true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return "", false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *File) isMain() bool {
 | 
				
			||||||
 | 
						if f.AST.Name.Name == "main" {
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const directiveSpecifyDisableReason = "specify-disable-reason"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *File) lint(rules []Rule, config Config, failures chan Failure) {
 | 
				
			||||||
 | 
						rulesConfig := config.Rules
 | 
				
			||||||
 | 
						_, mustSpecifyDisableReason := config.Directives[directiveSpecifyDisableReason]
 | 
				
			||||||
 | 
						disabledIntervals := f.disabledIntervals(rules, mustSpecifyDisableReason, failures)
 | 
				
			||||||
 | 
						for _, currentRule := range rules {
 | 
				
			||||||
 | 
							ruleConfig := rulesConfig[currentRule.Name()]
 | 
				
			||||||
 | 
							currentFailures := currentRule.Apply(f, ruleConfig.Arguments)
 | 
				
			||||||
 | 
							for idx, failure := range currentFailures {
 | 
				
			||||||
 | 
								if failure.RuleName == "" {
 | 
				
			||||||
 | 
									failure.RuleName = currentRule.Name()
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if failure.Node != nil {
 | 
				
			||||||
 | 
									failure.Position = ToFailurePosition(failure.Node.Pos(), failure.Node.End(), f)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								currentFailures[idx] = failure
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							currentFailures = f.filterFailures(currentFailures, disabledIntervals)
 | 
				
			||||||
 | 
							for _, failure := range currentFailures {
 | 
				
			||||||
 | 
								if failure.Confidence >= config.Confidence {
 | 
				
			||||||
 | 
									failures <- failure
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type enableDisableConfig struct {
 | 
				
			||||||
 | 
						enabled  bool
 | 
				
			||||||
 | 
						position int
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const directiveRE = `^//[\s]*revive:(enable|disable)(?:-(line|next-line))?(?::([^\s]+))?[\s]*(?: (.+))?$`
 | 
				
			||||||
 | 
					const directivePos = 1
 | 
				
			||||||
 | 
					const modifierPos = 2
 | 
				
			||||||
 | 
					const rulesPos = 3
 | 
				
			||||||
 | 
					const reasonPos = 4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var re = regexp.MustCompile(directiveRE)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *File) disabledIntervals(rules []Rule, mustSpecifyDisableReason bool, failures chan Failure) disabledIntervalsMap {
 | 
				
			||||||
 | 
						enabledDisabledRulesMap := make(map[string][]enableDisableConfig)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						getEnabledDisabledIntervals := func() disabledIntervalsMap {
 | 
				
			||||||
 | 
							result := make(disabledIntervalsMap)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							for ruleName, disabledArr := range enabledDisabledRulesMap {
 | 
				
			||||||
 | 
								ruleResult := []DisabledInterval{}
 | 
				
			||||||
 | 
								for i := 0; i < len(disabledArr); i++ {
 | 
				
			||||||
 | 
									interval := DisabledInterval{
 | 
				
			||||||
 | 
										RuleName: ruleName,
 | 
				
			||||||
 | 
										From: token.Position{
 | 
				
			||||||
 | 
											Filename: f.Name,
 | 
				
			||||||
 | 
											Line:     disabledArr[i].position,
 | 
				
			||||||
 | 
										},
 | 
				
			||||||
 | 
										To: token.Position{
 | 
				
			||||||
 | 
											Filename: f.Name,
 | 
				
			||||||
 | 
											Line:     math.MaxInt32,
 | 
				
			||||||
 | 
										},
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									if i%2 == 0 {
 | 
				
			||||||
 | 
										ruleResult = append(ruleResult, interval)
 | 
				
			||||||
 | 
									} else {
 | 
				
			||||||
 | 
										ruleResult[len(ruleResult)-1].To.Line = disabledArr[i].position
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								result[ruleName] = ruleResult
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return result
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						handleConfig := func(isEnabled bool, line int, name string) {
 | 
				
			||||||
 | 
							existing, ok := enabledDisabledRulesMap[name]
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								existing = []enableDisableConfig{}
 | 
				
			||||||
 | 
								enabledDisabledRulesMap[name] = existing
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (len(existing) > 1 && existing[len(existing)-1].enabled == isEnabled) ||
 | 
				
			||||||
 | 
								(len(existing) == 0 && isEnabled) {
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							existing = append(existing, enableDisableConfig{
 | 
				
			||||||
 | 
								enabled:  isEnabled,
 | 
				
			||||||
 | 
								position: line,
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
							enabledDisabledRulesMap[name] = existing
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						handleRules := func(filename, modifier string, isEnabled bool, line int, ruleNames []string) []DisabledInterval {
 | 
				
			||||||
 | 
							var result []DisabledInterval
 | 
				
			||||||
 | 
							for _, name := range ruleNames {
 | 
				
			||||||
 | 
								if modifier == "line" {
 | 
				
			||||||
 | 
									handleConfig(isEnabled, line, name)
 | 
				
			||||||
 | 
									handleConfig(!isEnabled, line, name)
 | 
				
			||||||
 | 
								} else if modifier == "next-line" {
 | 
				
			||||||
 | 
									handleConfig(isEnabled, line+1, name)
 | 
				
			||||||
 | 
									handleConfig(!isEnabled, line+1, name)
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									handleConfig(isEnabled, line, name)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return result
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						handleComment := func(filename string, c *ast.CommentGroup, line int) {
 | 
				
			||||||
 | 
							comments := c.List
 | 
				
			||||||
 | 
							for _, c := range comments {
 | 
				
			||||||
 | 
								match := re.FindStringSubmatch(c.Text)
 | 
				
			||||||
 | 
								if len(match) == 0 {
 | 
				
			||||||
 | 
									return
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								ruleNames := []string{}
 | 
				
			||||||
 | 
								tempNames := strings.Split(match[rulesPos], ",")
 | 
				
			||||||
 | 
								for _, name := range tempNames {
 | 
				
			||||||
 | 
									name = strings.Trim(name, "\n")
 | 
				
			||||||
 | 
									if len(name) > 0 {
 | 
				
			||||||
 | 
										ruleNames = append(ruleNames, name)
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								mustCheckDisablingReason := mustSpecifyDisableReason && match[directivePos] == "disable"
 | 
				
			||||||
 | 
								if mustCheckDisablingReason && strings.Trim(match[reasonPos], " ") == "" {
 | 
				
			||||||
 | 
									failures <- Failure{
 | 
				
			||||||
 | 
										Confidence: 1,
 | 
				
			||||||
 | 
										RuleName:   directiveSpecifyDisableReason,
 | 
				
			||||||
 | 
										Failure:    "reason of lint disabling not found",
 | 
				
			||||||
 | 
										Position:   ToFailurePosition(c.Pos(), c.End(), f),
 | 
				
			||||||
 | 
										Node:       c,
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									continue // skip this linter disabling directive
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								// TODO: optimize
 | 
				
			||||||
 | 
								if len(ruleNames) == 0 {
 | 
				
			||||||
 | 
									for _, rule := range rules {
 | 
				
			||||||
 | 
										ruleNames = append(ruleNames, rule.Name())
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								handleRules(filename, match[modifierPos], match[directivePos] == "enable", line, ruleNames)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						comments := f.AST.Comments
 | 
				
			||||||
 | 
						for _, c := range comments {
 | 
				
			||||||
 | 
							handleComment(f.Name, c, f.ToPosition(c.End()).Line)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return getEnabledDisabledIntervals()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (f *File) filterFailures(failures []Failure, disabledIntervals disabledIntervalsMap) []Failure {
 | 
				
			||||||
 | 
						result := []Failure{}
 | 
				
			||||||
 | 
						for _, failure := range failures {
 | 
				
			||||||
 | 
							fStart := failure.Position.Start.Line
 | 
				
			||||||
 | 
							fEnd := failure.Position.End.Line
 | 
				
			||||||
 | 
							intervals, ok := disabledIntervals[failure.RuleName]
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								result = append(result, failure)
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								include := true
 | 
				
			||||||
 | 
								for _, interval := range intervals {
 | 
				
			||||||
 | 
									intStart := interval.From.Line
 | 
				
			||||||
 | 
									intEnd := interval.To.Line
 | 
				
			||||||
 | 
									if (fStart >= intStart && fStart <= intEnd) ||
 | 
				
			||||||
 | 
										(fEnd >= intStart && fEnd <= intEnd) {
 | 
				
			||||||
 | 
										include = false
 | 
				
			||||||
 | 
										break
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if include {
 | 
				
			||||||
 | 
									result = append(result, failure)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return result
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										14
									
								
								vendor/github.com/mgechev/revive/lint/formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								vendor/github.com/mgechev/revive/lint/formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FormatterMetadata configuration of a formatter
 | 
				
			||||||
 | 
					type FormatterMetadata struct {
 | 
				
			||||||
 | 
						Name        string
 | 
				
			||||||
 | 
						Description string
 | 
				
			||||||
 | 
						Sample      string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Formatter defines an interface for failure formatters
 | 
				
			||||||
 | 
					type Formatter interface {
 | 
				
			||||||
 | 
						Format(<-chan Failure, Config) (string, error)
 | 
				
			||||||
 | 
						Name() string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										99
									
								
								vendor/github.com/mgechev/revive/lint/linter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								vendor/github.com/mgechev/revive/lint/linter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,99 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"bufio"
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
 | 
						"sync"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ReadFile defines an abstraction for reading files.
 | 
				
			||||||
 | 
					type ReadFile func(path string) (result []byte, err error)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type disabledIntervalsMap = map[string][]DisabledInterval
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Linter is used for linting set of files.
 | 
				
			||||||
 | 
					type Linter struct {
 | 
				
			||||||
 | 
						reader ReadFile
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// New creates a new Linter
 | 
				
			||||||
 | 
					func New(reader ReadFile) Linter {
 | 
				
			||||||
 | 
						return Linter{reader: reader}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						genHdr = []byte("// Code generated ")
 | 
				
			||||||
 | 
						genFtr = []byte(" DO NOT EDIT.")
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Lint lints a set of files with the specified rule.
 | 
				
			||||||
 | 
					func (l *Linter) Lint(packages [][]string, ruleSet []Rule, config Config) (<-chan Failure, error) {
 | 
				
			||||||
 | 
						failures := make(chan Failure)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var wg sync.WaitGroup
 | 
				
			||||||
 | 
						for _, pkg := range packages {
 | 
				
			||||||
 | 
							wg.Add(1)
 | 
				
			||||||
 | 
							go func(pkg []string) {
 | 
				
			||||||
 | 
								if err := l.lintPackage(pkg, ruleSet, config, failures); err != nil {
 | 
				
			||||||
 | 
									fmt.Fprintln(os.Stderr, err)
 | 
				
			||||||
 | 
									os.Exit(1)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								defer wg.Done()
 | 
				
			||||||
 | 
							}(pkg)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						go func() {
 | 
				
			||||||
 | 
							wg.Wait()
 | 
				
			||||||
 | 
							close(failures)
 | 
				
			||||||
 | 
						}()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (l *Linter) lintPackage(filenames []string, ruleSet []Rule, config Config, failures chan Failure) error {
 | 
				
			||||||
 | 
						pkg := &Package{
 | 
				
			||||||
 | 
							fset:  token.NewFileSet(),
 | 
				
			||||||
 | 
							files: map[string]*File{},
 | 
				
			||||||
 | 
							mu:    sync.Mutex{},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, filename := range filenames {
 | 
				
			||||||
 | 
							content, err := l.reader(filename)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if isGenerated(content) && !config.IgnoreGeneratedHeader {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							file, err := NewFile(filename, content, pkg)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							pkg.files[filename] = file
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(pkg.files) == 0 {
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pkg.lint(ruleSet, config, failures)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// isGenerated reports whether the source file is generated code
 | 
				
			||||||
 | 
					// according the rules from https://golang.org/s/generatedcode.
 | 
				
			||||||
 | 
					// This is inherited from the original go lint.
 | 
				
			||||||
 | 
					func isGenerated(src []byte) bool {
 | 
				
			||||||
 | 
						sc := bufio.NewScanner(bytes.NewReader(src))
 | 
				
			||||||
 | 
						for sc.Scan() {
 | 
				
			||||||
 | 
							b := sc.Bytes()
 | 
				
			||||||
 | 
							if bytes.HasPrefix(b, genHdr) && bytes.HasSuffix(b, genFtr) && len(b) >= len(genHdr)+len(genFtr) {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										178
									
								
								vendor/github.com/mgechev/revive/lint/package.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								vendor/github.com/mgechev/revive/lint/package.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,178 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"go/types"
 | 
				
			||||||
 | 
						"sync"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"golang.org/x/tools/go/gcexportdata"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Package represents a package in the project.
 | 
				
			||||||
 | 
					type Package struct {
 | 
				
			||||||
 | 
						fset  *token.FileSet
 | 
				
			||||||
 | 
						files map[string]*File
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						TypesPkg  *types.Package
 | 
				
			||||||
 | 
						TypesInfo *types.Info
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// sortable is the set of types in the package that implement sort.Interface.
 | 
				
			||||||
 | 
						Sortable map[string]bool
 | 
				
			||||||
 | 
						// main is whether this is a "main" package.
 | 
				
			||||||
 | 
						main int
 | 
				
			||||||
 | 
						mu   sync.Mutex
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var newImporter = func(fset *token.FileSet) types.ImporterFrom {
 | 
				
			||||||
 | 
						return gcexportdata.NewImporter(fset, make(map[string]*types.Package))
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						trueValue  = 1
 | 
				
			||||||
 | 
						falseValue = 2
 | 
				
			||||||
 | 
						notSet     = 3
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// IsMain returns if that's the main package.
 | 
				
			||||||
 | 
					func (p *Package) IsMain() bool {
 | 
				
			||||||
 | 
						if p.main == trueValue {
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						} else if p.main == falseValue {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, f := range p.files {
 | 
				
			||||||
 | 
							if f.isMain() {
 | 
				
			||||||
 | 
								p.main = trueValue
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						p.main = falseValue
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TypeCheck performs type checking for given package.
 | 
				
			||||||
 | 
					func (p *Package) TypeCheck() error {
 | 
				
			||||||
 | 
						p.mu.Lock()
 | 
				
			||||||
 | 
						// If type checking has already been performed
 | 
				
			||||||
 | 
						// skip it.
 | 
				
			||||||
 | 
						if p.TypesInfo != nil || p.TypesPkg != nil {
 | 
				
			||||||
 | 
							p.mu.Unlock()
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						config := &types.Config{
 | 
				
			||||||
 | 
							// By setting a no-op error reporter, the type checker does as much work as possible.
 | 
				
			||||||
 | 
							Error:    func(error) {},
 | 
				
			||||||
 | 
							Importer: newImporter(p.fset),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						info := &types.Info{
 | 
				
			||||||
 | 
							Types:  make(map[ast.Expr]types.TypeAndValue),
 | 
				
			||||||
 | 
							Defs:   make(map[*ast.Ident]types.Object),
 | 
				
			||||||
 | 
							Uses:   make(map[*ast.Ident]types.Object),
 | 
				
			||||||
 | 
							Scopes: make(map[ast.Node]*types.Scope),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						var anyFile *File
 | 
				
			||||||
 | 
						var astFiles []*ast.File
 | 
				
			||||||
 | 
						for _, f := range p.files {
 | 
				
			||||||
 | 
							anyFile = f
 | 
				
			||||||
 | 
							astFiles = append(astFiles, f.AST)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						typesPkg, err := check(config, anyFile.AST.Name.Name, p.fset, astFiles, info)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Remember the typechecking info, even if config.Check failed,
 | 
				
			||||||
 | 
						// since we will get partial information.
 | 
				
			||||||
 | 
						p.TypesPkg = typesPkg
 | 
				
			||||||
 | 
						p.TypesInfo = info
 | 
				
			||||||
 | 
						p.mu.Unlock()
 | 
				
			||||||
 | 
						return err
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// check function encapsulates the call to go/types.Config.Check method and
 | 
				
			||||||
 | 
					// recovers if the called method panics (see issue #59)
 | 
				
			||||||
 | 
					func check(config *types.Config, n string, fset *token.FileSet, astFiles []*ast.File, info *types.Info) (p *types.Package, err error) {
 | 
				
			||||||
 | 
						defer func() {
 | 
				
			||||||
 | 
							if r := recover(); r != nil {
 | 
				
			||||||
 | 
								err, _ = r.(error)
 | 
				
			||||||
 | 
								p = nil
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return config.Check(n, fset, astFiles, info)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TypeOf returns the type of an expression.
 | 
				
			||||||
 | 
					func (p *Package) TypeOf(expr ast.Expr) types.Type {
 | 
				
			||||||
 | 
						if p.TypesInfo == nil {
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return p.TypesInfo.TypeOf(expr)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type walker struct {
 | 
				
			||||||
 | 
						nmap map[string]int
 | 
				
			||||||
 | 
						has  map[string]int
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *walker) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fn, ok := n.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if !ok || fn.Recv == nil || len(fn.Recv.List) == 0 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// TODO(dsymonds): We could check the signature to be more precise.
 | 
				
			||||||
 | 
						recv := receiverType(fn)
 | 
				
			||||||
 | 
						if i, ok := w.nmap[fn.Name.Name]; ok {
 | 
				
			||||||
 | 
							w.has[recv] |= i
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (p *Package) scanSortable() {
 | 
				
			||||||
 | 
						p.Sortable = make(map[string]bool)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// bitfield for which methods exist on each type.
 | 
				
			||||||
 | 
						const (
 | 
				
			||||||
 | 
							Len = 1 << iota
 | 
				
			||||||
 | 
							Less
 | 
				
			||||||
 | 
							Swap
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						nmap := map[string]int{"Len": Len, "Less": Less, "Swap": Swap}
 | 
				
			||||||
 | 
						has := make(map[string]int)
 | 
				
			||||||
 | 
						for _, f := range p.files {
 | 
				
			||||||
 | 
							ast.Walk(&walker{nmap, has}, f.AST)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for typ, ms := range has {
 | 
				
			||||||
 | 
							if ms == Len|Less|Swap {
 | 
				
			||||||
 | 
								p.Sortable[typ] = true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// receiverType returns the named type of the method receiver, sans "*",
 | 
				
			||||||
 | 
					// or "invalid-type" if fn.Recv is ill formed.
 | 
				
			||||||
 | 
					func receiverType(fn *ast.FuncDecl) string {
 | 
				
			||||||
 | 
						switch e := fn.Recv.List[0].Type.(type) {
 | 
				
			||||||
 | 
						case *ast.Ident:
 | 
				
			||||||
 | 
							return e.Name
 | 
				
			||||||
 | 
						case *ast.StarExpr:
 | 
				
			||||||
 | 
							if id, ok := e.X.(*ast.Ident); ok {
 | 
				
			||||||
 | 
								return id.Name
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// The parser accepts much more than just the legal forms.
 | 
				
			||||||
 | 
						return "invalid-type"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (p *Package) lint(rules []Rule, config Config, failures chan Failure) {
 | 
				
			||||||
 | 
						p.scanSortable()
 | 
				
			||||||
 | 
						var wg sync.WaitGroup
 | 
				
			||||||
 | 
						for _, file := range p.files {
 | 
				
			||||||
 | 
							wg.Add(1)
 | 
				
			||||||
 | 
							go (func(file *File) {
 | 
				
			||||||
 | 
								file.lint(rules, config, failures)
 | 
				
			||||||
 | 
								defer wg.Done()
 | 
				
			||||||
 | 
							})(file)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						wg.Wait()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										31
									
								
								vendor/github.com/mgechev/revive/lint/rule.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								vendor/github.com/mgechev/revive/lint/rule.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,31 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DisabledInterval contains a single disabled interval and the associated rule name.
 | 
				
			||||||
 | 
					type DisabledInterval struct {
 | 
				
			||||||
 | 
						From     token.Position
 | 
				
			||||||
 | 
						To       token.Position
 | 
				
			||||||
 | 
						RuleName string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Rule defines an abstract rule interaface
 | 
				
			||||||
 | 
					type Rule interface {
 | 
				
			||||||
 | 
						Name() string
 | 
				
			||||||
 | 
						Apply(*File, Arguments) []Failure
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// AbstractRule defines an abstract rule.
 | 
				
			||||||
 | 
					type AbstractRule struct {
 | 
				
			||||||
 | 
						Failures []Failure
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ToFailurePosition returns the failure position.
 | 
				
			||||||
 | 
					func ToFailurePosition(start token.Pos, end token.Pos, file *File) FailurePosition {
 | 
				
			||||||
 | 
						return FailurePosition{
 | 
				
			||||||
 | 
							Start: file.ToPosition(start),
 | 
				
			||||||
 | 
							End:   file.ToPosition(end),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										128
									
								
								vendor/github.com/mgechev/revive/lint/utils.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								vendor/github.com/mgechev/revive/lint/utils.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,128 @@
 | 
				
			|||||||
 | 
					package lint
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
						"unicode"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns a different name if it should be different.
 | 
				
			||||||
 | 
					func Name(name string, whitelist, blacklist []string) (should string) {
 | 
				
			||||||
 | 
						// Fast path for simple cases: "_" and all lowercase.
 | 
				
			||||||
 | 
						if name == "_" {
 | 
				
			||||||
 | 
							return name
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						allLower := true
 | 
				
			||||||
 | 
						for _, r := range name {
 | 
				
			||||||
 | 
							if !unicode.IsLower(r) {
 | 
				
			||||||
 | 
								allLower = false
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if allLower {
 | 
				
			||||||
 | 
							return name
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Split camelCase at any lower->upper transition, and split on underscores.
 | 
				
			||||||
 | 
						// Check each word for common initialisms.
 | 
				
			||||||
 | 
						runes := []rune(name)
 | 
				
			||||||
 | 
						w, i := 0, 0 // index of start of word, scan
 | 
				
			||||||
 | 
						for i+1 <= len(runes) {
 | 
				
			||||||
 | 
							eow := false // whether we hit the end of a word
 | 
				
			||||||
 | 
							if i+1 == len(runes) {
 | 
				
			||||||
 | 
								eow = true
 | 
				
			||||||
 | 
							} else if runes[i+1] == '_' {
 | 
				
			||||||
 | 
								// underscore; shift the remainder forward over any run of underscores
 | 
				
			||||||
 | 
								eow = true
 | 
				
			||||||
 | 
								n := 1
 | 
				
			||||||
 | 
								for i+n+1 < len(runes) && runes[i+n+1] == '_' {
 | 
				
			||||||
 | 
									n++
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								// Leave at most one underscore if the underscore is between two digits
 | 
				
			||||||
 | 
								if i+n+1 < len(runes) && unicode.IsDigit(runes[i]) && unicode.IsDigit(runes[i+n+1]) {
 | 
				
			||||||
 | 
									n--
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								copy(runes[i+1:], runes[i+n+1:])
 | 
				
			||||||
 | 
								runes = runes[:len(runes)-n]
 | 
				
			||||||
 | 
							} else if unicode.IsLower(runes[i]) && !unicode.IsLower(runes[i+1]) {
 | 
				
			||||||
 | 
								// lower->non-lower
 | 
				
			||||||
 | 
								eow = true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							i++
 | 
				
			||||||
 | 
							if !eow {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// [w,i) is a word.
 | 
				
			||||||
 | 
							word := string(runes[w:i])
 | 
				
			||||||
 | 
							ignoreInitWarnings := map[string]bool{}
 | 
				
			||||||
 | 
							for _, i := range whitelist {
 | 
				
			||||||
 | 
								ignoreInitWarnings[i] = true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							extraInits := map[string]bool{}
 | 
				
			||||||
 | 
							for _, i := range blacklist {
 | 
				
			||||||
 | 
								extraInits[i] = true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if u := strings.ToUpper(word); (commonInitialisms[u] || extraInits[u]) && !ignoreInitWarnings[u] {
 | 
				
			||||||
 | 
								// Keep consistent case, which is lowercase only at the start.
 | 
				
			||||||
 | 
								if w == 0 && unicode.IsLower(runes[w]) {
 | 
				
			||||||
 | 
									u = strings.ToLower(u)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								// All the common initialisms are ASCII,
 | 
				
			||||||
 | 
								// so we can replace the bytes exactly.
 | 
				
			||||||
 | 
								copy(runes[w:], []rune(u))
 | 
				
			||||||
 | 
							} else if w > 0 && strings.ToLower(word) == word {
 | 
				
			||||||
 | 
								// already all lowercase, and not the first word, so uppercase the first character.
 | 
				
			||||||
 | 
								runes[w] = unicode.ToUpper(runes[w])
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							w = i
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return string(runes)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// commonInitialisms is a set of common initialisms.
 | 
				
			||||||
 | 
					// Only add entries that are highly unlikely to be non-initialisms.
 | 
				
			||||||
 | 
					// For instance, "ID" is fine (Freudian code is rare), but "AND" is not.
 | 
				
			||||||
 | 
					var commonInitialisms = map[string]bool{
 | 
				
			||||||
 | 
						"ACL":   true,
 | 
				
			||||||
 | 
						"API":   true,
 | 
				
			||||||
 | 
						"ASCII": true,
 | 
				
			||||||
 | 
						"CPU":   true,
 | 
				
			||||||
 | 
						"CSS":   true,
 | 
				
			||||||
 | 
						"DNS":   true,
 | 
				
			||||||
 | 
						"EOF":   true,
 | 
				
			||||||
 | 
						"GUID":  true,
 | 
				
			||||||
 | 
						"HTML":  true,
 | 
				
			||||||
 | 
						"HTTP":  true,
 | 
				
			||||||
 | 
						"HTTPS": true,
 | 
				
			||||||
 | 
						"ID":    true,
 | 
				
			||||||
 | 
						"IP":    true,
 | 
				
			||||||
 | 
						"JSON":  true,
 | 
				
			||||||
 | 
						"LHS":   true,
 | 
				
			||||||
 | 
						"QPS":   true,
 | 
				
			||||||
 | 
						"RAM":   true,
 | 
				
			||||||
 | 
						"RHS":   true,
 | 
				
			||||||
 | 
						"RPC":   true,
 | 
				
			||||||
 | 
						"SLA":   true,
 | 
				
			||||||
 | 
						"SMTP":  true,
 | 
				
			||||||
 | 
						"SQL":   true,
 | 
				
			||||||
 | 
						"SSH":   true,
 | 
				
			||||||
 | 
						"TCP":   true,
 | 
				
			||||||
 | 
						"TLS":   true,
 | 
				
			||||||
 | 
						"TTL":   true,
 | 
				
			||||||
 | 
						"UDP":   true,
 | 
				
			||||||
 | 
						"UI":    true,
 | 
				
			||||||
 | 
						"UID":   true,
 | 
				
			||||||
 | 
						"UUID":  true,
 | 
				
			||||||
 | 
						"URI":   true,
 | 
				
			||||||
 | 
						"URL":   true,
 | 
				
			||||||
 | 
						"UTF8":  true,
 | 
				
			||||||
 | 
						"VM":    true,
 | 
				
			||||||
 | 
						"XML":   true,
 | 
				
			||||||
 | 
						"XMPP":  true,
 | 
				
			||||||
 | 
						"XSRF":  true,
 | 
				
			||||||
 | 
						"XSS":   true,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										151
									
								
								vendor/github.com/mgechev/revive/rule/add-constant.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								vendor/github.com/mgechev/revive/rule/add-constant.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,151 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						defaultStrLitLimit = 2
 | 
				
			||||||
 | 
						kindFLOAT          = "FLOAT"
 | 
				
			||||||
 | 
						kindINT            = "INT"
 | 
				
			||||||
 | 
						kindSTRING         = "STRING"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type whiteList map[string]map[string]bool
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func newWhiteList() whiteList {
 | 
				
			||||||
 | 
						return map[string]map[string]bool{kindINT: map[string]bool{}, kindFLOAT: map[string]bool{}, kindSTRING: map[string]bool{}}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (wl whiteList) add(kind string, list string) {
 | 
				
			||||||
 | 
						elems := strings.Split(list, ",")
 | 
				
			||||||
 | 
						for _, e := range elems {
 | 
				
			||||||
 | 
							wl[kind][e] = true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// AddConstantRule lints unused params in functions.
 | 
				
			||||||
 | 
					type AddConstantRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *AddConstantRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						strLitLimit := defaultStrLitLimit
 | 
				
			||||||
 | 
						var whiteList = newWhiteList()
 | 
				
			||||||
 | 
						if len(arguments) > 0 {
 | 
				
			||||||
 | 
							args, ok := arguments[0].(map[string]interface{})
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								panic(fmt.Sprintf("Invalid argument to the add-constant rule. Expecting a k,v map, got %T", arguments[0]))
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							for k, v := range args {
 | 
				
			||||||
 | 
								kind := ""
 | 
				
			||||||
 | 
								switch k {
 | 
				
			||||||
 | 
								case "allowFloats":
 | 
				
			||||||
 | 
									kind = kindFLOAT
 | 
				
			||||||
 | 
									fallthrough
 | 
				
			||||||
 | 
								case "allowInts":
 | 
				
			||||||
 | 
									if kind == "" {
 | 
				
			||||||
 | 
										kind = kindINT
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									fallthrough
 | 
				
			||||||
 | 
								case "allowStrs":
 | 
				
			||||||
 | 
									if kind == "" {
 | 
				
			||||||
 | 
										kind = kindSTRING
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									list, ok := v.(string)
 | 
				
			||||||
 | 
									if !ok {
 | 
				
			||||||
 | 
										panic(fmt.Sprintf("Invalid argument to the add-constant rule, string expected. Got '%v' (%T)", v, v))
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									whiteList.add(kind, list)
 | 
				
			||||||
 | 
								case "maxLitCount":
 | 
				
			||||||
 | 
									sl, ok := v.(string)
 | 
				
			||||||
 | 
									if !ok {
 | 
				
			||||||
 | 
										panic(fmt.Sprintf("Invalid argument to the add-constant rule, expecting string representation of an integer. Got '%v' (%T)", v, v))
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									limit, err := strconv.Atoi(sl)
 | 
				
			||||||
 | 
									if err != nil {
 | 
				
			||||||
 | 
										panic(fmt.Sprintf("Invalid argument to the add-constant rule, expecting string representation of an integer. Got '%v'", v))
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									strLitLimit = limit
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintAddConstantRule{onFailure: onFailure, strLits: make(map[string]int, 0), strLitLimit: strLitLimit, whiteLst: whiteList}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *AddConstantRule) Name() string {
 | 
				
			||||||
 | 
						return "add-constant"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintAddConstantRule struct {
 | 
				
			||||||
 | 
						onFailure   func(lint.Failure)
 | 
				
			||||||
 | 
						strLits     map[string]int
 | 
				
			||||||
 | 
						strLitLimit int
 | 
				
			||||||
 | 
						whiteLst    whiteList
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintAddConstantRule) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := node.(type) {
 | 
				
			||||||
 | 
						case *ast.GenDecl:
 | 
				
			||||||
 | 
							return nil // skip declarations
 | 
				
			||||||
 | 
						case *ast.BasicLit:
 | 
				
			||||||
 | 
							switch kind := n.Kind.String(); kind {
 | 
				
			||||||
 | 
							case kindFLOAT, kindINT:
 | 
				
			||||||
 | 
								w.checkNumLit(kind, n)
 | 
				
			||||||
 | 
							case kindSTRING:
 | 
				
			||||||
 | 
								w.checkStrLit(n)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintAddConstantRule) checkStrLit(n *ast.BasicLit) {
 | 
				
			||||||
 | 
						if w.whiteLst[kindSTRING][n.Value] {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						count := w.strLits[n.Value]
 | 
				
			||||||
 | 
						if count >= 0 {
 | 
				
			||||||
 | 
							w.strLits[n.Value] = count + 1
 | 
				
			||||||
 | 
							if w.strLits[n.Value] > w.strLitLimit {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Node:       n,
 | 
				
			||||||
 | 
									Category:   "style",
 | 
				
			||||||
 | 
									Failure:    fmt.Sprintf("string literal %s appears, at least, %d times, create a named constant for it", n.Value, w.strLits[n.Value]),
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								w.strLits[n.Value] = -1 // mark it to avoid failing again on the same literal
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintAddConstantRule) checkNumLit(kind string, n *ast.BasicLit) {
 | 
				
			||||||
 | 
						if w.whiteLst[kind][n.Value] {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Node:       n,
 | 
				
			||||||
 | 
							Category:   "style",
 | 
				
			||||||
 | 
							Failure:    fmt.Sprintf("avoid magic numbers like '%s', create a named constant for it", n.Value),
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										67
									
								
								vendor/github.com/mgechev/revive/rule/argument-limit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								vendor/github.com/mgechev/revive/rule/argument-limit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,67 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ArgumentsLimitRule lints given else constructs.
 | 
				
			||||||
 | 
					type ArgumentsLimitRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ArgumentsLimitRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						if len(arguments) != 1 {
 | 
				
			||||||
 | 
							panic(`invalid configuration for "argument-limit"`)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						total, ok := arguments[0].(int64) // Alt. non panicking version
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							panic(`invalid value passed as argument number to the "argument-list" rule`)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						walker := lintArgsNum{
 | 
				
			||||||
 | 
							total: int(total),
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, file.AST)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ArgumentsLimitRule) Name() string {
 | 
				
			||||||
 | 
						return "argument-limit"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintArgsNum struct {
 | 
				
			||||||
 | 
						total     int
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintArgsNum) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						node, ok := n.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if ok {
 | 
				
			||||||
 | 
							num := 0
 | 
				
			||||||
 | 
							for _, l := range node.Type.Params.List {
 | 
				
			||||||
 | 
								for range l.Names {
 | 
				
			||||||
 | 
									num++
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if num > w.total {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Failure:    fmt.Sprintf("maximum number of arguments per function exceeded; max %d but got %d", w.total, num),
 | 
				
			||||||
 | 
									Node:       node.Type,
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										94
									
								
								vendor/github.com/mgechev/revive/rule/atomic.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								vendor/github.com/mgechev/revive/rule/atomic.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,94 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"go/types"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// AtomicRule lints given else constructs.
 | 
				
			||||||
 | 
					type AtomicRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *AtomicRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
						walker := atomic{
 | 
				
			||||||
 | 
							pkgTypesInfo: file.Pkg.TypesInfo,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, file.AST)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *AtomicRule) Name() string {
 | 
				
			||||||
 | 
						return "atomic"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type atomic struct {
 | 
				
			||||||
 | 
						pkgTypesInfo *types.Info
 | 
				
			||||||
 | 
						onFailure    func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w atomic) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						n, ok := node.(*ast.AssignStmt)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(n.Lhs) != len(n.Rhs) {
 | 
				
			||||||
 | 
							return nil // skip assignment sub-tree
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if len(n.Lhs) == 1 && n.Tok == token.DEFINE {
 | 
				
			||||||
 | 
							return nil // skip assignment sub-tree
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for i, right := range n.Rhs {
 | 
				
			||||||
 | 
							call, ok := right.(*ast.CallExpr)
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							sel, ok := call.Fun.(*ast.SelectorExpr)
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							pkgIdent, _ := sel.X.(*ast.Ident)
 | 
				
			||||||
 | 
							if w.pkgTypesInfo != nil {
 | 
				
			||||||
 | 
								pkgName, ok := w.pkgTypesInfo.Uses[pkgIdent].(*types.PkgName)
 | 
				
			||||||
 | 
								if !ok || pkgName.Imported().Path() != "sync/atomic" {
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							switch sel.Sel.Name {
 | 
				
			||||||
 | 
							case "AddInt32", "AddInt64", "AddUint32", "AddUint64", "AddUintptr":
 | 
				
			||||||
 | 
								left := n.Lhs[i]
 | 
				
			||||||
 | 
								if len(call.Args) != 2 {
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								arg := call.Args[0]
 | 
				
			||||||
 | 
								broken := false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								if uarg, ok := arg.(*ast.UnaryExpr); ok && uarg.Op == token.AND {
 | 
				
			||||||
 | 
									broken = gofmt(left) == gofmt(uarg.X)
 | 
				
			||||||
 | 
								} else if star, ok := left.(*ast.StarExpr); ok {
 | 
				
			||||||
 | 
									broken = gofmt(star.X) == gofmt(arg)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								if broken {
 | 
				
			||||||
 | 
									w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
										Confidence: 1,
 | 
				
			||||||
 | 
										Failure:    "direct assignment to atomic value",
 | 
				
			||||||
 | 
										Node:       n,
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										84
									
								
								vendor/github.com/mgechev/revive/rule/bare-return.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								vendor/github.com/mgechev/revive/rule/bare-return.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,84 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// BareReturnRule lints given else constructs.
 | 
				
			||||||
 | 
					type BareReturnRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *BareReturnRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintBareReturnRule{onFailure: onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *BareReturnRule) Name() string {
 | 
				
			||||||
 | 
						return "bare-return"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintBareReturnRule struct {
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintBareReturnRule) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := node.(type) {
 | 
				
			||||||
 | 
						case *ast.FuncDecl:
 | 
				
			||||||
 | 
							w.checkFunc(n.Type.Results, n.Body)
 | 
				
			||||||
 | 
						case *ast.FuncLit: // to cope with deferred functions and go-routines
 | 
				
			||||||
 | 
							w.checkFunc(n.Type.Results, n.Body)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// checkFunc will verify if the given function has named result and bare returns
 | 
				
			||||||
 | 
					func (w lintBareReturnRule) checkFunc(results *ast.FieldList, body *ast.BlockStmt) {
 | 
				
			||||||
 | 
						hasNamedResults := results != nil && len(results.List) > 0 && results.List[0].Names != nil
 | 
				
			||||||
 | 
						if !hasNamedResults || body == nil {
 | 
				
			||||||
 | 
							return // nothing to do
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						brf := bareReturnFinder{w.onFailure}
 | 
				
			||||||
 | 
						ast.Walk(brf, body)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type bareReturnFinder struct {
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w bareReturnFinder) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						_, ok := node.(*ast.FuncLit)
 | 
				
			||||||
 | 
						if ok {
 | 
				
			||||||
 | 
							// skip analysing function literals
 | 
				
			||||||
 | 
							// they will analyzed by the lintBareReturnRule.Visit method
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						rs, ok := node.(*ast.ReturnStmt)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(rs.Results) > 0 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Node:       rs,
 | 
				
			||||||
 | 
							Failure:    "avoid using bare returns, please add return expressions",
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										74
									
								
								vendor/github.com/mgechev/revive/rule/blank-imports.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								vendor/github.com/mgechev/revive/rule/blank-imports.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,74 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// BlankImportsRule lints given else constructs.
 | 
				
			||||||
 | 
					type BlankImportsRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *BlankImportsRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintBlankImports{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *BlankImportsRule) Name() string {
 | 
				
			||||||
 | 
						return "blank-imports"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintBlankImports struct {
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintBlankImports) Visit(_ ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						// In package main and in tests, we don't complain about blank imports.
 | 
				
			||||||
 | 
						if w.file.Pkg.IsMain() || w.file.IsTest() {
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// The first element of each contiguous group of blank imports should have
 | 
				
			||||||
 | 
						// an explanatory comment of some kind.
 | 
				
			||||||
 | 
						for i, imp := range w.fileAst.Imports {
 | 
				
			||||||
 | 
							pos := w.file.ToPosition(imp.Pos())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if !isBlank(imp.Name) {
 | 
				
			||||||
 | 
								continue // Ignore non-blank imports.
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if i > 0 {
 | 
				
			||||||
 | 
								prev := w.fileAst.Imports[i-1]
 | 
				
			||||||
 | 
								prevPos := w.file.ToPosition(prev.Pos())
 | 
				
			||||||
 | 
								if isBlank(prev.Name) && prevPos.Line+1 == pos.Line {
 | 
				
			||||||
 | 
									continue // A subsequent blank in a group.
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// This is the first blank import of a group.
 | 
				
			||||||
 | 
							if imp.Doc == nil && imp.Comment == nil {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Node:       imp,
 | 
				
			||||||
 | 
									Failure:    "a blank import should be only in a main or test package, or have a comment justifying it",
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Category:   "imports",
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										73
									
								
								vendor/github.com/mgechev/revive/rule/bool-literal-in-expr.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								vendor/github.com/mgechev/revive/rule/bool-literal-in-expr.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,73 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// BoolLiteralRule warns when logic expressions contains Boolean literals.
 | 
				
			||||||
 | 
					type BoolLiteralRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *BoolLiteralRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						astFile := file.AST
 | 
				
			||||||
 | 
						w := &lintBoolLiteral{astFile, onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, astFile)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *BoolLiteralRule) Name() string {
 | 
				
			||||||
 | 
						return "bool-literal-in-expr"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintBoolLiteral struct {
 | 
				
			||||||
 | 
						file      *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintBoolLiteral) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := node.(type) {
 | 
				
			||||||
 | 
						case *ast.BinaryExpr:
 | 
				
			||||||
 | 
							if !isBoolOp(n.Op) {
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							lexeme, ok := isExprABooleanLit(n.X)
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								lexeme, ok = isExprABooleanLit(n.Y)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								if !ok {
 | 
				
			||||||
 | 
									return w
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							isConstant := (n.Op == token.LAND && lexeme == "false") || (n.Op == token.LOR && lexeme == "true")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if isConstant {
 | 
				
			||||||
 | 
								w.addFailure(n, "Boolean expression seems to always evaluate to "+lexeme, "logic")
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								w.addFailure(n, "omit Boolean literal in expression", "style")
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintBoolLiteral) addFailure(node ast.Node, msg string, cat string) {
 | 
				
			||||||
 | 
						w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Node:       node,
 | 
				
			||||||
 | 
							Category:   cat,
 | 
				
			||||||
 | 
							Failure:    msg,
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										70
									
								
								vendor/github.com/mgechev/revive/rule/call-to-gc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								vendor/github.com/mgechev/revive/rule/call-to-gc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,70 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CallToGCRule lints calls to the garbage collector.
 | 
				
			||||||
 | 
					type CallToGCRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *CallToGCRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var gcTriggeringFunctions = map[string]map[string]bool{
 | 
				
			||||||
 | 
							"runtime": map[string]bool{"GC": true},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintCallToGC{onFailure, gcTriggeringFunctions}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *CallToGCRule) Name() string {
 | 
				
			||||||
 | 
						return "call-to-gc"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintCallToGC struct {
 | 
				
			||||||
 | 
						onFailure             func(lint.Failure)
 | 
				
			||||||
 | 
						gcTriggeringFunctions map[string]map[string]bool
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintCallToGC) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						ce, ok := node.(*ast.CallExpr)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w // nothing to do, the node is not a call
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fc, ok := ce.Fun.(*ast.SelectorExpr)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return nil // nothing to do, the call is not of the form pkg.func(...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						id, ok := fc.X.(*ast.Ident)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return nil // in case X is not an id (it should be!)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fn := fc.Sel.Name
 | 
				
			||||||
 | 
						pkg := id.Name
 | 
				
			||||||
 | 
						if !w.gcTriggeringFunctions[pkg][fn] {
 | 
				
			||||||
 | 
							return nil // it isn't a call to a GC triggering function
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Node:       node,
 | 
				
			||||||
 | 
							Category:   "bad practice",
 | 
				
			||||||
 | 
							Failure:    "explicit call to the garbage collector",
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										195
									
								
								vendor/github.com/mgechev/revive/rule/cognitive-complexity.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										195
									
								
								vendor/github.com/mgechev/revive/rule/cognitive-complexity.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,195 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"golang.org/x/tools/go/ast/astutil"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CognitiveComplexityRule lints given else constructs.
 | 
				
			||||||
 | 
					type CognitiveComplexityRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *CognitiveComplexityRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						const expectedArgumentsCount = 1
 | 
				
			||||||
 | 
						if len(arguments) < expectedArgumentsCount {
 | 
				
			||||||
 | 
							panic(fmt.Sprintf("not enough arguments for cognitive-complexity, expected %d, got %d", expectedArgumentsCount, len(arguments)))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						complexity, ok := arguments[0].(int64)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							panic(fmt.Sprintf("invalid argument type for cognitive-complexity, expected int64, got %T", arguments[0]))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						linter := cognitiveComplexityLinter{
 | 
				
			||||||
 | 
							file:          file,
 | 
				
			||||||
 | 
							maxComplexity: int(complexity),
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						linter.lint()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *CognitiveComplexityRule) Name() string {
 | 
				
			||||||
 | 
						return "cognitive-complexity"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type cognitiveComplexityLinter struct {
 | 
				
			||||||
 | 
						file          *lint.File
 | 
				
			||||||
 | 
						maxComplexity int
 | 
				
			||||||
 | 
						onFailure     func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w cognitiveComplexityLinter) lint() {
 | 
				
			||||||
 | 
						f := w.file
 | 
				
			||||||
 | 
						for _, decl := range f.AST.Decls {
 | 
				
			||||||
 | 
							if fn, ok := decl.(*ast.FuncDecl); ok {
 | 
				
			||||||
 | 
								v := cognitiveComplexityVisitor{}
 | 
				
			||||||
 | 
								c := v.subTreeComplexity(fn.Body)
 | 
				
			||||||
 | 
								if c > w.maxComplexity {
 | 
				
			||||||
 | 
									w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
										Confidence: 1,
 | 
				
			||||||
 | 
										Category:   "maintenance",
 | 
				
			||||||
 | 
										Failure:    fmt.Sprintf("function %s has cognitive complexity %d (> max enabled %d)", funcName(fn), c, w.maxComplexity),
 | 
				
			||||||
 | 
										Node:       fn,
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type cognitiveComplexityVisitor struct {
 | 
				
			||||||
 | 
						complexity   int
 | 
				
			||||||
 | 
						nestingLevel int
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// subTreeComplexity calculates the cognitive complexity of an AST-subtree.
 | 
				
			||||||
 | 
					func (v cognitiveComplexityVisitor) subTreeComplexity(n ast.Node) int {
 | 
				
			||||||
 | 
						ast.Walk(&v, n)
 | 
				
			||||||
 | 
						return v.complexity
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Visit implements the ast.Visitor interface.
 | 
				
			||||||
 | 
					func (v *cognitiveComplexityVisitor) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := n.(type) {
 | 
				
			||||||
 | 
						case *ast.IfStmt:
 | 
				
			||||||
 | 
							targets := []ast.Node{n.Cond, n.Body, n.Else}
 | 
				
			||||||
 | 
							v.walk(1, targets...)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.ForStmt:
 | 
				
			||||||
 | 
							targets := []ast.Node{n.Cond, n.Body}
 | 
				
			||||||
 | 
							v.walk(1, targets...)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.RangeStmt:
 | 
				
			||||||
 | 
							v.walk(1, n.Body)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.SelectStmt:
 | 
				
			||||||
 | 
							v.walk(1, n.Body)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.SwitchStmt:
 | 
				
			||||||
 | 
							v.walk(1, n.Body)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.TypeSwitchStmt:
 | 
				
			||||||
 | 
							v.walk(1, n.Body)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.FuncLit:
 | 
				
			||||||
 | 
							v.walk(0, n.Body) // do not increment the complexity, just do the nesting
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.BinaryExpr:
 | 
				
			||||||
 | 
							v.complexity += v.binExpComplexity(n)
 | 
				
			||||||
 | 
							return nil // skip visiting binexp sub-tree (already visited by binExpComplexity)
 | 
				
			||||||
 | 
						case *ast.BranchStmt:
 | 
				
			||||||
 | 
							if n.Label != nil {
 | 
				
			||||||
 | 
								v.complexity += 1
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// TODO handle (at least) direct recursion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return v
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (v *cognitiveComplexityVisitor) walk(complexityIncrement int, targets ...ast.Node) {
 | 
				
			||||||
 | 
						v.complexity += complexityIncrement + v.nestingLevel
 | 
				
			||||||
 | 
						nesting := v.nestingLevel
 | 
				
			||||||
 | 
						v.nestingLevel++
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, t := range targets {
 | 
				
			||||||
 | 
							if t == nil {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ast.Walk(v, t)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						v.nestingLevel = nesting
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (cognitiveComplexityVisitor) binExpComplexity(n *ast.BinaryExpr) int {
 | 
				
			||||||
 | 
						calculator := binExprComplexityCalculator{opsStack: []token.Token{}}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						astutil.Apply(n, calculator.pre, calculator.post)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return calculator.complexity
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type binExprComplexityCalculator struct {
 | 
				
			||||||
 | 
						complexity    int
 | 
				
			||||||
 | 
						opsStack      []token.Token // stack of bool operators
 | 
				
			||||||
 | 
						subexpStarted bool
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (becc *binExprComplexityCalculator) pre(c *astutil.Cursor) bool {
 | 
				
			||||||
 | 
						switch n := c.Node().(type) {
 | 
				
			||||||
 | 
						case *ast.BinaryExpr:
 | 
				
			||||||
 | 
							isBoolOp := n.Op == token.LAND || n.Op == token.LOR
 | 
				
			||||||
 | 
							if !isBoolOp {
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ops := len(becc.opsStack)
 | 
				
			||||||
 | 
							// if
 | 
				
			||||||
 | 
							// 		is the first boolop in the expression OR
 | 
				
			||||||
 | 
							// 		is the first boolop inside a subexpression (...) OR
 | 
				
			||||||
 | 
							//		is not the same to the previous one
 | 
				
			||||||
 | 
							// then
 | 
				
			||||||
 | 
							//      increment complexity
 | 
				
			||||||
 | 
							if ops == 0 || becc.subexpStarted || n.Op != becc.opsStack[ops-1] {
 | 
				
			||||||
 | 
								becc.complexity++
 | 
				
			||||||
 | 
								becc.subexpStarted = false
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							becc.opsStack = append(becc.opsStack, n.Op)
 | 
				
			||||||
 | 
						case *ast.ParenExpr:
 | 
				
			||||||
 | 
							becc.subexpStarted = true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return true
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (becc *binExprComplexityCalculator) post(c *astutil.Cursor) bool {
 | 
				
			||||||
 | 
						switch n := c.Node().(type) {
 | 
				
			||||||
 | 
						case *ast.BinaryExpr:
 | 
				
			||||||
 | 
							isBoolOp := n.Op == token.LAND || n.Op == token.LOR
 | 
				
			||||||
 | 
							if !isBoolOp {
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ops := len(becc.opsStack)
 | 
				
			||||||
 | 
							if ops > 0 {
 | 
				
			||||||
 | 
								becc.opsStack = becc.opsStack[:ops-1]
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						case *ast.ParenExpr:
 | 
				
			||||||
 | 
							becc.subexpStarted = false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return true
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										190
									
								
								vendor/github.com/mgechev/revive/rule/confusing-naming.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										190
									
								
								vendor/github.com/mgechev/revive/rule/confusing-naming.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,190 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
						"sync"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type referenceMethod struct {
 | 
				
			||||||
 | 
						fileName string
 | 
				
			||||||
 | 
						id       *ast.Ident
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type pkgMethods struct {
 | 
				
			||||||
 | 
						pkg     *lint.Package
 | 
				
			||||||
 | 
						methods map[string]map[string]*referenceMethod
 | 
				
			||||||
 | 
						mu      *sync.Mutex
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type packages struct {
 | 
				
			||||||
 | 
						pkgs []pkgMethods
 | 
				
			||||||
 | 
						mu   sync.Mutex
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (ps *packages) methodNames(lp *lint.Package) pkgMethods {
 | 
				
			||||||
 | 
						ps.mu.Lock()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, pkg := range ps.pkgs {
 | 
				
			||||||
 | 
							if pkg.pkg == lp {
 | 
				
			||||||
 | 
								ps.mu.Unlock()
 | 
				
			||||||
 | 
								return pkg
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pkgm := pkgMethods{pkg: lp, methods: make(map[string]map[string]*referenceMethod), mu: &sync.Mutex{}}
 | 
				
			||||||
 | 
						ps.pkgs = append(ps.pkgs, pkgm)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ps.mu.Unlock()
 | 
				
			||||||
 | 
						return pkgm
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var allPkgs = packages{pkgs: make([]pkgMethods, 1)}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ConfusingNamingRule lints method names that differ only by capitalization
 | 
				
			||||||
 | 
					type ConfusingNamingRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ConfusingNamingRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						pkgm := allPkgs.methodNames(file.Pkg)
 | 
				
			||||||
 | 
						walker := lintConfusingNames{
 | 
				
			||||||
 | 
							fileName: file.Name,
 | 
				
			||||||
 | 
							pkgm:     pkgm,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(&walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ConfusingNamingRule) Name() string {
 | 
				
			||||||
 | 
						return "confusing-naming"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//checkMethodName checks if a given method/function name is similar (just case differences) to other method/function of the same struct/file.
 | 
				
			||||||
 | 
					func checkMethodName(holder string, id *ast.Ident, w *lintConfusingNames) {
 | 
				
			||||||
 | 
						if id.Name == "init" && holder == defaultStructName {
 | 
				
			||||||
 | 
							// ignore init functions
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pkgm := w.pkgm
 | 
				
			||||||
 | 
						name := strings.ToUpper(id.Name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pkgm.mu.Lock()
 | 
				
			||||||
 | 
						defer pkgm.mu.Unlock()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if pkgm.methods[holder] != nil {
 | 
				
			||||||
 | 
							if pkgm.methods[holder][name] != nil {
 | 
				
			||||||
 | 
								refMethod := pkgm.methods[holder][name]
 | 
				
			||||||
 | 
								// confusing names
 | 
				
			||||||
 | 
								var kind string
 | 
				
			||||||
 | 
								if holder == defaultStructName {
 | 
				
			||||||
 | 
									kind = "function"
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									kind = "method"
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								var fileName string
 | 
				
			||||||
 | 
								if w.fileName == refMethod.fileName {
 | 
				
			||||||
 | 
									fileName = "the same source file"
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									fileName = refMethod.fileName
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Failure:    fmt.Sprintf("Method '%s' differs only by capitalization to %s '%s' in %s", id.Name, kind, refMethod.id.Name, fileName),
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Node:       id,
 | 
				
			||||||
 | 
									Category:   "naming",
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							pkgm.methods[holder] = make(map[string]*referenceMethod, 1)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// update the black list
 | 
				
			||||||
 | 
						if pkgm.methods[holder] == nil {
 | 
				
			||||||
 | 
							println("no entry for '", holder, "'")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						pkgm.methods[holder][name] = &referenceMethod{fileName: w.fileName, id: id}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintConfusingNames struct {
 | 
				
			||||||
 | 
						fileName  string
 | 
				
			||||||
 | 
						pkgm      pkgMethods
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const defaultStructName = "_" // used to map functions
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//getStructName of a function receiver. Defaults to defaultStructName
 | 
				
			||||||
 | 
					func getStructName(r *ast.FieldList) string {
 | 
				
			||||||
 | 
						result := defaultStructName
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if r == nil || len(r.List) < 1 {
 | 
				
			||||||
 | 
							return result
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						t := r.List[0].Type
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if p, _ := t.(*ast.StarExpr); p != nil { // if a pointer receiver => dereference pointer receiver types
 | 
				
			||||||
 | 
							t = p.X
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if p, _ := t.(*ast.Ident); p != nil {
 | 
				
			||||||
 | 
							result = p.Name
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return result
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func checkStructFields(fields *ast.FieldList, structName string, w *lintConfusingNames) {
 | 
				
			||||||
 | 
						bl := make(map[string]bool, len(fields.List))
 | 
				
			||||||
 | 
						for _, f := range fields.List {
 | 
				
			||||||
 | 
							for _, id := range f.Names {
 | 
				
			||||||
 | 
								normName := strings.ToUpper(id.Name)
 | 
				
			||||||
 | 
								if bl[normName] {
 | 
				
			||||||
 | 
									w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
										Failure:    fmt.Sprintf("Field '%s' differs only by capitalization to other field in the struct type %s", id.Name, structName),
 | 
				
			||||||
 | 
										Confidence: 1,
 | 
				
			||||||
 | 
										Node:       id,
 | 
				
			||||||
 | 
										Category:   "naming",
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									bl[normName] = true
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintConfusingNames) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch v := n.(type) {
 | 
				
			||||||
 | 
						case *ast.FuncDecl:
 | 
				
			||||||
 | 
							// Exclude naming warnings for functions that are exported to C but
 | 
				
			||||||
 | 
							// not exported in the Go API.
 | 
				
			||||||
 | 
							// See https://github.com/golang/lint/issues/144.
 | 
				
			||||||
 | 
							if ast.IsExported(v.Name.Name) || !isCgoExported(v) {
 | 
				
			||||||
 | 
								checkMethodName(getStructName(v.Recv), v.Name, w)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						case *ast.TypeSpec:
 | 
				
			||||||
 | 
							if s, ok := v.Type.(*ast.StructType); ok {
 | 
				
			||||||
 | 
								checkStructFields(s.Fields, v.Name.Name, w)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							// will add other checks like field names, struct names, etc.
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										67
									
								
								vendor/github.com/mgechev/revive/rule/confusing-results.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								vendor/github.com/mgechev/revive/rule/confusing-results.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,67 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ConfusingResultsRule lints given function declarations
 | 
				
			||||||
 | 
					type ConfusingResultsRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ConfusingResultsRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintConfusingResults{
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ConfusingResultsRule) Name() string {
 | 
				
			||||||
 | 
						return "confusing-results"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintConfusingResults struct {
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintConfusingResults) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fn, ok := n.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if !ok || fn.Type.Results == nil || len(fn.Type.Results.List) < 2 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						lastType := ""
 | 
				
			||||||
 | 
						for _, result := range fn.Type.Results.List {
 | 
				
			||||||
 | 
							if len(result.Names) > 0 {
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							t, ok := result.Type.(*ast.Ident)
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if t.Name == lastType {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Node:       n,
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Category:   "naming",
 | 
				
			||||||
 | 
									Failure:    "unnamed results of the same type may be confusing, consider using named results",
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							lastType = t.Name
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										88
									
								
								vendor/github.com/mgechev/revive/rule/constant-logical-expr.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								vendor/github.com/mgechev/revive/rule/constant-logical-expr.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,88 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ConstantLogicalExprRule warns on constant logical expressions.
 | 
				
			||||||
 | 
					type ConstantLogicalExprRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ConstantLogicalExprRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						astFile := file.AST
 | 
				
			||||||
 | 
						w := &lintConstantLogicalExpr{astFile, onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, astFile)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ConstantLogicalExprRule) Name() string {
 | 
				
			||||||
 | 
						return "constant-logical-expr"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintConstantLogicalExpr struct {
 | 
				
			||||||
 | 
						file      *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintConstantLogicalExpr) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := node.(type) {
 | 
				
			||||||
 | 
						case *ast.BinaryExpr:
 | 
				
			||||||
 | 
							if !w.isOperatorWithLogicalResult(n.Op) {
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if gofmt(n.X) != gofmt(n.Y) { // check if subexpressions are the same
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if n.Op == token.EQL {
 | 
				
			||||||
 | 
								w.newFailure(n, "expression always evaluates to true")
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if w.isInequalityOperator(n.Op) {
 | 
				
			||||||
 | 
								w.newFailure(n, "expression always evaluates to false")
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							w.newFailure(n, "left and right hand-side sub-expressions are the same")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintConstantLogicalExpr) isOperatorWithLogicalResult(t token.Token) bool {
 | 
				
			||||||
 | 
						switch t {
 | 
				
			||||||
 | 
						case token.LAND, token.LOR, token.EQL, token.LSS, token.GTR, token.NEQ, token.LEQ, token.GEQ:
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintConstantLogicalExpr) isInequalityOperator(t token.Token) bool {
 | 
				
			||||||
 | 
						switch t {
 | 
				
			||||||
 | 
						case token.LSS, token.GTR, token.NEQ, token.LEQ, token.GEQ:
 | 
				
			||||||
 | 
							return true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintConstantLogicalExpr) newFailure(node ast.Node, msg string) {
 | 
				
			||||||
 | 
						w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Node:       node,
 | 
				
			||||||
 | 
							Category:   "logic",
 | 
				
			||||||
 | 
							Failure:    msg,
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										60
									
								
								vendor/github.com/mgechev/revive/rule/context-as-argument.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								vendor/github.com/mgechev/revive/rule/context-as-argument.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,60 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ContextAsArgumentRule lints given else constructs.
 | 
				
			||||||
 | 
					type ContextAsArgumentRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ContextAsArgumentRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintContextArguments{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ContextAsArgumentRule) Name() string {
 | 
				
			||||||
 | 
						return "context-as-argument"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintContextArguments struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintContextArguments) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fn, ok := n.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if !ok || len(fn.Type.Params.List) <= 1 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// A context.Context should be the first parameter of a function.
 | 
				
			||||||
 | 
						// Flag any that show up after the first.
 | 
				
			||||||
 | 
						for _, arg := range fn.Type.Params.List[1:] {
 | 
				
			||||||
 | 
							if isPkgDot(arg.Type, "context", "Context") {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Node:       fn,
 | 
				
			||||||
 | 
									Category:   "arg-order",
 | 
				
			||||||
 | 
									Failure:    "context.Context should be the first parameter of a function",
 | 
				
			||||||
 | 
									Confidence: 0.9,
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								break // only flag one
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										81
									
								
								vendor/github.com/mgechev/revive/rule/context-keys-type.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								vendor/github.com/mgechev/revive/rule/context-keys-type.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,81 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/types"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ContextKeysType lints given else constructs.
 | 
				
			||||||
 | 
					type ContextKeysType struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ContextKeysType) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintContextKeyTypes{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						file.Pkg.TypeCheck()
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ContextKeysType) Name() string {
 | 
				
			||||||
 | 
						return "context-keys-type"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintContextKeyTypes struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintContextKeyTypes) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := n.(type) {
 | 
				
			||||||
 | 
						case *ast.CallExpr:
 | 
				
			||||||
 | 
							checkContextKeyType(w, n)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func checkContextKeyType(w lintContextKeyTypes, x *ast.CallExpr) {
 | 
				
			||||||
 | 
						f := w.file
 | 
				
			||||||
 | 
						sel, ok := x.Fun.(*ast.SelectorExpr)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						pkg, ok := sel.X.(*ast.Ident)
 | 
				
			||||||
 | 
						if !ok || pkg.Name != "context" {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if sel.Sel.Name != "WithValue" {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// key is second argument to context.WithValue
 | 
				
			||||||
 | 
						if len(x.Args) != 3 {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						key := f.Pkg.TypesInfo.Types[x.Args[1]]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if ktyp, ok := key.Type.(*types.Basic); ok && ktyp.Kind() != types.Invalid {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       x,
 | 
				
			||||||
 | 
								Category:   "content",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("should not use basic type %s as key in context.WithValue", key.Type),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										115
									
								
								vendor/github.com/mgechev/revive/rule/cyclomatic.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								vendor/github.com/mgechev/revive/rule/cyclomatic.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,115 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Based on https://github.com/fzipp/gocyclo
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CyclomaticRule lints given else constructs.
 | 
				
			||||||
 | 
					type CyclomaticRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *CyclomaticRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						complexity, ok := arguments[0].(int64) // Alt. non panicking version
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							panic("invalid argument for cyclomatic complexity")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintCyclomatic{
 | 
				
			||||||
 | 
							file:       file,
 | 
				
			||||||
 | 
							complexity: int(complexity),
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *CyclomaticRule) Name() string {
 | 
				
			||||||
 | 
						return "cyclomatic"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintCyclomatic struct {
 | 
				
			||||||
 | 
						file       *lint.File
 | 
				
			||||||
 | 
						complexity int
 | 
				
			||||||
 | 
						onFailure  func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintCyclomatic) Visit(_ ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						f := w.file
 | 
				
			||||||
 | 
						for _, decl := range f.AST.Decls {
 | 
				
			||||||
 | 
							if fn, ok := decl.(*ast.FuncDecl); ok {
 | 
				
			||||||
 | 
								c := complexity(fn)
 | 
				
			||||||
 | 
								if c > w.complexity {
 | 
				
			||||||
 | 
									w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
										Confidence: 1,
 | 
				
			||||||
 | 
										Category:   "maintenance",
 | 
				
			||||||
 | 
										Failure:    fmt.Sprintf("function %s has cyclomatic complexity %d", funcName(fn), c),
 | 
				
			||||||
 | 
										Node:       fn,
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// funcName returns the name representation of a function or method:
 | 
				
			||||||
 | 
					// "(Type).Name" for methods or simply "Name" for functions.
 | 
				
			||||||
 | 
					func funcName(fn *ast.FuncDecl) string {
 | 
				
			||||||
 | 
						if fn.Recv != nil {
 | 
				
			||||||
 | 
							if fn.Recv.NumFields() > 0 {
 | 
				
			||||||
 | 
								typ := fn.Recv.List[0].Type
 | 
				
			||||||
 | 
								return fmt.Sprintf("(%s).%s", recvString(typ), fn.Name)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return fn.Name.Name
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// recvString returns a string representation of recv of the
 | 
				
			||||||
 | 
					// form "T", "*T", or "BADRECV" (if not a proper receiver type).
 | 
				
			||||||
 | 
					func recvString(recv ast.Expr) string {
 | 
				
			||||||
 | 
						switch t := recv.(type) {
 | 
				
			||||||
 | 
						case *ast.Ident:
 | 
				
			||||||
 | 
							return t.Name
 | 
				
			||||||
 | 
						case *ast.StarExpr:
 | 
				
			||||||
 | 
							return "*" + recvString(t.X)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return "BADRECV"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// complexity calculates the cyclomatic complexity of a function.
 | 
				
			||||||
 | 
					func complexity(fn *ast.FuncDecl) int {
 | 
				
			||||||
 | 
						v := complexityVisitor{}
 | 
				
			||||||
 | 
						ast.Walk(&v, fn)
 | 
				
			||||||
 | 
						return v.Complexity
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type complexityVisitor struct {
 | 
				
			||||||
 | 
						// Complexity is the cyclomatic complexity
 | 
				
			||||||
 | 
						Complexity int
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Visit implements the ast.Visitor interface.
 | 
				
			||||||
 | 
					func (v *complexityVisitor) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch n := n.(type) {
 | 
				
			||||||
 | 
						case *ast.FuncDecl, *ast.IfStmt, *ast.ForStmt, *ast.RangeStmt, *ast.CaseClause, *ast.CommClause:
 | 
				
			||||||
 | 
							v.Complexity++
 | 
				
			||||||
 | 
						case *ast.BinaryExpr:
 | 
				
			||||||
 | 
							if n.Op == token.LAND || n.Op == token.LOR {
 | 
				
			||||||
 | 
								v.Complexity++
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return v
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										94
									
								
								vendor/github.com/mgechev/revive/rule/deep-exit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								vendor/github.com/mgechev/revive/rule/deep-exit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,94 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DeepExitRule lints program exit at functions other than main or init.
 | 
				
			||||||
 | 
					type DeepExitRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *DeepExitRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var exitFunctions = map[string]map[string]bool{
 | 
				
			||||||
 | 
							"os":      map[string]bool{"Exit": true},
 | 
				
			||||||
 | 
							"syscall": map[string]bool{"Exit": true},
 | 
				
			||||||
 | 
							"log": map[string]bool{
 | 
				
			||||||
 | 
								"Fatal":   true,
 | 
				
			||||||
 | 
								"Fatalf":  true,
 | 
				
			||||||
 | 
								"Fatalln": true,
 | 
				
			||||||
 | 
								"Panic":   true,
 | 
				
			||||||
 | 
								"Panicf":  true,
 | 
				
			||||||
 | 
								"Panicln": true,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintDeepExit{onFailure, exitFunctions, file.IsTest()}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *DeepExitRule) Name() string {
 | 
				
			||||||
 | 
						return "deep-exit"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintDeepExit struct {
 | 
				
			||||||
 | 
						onFailure     func(lint.Failure)
 | 
				
			||||||
 | 
						exitFunctions map[string]map[string]bool
 | 
				
			||||||
 | 
						isTestFile    bool
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintDeepExit) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						if fd, ok := node.(*ast.FuncDecl); ok {
 | 
				
			||||||
 | 
							if w.mustIgnore(fd) {
 | 
				
			||||||
 | 
								return nil // skip analysis of this function
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						se, ok := node.(*ast.ExprStmt)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						ce, ok := se.X.(*ast.CallExpr)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fc, ok := ce.Fun.(*ast.SelectorExpr)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						id, ok := fc.X.(*ast.Ident)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fn := fc.Sel.Name
 | 
				
			||||||
 | 
						pkg := id.Name
 | 
				
			||||||
 | 
						if w.exitFunctions[pkg] != nil && w.exitFunctions[pkg][fn] { // it's a call to an exit function
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       ce,
 | 
				
			||||||
 | 
								Category:   "bad practice",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("calls to %s.%s only in main() or init() functions", pkg, fn),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintDeepExit) mustIgnore(fd *ast.FuncDecl) bool {
 | 
				
			||||||
 | 
						fn := fd.Name.Name
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fn == "init" || fn == "main" || (w.isTestFile && fn == "TestMain")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										54
									
								
								vendor/github.com/mgechev/revive/rule/dot-imports.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								vendor/github.com/mgechev/revive/rule/dot-imports.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,54 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DotImportsRule lints given else constructs.
 | 
				
			||||||
 | 
					type DotImportsRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *DotImportsRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintImports{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *DotImportsRule) Name() string {
 | 
				
			||||||
 | 
						return "dot-imports"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintImports struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintImports) Visit(_ ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						for i, is := range w.fileAst.Imports {
 | 
				
			||||||
 | 
							_ = i
 | 
				
			||||||
 | 
							if is.Name != nil && is.Name.Name == "." && !w.file.IsTest() {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Failure:    "should not use dot imports",
 | 
				
			||||||
 | 
									Node:       is,
 | 
				
			||||||
 | 
									Category:   "imports",
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										39
									
								
								vendor/github.com/mgechev/revive/rule/duplicated-imports.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								vendor/github.com/mgechev/revive/rule/duplicated-imports.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DuplicatedImportsRule lints given else constructs.
 | 
				
			||||||
 | 
					type DuplicatedImportsRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *DuplicatedImportsRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						impPaths := map[string]struct{}{}
 | 
				
			||||||
 | 
						for _, imp := range file.AST.Imports {
 | 
				
			||||||
 | 
							path := imp.Path.Value
 | 
				
			||||||
 | 
							_, ok := impPaths[path]
 | 
				
			||||||
 | 
							if ok {
 | 
				
			||||||
 | 
								failures = append(failures, lint.Failure{
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Failure:    fmt.Sprintf("Package %s already imported", path),
 | 
				
			||||||
 | 
									Node:       imp,
 | 
				
			||||||
 | 
									Category:   "imports",
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							impPaths[path] = struct{}{}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *DuplicatedImportsRule) Name() string {
 | 
				
			||||||
 | 
						return "duplicated-imports"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										76
									
								
								vendor/github.com/mgechev/revive/rule/empty-block.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								vendor/github.com/mgechev/revive/rule/empty-block.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,76 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// EmptyBlockRule lints given else constructs.
 | 
				
			||||||
 | 
					type EmptyBlockRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *EmptyBlockRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintEmptyBlock{make([]*ast.BlockStmt, 0), onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *EmptyBlockRule) Name() string {
 | 
				
			||||||
 | 
						return "empty-block"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintEmptyBlock struct {
 | 
				
			||||||
 | 
						ignore    []*ast.BlockStmt
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintEmptyBlock) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fd, ok := node.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if ok {
 | 
				
			||||||
 | 
							w.ignore = append(w.ignore, fd.Body)
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fl, ok := node.(*ast.FuncLit)
 | 
				
			||||||
 | 
						if ok {
 | 
				
			||||||
 | 
							w.ignore = append(w.ignore, fl.Body)
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						block, ok := node.(*ast.BlockStmt)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if mustIgnore(block, w.ignore) {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(block.List) == 0 {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       block,
 | 
				
			||||||
 | 
								Category:   "logic",
 | 
				
			||||||
 | 
								Failure:    "this block is empty, you can remove it",
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func mustIgnore(block *ast.BlockStmt, blackList []*ast.BlockStmt) bool {
 | 
				
			||||||
 | 
						for _, b := range blackList {
 | 
				
			||||||
 | 
							if b == block {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										113
									
								
								vendor/github.com/mgechev/revive/rule/empty-lines.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								vendor/github.com/mgechev/revive/rule/empty-lines.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,113 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// EmptyLinesRule lints empty lines in blocks.
 | 
				
			||||||
 | 
					type EmptyLinesRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *EmptyLinesRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintEmptyLines{file, file.CommentMap(), onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *EmptyLinesRule) Name() string {
 | 
				
			||||||
 | 
						return "empty-lines"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintEmptyLines struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						cmap      ast.CommentMap
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintEmptyLines) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						block, ok := node.(*ast.BlockStmt)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.checkStart(block)
 | 
				
			||||||
 | 
						w.checkEnd(block)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintEmptyLines) checkStart(block *ast.BlockStmt) {
 | 
				
			||||||
 | 
						if len(block.List) == 0 {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						start := w.position(block.Lbrace)
 | 
				
			||||||
 | 
						firstNode := block.List[0]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if w.commentBetween(start, firstNode) {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						first := w.position(firstNode.Pos())
 | 
				
			||||||
 | 
						if first.Line-start.Line > 1 {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       block,
 | 
				
			||||||
 | 
								Category:   "style",
 | 
				
			||||||
 | 
								Failure:    "extra empty line at the start of a block",
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintEmptyLines) checkEnd(block *ast.BlockStmt) {
 | 
				
			||||||
 | 
						if len(block.List) < 1 {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						end := w.position(block.Rbrace)
 | 
				
			||||||
 | 
						lastNode := block.List[len(block.List)-1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if w.commentBetween(end, lastNode) {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						last := w.position(lastNode.End())
 | 
				
			||||||
 | 
						if end.Line-last.Line > 1 {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       lastNode,
 | 
				
			||||||
 | 
								Category:   "style",
 | 
				
			||||||
 | 
								Failure:    "extra empty line at the end of a block",
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintEmptyLines) commentBetween(position token.Position, node ast.Node) bool {
 | 
				
			||||||
 | 
						comments := w.cmap.Filter(node).Comments()
 | 
				
			||||||
 | 
						if len(comments) == 0 {
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, comment := range comments {
 | 
				
			||||||
 | 
							start, end := w.position(comment.Pos()), w.position(comment.End())
 | 
				
			||||||
 | 
							if start.Line-position.Line == 1 || position.Line-end.Line == 1 {
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintEmptyLines) position(pos token.Pos) token.Position {
 | 
				
			||||||
 | 
						return w.file.ToPosition(pos)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										79
									
								
								vendor/github.com/mgechev/revive/rule/error-naming.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								vendor/github.com/mgechev/revive/rule/error-naming.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,79 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ErrorNamingRule lints given else constructs.
 | 
				
			||||||
 | 
					type ErrorNamingRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ErrorNamingRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintErrors{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ErrorNamingRule) Name() string {
 | 
				
			||||||
 | 
						return "error-naming"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintErrors struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintErrors) Visit(_ ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						for _, decl := range w.fileAst.Decls {
 | 
				
			||||||
 | 
							gd, ok := decl.(*ast.GenDecl)
 | 
				
			||||||
 | 
							if !ok || gd.Tok != token.VAR {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							for _, spec := range gd.Specs {
 | 
				
			||||||
 | 
								spec := spec.(*ast.ValueSpec)
 | 
				
			||||||
 | 
								if len(spec.Names) != 1 || len(spec.Values) != 1 {
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								ce, ok := spec.Values[0].(*ast.CallExpr)
 | 
				
			||||||
 | 
								if !ok {
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if !isPkgDot(ce.Fun, "errors", "New") && !isPkgDot(ce.Fun, "fmt", "Errorf") {
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								id := spec.Names[0]
 | 
				
			||||||
 | 
								prefix := "err"
 | 
				
			||||||
 | 
								if id.IsExported() {
 | 
				
			||||||
 | 
									prefix = "Err"
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if !strings.HasPrefix(id.Name, prefix) {
 | 
				
			||||||
 | 
									w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
										Node:       id,
 | 
				
			||||||
 | 
										Confidence: 0.9,
 | 
				
			||||||
 | 
										Category:   "naming",
 | 
				
			||||||
 | 
										Failure:    fmt.Sprintf("error var %s should have name of the form %sFoo", id.Name, prefix),
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										67
									
								
								vendor/github.com/mgechev/revive/rule/error-return.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								vendor/github.com/mgechev/revive/rule/error-return.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,67 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ErrorReturnRule lints given else constructs.
 | 
				
			||||||
 | 
					type ErrorReturnRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ErrorReturnRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintErrorReturn{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ErrorReturnRule) Name() string {
 | 
				
			||||||
 | 
						return "error-return"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintErrorReturn struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintErrorReturn) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fn, ok := n.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if !ok || fn.Type.Results == nil {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						ret := fn.Type.Results.List
 | 
				
			||||||
 | 
						if len(ret) <= 1 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if isIdent(ret[len(ret)-1].Type, "error") {
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// An error return parameter should be the last parameter.
 | 
				
			||||||
 | 
						// Flag any error parameters found before the last.
 | 
				
			||||||
 | 
						for _, r := range ret[:len(ret)-1] {
 | 
				
			||||||
 | 
							if isIdent(r.Type, "error") {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Category:   "arg-order",
 | 
				
			||||||
 | 
									Confidence: 0.9,
 | 
				
			||||||
 | 
									Node:       fn,
 | 
				
			||||||
 | 
									Failure:    "error should be the last type when returning multiple items",
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								break // only flag one
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										98
									
								
								vendor/github.com/mgechev/revive/rule/error-strings.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								vendor/github.com/mgechev/revive/rule/error-strings.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,98 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
 | 
						"unicode"
 | 
				
			||||||
 | 
						"unicode/utf8"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ErrorStringsRule lints given else constructs.
 | 
				
			||||||
 | 
					type ErrorStringsRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ErrorStringsRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintErrorStrings{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ErrorStringsRule) Name() string {
 | 
				
			||||||
 | 
						return "error-strings"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintErrorStrings struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintErrorStrings) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						ce, ok := n.(*ast.CallExpr)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !isPkgDot(ce.Fun, "errors", "New") && !isPkgDot(ce.Fun, "fmt", "Errorf") {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if len(ce.Args) < 1 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						str, ok := ce.Args[0].(*ast.BasicLit)
 | 
				
			||||||
 | 
						if !ok || str.Kind != token.STRING {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						s, _ := strconv.Unquote(str.Value) // can assume well-formed Go
 | 
				
			||||||
 | 
						if s == "" {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						clean, conf := lintErrorString(s)
 | 
				
			||||||
 | 
						if clean {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Node:       str,
 | 
				
			||||||
 | 
							Confidence: conf,
 | 
				
			||||||
 | 
							Category:   "errors",
 | 
				
			||||||
 | 
							Failure:    "error strings should not be capitalized or end with punctuation or a newline",
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func lintErrorString(s string) (isClean bool, conf float64) {
 | 
				
			||||||
 | 
						const basicConfidence = 0.8
 | 
				
			||||||
 | 
						const capConfidence = basicConfidence - 0.2
 | 
				
			||||||
 | 
						first, firstN := utf8.DecodeRuneInString(s)
 | 
				
			||||||
 | 
						last, _ := utf8.DecodeLastRuneInString(s)
 | 
				
			||||||
 | 
						if last == '.' || last == ':' || last == '!' || last == '\n' {
 | 
				
			||||||
 | 
							return false, basicConfidence
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if unicode.IsUpper(first) {
 | 
				
			||||||
 | 
							// People use proper nouns and exported Go identifiers in error strings,
 | 
				
			||||||
 | 
							// so decrease the confidence of warnings for capitalization.
 | 
				
			||||||
 | 
							if len(s) <= firstN {
 | 
				
			||||||
 | 
								return false, capConfidence
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							// Flag strings starting with something that doesn't look like an initialism.
 | 
				
			||||||
 | 
							if second, _ := utf8.DecodeRuneInString(s[firstN:]); !unicode.IsUpper(second) {
 | 
				
			||||||
 | 
								return false, capConfidence
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return true, 0
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										93
									
								
								vendor/github.com/mgechev/revive/rule/errorf.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								vendor/github.com/mgechev/revive/rule/errorf.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,93 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"regexp"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ErrorfRule lints given else constructs.
 | 
				
			||||||
 | 
					type ErrorfRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ErrorfRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintErrorf{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						file.Pkg.TypeCheck()
 | 
				
			||||||
 | 
						ast.Walk(walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ErrorfRule) Name() string {
 | 
				
			||||||
 | 
						return "errorf"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintErrorf struct {
 | 
				
			||||||
 | 
						file      *lint.File
 | 
				
			||||||
 | 
						fileAst   *ast.File
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintErrorf) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						ce, ok := n.(*ast.CallExpr)
 | 
				
			||||||
 | 
						if !ok || len(ce.Args) != 1 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						isErrorsNew := isPkgDot(ce.Fun, "errors", "New")
 | 
				
			||||||
 | 
						var isTestingError bool
 | 
				
			||||||
 | 
						se, ok := ce.Fun.(*ast.SelectorExpr)
 | 
				
			||||||
 | 
						if ok && se.Sel.Name == "Error" {
 | 
				
			||||||
 | 
							if typ := w.file.Pkg.TypeOf(se.X); typ != nil {
 | 
				
			||||||
 | 
								isTestingError = typ.String() == "*testing.T"
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !isErrorsNew && !isTestingError {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						arg := ce.Args[0]
 | 
				
			||||||
 | 
						ce, ok = arg.(*ast.CallExpr)
 | 
				
			||||||
 | 
						if !ok || !isPkgDot(ce.Fun, "fmt", "Sprintf") {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						errorfPrefix := "fmt"
 | 
				
			||||||
 | 
						if isTestingError {
 | 
				
			||||||
 | 
							errorfPrefix = w.file.Render(se.X)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						failure := lint.Failure{
 | 
				
			||||||
 | 
							Category:   "errors",
 | 
				
			||||||
 | 
							Node:       n,
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Failure:    fmt.Sprintf("should replace %s(fmt.Sprintf(...)) with %s.Errorf(...)", w.file.Render(se), errorfPrefix),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						m := srcLineWithMatch(w.file, ce, `^(.*)`+w.file.Render(se)+`\(fmt\.Sprintf\((.*)\)\)(.*)$`)
 | 
				
			||||||
 | 
						if m != nil {
 | 
				
			||||||
 | 
							failure.ReplacementLine = m[1] + errorfPrefix + ".Errorf(" + m[2] + ")" + m[3]
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.onFailure(failure)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func srcLineWithMatch(file *lint.File, node ast.Node, pattern string) (m []string) {
 | 
				
			||||||
 | 
						line := srcLine(file.Content(), file.ToPosition(node.Pos()))
 | 
				
			||||||
 | 
						line = strings.TrimSuffix(line, "\n")
 | 
				
			||||||
 | 
						rx := regexp.MustCompile(pattern)
 | 
				
			||||||
 | 
						return rx.FindStringSubmatch(line)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										272
									
								
								vendor/github.com/mgechev/revive/rule/exported.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										272
									
								
								vendor/github.com/mgechev/revive/rule/exported.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,272 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"go/token"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
						"unicode"
 | 
				
			||||||
 | 
						"unicode/utf8"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ExportedRule lints given else constructs.
 | 
				
			||||||
 | 
					type ExportedRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *ExportedRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if isTest(file) {
 | 
				
			||||||
 | 
							return failures
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fileAst := file.AST
 | 
				
			||||||
 | 
						walker := lintExported{
 | 
				
			||||||
 | 
							file:    file,
 | 
				
			||||||
 | 
							fileAst: fileAst,
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							genDeclMissingComments: make(map[*ast.GenDecl]bool),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(&walker, fileAst)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *ExportedRule) Name() string {
 | 
				
			||||||
 | 
						return "exported"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintExported struct {
 | 
				
			||||||
 | 
						file                   *lint.File
 | 
				
			||||||
 | 
						fileAst                *ast.File
 | 
				
			||||||
 | 
						lastGen                *ast.GenDecl
 | 
				
			||||||
 | 
						genDeclMissingComments map[*ast.GenDecl]bool
 | 
				
			||||||
 | 
						onFailure              func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintExported) lintFuncDoc(fn *ast.FuncDecl) {
 | 
				
			||||||
 | 
						if !ast.IsExported(fn.Name.Name) {
 | 
				
			||||||
 | 
							// func is unexported
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						kind := "function"
 | 
				
			||||||
 | 
						name := fn.Name.Name
 | 
				
			||||||
 | 
						if fn.Recv != nil && len(fn.Recv.List) > 0 {
 | 
				
			||||||
 | 
							// method
 | 
				
			||||||
 | 
							kind = "method"
 | 
				
			||||||
 | 
							recv := receiverType(fn)
 | 
				
			||||||
 | 
							if !ast.IsExported(recv) {
 | 
				
			||||||
 | 
								// receiver is unexported
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if commonMethods[name] {
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							switch name {
 | 
				
			||||||
 | 
							case "Len", "Less", "Swap":
 | 
				
			||||||
 | 
								if w.file.Pkg.Sortable[recv] {
 | 
				
			||||||
 | 
									return
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							name = recv + "." + name
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if fn.Doc == nil {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Node:       fn,
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Category:   "comments",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("exported %s %s should have comment or be unexported", kind, name),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						s := normalizeText(fn.Doc.Text())
 | 
				
			||||||
 | 
						prefix := fn.Name.Name + " "
 | 
				
			||||||
 | 
						if !strings.HasPrefix(s, prefix) {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Node:       fn.Doc,
 | 
				
			||||||
 | 
								Confidence: 0.8,
 | 
				
			||||||
 | 
								Category:   "comments",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf(`comment on exported %s %s should be of the form "%s..."`, kind, name, prefix),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintExported) checkStutter(id *ast.Ident, thing string) {
 | 
				
			||||||
 | 
						pkg, name := w.fileAst.Name.Name, id.Name
 | 
				
			||||||
 | 
						if !ast.IsExported(name) {
 | 
				
			||||||
 | 
							// unexported name
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// A name stutters if the package name is a strict prefix
 | 
				
			||||||
 | 
						// and the next character of the name starts a new word.
 | 
				
			||||||
 | 
						if len(name) <= len(pkg) {
 | 
				
			||||||
 | 
							// name is too short to stutter.
 | 
				
			||||||
 | 
							// This permits the name to be the same as the package name.
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !strings.EqualFold(pkg, name[:len(pkg)]) {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// We can assume the name is well-formed UTF-8.
 | 
				
			||||||
 | 
						// If the next rune after the package name is uppercase or an underscore
 | 
				
			||||||
 | 
						// the it's starting a new word and thus this name stutters.
 | 
				
			||||||
 | 
						rem := name[len(pkg):]
 | 
				
			||||||
 | 
						if next, _ := utf8.DecodeRuneInString(rem); next == '_' || unicode.IsUpper(next) {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Node:       id,
 | 
				
			||||||
 | 
								Confidence: 0.8,
 | 
				
			||||||
 | 
								Category:   "naming",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("%s name will be used as %s.%s by other packages, and that stutters; consider calling this %s", thing, pkg, name, rem),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintExported) lintTypeDoc(t *ast.TypeSpec, doc *ast.CommentGroup) {
 | 
				
			||||||
 | 
						if !ast.IsExported(t.Name.Name) {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if doc == nil {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Node:       t,
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Category:   "comments",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("exported type %v should have comment or be unexported", t.Name),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						s := normalizeText(doc.Text())
 | 
				
			||||||
 | 
						articles := [...]string{"A", "An", "The", "This"}
 | 
				
			||||||
 | 
						for _, a := range articles {
 | 
				
			||||||
 | 
							if t.Name.Name == a {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if strings.HasPrefix(s, a+" ") {
 | 
				
			||||||
 | 
								s = s[len(a)+1:]
 | 
				
			||||||
 | 
								break
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !strings.HasPrefix(s, t.Name.Name+" ") {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Node:       doc,
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Category:   "comments",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf(`comment on exported type %v should be of the form "%v ..." (with optional leading article)`, t.Name, t.Name),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintExported) lintValueSpecDoc(vs *ast.ValueSpec, gd *ast.GenDecl, genDeclMissingComments map[*ast.GenDecl]bool) {
 | 
				
			||||||
 | 
						kind := "var"
 | 
				
			||||||
 | 
						if gd.Tok == token.CONST {
 | 
				
			||||||
 | 
							kind = "const"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(vs.Names) > 1 {
 | 
				
			||||||
 | 
							// Check that none are exported except for the first.
 | 
				
			||||||
 | 
							for _, n := range vs.Names[1:] {
 | 
				
			||||||
 | 
								if ast.IsExported(n.Name) {
 | 
				
			||||||
 | 
									w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
										Category:   "comments",
 | 
				
			||||||
 | 
										Confidence: 1,
 | 
				
			||||||
 | 
										Failure:    fmt.Sprintf("exported %s %s should have its own declaration", kind, n.Name),
 | 
				
			||||||
 | 
										Node:       vs,
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
									return
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Only one name.
 | 
				
			||||||
 | 
						name := vs.Names[0].Name
 | 
				
			||||||
 | 
						if !ast.IsExported(name) {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if vs.Doc == nil && gd.Doc == nil {
 | 
				
			||||||
 | 
							if genDeclMissingComments[gd] {
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							block := ""
 | 
				
			||||||
 | 
							if kind == "const" && gd.Lparen.IsValid() {
 | 
				
			||||||
 | 
								block = " (or a comment on this block)"
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       vs,
 | 
				
			||||||
 | 
								Category:   "comments",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("exported %s %s should have comment%s or be unexported", kind, name, block),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
							genDeclMissingComments[gd] = true
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// If this GenDecl has parens and a comment, we don't check its comment form.
 | 
				
			||||||
 | 
						if gd.Lparen.IsValid() && gd.Doc != nil {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// The relevant text to check will be on either vs.Doc or gd.Doc.
 | 
				
			||||||
 | 
						// Use vs.Doc preferentially.
 | 
				
			||||||
 | 
						doc := vs.Doc
 | 
				
			||||||
 | 
						if doc == nil {
 | 
				
			||||||
 | 
							doc = gd.Doc
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						prefix := name + " "
 | 
				
			||||||
 | 
						s := normalizeText(doc.Text())
 | 
				
			||||||
 | 
						if !strings.HasPrefix(s, prefix) {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Node:       doc,
 | 
				
			||||||
 | 
								Category:   "comments",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf(`comment on exported %s %s should be of the form "%s..."`, kind, name, prefix),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// normalizeText is a helper function that normalizes comment strings by:
 | 
				
			||||||
 | 
					// * removing one leading space
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// This function is needed because ast.CommentGroup.Text() does not handle //-style and /*-style comments uniformly
 | 
				
			||||||
 | 
					func normalizeText(t string) string {
 | 
				
			||||||
 | 
						return strings.TrimPrefix(t, " ")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w *lintExported) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						switch v := n.(type) {
 | 
				
			||||||
 | 
						case *ast.GenDecl:
 | 
				
			||||||
 | 
							if v.Tok == token.IMPORT {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							// token.CONST, token.TYPE or token.VAR
 | 
				
			||||||
 | 
							w.lastGen = v
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						case *ast.FuncDecl:
 | 
				
			||||||
 | 
							w.lintFuncDoc(v)
 | 
				
			||||||
 | 
							if v.Recv == nil {
 | 
				
			||||||
 | 
								// Only check for stutter on functions, not methods.
 | 
				
			||||||
 | 
								// Method names are not used package-qualified.
 | 
				
			||||||
 | 
								w.checkStutter(v.Name, "func")
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							// Don't proceed inside funcs.
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.TypeSpec:
 | 
				
			||||||
 | 
							// inside a GenDecl, which usually has the doc
 | 
				
			||||||
 | 
							doc := v.Doc
 | 
				
			||||||
 | 
							if doc == nil {
 | 
				
			||||||
 | 
								doc = w.lastGen.Doc
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							w.lintTypeDoc(v, doc)
 | 
				
			||||||
 | 
							w.checkStutter(v.Name, "type")
 | 
				
			||||||
 | 
							// Don't proceed inside types.
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						case *ast.ValueSpec:
 | 
				
			||||||
 | 
							w.lintValueSpecDoc(v, w.lastGen, w.genDeclMissingComments)
 | 
				
			||||||
 | 
							return nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										69
									
								
								vendor/github.com/mgechev/revive/rule/file-header.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								vendor/github.com/mgechev/revive/rule/file-header.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,69 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"regexp"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FileHeaderRule lints given else constructs.
 | 
				
			||||||
 | 
					type FileHeaderRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var (
 | 
				
			||||||
 | 
						multiRegexp  = regexp.MustCompile("^/\\*")
 | 
				
			||||||
 | 
						singleRegexp = regexp.MustCompile("^//")
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *FileHeaderRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						if len(arguments) != 1 {
 | 
				
			||||||
 | 
							panic(`invalid configuration for "file-header" rule`)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						header, ok := arguments[0].(string)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							panic(`invalid argument for "file-header" rule: first argument should be a string`)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						failure := []lint.Failure{
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								Node:       file.AST,
 | 
				
			||||||
 | 
								Confidence: 1,
 | 
				
			||||||
 | 
								Failure:    "the file doesn't have an appropriate header",
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(file.AST.Comments) == 0 {
 | 
				
			||||||
 | 
							return failure
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						g := file.AST.Comments[0]
 | 
				
			||||||
 | 
						if g == nil {
 | 
				
			||||||
 | 
							return failure
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						comment := ""
 | 
				
			||||||
 | 
						for _, c := range g.List {
 | 
				
			||||||
 | 
							text := c.Text
 | 
				
			||||||
 | 
							if multiRegexp.Match([]byte(text)) {
 | 
				
			||||||
 | 
								text = text[2 : len(text)-2]
 | 
				
			||||||
 | 
							} else if singleRegexp.Match([]byte(text)) {
 | 
				
			||||||
 | 
								text = text[2:]
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							comment += text
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						regex, err := regexp.Compile(header)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							panic(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if !regex.Match([]byte(comment)) {
 | 
				
			||||||
 | 
							return failure
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *FileHeaderRule) Name() string {
 | 
				
			||||||
 | 
						return "file-header"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										104
									
								
								vendor/github.com/mgechev/revive/rule/flag-param.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								vendor/github.com/mgechev/revive/rule/flag-param.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,104 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FlagParamRule lints given else constructs.
 | 
				
			||||||
 | 
					type FlagParamRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *FlagParamRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintFlagParamRule{onFailure: onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *FlagParamRule) Name() string {
 | 
				
			||||||
 | 
						return "flag-parameter"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintFlagParamRule struct {
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintFlagParamRule) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fd, ok := node.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if fd.Body == nil {
 | 
				
			||||||
 | 
							return nil // skip whole function declaration
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, p := range fd.Type.Params.List {
 | 
				
			||||||
 | 
							t := p.Type
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							id, ok := t.(*ast.Ident)
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if id.Name != "bool" {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							cv := conditionVisitor{p.Names, fd, w}
 | 
				
			||||||
 | 
							ast.Walk(cv, fd.Body)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type conditionVisitor struct {
 | 
				
			||||||
 | 
						ids    []*ast.Ident
 | 
				
			||||||
 | 
						fd     *ast.FuncDecl
 | 
				
			||||||
 | 
						linter lintFlagParamRule
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w conditionVisitor) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						ifStmt, ok := node.(*ast.IfStmt)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fselect := func(n ast.Node) bool {
 | 
				
			||||||
 | 
							ident, ok := n.(*ast.Ident)
 | 
				
			||||||
 | 
							if !ok {
 | 
				
			||||||
 | 
								return false
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							for _, id := range w.ids {
 | 
				
			||||||
 | 
								if ident.Name == id.Name {
 | 
				
			||||||
 | 
									return true
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return false
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						uses := pick(ifStmt.Cond, fselect, nil)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(uses) < 1 {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w.linter.onFailure(lint.Failure{
 | 
				
			||||||
 | 
							Confidence: 1,
 | 
				
			||||||
 | 
							Node:       w.fd.Type.Params,
 | 
				
			||||||
 | 
							Category:   "bad practice",
 | 
				
			||||||
 | 
							Failure:    fmt.Sprintf("parameter '%s' seems to be a control flag, avoid control coupling", uses[0]),
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										68
									
								
								vendor/github.com/mgechev/revive/rule/function-result-limit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								vendor/github.com/mgechev/revive/rule/function-result-limit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,68 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FunctionResultsLimitRule lints given else constructs.
 | 
				
			||||||
 | 
					type FunctionResultsLimitRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *FunctionResultsLimitRule) Apply(file *lint.File, arguments lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						if len(arguments) != 1 {
 | 
				
			||||||
 | 
							panic(`invalid configuration for "function-result-limit"`)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						max, ok := arguments[0].(int64) // Alt. non panicking version
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							panic(fmt.Sprintf(`invalid value passed as return results number to the "function-result-limit" rule; need int64 but got %T`, arguments[0]))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if max < 0 {
 | 
				
			||||||
 | 
							panic(`the value passed as return results number to the "function-result-limit" rule cannot be negative`)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						walker := lintFunctionResultsNum{
 | 
				
			||||||
 | 
							max: int(max),
 | 
				
			||||||
 | 
							onFailure: func(failure lint.Failure) {
 | 
				
			||||||
 | 
								failures = append(failures, failure)
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ast.Walk(walker, file.AST)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *FunctionResultsLimitRule) Name() string {
 | 
				
			||||||
 | 
						return "function-result-limit"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintFunctionResultsNum struct {
 | 
				
			||||||
 | 
						max       int
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintFunctionResultsNum) Visit(n ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						node, ok := n.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if ok {
 | 
				
			||||||
 | 
							num := 0
 | 
				
			||||||
 | 
							if node.Type.Results != nil {
 | 
				
			||||||
 | 
								num = node.Type.Results.NumFields()
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if num > w.max {
 | 
				
			||||||
 | 
								w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
									Confidence: 1,
 | 
				
			||||||
 | 
									Failure:    fmt.Sprintf("maximum number of return results per function exceeded; max %d but got %d", w.max, num),
 | 
				
			||||||
 | 
									Node:       node.Type,
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
								return w
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										70
									
								
								vendor/github.com/mgechev/revive/rule/get-return.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								vendor/github.com/mgechev/revive/rule/get-return.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,70 @@
 | 
				
			|||||||
 | 
					package rule
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"go/ast"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/mgechev/revive/lint"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GetReturnRule lints given else constructs.
 | 
				
			||||||
 | 
					type GetReturnRule struct{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Apply applies the rule to given file.
 | 
				
			||||||
 | 
					func (r *GetReturnRule) Apply(file *lint.File, _ lint.Arguments) []lint.Failure {
 | 
				
			||||||
 | 
						var failures []lint.Failure
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						onFailure := func(failure lint.Failure) {
 | 
				
			||||||
 | 
							failures = append(failures, failure)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						w := lintReturnRule{onFailure}
 | 
				
			||||||
 | 
						ast.Walk(w, file.AST)
 | 
				
			||||||
 | 
						return failures
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Name returns the rule name.
 | 
				
			||||||
 | 
					func (r *GetReturnRule) Name() string {
 | 
				
			||||||
 | 
						return "get-return"
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type lintReturnRule struct {
 | 
				
			||||||
 | 
						onFailure func(lint.Failure)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func isGetter(name string) bool {
 | 
				
			||||||
 | 
						if strings.HasPrefix(strings.ToUpper(name), "GET") {
 | 
				
			||||||
 | 
							if len(name) > 3 {
 | 
				
			||||||
 | 
								c := name[3]
 | 
				
			||||||
 | 
								return !(c >= 'a' && c <= 'z')
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func hasResults(rs *ast.FieldList) bool {
 | 
				
			||||||
 | 
						return rs != nil && len(rs.List) > 0
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (w lintReturnRule) Visit(node ast.Node) ast.Visitor {
 | 
				
			||||||
 | 
						fd, ok := node.(*ast.FuncDecl)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if !isGetter(fd.Name.Name) {
 | 
				
			||||||
 | 
							return w
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !hasResults(fd.Type.Results) {
 | 
				
			||||||
 | 
							w.onFailure(lint.Failure{
 | 
				
			||||||
 | 
								Confidence: 0.8,
 | 
				
			||||||
 | 
								Node:       fd,
 | 
				
			||||||
 | 
								Category:   "logic",
 | 
				
			||||||
 | 
								Failure:    fmt.Sprintf("function '%s' seems to be a getter but it does not return any result", fd.Name.Name),
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return w
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user