├── .gitignore ├── docs ├── images │ ├── diagram.png │ └── diagram1000.png ├── redis-url.md ├── cobra │ ├── redis-inventory.md │ ├── redis-inventory_display.md │ ├── redis-inventory_index.md │ └── redis-inventory_inventory.md ├── installation.md └── usage.md ├── main.go ├── cmd ├── app │ ├── vars.go │ ├── display.go │ ├── root.go │ ├── inventory.go │ └── index.go └── docgen │ └── main.go ├── docker-compose.yml ├── Dockerfile ├── .pre-commit-config.yaml ├── makefile ├── src ├── logger │ └── logger.go ├── server │ ├── server_test.go │ └── server.go ├── trie │ ├── aggregator.go │ ├── invparam_test.go │ ├── invparam.go │ ├── keysplitter.go │ ├── node_test.go │ ├── keysplitter_test.go │ ├── trie.go │ ├── node.go │ └── trie_test.go ├── renderer │ ├── renderer.go │ ├── json.go │ ├── renderer_test.go │ ├── json_test.go │ ├── table.go │ ├── chart_test.go │ ├── table_test.go │ └── chart.go ├── adapter │ ├── rservice_test.go │ ├── progress.go │ ├── progress_mock_test.go │ ├── progress_test.go │ └── rservice.go └── scanner │ ├── scanner.go │ └── scanner_test.go ├── go.mod ├── .github └── dependabot.yml ├── .travis.yml ├── .devcontainer └── devcontainer.json ├── LICENSE ├── README.md └── go.sum /.gitignore: -------------------------------------------------------------------------------- 1 | /vendor 2 | /.idea 3 | /build 4 | -------------------------------------------------------------------------------- /docs/images/diagram.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/genius/redis-inventory/master/docs/images/diagram.png -------------------------------------------------------------------------------- /docs/images/diagram1000.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/genius/redis-inventory/master/docs/images/diagram1000.png -------------------------------------------------------------------------------- /main.go: -------------------------------------------------------------------------------- 1 | package main 2 | 3 | import ( 4 | "github.com/obukhov/redis-inventory/cmd/app" 5 | ) 6 | 7 | func main() { 8 | app.Execute() 9 | } 10 | -------------------------------------------------------------------------------- /cmd/app/vars.go: -------------------------------------------------------------------------------- 1 | package app 2 | 3 | var ( 4 | output, outputParams, separators, pattern string 5 | maxChildren, scanCount, throttleNs, samplePerc int 6 | logLevel string 7 | ) 8 | -------------------------------------------------------------------------------- /cmd/docgen/main.go: -------------------------------------------------------------------------------- 1 | package main 2 | 3 | import ( 4 | "github.com/obukhov/redis-inventory/cmd/app" 5 | "github.com/spf13/cobra/doc" 6 | "log" 7 | ) 8 | 9 | func main() { 10 | err := doc.GenMarkdownTree(app.RootCmd, "./docs/cobra") 11 | if err != nil { 12 | log.Fatal(err) 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /docker-compose.yml: -------------------------------------------------------------------------------- 1 | version: '3.7' 2 | services: 3 | redis: 4 | image: redis:6.2-alpine 5 | command: redis-server 6 | ports: 7 | - "${REDIS_PORT:-63795}:6379" 8 | redis-auth: 9 | image: redis:6.2-alpine 10 | command: redis-server --requirepass 12345 11 | ports: 12 | - "${REDIS_PORT:-63796}:6379" 13 | -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | FROM golang:1.17.7-alpine as build 2 | 3 | WORKDIR /go/src/ 4 | 5 | COPY go.mod go.sum ./ 6 | RUN go mod download 7 | 8 | COPY . . 9 | 10 | RUN go build -o /go/bin/redis-inventory 11 | 12 | ENTRYPOINT ["/go/bin/redis-inventory"] 13 | 14 | FROM alpine:3.15.1 AS dist 15 | 16 | WORKDIR /go/bin/ 17 | 18 | COPY --from=build /go/bin/ . 19 | 20 | ENTRYPOINT ["/go/bin/redis-inventory"] 21 | -------------------------------------------------------------------------------- /.pre-commit-config.yaml: -------------------------------------------------------------------------------- 1 | repos: 2 | - repo: https://github.com/pre-commit/pre-commit-hooks 3 | rev: v2.3.0 4 | hooks: 5 | - id: check-yaml 6 | - id: end-of-file-fixer 7 | - id: trailing-whitespace 8 | - repo: git://github.com/dnephin/pre-commit-golang 9 | rev: master 10 | hooks: 11 | - id: go-fmt 12 | - id: go-vet 13 | - id: go-lint 14 | - id: go-imports 15 | -------------------------------------------------------------------------------- /makefile: -------------------------------------------------------------------------------- 1 | #!make 2 | .PHONY: help docs build test 3 | .DEFAULT_GOAL := help 4 | 5 | help:## Show this help. 6 | @grep -E '^[a-zA-Z_-]+:.*?##\s*.*$$' $(MAKEFILE_LIST) | sort | awk 'BEGIN {FS = ":.*?##\\s*"}; {printf "\033[36m%-30s\033[0m %s\n", $$1, $$2}' 7 | 8 | test:## Run tests 9 | go test ./... 10 | 11 | docs:## Generate docs for cobra commands 12 | go run cmd/docgen/main.go 13 | 14 | build:## Build docker services 15 | go build -o build/redis-inventory main.go 16 | -------------------------------------------------------------------------------- /docs/redis-url.md: -------------------------------------------------------------------------------- 1 | Redis URL 2 | ======== 3 | 4 | Redis URL can be provided in one of two formats: 5 | 6 | - `:` - simplified format, 7 | - `redis://[:@]:[/]` - in case if you want to specify password or DB Index for the connection. 8 | 9 | Examples (can be used with provided docker-compose): 10 | - `redis-inventory localhost:63795` 11 | - `redis-inventory redis://localhost:63795/1` 12 | - `redis-inventory redis://:12345@localhost:63795` 13 | - `redis-inventory redis://:12345@localhost:63795/3` -------------------------------------------------------------------------------- /src/logger/logger.go: -------------------------------------------------------------------------------- 1 | package logger 2 | 3 | import ( 4 | "github.com/rs/zerolog" 5 | "os" 6 | ) 7 | 8 | // NewConsoleLogger creates cli friendly logger with given minimal logLevel 9 | func NewConsoleLogger(logLevel string) zerolog.Logger { 10 | 11 | logger := zerolog.New(zerolog.ConsoleWriter{Out: os.Stderr}).With().Timestamp().Logger() 12 | parsedLevel, err := zerolog.ParseLevel(logLevel) 13 | if err != nil { 14 | logger.Fatal().Msgf("Cannot parse log level: %s", logLevel) 15 | } 16 | 17 | return logger.Level(parsedLevel) 18 | } 19 | -------------------------------------------------------------------------------- /go.mod: -------------------------------------------------------------------------------- 1 | module github.com/obukhov/redis-inventory 2 | 3 | go 1.16 4 | 5 | require ( 6 | code.cloudfoundry.org/bytefmt v0.0.0-20210608160410-67692ebc98de 7 | github.com/alicebob/miniredis/v2 v2.18.0 8 | github.com/hetiansu5/urlquery v1.2.7 9 | github.com/jedib0t/go-pretty/v6 v6.2.7 10 | github.com/mediocregopher/radix/v4 v4.0.0 11 | github.com/mitchellh/go-homedir v1.1.0 12 | github.com/rs/zerolog v1.26.1 13 | github.com/spf13/cobra v1.4.0 14 | github.com/spf13/viper v1.10.1 15 | github.com/stretchr/testify v1.7.1 16 | golang.org/x/text v0.3.7 17 | ) 18 | -------------------------------------------------------------------------------- /.github/dependabot.yml: -------------------------------------------------------------------------------- 1 | # https://docs.github.com/en/github/administering-a-repository/configuration-options-for-dependency-updates 2 | version: 2 3 | updates: 4 | - package-ecosystem: "docker" 5 | directory: "/" 6 | pull-request-branch-name: 7 | separator: "-" 8 | reviewers: ["obukhov"] 9 | schedule: 10 | interval: "daily" 11 | open-pull-requests-limit: 2 12 | 13 | - package-ecosystem: "gomod" 14 | directory: "/" 15 | pull-request-branch-name: 16 | separator: "-" 17 | reviewers: ["obukhov"] 18 | schedule: 19 | interval: "daily" 20 | open-pull-requests-limit: 5 21 | -------------------------------------------------------------------------------- /docs/cobra/redis-inventory.md: -------------------------------------------------------------------------------- 1 | ## redis-inventory 2 | 3 | 4 | 5 | ### Synopsis 6 | 7 | 8 | 9 | ### Options 10 | 11 | ``` 12 | -h, --help help for redis-inventory 13 | ``` 14 | 15 | ### SEE ALSO 16 | 17 | * [redis-inventory display](redis-inventory_display.md) - Reads cached usage data created with `index` command and displays according to output and output params 18 | * [redis-inventory index](redis-inventory_index.md) - Scan keys and save prefix tree in a temporary file for further rendering with display command 19 | * [redis-inventory inventory](redis-inventory_inventory.md) - Scan keys and display summary right away with selected output and output params 20 | 21 | ###### Auto generated by spf13/cobra on 15-Sep-2021 22 | -------------------------------------------------------------------------------- /src/server/server_test.go: -------------------------------------------------------------------------------- 1 | package server 2 | 3 | import ( 4 | "github.com/rs/zerolog" 5 | "github.com/stretchr/testify/suite" 6 | "io/ioutil" 7 | "net/http" 8 | "net/http/httptest" 9 | "testing" 10 | "time" 11 | ) 12 | 13 | type ServerTestSuite struct { 14 | suite.Suite 15 | } 16 | 17 | func (suite *ServerTestSuite) TestRenderParam() { 18 | srv := NewServer(zerolog.Nop()) 19 | 20 | go srv.Serve(9999, "test content to serve") 21 | time.Sleep(time.Millisecond) 22 | 23 | req, _ := http.NewRequest("GET", "/", nil) 24 | 25 | recorder := httptest.NewRecorder() 26 | srv.ServeHTTP(recorder, req) 27 | 28 | body, _ := ioutil.ReadAll(recorder.Result().Body) 29 | suite.Assert().Equal("test content to serve", string(body)) 30 | } 31 | 32 | func TestServerTestSuite(t *testing.T) { 33 | suite.Run(t, new(ServerTestSuite)) 34 | } 35 | -------------------------------------------------------------------------------- /src/trie/aggregator.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | // ParamValue value for inventory param 4 | type ParamValue struct { 5 | Param InvParam 6 | Value int64 7 | } 8 | 9 | // NewAggregator creates Aggregator 10 | func NewAggregator() *Aggregator { 11 | return &Aggregator{ 12 | Params: make(map[InvParam]int64), 13 | } 14 | } 15 | 16 | // Aggregator struct holding various inventory param values 17 | type Aggregator struct { 18 | Params map[InvParam]int64 19 | } 20 | 21 | // Add adds inv parameter value to aggregation 22 | func (a *Aggregator) Add(param InvParam, val int64) { 23 | a.Params[param] += val 24 | } 25 | 26 | // Clone creates a copy of aggregator 27 | func (a *Aggregator) Clone() *Aggregator { 28 | cloned := NewAggregator() 29 | for k, v := range a.Params { 30 | cloned.Params[k] = v 31 | } 32 | 33 | return cloned 34 | } 35 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | dist: trusty 2 | 3 | jobs: 4 | include: 5 | - stage: run tests 6 | language: go 7 | go: 8 | - 1.16.x 9 | before_install: 10 | - go get github.com/mattn/goveralls 11 | script: 12 | - $GOPATH/bin/goveralls -service=travis-ci 13 | - stage: build docker image 14 | if: type != pull_request AND branch = master 15 | script: 16 | - echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin 17 | - docker build -t redis-inventory . 18 | - docker images 19 | - docker tag redis-inventory $DOCKER_USERNAME/redis-inventory 20 | - docker push $DOCKER_USERNAME/redis-inventory 21 | - stage: test docker image 22 | if: type != pull_request AND branch = master 23 | script: docker run --rm $DOCKER_USERNAME/redis-inventory help 24 | -------------------------------------------------------------------------------- /docs/cobra/redis-inventory_display.md: -------------------------------------------------------------------------------- 1 | ## redis-inventory display 2 | 3 | Reads cached usage data created with `index` command and displays according to output and output params 4 | 5 | ### Synopsis 6 | 7 | It can be useful to play with different display params, for example: depth, padding, human readable formatting 8 | 9 | ``` 10 | redis-inventory display [flags] 11 | ``` 12 | 13 | ### Options 14 | 15 | ``` 16 | -h, --help help for display 17 | -l, --logLevel string Level of logs to be displayed (default "info") 18 | -o, --output string One of possible outputs: json, jsonp, table (default "table") 19 | -p, --output-params string Parameters specific for output type 20 | ``` 21 | 22 | ### SEE ALSO 23 | 24 | * [redis-inventory](redis-inventory.md) - 25 | 26 | ###### Auto generated by spf13/cobra on 15-Sep-2021 27 | -------------------------------------------------------------------------------- /.devcontainer/devcontainer.json: -------------------------------------------------------------------------------- 1 | // For format details, see https://aka.ms/devcontainer.json. For config options, see the 2 | // README at: https://github.com/devcontainers/templates/tree/main/src/go 3 | { 4 | "name": "Go", 5 | // Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile 6 | "image": "mcr.microsoft.com/devcontainers/go:1-1.21-bullseye" 7 | 8 | // Features to add to the dev container. More info: https://containers.dev/features. 9 | // "features": {}, 10 | 11 | // Use 'forwardPorts' to make a list of ports inside the container available locally. 12 | // "forwardPorts": [], 13 | 14 | // Use 'postCreateCommand' to run commands after the container is created. 15 | // "postCreateCommand": "go version", 16 | 17 | // Configure tool-specific properties. 18 | // "customizations": {}, 19 | 20 | // Uncomment to connect as root instead. More info: https://aka.ms/dev-containers-non-root. 21 | // "remoteUser": "root" 22 | } 23 | -------------------------------------------------------------------------------- /src/trie/invparam_test.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | import ( 4 | "encoding/json" 5 | "github.com/stretchr/testify/suite" 6 | "testing" 7 | ) 8 | 9 | type InvParamTestSuite struct { 10 | suite.Suite 11 | } 12 | 13 | func (suite *InvParamTestSuite) TestStringPanic() { 14 | suite.Assert().Panics(func() { 15 | p := InvParam(9999) 16 | _ = p.String() 17 | }) 18 | } 19 | 20 | func (suite *InvParamTestSuite) TestUnmarshalText() { 21 | jsonString := `["BytesSize", "KeysCount"]` 22 | var result []InvParam 23 | 24 | err := json.Unmarshal([]byte(jsonString), &result) 25 | 26 | suite.Assert().Nil(err) 27 | suite.Assert().Equal([]InvParam{BytesSize, KeysCount}, result) 28 | } 29 | 30 | func (suite *InvParamTestSuite) TestUnmarshalTextError() { 31 | jsonString := `["BytesSizeFoo", "KeysCount"]` 32 | var result []InvParam 33 | 34 | err := json.Unmarshal([]byte(jsonString), &result) 35 | 36 | suite.Assert().NotNil(err) 37 | } 38 | 39 | func TestInvParamTestSuite(t *testing.T) { 40 | suite.Run(t, new(InvParamTestSuite)) 41 | } 42 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2021 Aleksandr Obukhov 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /src/trie/invparam.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | import ( 4 | "errors" 5 | "strconv" 6 | ) 7 | 8 | // InvParam represent type or inventory parameter 9 | type InvParam uint 10 | 11 | const ( 12 | // BytesSize size of the values in bytes 13 | BytesSize InvParam = iota 14 | // KeysCount number of keys 15 | KeysCount 16 | ) 17 | 18 | // String implements stringer interface to display InvParam as a string 19 | func (p InvParam) String() string { 20 | switch p { 21 | case BytesSize: 22 | return "BytesSize" 23 | case KeysCount: 24 | return "KeysCount" 25 | } 26 | 27 | panic("Unknown InvParam: " + strconv.Itoa(int(p))) 28 | } 29 | 30 | // MarshalText renders InvParam as a string when marshalling 31 | func (p InvParam) MarshalText() (text []byte, err error) { 32 | return []byte(p.String()), nil 33 | } 34 | 35 | // UnmarshalText decodes InvParam value from string 36 | func (p *InvParam) UnmarshalText(text []byte) error { 37 | var tmp InvParam 38 | switch string(text) { 39 | case "BytesSize": 40 | tmp = BytesSize 41 | case "KeysCount": 42 | tmp = KeysCount 43 | default: 44 | return errors.New("Unknown InvParam " + string(text)) 45 | } 46 | 47 | *p = tmp 48 | 49 | return nil 50 | } 51 | -------------------------------------------------------------------------------- /src/renderer/renderer.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "errors" 5 | "github.com/obukhov/redis-inventory/src/server" 6 | "github.com/obukhov/redis-inventory/src/trie" 7 | "github.com/rs/zerolog" 8 | "os" 9 | ) 10 | 11 | // Renderer abstraction for rendering trie to a given output 12 | type Renderer interface { 13 | // Render executes rendering 14 | Render(root *trie.Node) error 15 | } 16 | 17 | // NewRenderer creates Renderer implementation by type and set of params 18 | func NewRenderer(output, paramsString string, logger zerolog.Logger) (Renderer, error) { 19 | switch output { 20 | case "table": 21 | params, err := NewTableRendererParams(paramsString) 22 | if err != nil { 23 | return nil, err 24 | } 25 | 26 | return TableRenderer{os.Stdout, params}, nil 27 | case "json": 28 | params, err := NewJSONRendererParams(paramsString) 29 | if err != nil { 30 | return nil, err 31 | } 32 | 33 | return JSONRenderer{os.Stdout, params}, nil 34 | 35 | case "chart": 36 | params, err := NewChartRendererParams(paramsString) 37 | if err != nil { 38 | return nil, err 39 | } 40 | 41 | return NewChartRenderer(server.NewServer(logger), params), nil 42 | default: 43 | return nil, errors.New("unknown render type") 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /src/trie/keysplitter.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | // Splitter abstraction to split string in fragments 4 | type Splitter interface { 5 | // Split splits string key to fragments with given strategy 6 | Split(in string) []string 7 | } 8 | 9 | // PunctuationSplitter splitting keys by a specific set of symbols (i.e. punctuation) 10 | type PunctuationSplitter struct { 11 | dividers map[rune]bool 12 | } 13 | 14 | // NewPunctuationSplitter creates PunctuationSplitter 15 | func NewPunctuationSplitter(punctuation ...rune) *PunctuationSplitter { 16 | m := make(map[rune]bool) 17 | for _, r := range punctuation { 18 | m[r] = true 19 | } 20 | 21 | return &PunctuationSplitter{dividers: m} 22 | } 23 | 24 | // Split splits string key to fragments with given strategy 25 | func (s *PunctuationSplitter) Split(in string) []string { 26 | result := make([]string, 0) 27 | 28 | cur := "" 29 | isPunkt := false 30 | 31 | for _, c := range in { 32 | if _, found := s.dividers[c]; found { 33 | isPunkt = true 34 | cur = cur + string(c) 35 | } else { 36 | if isPunkt { 37 | result = append(result, cur) 38 | cur = string(c) 39 | isPunkt = false 40 | } else { 41 | cur = cur + string(c) 42 | } 43 | } 44 | } 45 | 46 | result = append(result, cur) 47 | 48 | return result 49 | } 50 | -------------------------------------------------------------------------------- /docs/cobra/redis-inventory_index.md: -------------------------------------------------------------------------------- 1 | ## redis-inventory index 2 | 3 | Scan keys and save prefix tree in a temporary file for further rendering with display command 4 | 5 | ### Synopsis 6 | 7 | Keep in mind that some options are scanning (index) options that cannot be redefined later. For example, `maxChildren` changes the way index data is built, unlike `depth` parameter only influencing rendering 8 | 9 | ``` 10 | redis-inventory index redis://[:@]:[/] [flags] 11 | ``` 12 | 13 | ### Options 14 | 15 | ``` 16 | -h, --help help for index 17 | -l, --logLevel string Level of logs to be displayed (default "info") 18 | -m, --maxChildren int Maximum children node can have before start aggregating (default 10) 19 | -k, --pattern string Glob pattern limiting the keys to be aggregated (default "*") 20 | -c, --scanCount int Number of keys to be scanned in one iteration (argument of scan command) (default 1000) 21 | -s, --separators string Symbols that logically separate levels of the key (default ":") 22 | -t, --throttle int Throttle: number of nanoseconds to sleep between keys 23 | ``` 24 | 25 | ### SEE ALSO 26 | 27 | * [redis-inventory](redis-inventory.md) - 28 | 29 | ###### Auto generated by spf13/cobra on 15-Sep-2021 30 | -------------------------------------------------------------------------------- /docs/cobra/redis-inventory_inventory.md: -------------------------------------------------------------------------------- 1 | ## redis-inventory inventory 2 | 3 | Scan keys and display summary right away with selected output and output params 4 | 5 | ### Synopsis 6 | 7 | Scan command builds prefix tree in memory and then displays the usage summary. To avoid scanning redis instance when trying different output formats use `index` and `display` commands 8 | 9 | ``` 10 | redis-inventory inventory redis://[:@]:[/] [flags] 11 | ``` 12 | 13 | ### Options 14 | 15 | ``` 16 | -h, --help help for inventory 17 | -l, --logLevel string Level of logs to be displayed (default "info") 18 | -m, --maxChildren int Maximum children node can have before start aggregating (default 10) 19 | -o, --output string One of possible outputs: json, jsonp, table (default "table") 20 | -p, --output-params string Parameters specific for output type 21 | -k, --pattern string Glob pattern limiting the keys to be aggregated (default "*") 22 | -c, --scanCount int Number of keys to be scanned in one iteration (argument of scan command) (default 1000) 23 | -s, --separators string Symbols that logically separate levels of the key (default ":") 24 | -t, --throttle int Throttle: number of nanoseconds to sleep between keys 25 | ``` 26 | 27 | ### SEE ALSO 28 | 29 | * [redis-inventory](redis-inventory.md) - 30 | 31 | ###### Auto generated by spf13/cobra on 15-Sep-2021 32 | -------------------------------------------------------------------------------- /src/renderer/json.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "encoding/json" 5 | "github.com/hetiansu5/urlquery" 6 | "github.com/obukhov/redis-inventory/src/trie" 7 | "io" 8 | "strings" 9 | ) 10 | 11 | // NewJSONRendererParams creates JSONRendererParams 12 | func NewJSONRendererParams(paramsSerialized string) (JSONRendererParams, error) { 13 | params := JSONRendererParams{} 14 | 15 | err := urlquery.Unmarshal([]byte(paramsSerialized), ¶ms) 16 | if err != nil { 17 | return params, err 18 | } 19 | 20 | return params, nil 21 | } 22 | 23 | // JSONRendererParams represents rendering params for Json renderer 24 | type JSONRendererParams struct { 25 | Padding string `query:"padding"` 26 | PaddingSpaceCount int `query:"padSpaces"` 27 | } 28 | 29 | // NewJSONRenderer creates JSONRenderer 30 | func NewJSONRenderer(output io.Writer, params JSONRendererParams) JSONRenderer { 31 | return JSONRenderer{ 32 | output: output, 33 | params: params, 34 | } 35 | } 36 | 37 | // JSONRenderer renders trie in the JSON format 38 | type JSONRenderer struct { 39 | output io.Writer 40 | params JSONRendererParams 41 | } 42 | 43 | // Render executes rendering 44 | func (o JSONRenderer) Render(root *trie.Node) error { 45 | encoder := json.NewEncoder(o.output) 46 | 47 | indent := o.params.Padding + strings.Repeat(" ", o.params.PaddingSpaceCount) 48 | if indent != "" { 49 | encoder.SetIndent("", indent) 50 | } 51 | 52 | return encoder.Encode(root) 53 | } 54 | -------------------------------------------------------------------------------- /src/server/server.go: -------------------------------------------------------------------------------- 1 | package server 2 | 3 | import ( 4 | "bufio" 5 | "fmt" 6 | "github.com/rs/zerolog" 7 | "net/http" 8 | "os" 9 | ) 10 | 11 | // SimpleServerInterface abstraction for simplified web server 12 | type SimpleServerInterface interface { 13 | Serve(port int, content string) 14 | } 15 | 16 | // NewServer creates SimpleServer 17 | func NewServer(logger zerolog.Logger) *SimpleServer { 18 | return &SimpleServer{ 19 | logger: logger, 20 | } 21 | } 22 | 23 | // SimpleServer is SimpleServerInterface implementation 24 | type SimpleServer struct { 25 | content string 26 | logger zerolog.Logger 27 | } 28 | 29 | // Serve initiates server, block for input on stdin 30 | func (s *SimpleServer) Serve(port int, content string) { 31 | s.content = content 32 | go func() { 33 | s.logger.Info().Msgf("Listening on port %d: http://localhost:%d/", port, port) 34 | s.logger.Info().Msg("Press enter to exit") 35 | err := http.ListenAndServe(fmt.Sprintf(":%d", port), s) 36 | s.logger.Fatal().Err(err).Msg("Error serving content") 37 | }() 38 | 39 | input := bufio.NewScanner(os.Stdin) 40 | input.Scan() 41 | } 42 | 43 | // ServeHTTP handler function 44 | func (s *SimpleServer) ServeHTTP(w http.ResponseWriter, r *http.Request) { 45 | countBytes, err := w.Write([]byte(s.content)) 46 | if err != nil { 47 | s.logger.Fatal().Err(err).Msg("Error handling request") 48 | } else { 49 | s.logger.Debug().Msgf("Handled request with %d bytes", countBytes) 50 | } 51 | } 52 | -------------------------------------------------------------------------------- /src/trie/node_test.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | import ( 4 | "github.com/stretchr/testify/suite" 5 | "testing" 6 | ) 7 | 8 | type NodeTestSuite struct { 9 | suite.Suite 10 | } 11 | 12 | func (suite *NodeTestSuite) TestFirstChildPanic() { 13 | node := NewNode() 14 | suite.Assert().Panics(func() { 15 | node.FirstChild() 16 | }) 17 | } 18 | 19 | func (suite *NodeTestSuite) TestFirstChildWithKeyPanic() { 20 | node := NewNode() 21 | suite.Assert().Panics(func() { 22 | node.FirstChildWithKey() 23 | }) 24 | } 25 | 26 | func (suite *NodeTestSuite) TestFindNextAggregatedNode() { 27 | node := NewNode() 28 | 29 | level1 := NewNode() 30 | level2 := NewNode() 31 | level3one := NewNode() 32 | level3two := NewNode() 33 | 34 | level2.AddChild("one", level3one) 35 | level2.AddChild("two", level3two) 36 | level2.AddAggregator(NewAggregator()) 37 | 38 | level1.AddChild("bar", level2) 39 | node.AddChild("foo", level1) 40 | 41 | actual := node.FindNextAggregatedNode() 42 | 43 | suite.Assert().Equal(level2, actual) 44 | } 45 | 46 | func (suite *NodeTestSuite) TestFindNextAggregatedNodeWithPath() { 47 | node := NewNode() 48 | 49 | level1 := NewNode() 50 | level2 := NewNode() 51 | level3one := NewNode() 52 | level3two := NewNode() 53 | 54 | level2.AddChild("one", level3one) 55 | level2.AddChild("two", level3two) 56 | level2.AddAggregator(NewAggregator()) 57 | 58 | level1.AddChild("bar", level2) 59 | node.AddChild("foo", level1) 60 | 61 | actгalKeys, actual := node.FindNextAggregatedNodeWithKey() 62 | 63 | suite.Assert().Equal(level2, actual) 64 | suite.Assert().Equal([]string{"foo", "bar"}, actгalKeys) 65 | } 66 | func TestNodeTestSuiteTestSuite(t *testing.T) { 67 | suite.Run(t, new(NodeTestSuite)) 68 | } 69 | -------------------------------------------------------------------------------- /src/renderer/renderer_test.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "github.com/rs/zerolog" 5 | "github.com/stretchr/testify/suite" 6 | "testing" 7 | ) 8 | 9 | type RendererTestSuite struct { 10 | suite.Suite 11 | } 12 | 13 | func (suite *RendererTestSuite) TestNewRender() { 14 | for _, t := range []struct { 15 | output string 16 | outputParams string 17 | expectedType Renderer 18 | }{ 19 | { 20 | "table", 21 | "depth=2", 22 | TableRenderer{}, 23 | }, 24 | { 25 | "json", 26 | "", 27 | JSONRenderer{}, 28 | }, 29 | { 30 | "chart", 31 | "", 32 | ChartRenderer{}, 33 | }, 34 | } { 35 | suite.Run(t.outputParams, func() { 36 | renderer, err := NewRenderer(t.output, t.outputParams, zerolog.Nop()) 37 | 38 | suite.Assert().Nil(err) 39 | suite.Assert().IsTypef(t.expectedType, renderer, "Unexpected type") 40 | }) 41 | } 42 | } 43 | 44 | func (suite *RendererTestSuite) TestNewRenderWithError() { 45 | for _, t := range []struct { 46 | output string 47 | outputParams string 48 | }{ 49 | { 50 | "table", 51 | "padSpaces=asd", 52 | }, 53 | { 54 | "json", 55 | "padSpaces=asd", 56 | }, 57 | { 58 | "chart", 59 | "port=-1", 60 | }, 61 | } { 62 | suite.Run(t.outputParams, func() { 63 | renderer, err := NewRenderer(t.output, t.outputParams, zerolog.Nop()) 64 | 65 | suite.Assert().Nil(renderer) 66 | suite.Assert().Error(err) 67 | }) 68 | } 69 | } 70 | func (suite *RendererTestSuite) TestNewRenderError() { 71 | renderer, err := NewRenderer("foo", "", zerolog.Nop()) 72 | 73 | suite.Assert().NotNil(err) 74 | suite.Assert().Nil(renderer) 75 | } 76 | 77 | func TestRendererTestSuite(t *testing.T) { 78 | suite.Run(t, new(RendererTestSuite)) 79 | } 80 | -------------------------------------------------------------------------------- /src/trie/keysplitter_test.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | import ( 4 | "github.com/stretchr/testify/suite" 5 | "testing" 6 | ) 7 | 8 | type PunctuationSplitterTestSuite struct { 9 | suite.Suite 10 | splitter Splitter 11 | } 12 | 13 | func (suite *PunctuationSplitterTestSuite) SetupTest() { 14 | suite.splitter = NewPunctuationSplitter('_', ':') 15 | } 16 | 17 | func (suite *PunctuationSplitterTestSuite) TestSplit() { 18 | for _, testcase := range []struct { 19 | in string 20 | out []string 21 | message string 22 | }{ 23 | { 24 | "helloWorld", 25 | []string{"helloWorld"}, 26 | "no punctuation", 27 | }, 28 | { 29 | "hello_world", 30 | []string{"hello_", "world"}, 31 | "basic split", 32 | }, 33 | { 34 | "hello___world", 35 | []string{"hello___", "world"}, 36 | "multiple punctuation in a single split", 37 | }, 38 | { 39 | "hello_my:dear_:world", 40 | []string{"hello_", "my:", "dear_:", "world"}, 41 | "basic split", 42 | }, 43 | { 44 | "_hello_world", 45 | []string{"_", "hello_", "world"}, 46 | "starting with punctuation", 47 | }, 48 | { 49 | "___hello_world", 50 | []string{"___", "hello_", "world"}, 51 | "starting with multiple punctuation", 52 | }, 53 | { 54 | "hello_world___", 55 | []string{"hello_", "world___"}, 56 | "ending with multiple punctuation", 57 | }, 58 | { 59 | "_", 60 | []string{"_"}, 61 | "just punctuation", 62 | }, 63 | { 64 | "", 65 | []string{""}, 66 | "empty string", 67 | }, 68 | } { 69 | suite.Equal(suite.splitter.Split(testcase.in), testcase.out, testcase.message) 70 | } 71 | } 72 | 73 | func TestPunctuationSplitterTestSuite(t *testing.T) { 74 | suite.Run(t, new(PunctuationSplitterTestSuite)) 75 | } 76 | -------------------------------------------------------------------------------- /src/adapter/rservice_test.go: -------------------------------------------------------------------------------- 1 | package adapter 2 | 3 | import ( 4 | "context" 5 | "testing" 6 | 7 | "github.com/alicebob/miniredis/v2" 8 | "github.com/mediocregopher/radix/v4" 9 | "github.com/stretchr/testify/suite" 10 | ) 11 | 12 | type RedisServiceTestSuite struct { 13 | suite.Suite 14 | service RedisService 15 | miniredis *miniredis.Miniredis 16 | } 17 | 18 | func (suite *RedisServiceTestSuite) createRedis() (RedisService, *miniredis.Miniredis) { 19 | var err error 20 | m, err := miniredis.Run() 21 | if err != nil { 22 | panic(err) 23 | } 24 | 25 | m.Set("dev:key1", "bar") 26 | m.Set("dev:key2", "foobar") 27 | 28 | client, _ := (radix.PoolConfig{}).New(context.Background(), "tcp", m.Addr()) 29 | 30 | service := NewRedisService(client) 31 | 32 | return service, m 33 | } 34 | 35 | func (suite *RedisServiceTestSuite) TestCountKeys() { 36 | service, m := suite.createRedis() 37 | 38 | count, err := service.GetKeysCount(context.Background()) 39 | 40 | suite.Assert().Nil(err) 41 | suite.Assert().Equal(int64(2), count) 42 | 43 | m.Close() 44 | } 45 | 46 | func (suite *RedisServiceTestSuite) TestScan() { 47 | service, m := suite.createRedis() 48 | 49 | res := service.ScanKeys(context.Background(), ScanOptions{ScanCount: 1000}) 50 | 51 | key1 := <-res 52 | key2 := <-res 53 | 54 | suite.Assert().Equal("dev:key1", key1) 55 | suite.Assert().Equal("dev:key2", key2) 56 | 57 | m.Close() 58 | } 59 | 60 | func (suite *RedisServiceTestSuite) TestScanMatch() { 61 | service, m := suite.createRedis() 62 | 63 | res := service.ScanKeys(context.Background(), ScanOptions{ScanCount: 1000, Pattern: "*:key1"}) 64 | 65 | key1 := <-res 66 | suite.Assert().Equal("dev:key1", key1) 67 | 68 | m.Close() 69 | 70 | } 71 | 72 | func TestRedisServiceTestSuite(t *testing.T) { 73 | suite.Run(t, new(RedisServiceTestSuite)) 74 | } 75 | -------------------------------------------------------------------------------- /cmd/app/display.go: -------------------------------------------------------------------------------- 1 | package app 2 | 3 | import ( 4 | "encoding/json" 5 | "os" 6 | 7 | "github.com/obukhov/redis-inventory/src/logger" 8 | "github.com/obukhov/redis-inventory/src/renderer" 9 | "github.com/obukhov/redis-inventory/src/trie" 10 | "github.com/spf13/cobra" 11 | ) 12 | 13 | var displayCmd = &cobra.Command{ 14 | Use: "display", 15 | Short: "Reads cached usage data created with `index` command and displays according to output and output params", 16 | Long: "It can be useful to play with different display params, for example: depth, padding, human readable formatting", 17 | Args: cobra.MinimumNArgs(0), 18 | Run: func(cmd *cobra.Command, args []string) { 19 | consoleLogger := logger.NewConsoleLogger(logLevel) 20 | consoleLogger.Info().Msg("Loading index") 21 | 22 | var indexFileName string 23 | if len(args) > 0 { 24 | indexFileName = args[0] 25 | } else { 26 | indexFileName = os.TempDir() + "/redis-inventory.json" 27 | } 28 | 29 | f, err := os.Open(indexFileName) 30 | if err != nil { 31 | consoleLogger.Fatal().Err(err).Msg("Can't create renderer") 32 | } 33 | 34 | var root *trie.Node 35 | dec := json.NewDecoder(f) 36 | err = dec.Decode(&root) 37 | if err != nil { 38 | consoleLogger.Fatal().Err(err).Msg("Can't decode file") 39 | } 40 | 41 | r, err := renderer.NewRenderer(output, outputParams, consoleLogger) 42 | if err != nil { 43 | consoleLogger.Fatal().Err(err).Msg("Can't create renderer") 44 | } 45 | 46 | err = r.Render(root) 47 | if err != nil { 48 | consoleLogger.Fatal().Err(err).Msg("Can't render report") 49 | } 50 | 51 | consoleLogger.Info().Msg("Done") 52 | }, 53 | } 54 | 55 | func init() { 56 | RootCmd.AddCommand(displayCmd) 57 | displayCmd.Flags().StringVarP(&output, "output", "o", "table", "One of possible outputs: json, jsonp, table") 58 | displayCmd.Flags().StringVarP(&outputParams, "output-params", "p", "", "Parameters specific for output type") 59 | displayCmd.Flags().StringVarP(&logLevel, "logLevel", "l", "info", "Level of logs to be displayed") 60 | } 61 | -------------------------------------------------------------------------------- /src/scanner/scanner.go: -------------------------------------------------------------------------------- 1 | package scanner 2 | 3 | import ( 4 | "context" 5 | 6 | "github.com/obukhov/redis-inventory/src/adapter" 7 | "github.com/obukhov/redis-inventory/src/trie" 8 | "github.com/rs/zerolog" 9 | ) 10 | 11 | // RedisServiceInterface abstraction to access redis 12 | type RedisServiceInterface interface { 13 | ScanKeys(ctx context.Context, options adapter.ScanOptions) <-chan adapter.BulkKeyInfo 14 | GetKeysCount(ctx context.Context) (int64, error) 15 | } 16 | 17 | // RedisScanner scans redis keys and puts them in a trie 18 | type RedisScanner struct { 19 | redisService RedisServiceInterface 20 | scanProgress adapter.ProgressWriter 21 | logger zerolog.Logger 22 | } 23 | 24 | // NewScanner creates RedisScanner 25 | func NewScanner(redisService RedisServiceInterface, scanProgress adapter.ProgressWriter, logger zerolog.Logger) *RedisScanner { 26 | return &RedisScanner{ 27 | redisService: redisService, 28 | scanProgress: scanProgress, 29 | logger: logger, 30 | } 31 | } 32 | 33 | // Scan initiates scanning process 34 | func (s *RedisScanner) Scan(options adapter.ScanOptions, result *trie.Trie) { 35 | var totalCount int64 36 | if options.Pattern == "*" || options.Pattern == "" { 37 | totalCount = s.getKeysCount() 38 | } 39 | 40 | s.scanProgress.Start((totalCount * int64(options.SamplePerc)) / 100) 41 | for keyResult := range s.redisService.ScanKeys(context.Background(), options) { 42 | for index, key := range keyResult.Keys { 43 | s.scanProgress.Increment() 44 | 45 | result.Add( 46 | key, 47 | trie.ParamValue{Param: trie.BytesSize, Value: keyResult.Sizes[index] * int64(100/options.SamplePerc)}, 48 | trie.ParamValue{Param: trie.KeysCount, Value: int64(100 / options.SamplePerc)}, 49 | ) 50 | 51 | // if index%10 == 0 { 52 | // s.logger.Debug().Msgf("Dump %s value: %d", key, keyResult.Sizes[index]) 53 | // } 54 | } 55 | } 56 | s.scanProgress.Stop() 57 | } 58 | 59 | func (s *RedisScanner) getKeysCount() int64 { 60 | res, err := s.redisService.GetKeysCount(context.Background()) 61 | if err != nil { 62 | s.logger.Error().Err(err).Msgf("Error getting number of keys") 63 | return 0 64 | } 65 | 66 | return res 67 | } 68 | -------------------------------------------------------------------------------- /src/trie/trie.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | // NewTrie created Trie 4 | func NewTrie(splitter Splitter, maxChildren int) *Trie { 5 | node := NewNode() 6 | node.AddAggregator(NewAggregator()) 7 | 8 | return &Trie{ 9 | root: node, 10 | splitter: splitter, 11 | maxChildren: maxChildren, 12 | } 13 | } 14 | 15 | // Trie stores data about keys in a prefix tree 16 | type Trie struct { 17 | root *Node 18 | splitter Splitter 19 | maxChildren int 20 | } 21 | 22 | func digitPrefix(key string) int { 23 | lastDigit := -1 24 | for index, c := range key { 25 | if c < '0' || c > '9' { 26 | lastDigit = index 27 | break 28 | } 29 | } 30 | 31 | return lastDigit 32 | } 33 | 34 | // Add adds information about another key with set of params 35 | func (t *Trie) Add(key string, paramValues ...ParamValue) { 36 | curNode := t.root 37 | for _, keyPiece := range t.splitter.Split(key) { // change to zero allocation segmenter 38 | var nextNode *Node 39 | 40 | prefix := digitPrefix(keyPiece) 41 | if prefix > 0 { 42 | keyPiece = "" + keyPiece[prefix:] 43 | } else if prefix == -1 { 44 | keyPiece = "" 45 | } 46 | 47 | if childNode := curNode.GetChild(keyPiece); childNode == nil { 48 | if curNode.ChildCount() == 1 { 49 | // creating a fork in the trie 50 | nextAggregatedNode := curNode.FindNextAggregatedNode() 51 | curNode.AddAggregator(nextAggregatedNode.Aggregator().Clone()) 52 | } 53 | 54 | if curNode.ChildCount() < t.maxChildren { 55 | nextNode = NewNode() 56 | curNode.AddChild(keyPiece, nextNode) 57 | } else { 58 | curNode.OverflowChildrenCount++ 59 | break 60 | } 61 | 62 | } else { 63 | nextNode = childNode 64 | } 65 | 66 | if curNode.HasAggregator() { 67 | for _, p := range paramValues { 68 | curNode.Aggregator().Add(p.Param, p.Value) 69 | } 70 | } 71 | 72 | curNode = nextNode 73 | } 74 | 75 | if !curNode.HasAggregator() { 76 | if curNode.HasChildren() { 77 | curNode.AddAggregator(curNode.FindNextAggregatedNode().Aggregator().Clone()) 78 | } else { 79 | curNode.AddAggregator(NewAggregator()) 80 | } 81 | } 82 | 83 | for _, p := range paramValues { 84 | curNode.Aggregator().Add(p.Param, p.Value) 85 | } 86 | } 87 | 88 | // Root returns root of the trie 89 | func (t *Trie) Root() *Node { 90 | return t.root 91 | } 92 | -------------------------------------------------------------------------------- /src/adapter/progress.go: -------------------------------------------------------------------------------- 1 | package adapter 2 | 3 | import ( 4 | "github.com/jedib0t/go-pretty/v6/progress" 5 | "io" 6 | "time" 7 | ) 8 | 9 | // ProgressWriter abstraction of progress writer 10 | type ProgressWriter interface { 11 | // Start initiates progress writing progress, if total is unknown should be zero 12 | Start(total int64) 13 | // Increment increments progress 14 | Increment() 15 | // Stop labels progress as finished and stops updating progress 16 | Stop() 17 | } 18 | 19 | // NewPrettyProgressWriter creates PrettyProgressWriter 20 | func NewPrettyProgressWriter(output io.Writer) *PrettyProgressWriter { 21 | p := &PrettyProgressWriter{pw: progress.NewWriter()} 22 | p.init(output) 23 | 24 | return p 25 | } 26 | 27 | // PrettyProgressWriter progress writer using go-pretty/progress library 28 | type PrettyProgressWriter struct { 29 | pw progress.Writer 30 | tracker Tracker 31 | } 32 | 33 | // Tracker is abstraction over libraries "Tracker" struct 34 | type Tracker interface { 35 | Increment(value int64) 36 | MarkAsDone() 37 | } 38 | 39 | func (p *PrettyProgressWriter) init(output io.Writer) { 40 | p.pw.SetAutoStop(false) 41 | p.pw.SetTrackerLength(50) 42 | p.pw.ShowETA(true) 43 | p.pw.ShowOverallTracker(false) 44 | p.pw.ShowTime(true) 45 | p.pw.ShowTracker(true) 46 | p.pw.ShowValue(true) 47 | p.pw.SetMessageWidth(13) 48 | p.pw.SetNumTrackersExpected(1) 49 | p.pw.SetSortBy(progress.SortByPercentDsc) 50 | p.pw.SetStyle(progress.StyleDefault) 51 | p.pw.SetTrackerPosition(progress.PositionRight) 52 | p.pw.SetUpdateFrequency(time.Millisecond * 10) 53 | p.pw.Style().Colors = progress.StyleColorsExample 54 | p.pw.Style().Options.PercentFormat = "%4.1f%%" 55 | p.pw.SetOutputWriter(output) 56 | } 57 | 58 | // Start initiates progress writing progress, if total is unknown should be zero 59 | func (p *PrettyProgressWriter) Start(total int64) { 60 | scanningKeysTracker := &progress.Tracker{Message: "Scanning keys", Total: total, Units: progress.UnitsDefault} 61 | p.pw.AppendTracker(scanningKeysTracker) 62 | p.tracker = scanningKeysTracker 63 | 64 | go p.pw.Render() 65 | } 66 | 67 | // Increment increments progress 68 | func (p *PrettyProgressWriter) Increment() { 69 | p.tracker.Increment(1) 70 | } 71 | 72 | // Stop labels progress as finished and stops updating progress 73 | func (p *PrettyProgressWriter) Stop() { 74 | p.tracker.MarkAsDone() 75 | p.pw.Stop() 76 | } 77 | -------------------------------------------------------------------------------- /src/trie/node.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | // NewNode creates Node 4 | func NewNode() *Node { 5 | return &Node{ 6 | Children: make(map[string]*Node), 7 | Aggr: nil, 8 | } 9 | } 10 | 11 | // Node node of the trie 12 | type Node struct { 13 | Children map[string]*Node `json:"Children,omitempty"` 14 | Aggr *Aggregator `json:"Values,omitempty"` 15 | OverflowChildrenCount uint64 `json:"Overflow,omitempty"` 16 | } 17 | 18 | // HasAggregator returns if the node has aggregator attached 19 | func (n *Node) HasAggregator() bool { 20 | return n.Aggr != nil 21 | } 22 | 23 | // GetChild returns a child node by provided key, if key doesn't exist returns nil 24 | func (n *Node) GetChild(key string) *Node { 25 | return n.Children[key] 26 | } 27 | 28 | // Aggregator Returns aggregator attached to the node 29 | func (n *Node) Aggregator() *Aggregator { 30 | return n.Aggr 31 | } 32 | 33 | // HasChildren returns if the node has at least one child node 34 | func (n *Node) HasChildren() bool { 35 | return len(n.Children) > 0 36 | } 37 | 38 | // AddAggregator adds aggregator 39 | func (n *Node) AddAggregator(aggr *Aggregator) { 40 | n.Aggr = aggr 41 | } 42 | 43 | // AddChild add child nodes on provided key 44 | func (n *Node) AddChild(key string, node *Node) { 45 | n.Children[key] = node 46 | } 47 | 48 | // ChildCount return number of child nodes 49 | func (n *Node) ChildCount() int { 50 | return len(n.Children) 51 | } 52 | 53 | // FirstChild returns child node, panics if there are no child nodes 54 | func (n *Node) FirstChild() *Node { 55 | for _, child := range n.Children { 56 | return child 57 | } 58 | 59 | panic("No Children when called FirstChild") 60 | } 61 | 62 | // FirstChildWithKey return child node and its key, panics if there are no child nodes 63 | func (n *Node) FirstChildWithKey() (string, *Node) { 64 | for key, child := range n.Children { 65 | return key, child 66 | } 67 | 68 | panic("No Children when called FirstChildWithKey") 69 | } 70 | 71 | // FindNextAggregatedNode descends down trie branch till the next node with aggregator 72 | func (n *Node) FindNextAggregatedNode() *Node { 73 | nextNode := n.FirstChild() 74 | for !nextNode.HasAggregator() { 75 | nextNode = nextNode.FirstChild() 76 | } 77 | 78 | return nextNode 79 | } 80 | 81 | // FindNextAggregatedNodeWithKey descends down trie branch till the next node with aggregator and returns key path to it 82 | func (n *Node) FindNextAggregatedNodeWithKey() ([]string, *Node) { 83 | firstKey, nextNode := n.FirstChildWithKey() 84 | keys := []string{firstKey} 85 | for !nextNode.HasAggregator() { 86 | var key string 87 | key, nextNode = nextNode.FirstChildWithKey() 88 | keys = append(keys, key) 89 | } 90 | 91 | return keys, nextNode 92 | } 93 | -------------------------------------------------------------------------------- /cmd/app/root.go: -------------------------------------------------------------------------------- 1 | // Copyright © 2019 NAME HERE 2 | // 3 | // Licensed under the Apache License, Version 2.0 (the "License"); 4 | // you may not use this file except in compliance with the License. 5 | // You may obtain a copy of the License at 6 | // 7 | // http://www.apache.org/licenses/LICENSE-2.0 8 | // 9 | // Unless required by applicable law or agreed to in writing, software 10 | // distributed under the License is distributed on an "AS IS" BASIS, 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 | // See the License for the specific language governing permissions and 13 | // limitations under the License. 14 | 15 | package app 16 | 17 | import ( 18 | "fmt" 19 | "os" 20 | 21 | homedir "github.com/mitchellh/go-homedir" 22 | "github.com/spf13/cobra" 23 | "github.com/spf13/viper" 24 | ) 25 | 26 | var ( 27 | cfgFile string 28 | ) 29 | 30 | // RootCmd represents the base command when called without any subcommands 31 | var RootCmd = &cobra.Command{ 32 | Use: "redis-inventory", 33 | Short: "", 34 | Long: "", 35 | // Uncomment the following line if your bare application 36 | // has an action associated with it: 37 | // Run: func(cmd *cobra.Command, args []string) { }, 38 | } 39 | 40 | // Execute adds all child commands to the root command and sets flags appropriately. 41 | // This is called by main.main(). It only needs to happen once to the RootCmd. 42 | func Execute() { 43 | if err := RootCmd.Execute(); err != nil { 44 | fmt.Println(err) 45 | os.Exit(1) 46 | } 47 | } 48 | 49 | func init() { 50 | cobra.OnInitialize(initConfig) 51 | 52 | // Here you will define your flags and configuration settings. 53 | // Cobra supports persistent flags, which, if defined here, 54 | // will be global for your application. 55 | //RootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is $HOME/.go-redis-migrate.yaml)") 56 | 57 | // Cobra also supports local flags, which will only run 58 | // when this action is called directly. 59 | //RootCmd.Flags().BoolP("toggle", "t", false, "Help message for toggle") 60 | } 61 | 62 | // initConfig reads in config file and ENV variables if set. 63 | func initConfig() { 64 | if cfgFile != "" { 65 | // Use config file from the flag. 66 | viper.SetConfigFile(cfgFile) 67 | } else { 68 | // Find home directory. 69 | home, err := homedir.Dir() 70 | if err != nil { 71 | fmt.Println(err) 72 | os.Exit(1) 73 | } 74 | 75 | // Search config in home directory with name ".go-redis-migrate" (without extension). 76 | viper.AddConfigPath(home) 77 | viper.SetConfigName(".go-redis-migrate") 78 | } 79 | 80 | viper.AutomaticEnv() // read in environment variables that match 81 | 82 | // If a config file is found, read it in. 83 | if err := viper.ReadInConfig(); err == nil { 84 | fmt.Println("Using config file:", viper.ConfigFileUsed()) 85 | } 86 | } 87 | -------------------------------------------------------------------------------- /docs/installation.md: -------------------------------------------------------------------------------- 1 | ## Installation 2 | 3 | There are two ways to install the tool: 4 | 5 | - use docker image 6 | - building from sources 7 | 8 | ### Using docker 9 | 10 | To run the tool from a docker image, run the command: 11 | 12 | ```bash 13 | docker run --rm -v "$PWD:/tmp" -p 8888:8888 dclg/redis-inventory inventory 14 | ``` 15 | 16 | Mounting temp dir (`-v $PWD:/tmp`) makes it possible to run sequence of `index` and `display` while keeping the state 17 | between runs. 18 | 19 | Exposing a port (`-p 8888:8888`) is needed for `chart` output type. 20 | 21 | If you are using the tool regularly, create an alias: 22 | 23 | ```bash 24 | alias ri="docker run --rm -v \"$PWD:/tmp\" -p 8888:8888 dclg/redis-inventory" 25 | ``` 26 | 27 | So you can run it just like so: 28 | 29 | ```bash 30 | $ ri index : 31 | 11:52PM INF Start indexing 32 | Scanning keys ... done! [3.18K in 262ms] 33 | 11:52PM INF Finish scanning and saved index as a file /tmp/redis-inventory.json 34 | 35 | $ ri display --output-params="depth=2&padSpaces=2" 36 | 11:53PM INF Loading index 37 | +---------------------+----------+-----------+ 38 | | KEY | BYTESIZE | KEYSCOUNT | 39 | +---------------------+----------+-----------+ 40 | | dev: | 1053120 | 1587 | 41 | | article: | 142824 | 210 | 42 | | blogpost: | 153416 | 240 | 43 | | collections: | 145912 | 217 | 44 | | events: | 144360 | 217 | 45 | | friends:foobar: | 169040 | 249 | 46 | | news: | 124944 | 191 | 47 | | user: | 172624 | 263 | 48 | | prod: | 1078048 | 1589 | 49 | | article: | 157264 | 231 | 50 | | blogpost: | 145632 | 214 | 51 | | collections: | 165992 | 240 | 52 | | events: | 143536 | 219 | 53 | | friends:foobar: | 168864 | 251 | 54 | | news: | 132096 | 197 | 55 | | user: | 164664 | 237 | 56 | +---------------------+----------+-----------+ 57 | 11:53PM INF Done 58 | 59 | $ ri display --output-params="depth=1&human=1&padding=🔥" 60 | 11:57PM INF Loading index 61 | +---------+----------+-----------+ 62 | | KEY | BYTESIZE | KEYSCOUNT | 63 | +---------+----------+-----------+ 64 | | 🔥dev: | 1M | 1,587 | 65 | | 🔥prod: | 1M | 1,589 | 66 | +---------+----------+-----------+ 67 | ``` 68 | 69 | ### Building from sources 70 | 71 | You have to have [golang installed](https://golang.org/doc/install) on your computer. 72 | 73 | 1. Checkout the repository: 74 | 75 | ```bash 76 | git clone git@github.com:obukhov/redis-inventory.git 77 | ``` 78 | 79 | 2. Build the binary: 80 | 81 | ```bash 82 | cd redis-inventory 83 | go build -o redis-inventory main.go 84 | ``` 85 | 86 | 3. Run it or move it to one of your PATH directories: 87 | 88 | ```bash 89 | mv redis-inventory /usr/local/bin/ 90 | ``` 91 | -------------------------------------------------------------------------------- /src/adapter/progress_mock_test.go: -------------------------------------------------------------------------------- 1 | package adapter 2 | 3 | import ( 4 | "github.com/jedib0t/go-pretty/v6/progress" 5 | "github.com/stretchr/testify/mock" 6 | "io" 7 | "time" 8 | ) 9 | 10 | // Tracker mock 11 | 12 | type TrackerMock struct { 13 | mock.Mock 14 | } 15 | 16 | func (t *TrackerMock) Increment(value int64) { 17 | t.Called(value) 18 | } 19 | 20 | func (t *TrackerMock) MarkAsDone() { 21 | t.Called() 22 | } 23 | 24 | // Progress mock 25 | 26 | type ProgressMock struct { 27 | mock.Mock 28 | } 29 | 30 | func (p *ProgressMock) SetAutoStop(autoStop bool) { 31 | p.Called(autoStop) 32 | } 33 | 34 | func (p *ProgressMock) AppendTracker(tracker *progress.Tracker) { 35 | p.Called(tracker) 36 | } 37 | 38 | func (p *ProgressMock) AppendTrackers(trackers []*progress.Tracker) { 39 | p.Called(trackers) 40 | } 41 | 42 | func (p *ProgressMock) IsRenderInProgress() bool { 43 | arg := p.Called() 44 | return arg.Bool(0) 45 | } 46 | 47 | func (p *ProgressMock) Length() int { 48 | arg := p.Called() 49 | return arg.Int(0) 50 | } 51 | 52 | func (p *ProgressMock) LengthActive() int { 53 | arg := p.Called() 54 | return arg.Int(0) 55 | } 56 | 57 | func (p *ProgressMock) LengthDone() int { 58 | arg := p.Called() 59 | return arg.Int(0) 60 | } 61 | 62 | func (p *ProgressMock) LengthInQueue() int { 63 | arg := p.Called() 64 | return arg.Int(0) 65 | } 66 | 67 | func (p *ProgressMock) SetMessageWidth(width int) { 68 | p.Called(width) 69 | } 70 | 71 | func (p *ProgressMock) SetNumTrackersExpected(numTrackers int) { 72 | p.Called(numTrackers) 73 | } 74 | 75 | func (p *ProgressMock) SetOutputWriter(output io.Writer) { 76 | p.Called(output) 77 | } 78 | 79 | func (p *ProgressMock) SetSortBy(sortBy progress.SortBy) { 80 | p.Called(sortBy) 81 | } 82 | 83 | func (p *ProgressMock) SetStyle(style progress.Style) { 84 | p.Called(style) 85 | } 86 | 87 | func (p *ProgressMock) SetTrackerLength(length int) { 88 | p.Called(length) 89 | } 90 | 91 | func (p *ProgressMock) SetTrackerPosition(position progress.Position) { 92 | p.Called(position) 93 | } 94 | 95 | func (p *ProgressMock) ShowETA(show bool) { 96 | p.Called(show) 97 | } 98 | 99 | func (p *ProgressMock) ShowOverallTracker(show bool) { 100 | p.Called(show) 101 | } 102 | 103 | func (p *ProgressMock) ShowPercentage(show bool) { 104 | p.Called(show) 105 | } 106 | 107 | func (p *ProgressMock) ShowTime(show bool) { 108 | p.Called(show) 109 | } 110 | 111 | func (p *ProgressMock) ShowTracker(show bool) { 112 | p.Called(show) 113 | } 114 | 115 | func (p *ProgressMock) ShowValue(show bool) { 116 | p.Called(show) 117 | } 118 | 119 | func (p *ProgressMock) SetUpdateFrequency(frequency time.Duration) { 120 | p.Called(frequency) 121 | } 122 | 123 | func (p *ProgressMock) Stop() { 124 | p.Called() 125 | } 126 | 127 | func (p *ProgressMock) Style() *progress.Style { 128 | arg := p.Called() 129 | return arg.Get(0).(*progress.Style) 130 | } 131 | 132 | func (p *ProgressMock) Render() { 133 | p.Called() 134 | } 135 | -------------------------------------------------------------------------------- /src/adapter/progress_test.go: -------------------------------------------------------------------------------- 1 | package adapter 2 | 3 | import ( 4 | "bytes" 5 | "github.com/jedib0t/go-pretty/v6/progress" 6 | "github.com/stretchr/testify/mock" 7 | "github.com/stretchr/testify/suite" 8 | "testing" 9 | "time" 10 | ) 11 | 12 | type ProgressWriterTestSuite struct { 13 | suite.Suite 14 | } 15 | 16 | func (suite ProgressWriterTestSuite) TestStart() { 17 | pwMock := &ProgressMock{} 18 | prettyProgressWriter := &PrettyProgressWriter{ 19 | pw: pwMock, 20 | tracker: nil, 21 | } 22 | 23 | pwMock.On("AppendTracker", mock.Anything).Once().Run(func(args mock.Arguments) { 24 | tracker, ok := args.Get(0).(*progress.Tracker) 25 | 26 | suite.Assert().True(ok) 27 | suite.Assert().Equal(int64(42), tracker.Total) 28 | }) 29 | 30 | pwMock.On("Render").Once() 31 | 32 | prettyProgressWriter.Start(42) 33 | 34 | time.Sleep(time.Millisecond) 35 | pwMock.AssertExpectations(suite.T()) 36 | } 37 | 38 | func (suite ProgressWriterTestSuite) TestStop() { 39 | pwMock := &ProgressMock{} 40 | trackerMock := &TrackerMock{} 41 | prettyProgressWriter := &PrettyProgressWriter{ 42 | pw: pwMock, 43 | tracker: trackerMock, 44 | } 45 | 46 | pwMock.On("Stop").Once() 47 | trackerMock.On("MarkAsDone").Once() 48 | 49 | prettyProgressWriter.Stop() 50 | 51 | pwMock.AssertExpectations(suite.T()) 52 | trackerMock.AssertExpectations(suite.T()) 53 | } 54 | 55 | func (suite ProgressWriterTestSuite) TestIncrement() { 56 | pwMock := &ProgressMock{} 57 | trackerMock := &TrackerMock{} 58 | 59 | prettyProgressWriter := &PrettyProgressWriter{ 60 | pw: pwMock, 61 | tracker: trackerMock, 62 | } 63 | 64 | trackerMock.On("Increment", int64(1)).Once() 65 | 66 | prettyProgressWriter.Increment() 67 | 68 | pwMock.AssertExpectations(suite.T()) 69 | trackerMock.AssertExpectations(suite.T()) 70 | } 71 | 72 | // Dumbest test I've ever written 73 | func (suite ProgressWriterTestSuite) TestInit() { 74 | var buf bytes.Buffer 75 | pwMock := &ProgressMock{} 76 | style := progress.Style{} 77 | 78 | pwMock.On("SetAutoStop", mock.Anything) 79 | pwMock.On("SetTrackerLength", mock.Anything) 80 | pwMock.On("ShowETA", mock.Anything) 81 | pwMock.On("ShowOverallTracker", mock.Anything) 82 | pwMock.On("ShowTime", mock.Anything) 83 | pwMock.On("ShowTracker", mock.Anything) 84 | pwMock.On("ShowValue", mock.Anything) 85 | pwMock.On("SetMessageWidth", mock.Anything) 86 | pwMock.On("SetNumTrackersExpected", mock.Anything) 87 | pwMock.On("SetSortBy", mock.Anything) 88 | pwMock.On("SetStyle", mock.Anything) 89 | pwMock.On("SetTrackerPosition", mock.Anything) 90 | pwMock.On("SetUpdateFrequency", mock.Anything) 91 | pwMock.On("SetOutputWriter", &buf) 92 | pwMock.On("Style").Return(&style) 93 | 94 | prettyProgressWriter := &PrettyProgressWriter{ 95 | pw: pwMock, 96 | tracker: nil, 97 | } 98 | 99 | prettyProgressWriter.init(&buf) 100 | } 101 | 102 | func TestProgressWriterTestSuite(t *testing.T) { 103 | suite.Run(t, new(ProgressWriterTestSuite)) 104 | } 105 | -------------------------------------------------------------------------------- /docs/usage.md: -------------------------------------------------------------------------------- 1 | Usage 2 | ===== 3 | 4 | ## General interface 5 | 6 | ```bash 7 | redis-inventory inventory [--output=] [--output-params=] 8 | ``` 9 | 10 | `` can be provided in [one of two formats](docs/redis-url.md) 11 | 12 | ## Output type 13 | 14 | ### Table 15 | 16 | | Option name | Description | Default | 17 | |--------------|------------------------------------------------|-----------| 18 | | padSpaces | Number of spaces to indent the nested level | `0` | 19 | | padding | Use custom character to pad nested level | `""` | 20 | | depth | Maximum nesting level for keys before grouping | 10 | 21 | | human | Display numbers in human-friendly way (0 or 1) | 0 | 22 | 23 | If padding is not specified in either way, nested keys are displayed with full paths as following: 24 | 25 | ```bash 26 | +----------------------+----------+-----------+ 27 | | KEY | BYTESIZE | KEYSCOUNT | 28 | +----------------------+----------+-----------+ 29 | | dev: | 2.9M | 4,555 | 30 | | dev:article: | 413.7K | 616 | 31 | | dev:blogpost: | 408.5K | 630 | 32 | | dev:collections: | 426.7K | 627 | 33 | | dev:events: | 391.2K | 614 | 34 | | dev:friends:foobar: | 501.1K | 745 | 35 | | dev:news: | 388.8K | 593 | 36 | | dev:user: | 481K | 730 | 37 | ... 38 | ``` 39 | 40 | ### Json 41 | 42 | | Option name | Description | Default | 43 | |--------------|----------------------------------------------|-----------| 44 | | padSpaces | Number of spaces to indent the nested level | `0` | 45 | | padding | Use custom character to pad nested level | `""` | 46 | 47 | If padding is not specified in either way json is not pretty-printed. 48 | 49 | ### Chart 50 | 51 | | Option name | Description | Default | 52 | |--------------|-------------------------------------------------|-----------| 53 | | depth | Maximum nesting level for keys before grouping | `10` | 54 | | port | Use custom character to pad nested level | `8888` | 55 | 56 | ## Using cached index 57 | 58 | You can scan redis instance once and then use different visualisations. For scanning use `index` command. To display 59 | cached index use `display` command. 60 | 61 | ## Customising level separators 62 | 63 | It is quite common to use ":" as a level separator in redis. Sometimes, though, you want to use other punctuation (for 64 | example "_" or "-") to also be a level separator. Redis-inventory can use custom characters for splitting keys: 65 | 66 | ```bash 67 | redis-inventory inventory --separators=":_-" 68 | ``` 69 | 70 | `` can be provided in [one of two formats](docs/redis-url.md) 71 | 72 | It will work even if those characters sometimes are used not for level separation, as a node with just one child will 73 | automatically be merged with it. 74 | 75 | ## See more 76 | 77 | - [index command](cobra/redis-inventory_index.md) 78 | - [display command](cobra/redis-inventory_display.md) 79 | - [inventory command](cobra/redis-inventory_inventory.md) 80 | -------------------------------------------------------------------------------- /cmd/app/inventory.go: -------------------------------------------------------------------------------- 1 | package app 2 | 3 | import ( 4 | "context" 5 | "os" 6 | 7 | "github.com/mediocregopher/radix/v4" 8 | "github.com/obukhov/redis-inventory/src/adapter" 9 | "github.com/obukhov/redis-inventory/src/logger" 10 | "github.com/obukhov/redis-inventory/src/renderer" 11 | "github.com/obukhov/redis-inventory/src/scanner" 12 | "github.com/obukhov/redis-inventory/src/trie" 13 | "github.com/spf13/cobra" 14 | ) 15 | 16 | var scanCmd = &cobra.Command{ 17 | Use: "inventory redis://[:@]:[/]", 18 | Short: "Scan keys and display summary right away with selected output and output params", 19 | Long: "Scan command builds prefix tree in memory and then displays the usage summary. To avoid scanning redis instance when trying different output formats use `index` and `display` commands", 20 | Args: cobra.MinimumNArgs(0), 21 | Run: func(cmd *cobra.Command, args []string) { 22 | consoleLogger := logger.NewConsoleLogger(logLevel) 23 | consoleLogger.Info().Msg("Start scanning") 24 | 25 | var redisUrl string 26 | if len(args) > 0 { 27 | redisUrl = args[0] 28 | } else { 29 | redisUrl = os.Getenv("REDIS_URL") 30 | } 31 | 32 | clientSource, err := (radix.PoolConfig{}).New(context.Background(), "tcp", redisUrl) 33 | if err != nil { 34 | consoleLogger.Fatal().Err(err).Msg("Can't create redis client") 35 | } 36 | 37 | redisScanner := scanner.NewScanner( 38 | adapter.NewRedisService(clientSource), 39 | adapter.NewPrettyProgressWriter(os.Stdout), 40 | consoleLogger, 41 | ) 42 | 43 | resultTrie := trie.NewTrie(trie.NewPunctuationSplitter([]rune(separators)...), maxChildren) 44 | redisScanner.Scan( 45 | adapter.ScanOptions{ 46 | ScanCount: scanCount, 47 | Pattern: pattern, 48 | Throttle: throttleNs, 49 | SamplePerc: samplePerc, 50 | }, 51 | resultTrie, 52 | ) 53 | 54 | r, err := renderer.NewRenderer(output, outputParams, consoleLogger) 55 | if err != nil { 56 | consoleLogger.Fatal().Err(err).Msg("Can't create renderer") 57 | } 58 | 59 | err = r.Render(resultTrie.Root()) 60 | if err != nil { 61 | consoleLogger.Fatal().Err(err).Msg("Can't render report") 62 | } 63 | 64 | consoleLogger.Info().Msg("Finish scanning") 65 | }, 66 | } 67 | 68 | func init() { 69 | RootCmd.AddCommand(scanCmd) 70 | scanCmd.Flags().StringVarP(&output, "output", "o", "table", "One of possible outputs: json, jsonp, table") 71 | scanCmd.Flags().StringVarP(&outputParams, "output-params", "p", "", "Parameters specific for output type") 72 | scanCmd.Flags().StringVarP(&logLevel, "logLevel", "l", "info", "Level of logs to be displayed") 73 | scanCmd.Flags().StringVarP(&separators, "separators", "s", ":", "Symbols that logically separate levels of the key") 74 | scanCmd.Flags().IntVarP(&samplePerc, "samplePerc", "n", 10, "Percentage of returned keys to sample") 75 | scanCmd.Flags().IntVarP(&maxChildren, "maxChildren", "m", 10, "Maximum children node can have before start aggregating") 76 | scanCmd.Flags().StringVarP(&pattern, "pattern", "k", "*", "Glob pattern limiting the keys to be aggregated") 77 | scanCmd.Flags().IntVarP(&scanCount, "scanCount", "c", 1000, "Number of keys to be scanned in one iteration (argument of scan command)") 78 | scanCmd.Flags().IntVarP(&throttleNs, "throttle", "t", 0, "Throttle: number of nanoseconds to sleep between keys") 79 | } 80 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | Redis Inventory 2 | =============== 3 | 4 | [![Build Status](https://travis-ci.com/obukhov/redis-inventory.svg?branch=master)](https://travis-ci.com/obukhov/redis-inventory) 5 | [![Coverage Status](https://coveralls.io/repos/github/obukhov/redis-inventory/badge.svg?branch=master)](https://coveralls.io/github/obukhov/redis-inventory?branch=master) 6 | [![Go Report Card](https://goreportcard.com/badge/github.com/obukhov/redis-inventory)](https://goreportcard.com/report/github.com/obukhov/redis-inventory) 7 | [![Docker Pulls](https://img.shields.io/docker/pulls/dclg/redis-inventory)](https://hub.docker.com/repository/docker/dclg/redis-inventory) 8 | 9 | Redis inventory is a tool to analyse Redis memory usage by key patterns and displaying it hierarchically. The name is 10 | inspired by "Disk Inventory X" tool doing similar analysis for disk usage. 11 | 12 | [Blog post explaining how it works](https://blog.dclg.net/redis-inventory-analyzing-memory-usage-in-redis-by-key-pattern) 13 | 14 | Example: 15 | 16 | ```bash 17 | $ redis-inventory inventory --output=table --output-params="padSpaces=2&depth=2&human=1" 18 | ``` 19 | 20 | `` can be provided in [one of two formats](docs/redis-url.md) 21 | 22 | Outputs it as a nice table 23 | 24 | ```bash 25 | 12:39PM INF Start scanning 26 | +---------------------+----------+-----------+ 27 | | KEY | BYTESIZE | KEYSCOUNT | 28 | +---------------------+----------+-----------+ 29 | | dev: | 2.9M | 4,555 | 30 | | article: | 413.7K | 616 | 31 | | blogpost: | 408.5K | 630 | 32 | | collections: | 426.7K | 627 | 33 | | events: | 391.2K | 614 | 34 | | friends:foobar: | 501.1K | 745 | 35 | | news: | 388.8K | 593 | 36 | | user: | 481K | 730 | 37 | | prod: | 2.9M | 4,531 | 38 | | article: | 397.1K | 614 | 39 | | blogpost: | 409.4K | 627 | 40 | | collections: | 374.7K | 560 | 41 | | events: | 384.2K | 588 | 42 | | friends:foobar: | 503K | 755 | 43 | | news: | 407.9K | 618 | 44 | | user: | 492.3K | 769 | 45 | +---------------------+----------+-----------+ 46 | 12:39PM INF Finish scanning 47 | ``` 48 | 49 | It also can render sunburst diagrams to visualize it: 50 | ```bash 51 | $ redis-inventory inventory --output=chart --maxChildren=100 52 | ``` 53 | 54 | 55 | [![Full sunburst diagram](docs/images/diagram1000.png)](docs/images/diagram.png) 56 | 57 | Read more about [usage](docs/usage.md) 58 | 59 | ## Installation 60 | 61 | There are two ways to install the tool: 62 | 63 | - use docker image 64 | - build binary from sources 65 | 66 | ### Using docker 67 | 68 | To run the tool from a docker image, run the command: 69 | 70 | ```bash 71 | docker run --rm dclg/redis-inventory inventory 72 | ``` 73 | 74 | Read more about [installation](docs/installation.md) 75 | 76 | ### Kudos 77 | 78 | - [Radix](https://github.com/mediocregopher/radix) — Redis client for Go, 79 | - [Cobra](https://github.com/spf13/cobra) — CLI applications framework, 80 | - [Go-Pretty](https://github.com/jedib0t/go-pretty) — tools to render tables and progress bars, 81 | - [AnyChart](https://www.anychart.com/) — beautiful sunburst diagrams. 82 | -------------------------------------------------------------------------------- /cmd/app/index.go: -------------------------------------------------------------------------------- 1 | package app 2 | 3 | import ( 4 | "context" 5 | "os" 6 | 7 | "github.com/mediocregopher/radix/v4" 8 | "github.com/obukhov/redis-inventory/src/adapter" 9 | "github.com/obukhov/redis-inventory/src/logger" 10 | "github.com/obukhov/redis-inventory/src/renderer" 11 | "github.com/obukhov/redis-inventory/src/scanner" 12 | "github.com/obukhov/redis-inventory/src/trie" 13 | "github.com/spf13/cobra" 14 | ) 15 | 16 | var indexCmd = &cobra.Command{ 17 | Use: "index redis://[:@]:[/]", 18 | Short: "Scan keys and save prefix tree in a temporary file for further rendering with display command", 19 | Long: "Keep in mind that some options are scanning (index) options that cannot be redefined later. For example, `maxChildren` changes the way index data is built, unlike `depth` parameter only influencing rendering", 20 | Args: cobra.MinimumNArgs(0), 21 | Run: func(cmd *cobra.Command, args []string) { 22 | consoleLogger := logger.NewConsoleLogger(logLevel) 23 | consoleLogger.Info().Msg("Start indexing") 24 | 25 | var redisUrl string 26 | if len(args) > 0 { 27 | redisUrl = args[0] 28 | } else { 29 | redisUrl = os.Getenv("REDIS_URL") 30 | } 31 | 32 | if redisUrl == "" { 33 | consoleLogger.Fatal().Msg("No redis URL given") 34 | } 35 | 36 | clientSource, err := (radix.PoolConfig{}).New(context.Background(), "tcp", redisUrl) 37 | if err != nil { 38 | consoleLogger.Fatal().Err(err).Msg("Can't create redis client") 39 | } 40 | 41 | redisScanner := scanner.NewScanner( 42 | adapter.NewRedisService(clientSource), 43 | adapter.NewPrettyProgressWriter(os.Stdout), 44 | consoleLogger, 45 | ) 46 | 47 | resultTrie := trie.NewTrie(trie.NewPunctuationSplitter([]rune(separators)...), maxChildren) 48 | redisScanner.Scan( 49 | adapter.ScanOptions{ 50 | ScanCount: scanCount, 51 | Pattern: pattern, 52 | Throttle: throttleNs, 53 | SamplePerc: samplePerc, 54 | }, 55 | resultTrie, 56 | ) 57 | 58 | indexFileName := os.TempDir() + "/redis-inventory.json" 59 | f, err := os.Create(indexFileName) 60 | if err != nil { 61 | consoleLogger.Fatal().Err(err).Msg("Can't create renderer") 62 | } 63 | 64 | r := renderer.NewJSONRenderer(f, renderer.JSONRendererParams{}) 65 | 66 | err = r.Render(resultTrie.Root()) 67 | if err != nil { 68 | consoleLogger.Fatal().Err(err).Msg("Can't write to file") 69 | } 70 | 71 | consoleLogger.Info().Msgf("Finish scanning and saved index as a file %s", indexFileName) 72 | }, 73 | } 74 | 75 | func init() { 76 | RootCmd.AddCommand(indexCmd) 77 | indexCmd.Flags().StringVarP(&logLevel, "logLevel", "l", "info", "Level of logs to be displayed") 78 | indexCmd.Flags().StringVarP(&separators, "separators", "s", ":", "Symbols that logically separate levels of the key") 79 | indexCmd.Flags().IntVarP(&maxChildren, "maxChildren", "m", 50, "Maximum children node can have before start aggregating") 80 | indexCmd.Flags().IntVarP(&samplePerc, "samplePerc", "n", 10, "Percentage of returned keys to sample") 81 | indexCmd.Flags().StringVarP(&pattern, "pattern", "k", "*", "Glob pattern limiting the keys to be aggregated") 82 | indexCmd.Flags().IntVarP(&scanCount, "scanCount", "c", 5000, "Number of keys to be scanned in one iteration (argument of scan command)") 83 | indexCmd.Flags().IntVarP(&throttleNs, "throttle", "t", 0, "Throttle: number of nanoseconds to sleep between keys") 84 | } 85 | -------------------------------------------------------------------------------- /src/renderer/json_test.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "bytes" 5 | "github.com/obukhov/redis-inventory/src/trie" 6 | "github.com/stretchr/testify/suite" 7 | "testing" 8 | ) 9 | 10 | type JSONRendererTestSuite struct { 11 | suite.Suite 12 | trie *trie.Trie 13 | } 14 | 15 | func (suite *JSONRendererTestSuite) TestRender() { 16 | var buf bytes.Buffer 17 | 18 | r := JSONRenderer{&buf, JSONRendererParams{}} 19 | 20 | err := r.Render(suite.trie.Root()) 21 | suite.Assert().Nil(err, "Error rendering trie") 22 | 23 | suite.Assert().Equal( 24 | "{\"Children\":{\"dev:\":{\"Children\":{\"article:\":{\"Children\":{\"1\":{\"Values\":{\"Params\":{\"BytesSize\":100,\"KeysCount\":1}}},\"2\":{\"Values\":{\"Params\":{\"BytesSize\":100,\"KeysCount\":1}}}},\"Values\":{\"Params\":{\"BytesSize\":200,\"KeysCount\":2}}}}}},\"Values\":{\"Params\":{\"BytesSize\":200,\"KeysCount\":2}}}\n", 25 | buf.String(), 26 | ) 27 | } 28 | 29 | func (suite *JSONRendererTestSuite) TestRenderWithIndent() { 30 | var buf bytes.Buffer 31 | 32 | params, err := NewJSONRendererParams("padSpaces=2") 33 | suite.Assert().Nil(err) 34 | 35 | r := NewJSONRenderer(&buf, params) 36 | 37 | err = r.Render(suite.trie.Root()) 38 | suite.Assert().Nil(err, "Error rendering trie") 39 | 40 | suite.Assert().Equal( 41 | "{\n"+ 42 | " \"Children\": {\n"+ 43 | " \"dev:\": {\n"+ 44 | " \"Children\": {\n"+ 45 | " \"article:\": {\n"+ 46 | " \"Children\": {\n"+ 47 | " \"1\": {\n"+ 48 | " \"Values\": {\n"+ 49 | " \"Params\": {\n"+ 50 | " \"BytesSize\": 100,\n"+ 51 | " \"KeysCount\": 1\n"+ 52 | " }\n"+ 53 | " }\n"+ 54 | " },\n"+ 55 | " \"2\": {\n"+ 56 | " \"Values\": {\n"+ 57 | " \"Params\": {\n"+ 58 | " \"BytesSize\": 100,\n"+ 59 | " \"KeysCount\": 1\n"+ 60 | " }\n"+ 61 | " }\n"+ 62 | " }\n"+ 63 | " },\n"+ 64 | " \"Values\": {\n"+ 65 | " \"Params\": {\n"+ 66 | " \"BytesSize\": 200,\n"+ 67 | " \"KeysCount\": 2\n"+ 68 | " }\n"+ 69 | " }\n"+ 70 | " }\n"+ 71 | " }\n"+ 72 | " }\n"+ 73 | " },\n"+ 74 | " \"Values\": {\n"+ 75 | " \"Params\": {\n"+ 76 | " \"BytesSize\": 200,\n"+ 77 | " \"KeysCount\": 2\n"+ 78 | " }\n"+ 79 | " }\n"+ 80 | "}\n", 81 | buf.String(), 82 | ) 83 | } 84 | 85 | func (suite *JSONRendererTestSuite) TestNewJSONRendererParams() { 86 | _, err := NewJSONRendererParams("padSpaces=asd") 87 | suite.Assert().Error(err) 88 | } 89 | 90 | func (suite *JSONRendererTestSuite) SetupTest() { 91 | suite.trie = trie.NewTrie(trie.NewPunctuationSplitter(':'), 3) 92 | 93 | suite.setupTrieKey("dev:article:1", 100) 94 | suite.setupTrieKey("dev:article:2", 100) 95 | } 96 | 97 | func (suite *JSONRendererTestSuite) setupTrieKey(key string, value int64) { 98 | suite.trie.Add(key, trie.ParamValue{Param: trie.BytesSize, Value: value}, trie.ParamValue{Param: trie.KeysCount, Value: 1}) 99 | } 100 | 101 | func TestJsonRendererTestSuite(t *testing.T) { 102 | suite.Run(t, new(JSONRendererTestSuite)) 103 | } 104 | -------------------------------------------------------------------------------- /src/renderer/table.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "fmt" 5 | "io" 6 | "sort" 7 | "strconv" 8 | "strings" 9 | 10 | "code.cloudfoundry.org/bytefmt" 11 | "github.com/hetiansu5/urlquery" 12 | "github.com/jedib0t/go-pretty/v6/table" 13 | "github.com/jedib0t/go-pretty/v6/text" 14 | "github.com/obukhov/redis-inventory/src/trie" 15 | "golang.org/x/text/message" 16 | ) 17 | 18 | var p = message.NewPrinter(message.MatchLanguage("en")) 19 | 20 | // NewTableRendererParams Creates parameters structure from url-encoded string respecting some defaults 21 | func NewTableRendererParams(paramsString string) (TableRendererParams, error) { 22 | params := TableRendererParams{ 23 | Depth: 10, 24 | } 25 | 26 | err := urlquery.Unmarshal([]byte(paramsString), ¶ms) 27 | if err != nil { 28 | return params, err 29 | } 30 | 31 | params.indent = params.Padding + strings.Repeat(" ", params.PaddingSpaceCount) 32 | 33 | return params, nil 34 | } 35 | 36 | // TableRendererParams represents renderer parameters 37 | type TableRendererParams struct { 38 | Depth int `query:"depth"` 39 | Padding string `query:"padding"` 40 | PaddingSpaceCount int `query:"padSpaces"` 41 | HumanReadable bool `query:"human"` 42 | indent string 43 | } 44 | 45 | // TableRenderer renders trie as ascii-table to output (most probably stdout) 46 | type TableRenderer struct { 47 | output io.Writer 48 | params TableRendererParams 49 | } 50 | 51 | // Render executes rendering 52 | func (o TableRenderer) Render(root *trie.Node) error { 53 | t := table.NewWriter() 54 | t.AppendHeader(table.Row{"Key", "ByteSize", "KeysCount"}) 55 | o.appendLevel(t, root, 1, "") 56 | 57 | t.SetColumnConfigs([]table.ColumnConfig{ 58 | {Number: 1}, 59 | {Number: 2, Align: text.AlignRight, AlignHeader: text.AlignCenter}, 60 | {Number: 3, Align: text.AlignRight, AlignHeader: text.AlignCenter}, 61 | }) 62 | t.SetOutputMirror(o.output) 63 | t.Render() 64 | 65 | return nil 66 | } 67 | 68 | func (o TableRenderer) appendLevel(t table.Writer, node *trie.Node, level int, prefix string) { 69 | childKeys := make([]string, 0, len(node.Children)) 70 | for k := range node.Children { 71 | childKeys = append(childKeys, k) 72 | } 73 | 74 | sort.Strings(childKeys) 75 | for _, key := range childKeys { 76 | childNode := node.Children[key] 77 | nextLevel := level + 1 78 | if !childNode.HasAggregator() { 79 | var keys []string 80 | keys, childNode = childNode.FindNextAggregatedNodeWithKey() 81 | nextLevel += len(keys) 82 | key = key + strings.Join(keys, "") 83 | } 84 | 85 | t.AppendRow(table.Row{ 86 | o.displayKey(level, key, prefix), 87 | o.formatBytes(childNode.Aggregator().Params[trie.BytesSize]), 88 | o.formatNumber(childNode.Aggregator().Params[trie.KeysCount]), 89 | }) 90 | 91 | if level < o.params.Depth { 92 | o.appendLevel(t, childNode, nextLevel, prefix+key) 93 | } 94 | } 95 | 96 | if node.OverflowChildrenCount > 0 { 97 | t.AppendRow(table.Row{o.displayKey(level, fmt.Sprintf("( %d more keys )", node.OverflowChildrenCount), prefix)}) 98 | } 99 | } 100 | 101 | func (o TableRenderer) formatBytes(value int64) string { 102 | if o.params.HumanReadable { 103 | return bytefmt.ByteSize(uint64(value)) 104 | } 105 | 106 | return strconv.Itoa(int(value)) 107 | } 108 | 109 | func (o TableRenderer) formatNumber(value int64) string { 110 | 111 | if o.params.HumanReadable { 112 | return p.Sprint(value) 113 | } 114 | 115 | return strconv.Itoa(int(value)) 116 | } 117 | 118 | func (o TableRenderer) displayKey(level int, key string, prefix string) string { 119 | if o.params.indent != "" { 120 | return strings.Repeat(o.params.indent, level) + key 121 | } 122 | 123 | return prefix + key 124 | } 125 | -------------------------------------------------------------------------------- /src/adapter/rservice.go: -------------------------------------------------------------------------------- 1 | package adapter 2 | 3 | import ( 4 | "context" 5 | "errors" 6 | "fmt" 7 | "strconv" 8 | "time" 9 | 10 | "github.com/mediocregopher/radix/v4" 11 | 12 | "github.com/mediocregopher/radix/v4/resp" 13 | "github.com/mediocregopher/radix/v4/resp/resp3" 14 | ) 15 | 16 | // ScanOptions options for scanning keyspace 17 | type ScanOptions struct { 18 | Pattern string 19 | ScanCount int 20 | Throttle int 21 | SamplePerc int 22 | } 23 | 24 | type scanResult struct { 25 | cur string 26 | keys []string 27 | } 28 | 29 | func (s *scanResult) UnmarshalRESP(br resp.BufferedReader, o *resp.Opts) error { 30 | var ah resp3.ArrayHeader 31 | if err := ah.UnmarshalRESP(br, o); err != nil { 32 | return err 33 | } else if ah.NumElems != 2 { 34 | return errors.New("not enough parts returned") 35 | } 36 | 37 | var c resp3.BlobString 38 | if err := c.UnmarshalRESP(br, o); err != nil { 39 | return err 40 | } 41 | 42 | s.cur = c.S 43 | s.keys = s.keys[:0] 44 | 45 | return resp3.Unmarshal(br, &s.keys, o) 46 | } 47 | 48 | // NewRedisService creates RedisService 49 | func NewRedisService(client radix.Client) RedisService { 50 | return RedisService{ 51 | client: client, 52 | } 53 | } 54 | 55 | // RedisService implementation for iteration over redis 56 | type RedisService struct { 57 | client radix.Client 58 | } 59 | 60 | type BulkKeyInfo struct { 61 | Keys []string 62 | Sizes []int64 63 | } 64 | 65 | // ScanKeys scans keys asynchroniously and sends them to the returned channel 66 | func (s RedisService) ScanKeys(ctx context.Context, options ScanOptions) <-chan BulkKeyInfo { 67 | resultChan := make(chan BulkKeyInfo) 68 | 69 | // if options.Pattern != "*" && options.Pattern != "" { 70 | // scanOpts.Pattern = options.Pattern 71 | // } 72 | 73 | go func() { 74 | defer close(resultChan) 75 | scanRes := scanResult{cur: "0"} 76 | 77 | for { 78 | err := s.client.Do(ctx, radix.Cmd(&scanRes, "SCAN", scanRes.cur, "COUNT", strconv.Itoa(options.ScanCount))) 79 | if err != nil { 80 | fmt.Printf("Failed: %s\n", err.Error()) 81 | return 82 | } 83 | 84 | sampledKeyLength := len(scanRes.keys) * 100 / options.SamplePerc 85 | keyIndex := 0 86 | ret := BulkKeyInfo{ 87 | Keys: make([]string, sampledKeyLength), 88 | Sizes: make([]int64, sampledKeyLength), 89 | } 90 | p := radix.NewPipeline() 91 | for index, key := range scanRes.keys { 92 | if index%100 < options.SamplePerc { 93 | ret.Keys[keyIndex] = key 94 | p.Append(radix.Cmd(&ret.Sizes[keyIndex], "MEMORY", "USAGE", key)) 95 | keyIndex++ 96 | } 97 | } 98 | 99 | ret.Keys = ret.Keys[:keyIndex] 100 | ret.Sizes = ret.Sizes[:keyIndex] 101 | 102 | err = s.client.Do(ctx, p) 103 | if err != nil { 104 | fmt.Printf("Failed: %s\n", err.Error()) 105 | } 106 | 107 | resultChan <- ret 108 | 109 | if scanRes.cur == "0" { 110 | return 111 | } 112 | 113 | if options.Throttle > 0 { 114 | time.Sleep(time.Nanosecond * time.Duration(options.Throttle)) 115 | } 116 | } 117 | }() 118 | 119 | return resultChan 120 | } 121 | 122 | // GetKeysCount returns number of keys in the current database 123 | func (s RedisService) GetKeysCount(ctx context.Context) (int64, error) { 124 | var keysCount int64 125 | err := s.client.Do(context.Background(), radix.Cmd(&keysCount, "DBSIZE")) 126 | if err != nil { 127 | return 0, err 128 | } 129 | 130 | return keysCount, nil 131 | } 132 | 133 | // GetMemoryUsage returns memory usage of given key 134 | func (s RedisService) GetMemoryUsage(ctx context.Context, key string) (int64, error) { 135 | var res int64 136 | err := s.client.Do(context.Background(), radix.Cmd(&res, "MEMORY", "USAGE", key)) 137 | if err != nil { 138 | return 0, err 139 | } 140 | 141 | return res, nil 142 | } 143 | -------------------------------------------------------------------------------- /src/trie/trie_test.go: -------------------------------------------------------------------------------- 1 | package trie 2 | 3 | import ( 4 | "testing" 5 | 6 | "github.com/stretchr/testify/assert" 7 | "github.com/stretchr/testify/suite" 8 | ) 9 | 10 | type TrieTestSuite struct { 11 | suite.Suite 12 | splitter Splitter 13 | } 14 | 15 | func (suite *TrieTestSuite) TestAdd() { 16 | trie := NewTrie(NewPunctuationSplitter(':'), 100) 17 | 18 | trie.Add("foo:bar:lorem", ParamValue{BytesSize, 10}) 19 | trie.Add("foo:bar:ipsum", ParamValue{BytesSize, 20}) 20 | // root node 21 | assert.Equal(suite.T(), int64(30), trie.root.Aggr.Params[BytesSize], "Root node aggregated value is 30") 22 | // intermediate node 23 | assert.Nil(suite.T(), trie.root.Children["foo:"].Aggr, "Intermediate node skip aggregation if has just one child") 24 | // fork node 25 | assert.NotNil(suite.T(), trie.root.Children["foo:"].Children["bar:"].Aggr, "Fork node has aggregator") 26 | assert.Equal(suite.T(), int64(30), trie.root.Children["foo:"].Children["bar:"].Aggr.Params[BytesSize], "Fork node aggregated value is 30") 27 | } 28 | 29 | func (suite *TrieTestSuite) TestKeyEndingInFork() { 30 | trie := NewTrie(NewPunctuationSplitter(':'), 100) 31 | 32 | trie.Add("foo:bar:lorem", ParamValue{BytesSize, 10}) 33 | trie.Add("foo:bar:ipsum", ParamValue{BytesSize, 20}) 34 | trie.Add("foo:bar:", ParamValue{BytesSize, 100}) 35 | // root node 36 | assert.Equal(suite.T(), int64(130), trie.root.Aggr.Params[BytesSize], "Root node aggregated value is 130") 37 | // intermediate node 38 | assert.Nil(suite.T(), trie.root.Children["foo:"].Aggr, "Intermediate node skip aggregation if has just one child") 39 | // fork node 40 | assert.NotNil(suite.T(), trie.root.Children["foo:"].Children["bar:"].Aggr, "Fork node has where key ended") 41 | assert.Equal(suite.T(), int64(130), trie.root.Children["foo:"].Children["bar:"].Aggr.Params[BytesSize], "Fork node aggregated value is 130") 42 | } 43 | 44 | func (suite *TrieTestSuite) TestIDsEndingInFork() { 45 | trie := NewTrie(NewPunctuationSplitter(':'), 100) 46 | 47 | trie.Add("123123:bar:lorem", ParamValue{BytesSize, 10}) 48 | trie.Add("234234:bar:ipsum", ParamValue{BytesSize, 20}) 49 | trie.Add("234:bar:", ParamValue{BytesSize, 100}) 50 | // root node 51 | assert.Equal(suite.T(), int64(130), trie.root.Aggr.Params[BytesSize], "Root node aggregated value is 130") 52 | // intermediate node 53 | assert.Nil(suite.T(), trie.root.Children[":"].Aggr, "Intermediate node skip aggregation if has just one child") 54 | // fork node 55 | assert.NotNil(suite.T(), trie.root.Children[":"].Children["bar:"].Aggr, "Fork node has where key ended") 56 | assert.Equal(suite.T(), int64(130), trie.root.Children[":"].Children["bar:"].Aggr.Params[BytesSize], "Fork node aggregated value is 130") 57 | } 58 | 59 | func (suite *TrieTestSuite) TestKeyEndingInIntermediateNode() { 60 | trie := NewTrie(NewPunctuationSplitter(':'), 100) 61 | 62 | trie.Add("foo:bar:lorem", ParamValue{BytesSize, 10}) 63 | trie.Add("foo:bar:ipsum", ParamValue{BytesSize, 20}) 64 | trie.Add("foo:", ParamValue{BytesSize, 100}) 65 | // root node 66 | assert.Equal(suite.T(), int64(130), trie.root.Aggr.Params[BytesSize], "Root node aggregated value is 30") 67 | // intermediate node 68 | assert.NotNil(suite.T(), trie.root.Children["foo:"].Aggr, "Intermediate node has aggregator if key ends here") 69 | assert.Equal(suite.T(), int64(130), trie.root.Children["foo:"].Aggr.Params[BytesSize], "Fork node aggregated value is 130") 70 | // fork node 71 | assert.NotNil(suite.T(), trie.root.Children["foo:"].Children["bar:"].Aggr, "Fork node has aggregator") 72 | assert.Equal(suite.T(), int64(30), trie.root.Children["foo:"].Children["bar:"].Aggr.Params[BytesSize], "Fork node aggregated value is 30") 73 | } 74 | 75 | func (suite *TrieTestSuite) TestAddWithMaxChildrenLimit() { 76 | maxChildrenSavedLimit := 3 77 | 78 | trie := NewTrie(NewPunctuationSplitter(':'), maxChildrenSavedLimit) 79 | 80 | trie.Add("foo:bar:lorem", ParamValue{BytesSize, 10}) 81 | trie.Add("foo:bar:ipsum", ParamValue{BytesSize, 20}) 82 | trie.Add("foo:bar:dolor", ParamValue{BytesSize, 30}) 83 | trie.Add("foo:bar:sit", ParamValue{BytesSize, 40}) 84 | trie.Add("foo:bar:deep:key:nested", ParamValue{BytesSize, 50}) 85 | //trie.Add("foo:bar:", ParamValue{BytesSize, 100}) 86 | 87 | // root node 88 | assert.Equal(suite.T(), int64(150), trie.root.Aggr.Params[BytesSize], "Root node aggregated value is 30") 89 | 90 | // fork node 91 | assert.NotNil(suite.T(), trie.root.Children["foo:"].Children["bar:"].Aggr, "Fork node has aggregator") 92 | assert.Len(suite.T(), trie.root.Children["foo:"].Children["bar:"].Children, maxChildrenSavedLimit, "Max children saved is 3") 93 | assert.Equal(suite.T(), int64(150), trie.root.Children["foo:"].Children["bar:"].Aggr.Params[BytesSize], "Fork node aggregated value is 30") 94 | } 95 | 96 | func TestTrieTestSuite(t *testing.T) { 97 | suite.Run(t, new(TrieTestSuite)) 98 | } 99 | -------------------------------------------------------------------------------- /src/renderer/chart_test.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "encoding/xml" 5 | "github.com/obukhov/redis-inventory/src/trie" 6 | "github.com/stretchr/testify/mock" 7 | "github.com/stretchr/testify/suite" 8 | "io" 9 | "strings" 10 | "testing" 11 | ) 12 | 13 | type ChartRendererTestSuite struct { 14 | suite.Suite 15 | trie *trie.Trie 16 | } 17 | 18 | func (suite *ChartRendererTestSuite) TestRender() { 19 | srvMock := &MockServer{} 20 | srvMock.On("Serve", 123, "test page content").Once() 21 | 22 | rendererMock := &mockPageRenderer{} 23 | expectedRendererResult := Node{ 24 | Name: "Total", 25 | Value: 6500, 26 | ValueHuman: "6.3K", 27 | FullPath: "Total", 28 | KeysCount: 9, 29 | Children: []Node{ 30 | { 31 | Name: "dev:", 32 | Value: 2500, 33 | ValueHuman: "2.4K", 34 | FullPath: "dev:", 35 | KeysCount: 7, 36 | Children: []Node{ 37 | { 38 | Name: "article:", 39 | Value: 500, 40 | ValueHuman: "500B", 41 | FullPath: "dev:article:", 42 | KeysCount: 5, 43 | Children: []Node{ 44 | {Name: "1", Value: 100, ValueHuman: "100B", FullPath: "dev:article:1", KeysCount: 1}, 45 | {Name: "2", Value: 100, ValueHuman: "100B", FullPath: "dev:article:2", KeysCount: 1}, 46 | {Name: "3", Value: 100, ValueHuman: "100B", FullPath: "dev:article:3", KeysCount: 1}, 47 | }, 48 | }, { 49 | Name: "user:", 50 | Value: 2000, 51 | ValueHuman: "2K", 52 | FullPath: "dev:user:", 53 | KeysCount: 2, 54 | Children: []Node{ 55 | {Name: "bar", Value: 1000, ValueHuman: "1000B", FullPath: "dev:user:bar", KeysCount: 1}, 56 | {Name: "foo", Value: 1000, ValueHuman: "1000B", FullPath: "dev:user:foo", KeysCount: 1}, 57 | }, 58 | }, 59 | }, 60 | }, 61 | { 62 | Name: "prod:user:", 63 | Value: 4000, 64 | ValueHuman: "3.9K", 65 | FullPath: "prod:user:", 66 | KeysCount: 2, 67 | Children: []Node{ 68 | {Name: "bar", Value: 2000, ValueHuman: "2K", FullPath: "prod:user:bar", KeysCount: 1}, 69 | {Name: "foo", Value: 2000, ValueHuman: "2K", FullPath: "prod:user:foo", KeysCount: 1}, 70 | }, 71 | }, 72 | }, 73 | } 74 | rendererMock.On("render", expectedRendererResult).Once().Return("test page content", nil) 75 | 76 | renderer := NewChartRenderer( 77 | srvMock, 78 | ChartRendererParams{ 79 | Depth: 2, 80 | Port: 123, 81 | }, 82 | ) 83 | renderer.pageRenderer = rendererMock 84 | 85 | err := renderer.Render(suite.trie.Root()) 86 | 87 | suite.Assert().Nil(err) 88 | srvMock.AssertExpectations(suite.T()) 89 | rendererMock.AssertExpectations(suite.T()) 90 | } 91 | 92 | func (suite *ChartRendererTestSuite) TestAnychartRenderer() { 93 | renderer := anychartRenderer{} 94 | res, err := renderer.render(Node{ 95 | Name: "Total", 96 | Value: 6500, 97 | ValueHuman: "6.3K", 98 | FullPath: "Total", 99 | KeysCount: 9, 100 | Children: []Node{ 101 | { 102 | Name: "dev:", 103 | Value: 100, 104 | ValueHuman: "100B", 105 | FullPath: "dev:", 106 | KeysCount: 7, 107 | }, 108 | { 109 | Name: "prod:", 110 | Value: 2500, 111 | ValueHuman: "2.4K", 112 | FullPath: "dev:", 113 | KeysCount: 15, 114 | }, 115 | }, 116 | }) 117 | 118 | r := strings.NewReader(res) 119 | d := xml.NewDecoder(r) 120 | 121 | // Configure the decoder for HTML; leave off strict and autoclose for XHTML 122 | d.Strict = true 123 | //d.AutoClose = xml.HTMLAutoClose 124 | d.Entity = xml.HTMLEntity 125 | 126 | var errParse error 127 | for errParse == nil { 128 | _, errParse = d.Token() 129 | } 130 | 131 | suite.Assert().Equal(io.EOF, errParse, "Invalid html: %s", errParse) 132 | suite.Assert().Nil(err) 133 | } 134 | 135 | func (suite *ChartRendererTestSuite) SetupTest() { 136 | suite.trie = trie.NewTrie(trie.NewPunctuationSplitter(':'), 3) 137 | 138 | suite.setupTrieKey("dev:article:1", 100) 139 | suite.setupTrieKey("dev:article:2", 100) 140 | suite.setupTrieKey("dev:article:3", 100) 141 | suite.setupTrieKey("dev:article:4", 100) 142 | suite.setupTrieKey("dev:article:5", 100) 143 | suite.setupTrieKey("dev:user:bar", 1000) 144 | suite.setupTrieKey("dev:user:foo", 1000) 145 | suite.setupTrieKey("prod:user:bar", 2000) 146 | suite.setupTrieKey("prod:user:foo", 2000) 147 | } 148 | 149 | func (suite *ChartRendererTestSuite) setupTrieKey(key string, value int64) { 150 | suite.trie.Add(key, trie.ParamValue{Param: trie.BytesSize, Value: value}, trie.ParamValue{Param: trie.KeysCount, Value: 1}) 151 | } 152 | 153 | func TestChartRendererTestSuite(t *testing.T) { 154 | suite.Run(t, new(ChartRendererTestSuite)) 155 | } 156 | 157 | type MockServer struct { 158 | mock.Mock 159 | } 160 | 161 | func (m *MockServer) Serve(port int, content string) { 162 | m.Called(port, content) 163 | } 164 | 165 | type mockPageRenderer struct { 166 | mock.Mock 167 | } 168 | 169 | func (m *mockPageRenderer) render(result Node) (string, error) { 170 | args := m.Called(result) 171 | 172 | return args.String(0), args.Error(1) 173 | } 174 | -------------------------------------------------------------------------------- /src/renderer/table_test.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "bytes" 5 | "fmt" 6 | "github.com/obukhov/redis-inventory/src/trie" 7 | "github.com/stretchr/testify/suite" 8 | "testing" 9 | ) 10 | 11 | type TableRendererTestSuite struct { 12 | suite.Suite 13 | trie *trie.Trie 14 | } 15 | 16 | func (suite *TableRendererTestSuite) TestRenderSpacePadding() { 17 | var buf bytes.Buffer 18 | 19 | r := TableRenderer{&buf, TableRendererParams{10, "", 2, false, " "}} 20 | 21 | err := r.Render(suite.trie.Root()) 22 | suite.Assert().Nil(err, "Error rendering trie") 23 | 24 | suite.Assert().Equal( 25 | ""+ 26 | "+-----------------------+----------+-----------+\n"+ 27 | "| KEY | BYTESIZE | KEYSCOUNT |\n"+ 28 | "+-----------------------+----------+-----------+\n"+ 29 | "| dev: | 2500 | 7 |\n"+ 30 | "| article: | 500 | 5 |\n"+ 31 | "| 1 | 100 | 1 |\n"+ 32 | "| 2 | 100 | 1 |\n"+ 33 | "| 3 | 100 | 1 |\n"+ 34 | "| ( 2 more keys ) | | |\n"+ 35 | "| user: | 2000 | 2 |\n"+ 36 | "| bar | 1000 | 1 |\n"+ 37 | "| foo | 1000 | 1 |\n"+ 38 | "| prod:user: | 4000 | 2 |\n"+ 39 | "| bar | 2000 | 1 |\n"+ 40 | "| foo | 2000 | 1 |\n"+ 41 | "+-----------------------+----------+-----------+\n", 42 | buf.String(), 43 | ) 44 | } 45 | 46 | func (suite *TableRendererTestSuite) TestRenderFullPathAndDepthLimit() { 47 | var buf bytes.Buffer 48 | 49 | r := TableRenderer{&buf, TableRendererParams{Depth: 2}} 50 | 51 | err := r.Render(suite.trie.Root()) 52 | suite.Assert().Nil(err, "Error rendering trie") 53 | 54 | suite.Assert().Equal( 55 | ""+ 56 | "+---------------+----------+-----------+\n"+ 57 | "| KEY | BYTESIZE | KEYSCOUNT |\n"+ 58 | "+---------------+----------+-----------+\n"+ 59 | "| dev: | 2500 | 7 |\n"+ 60 | "| dev:article: | 500 | 5 |\n"+ 61 | "| dev:user: | 2000 | 2 |\n"+ 62 | "| prod:user: | 4000 | 2 |\n"+ 63 | "| prod:user:bar | 2000 | 1 |\n"+ 64 | "| prod:user:foo | 2000 | 1 |\n"+ 65 | "+---------------+----------+-----------+\n", 66 | buf.String(), 67 | ) 68 | } 69 | 70 | func (suite *TableRendererTestSuite) TestRenderHuman() { 71 | var buf bytes.Buffer 72 | 73 | r := TableRenderer{&buf, TableRendererParams{Depth: 2, HumanReadable: true}} 74 | 75 | for i := 0; i < 1000; i++ { 76 | suite.setupTrieKey(fmt.Sprintf("dev:blog:%d", i), 2) 77 | } 78 | 79 | err := r.Render(suite.trie.Root()) 80 | suite.Assert().Nil(err, "Error rendering trie") 81 | 82 | suite.Assert().Equal( 83 | ""+ 84 | "+---------------+----------+-----------+\n"+ 85 | "| KEY | BYTESIZE | KEYSCOUNT |\n"+ 86 | "+---------------+----------+-----------+\n"+ 87 | "| dev: | 4.4K | 1,007 |\n"+ 88 | "| dev:article: | 500B | 5 |\n"+ 89 | "| dev:blog: | 2K | 1,000 |\n"+ 90 | "| dev:user: | 2K | 2 |\n"+ 91 | "| prod:user: | 3.9K | 2 |\n"+ 92 | "| prod:user:bar | 2K | 1 |\n"+ 93 | "| prod:user:foo | 2K | 1 |\n"+ 94 | "+---------------+----------+-----------+\n", 95 | buf.String(), 96 | ) 97 | } 98 | 99 | func (suite *TableRendererTestSuite) TestRenderParam() { 100 | for _, t := range []struct { 101 | paramString string 102 | expectedParams TableRendererParams 103 | }{ 104 | { 105 | "", 106 | TableRendererParams{Depth: 10}}, 107 | { 108 | "depth=2&padSpaces=3", 109 | TableRendererParams{Depth: 2, PaddingSpaceCount: 3, indent: " "}, 110 | }, 111 | { 112 | "depth=5&padding=>>", 113 | TableRendererParams{Depth: 5, Padding: ">>", indent: ">>"}, 114 | }, 115 | } { 116 | suite.Run(t.paramString, func() { 117 | 118 | params, err := NewTableRendererParams(t.paramString) 119 | suite.Assert().Nil(err) 120 | suite.Assert().Equal(t.expectedParams, params) 121 | }) 122 | } 123 | } 124 | 125 | func (suite *TableRendererTestSuite) SetupTest() { 126 | suite.trie = trie.NewTrie(trie.NewPunctuationSplitter(':'), 3) 127 | 128 | suite.setupTrieKey("dev:article:1", 100) 129 | suite.setupTrieKey("dev:article:2", 100) 130 | suite.setupTrieKey("dev:article:3", 100) 131 | suite.setupTrieKey("dev:article:4", 100) 132 | suite.setupTrieKey("dev:article:5", 100) 133 | suite.setupTrieKey("dev:user:bar", 1000) 134 | suite.setupTrieKey("dev:user:foo", 1000) 135 | suite.setupTrieKey("prod:user:bar", 2000) 136 | suite.setupTrieKey("prod:user:foo", 2000) 137 | } 138 | 139 | func (suite *TableRendererTestSuite) setupTrieKey(key string, value int64) { 140 | suite.trie.Add(key, trie.ParamValue{Param: trie.BytesSize, Value: value}, trie.ParamValue{Param: trie.KeysCount, Value: 1}) 141 | } 142 | 143 | func TestTableRendererTestSuite(t *testing.T) { 144 | suite.Run(t, new(TableRendererTestSuite)) 145 | } 146 | -------------------------------------------------------------------------------- /src/scanner/scanner_test.go: -------------------------------------------------------------------------------- 1 | package scanner 2 | 3 | import ( 4 | "context" 5 | "errors" 6 | "testing" 7 | 8 | "github.com/obukhov/redis-inventory/src/adapter" 9 | 10 | "github.com/obukhov/redis-inventory/src/trie" 11 | "github.com/rs/zerolog" 12 | "github.com/stretchr/testify/mock" 13 | "github.com/stretchr/testify/suite" 14 | ) 15 | 16 | type ScannerTestSuite struct { 17 | suite.Suite 18 | } 19 | 20 | type RedisServiceMock struct { 21 | mock.Mock 22 | } 23 | 24 | func (m *RedisServiceMock) ScanKeys(ctx context.Context, options adapter.ScanOptions) <-chan adapter.BulkKeyInfo { 25 | args := m.Called(ctx, options) 26 | return args.Get(0).(chan adapter.BulkKeyInfo) 27 | } 28 | 29 | func (m *RedisServiceMock) GetKeysCount(ctx context.Context) (int64, error) { 30 | args := m.Called(ctx) 31 | return int64(args.Int(0)), args.Error(1) 32 | } 33 | 34 | type ProgressWriterMock struct { 35 | mock.Mock 36 | } 37 | 38 | func (m *ProgressWriterMock) Start(total int64) { 39 | m.Called(total) 40 | } 41 | 42 | func (m *ProgressWriterMock) Increment() { 43 | m.Called() 44 | } 45 | 46 | func (m *ProgressWriterMock) Stop() { 47 | m.Called() 48 | } 49 | 50 | func (suite *ScannerTestSuite) TestScan() { 51 | scanChannel := make(chan string, 5) 52 | 53 | redisMock := &RedisServiceMock{} 54 | redisMock. 55 | On("GetKeysCount", mock.Anything).Return(2, nil). 56 | On( 57 | "ScanKeys", 58 | mock.Anything, 59 | adapter.ScanOptions{ 60 | ScanCount: 1000, 61 | Throttle: 0, 62 | }, 63 | ). 64 | Return(scanChannel) 65 | 66 | progressMock := &ProgressWriterMock{} 67 | progressMock. 68 | On("Start", int64(2)).Once(). 69 | On("Stop").Once(). 70 | On("Increment").Times(2) 71 | 72 | scanChannel <- "key1" 73 | scanChannel <- "key2" 74 | close(scanChannel) 75 | 76 | scanner := NewScanner(redisMock, progressMock, zerolog.Nop()) 77 | scanner.Scan( 78 | adapter.ScanOptions{ 79 | ScanCount: 1000, 80 | Throttle: 0, 81 | }, 82 | trie.NewTrie(trie.NewPunctuationSplitter(':'), 5), 83 | ) 84 | 85 | redisMock.AssertExpectations(suite.T()) 86 | progressMock.AssertExpectations(suite.T()) 87 | } 88 | 89 | func (suite *ScannerTestSuite) TestScanWithPattern() { 90 | scanChannel := make(chan string, 5) 91 | 92 | redisMock := &RedisServiceMock{} 93 | redisMock. 94 | On( 95 | "ScanKeys", 96 | mock.Anything, 97 | adapter.ScanOptions{ 98 | ScanCount: 1000, 99 | Throttle: 0, 100 | Pattern: "dev:*", 101 | }, 102 | ). 103 | Return(scanChannel) 104 | 105 | progressMock := &ProgressWriterMock{} 106 | progressMock. 107 | On("Start", int64(0)).Once(). 108 | On("Stop").Once(). 109 | On("Increment").Times(2) 110 | 111 | scanChannel <- "key1" 112 | scanChannel <- "key2" 113 | close(scanChannel) 114 | 115 | scanner := NewScanner(redisMock, progressMock, zerolog.Nop()) 116 | scanner.Scan( 117 | adapter.ScanOptions{ 118 | ScanCount: 1000, 119 | Throttle: 0, 120 | Pattern: "dev:*", 121 | }, 122 | trie.NewTrie(trie.NewPunctuationSplitter(':'), 5), 123 | ) 124 | 125 | redisMock.AssertExpectations(suite.T()) 126 | progressMock.AssertExpectations(suite.T()) 127 | } 128 | 129 | func (suite *ScannerTestSuite) TestScanWithError() { 130 | scanChannel := make(chan string, 5) 131 | 132 | redisMock := &RedisServiceMock{} 133 | redisMock. 134 | On("GetKeysCount", mock.Anything).Return(2, nil). 135 | On("ScanKeys", mock.Anything, mock.Anything).Return(scanChannel) 136 | 137 | progressMock := &ProgressWriterMock{} 138 | progressMock. 139 | On("Start", int64(2)).Once(). 140 | On("Stop").Once(). 141 | On("Increment").Times(2) 142 | 143 | scanChannel <- "key1" 144 | scanChannel <- "key2" 145 | close(scanChannel) 146 | 147 | scanner := NewScanner(redisMock, progressMock, zerolog.Nop()) 148 | result := trie.NewTrie(trie.NewPunctuationSplitter(':'), 5) 149 | scanner.Scan( 150 | adapter.ScanOptions{ 151 | ScanCount: 1000, 152 | Throttle: 0, 153 | }, 154 | result, 155 | ) 156 | 157 | redisMock.AssertExpectations(suite.T()) 158 | progressMock.AssertExpectations(suite.T()) 159 | 160 | suite.Assert().Equal(int64(10), result.Root().Aggregator().Params[trie.BytesSize]) 161 | } 162 | 163 | func (suite *ScannerTestSuite) TestScanCantGetCountKeys() { 164 | scanChannel := make(chan string, 5) 165 | 166 | redisMock := &RedisServiceMock{} 167 | redisMock. 168 | On("GetKeysCount", mock.Anything).Return(2, errors.New("cannot get count keys")). 169 | On("ScanKeys", mock.Anything, mock.Anything).Return(scanChannel) 170 | 171 | progressMock := &ProgressWriterMock{} 172 | progressMock. 173 | On("Start", int64(0)).Once(). 174 | On("Stop").Once(). 175 | On("Increment").Times(2) 176 | 177 | scanChannel <- "key1" 178 | scanChannel <- "key2" 179 | close(scanChannel) 180 | 181 | scanner := NewScanner(redisMock, progressMock, zerolog.Nop()) 182 | scanner.Scan( 183 | adapter.ScanOptions{ 184 | ScanCount: 1000, 185 | Throttle: 0, 186 | }, 187 | trie.NewTrie(trie.NewPunctuationSplitter(':'), 5), 188 | ) 189 | 190 | redisMock.AssertExpectations(suite.T()) 191 | progressMock.AssertExpectations(suite.T()) 192 | } 193 | 194 | func TestScannerTestSuite(t *testing.T) { 195 | suite.Run(t, new(ScannerTestSuite)) 196 | } 197 | -------------------------------------------------------------------------------- /src/renderer/chart.go: -------------------------------------------------------------------------------- 1 | package renderer 2 | 3 | import ( 4 | "encoding/json" 5 | "errors" 6 | "sort" 7 | "strings" 8 | 9 | "code.cloudfoundry.org/bytefmt" 10 | "github.com/hetiansu5/urlquery" 11 | "github.com/obukhov/redis-inventory/src/server" 12 | "github.com/obukhov/redis-inventory/src/trie" 13 | ) 14 | 15 | // NewChartRendererParams creates ChartRendererParams 16 | func NewChartRendererParams(paramsSerialized string) (ChartRendererParams, error) { 17 | params := ChartRendererParams{Depth: 10, Port: 8888} 18 | 19 | err := urlquery.Unmarshal([]byte(paramsSerialized), ¶ms) 20 | if err != nil { 21 | return params, err 22 | } 23 | 24 | if params.Port <= 0 { 25 | return params, errors.New("port cannot be negative") 26 | } 27 | 28 | return params, nil 29 | } 30 | 31 | // ChartRendererParams represents rendering params for web renderer 32 | type ChartRendererParams struct { 33 | Depth int `query:"depth"` 34 | Port int `query:"port"` 35 | } 36 | 37 | // NewChartRenderer creates ChartRenderer 38 | func NewChartRenderer(srv server.SimpleServerInterface, params ChartRendererParams) ChartRenderer { 39 | return ChartRenderer{ 40 | server: srv, 41 | pageRenderer: anychartRenderer{}, 42 | params: params, 43 | } 44 | } 45 | 46 | // ChartRenderer renders trie in the JSON format 47 | type ChartRenderer struct { 48 | server server.SimpleServerInterface 49 | pageRenderer pageRenderer 50 | params ChartRendererParams 51 | } 52 | 53 | // Render executes rendering 54 | func (o ChartRenderer) Render(root *trie.Node) error { 55 | result := o.toNode(root, "Total", "") 56 | result.Children = o.convertChildren(root, 0, "") 57 | 58 | rendered, err := o.pageRenderer.render(result) 59 | if err != nil { 60 | return err 61 | } 62 | 63 | o.server.Serve(o.params.Port, rendered) 64 | 65 | return nil 66 | } 67 | 68 | func (o ChartRenderer) convertChildren(node *trie.Node, level int, prefix string) []Node { 69 | result := make([]Node, 0) 70 | 71 | childKeys := make([]string, 0, len(node.Children)) 72 | for k := range node.Children { 73 | childKeys = append(childKeys, k) 74 | } 75 | 76 | sort.Strings(childKeys) 77 | for _, key := range childKeys { 78 | childNode := node.Children[key] 79 | nextLevel := level + 1 80 | if !childNode.HasAggregator() { 81 | var keys []string 82 | keys, childNode = childNode.FindNextAggregatedNodeWithKey() 83 | nextLevel += len(keys) 84 | key = key + strings.Join(keys, "") 85 | } 86 | 87 | item := o.toNode(childNode, key, prefix) 88 | 89 | if level < o.params.Depth { 90 | item.Children = o.convertChildren(childNode, nextLevel, prefix+key) 91 | } 92 | 93 | result = append(result, item) 94 | } 95 | 96 | return result 97 | } 98 | 99 | func (o ChartRenderer) toNode(childNode *trie.Node, key string, prefix string) Node { 100 | value := childNode.Aggregator().Params[trie.BytesSize] 101 | item := Node{ 102 | Name: key, 103 | Value: value, 104 | KeysCount: childNode.Aggregator().Params[trie.KeysCount], 105 | ValueHuman: bytefmt.ByteSize(uint64(value)), 106 | FullPath: prefix + key, 107 | } 108 | return item 109 | } 110 | 111 | // Node structure for serialized json of anychart library 112 | type Node struct { 113 | Name string `json:"name"` 114 | Value int64 `json:"value"` 115 | ValueHuman string `json:"valueHuman"` 116 | FullPath string `json:"pathFull"` 117 | KeysCount int64 `json:"keys"` 118 | Children []Node `json:"children"` 119 | } 120 | 121 | type pageRenderer interface { 122 | render(result Node) (string, error) 123 | } 124 | 125 | type anychartRenderer struct{} 126 | 127 | func (o anychartRenderer) render(result Node) (string, error) { 128 | s, err := json.Marshal([]Node{result}) 129 | if err != nil { 130 | return "", err 131 | } 132 | 133 | rendered := ` 134 | 135 | 136 | 137 | 138 | 139 | 140 |
141 | 183 | 184 | 185 | ` 186 | 187 | return rendered, nil 188 | } 189 | -------------------------------------------------------------------------------- /go.sum: -------------------------------------------------------------------------------- 1 | cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= 2 | cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= 3 | cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= 4 | cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= 5 | cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= 6 | cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= 7 | cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0= 8 | cloud.google.com/go v0.50.0/go.mod h1:r9sluTvynVuxRIOHXQEHMFffphuXHOMZMycpNR5e6To= 9 | cloud.google.com/go v0.52.0/go.mod h1:pXajvRH/6o3+F9jDHZWQ5PbGhn+o8w9qiu/CffaVdO4= 10 | cloud.google.com/go v0.53.0/go.mod h1:fp/UouUEsRkN6ryDKNW/Upv/JBKnv6WDthjR6+vze6M= 11 | cloud.google.com/go v0.54.0/go.mod h1:1rq2OEkV3YMf6n/9ZvGWI3GWw0VoqH/1x2nd8Is/bPc= 12 | cloud.google.com/go v0.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKVk= 13 | cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= 14 | cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc= 15 | cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY= 16 | cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI= 17 | cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk= 18 | cloud.google.com/go v0.78.0/go.mod h1:QjdrLG0uq+YwhjoVOLsS1t7TW8fs36kLs4XO5R5ECHg= 19 | cloud.google.com/go v0.79.0/go.mod h1:3bzgcEeQlzbuEAYu4mrWhKqWjmpprinYgKJLgKHnbb8= 20 | cloud.google.com/go v0.81.0/go.mod h1:mk/AM35KwGk/Nm2YSeZbxXdrNK3KZOYHmLkOqC2V6E0= 21 | cloud.google.com/go v0.83.0/go.mod h1:Z7MJUsANfY0pYPdw0lbnivPx4/vhy/e2FEkSkF7vAVY= 22 | cloud.google.com/go v0.84.0/go.mod h1:RazrYuxIK6Kb7YrzzhPoLmCVzl7Sup4NrbKPg8KHSUM= 23 | cloud.google.com/go v0.87.0/go.mod h1:TpDYlFy7vuLzZMMZ+B6iRiELaY7z/gJPaqbMx6mlWcY= 24 | cloud.google.com/go v0.90.0/go.mod h1:kRX0mNRHe0e2rC6oNakvwQqzyDmg57xJ+SZU1eT2aDQ= 25 | cloud.google.com/go v0.93.3/go.mod h1:8utlLll2EF5XMAV15woO4lSbWQlk8rer9aLOfLh7+YI= 26 | cloud.google.com/go v0.94.1/go.mod h1:qAlAugsXlC+JWO+Bke5vCtc9ONxjQT3drlTTnAplMW4= 27 | cloud.google.com/go v0.97.0/go.mod h1:GF7l59pYBVlXQIBLx3a761cZ41F9bBH3JUlihCt2Udc= 28 | cloud.google.com/go v0.99.0/go.mod h1:w0Xx2nLzqWJPuozYQX+hFfCSI8WioryfRDzkoI/Y2ZA= 29 | cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= 30 | cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE= 31 | cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc= 32 | cloud.google.com/go/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg= 33 | cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc= 34 | cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ= 35 | cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= 36 | cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= 37 | cloud.google.com/go/firestore v1.6.1/go.mod h1:asNXNOzBdyVQmEU+ggO8UPodTkEVFW5Qx+rwHnAz+EY= 38 | cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= 39 | cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= 40 | cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA= 41 | cloud.google.com/go/pubsub v1.3.1/go.mod h1:i+ucay31+CNRpDW4Lu78I4xXG+O1r/MAHgjpRVR+TSU= 42 | cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw= 43 | cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0ZeosJ0Rtdos= 44 | cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohlUTyfDhBk= 45 | cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs= 46 | cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0= 47 | code.cloudfoundry.org/bytefmt v0.0.0-20210608160410-67692ebc98de h1:Gm/tSRP5CPFuoMUo7NDbRMbpN3h9fBNtD6bXC+Lye9I= 48 | code.cloudfoundry.org/bytefmt v0.0.0-20210608160410-67692ebc98de/go.mod h1:YOakoAiZWNbkynB2dKOKvdxVehYGn3EH4UXq/i99hYg= 49 | dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= 50 | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= 51 | github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= 52 | github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= 53 | github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= 54 | github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= 55 | github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= 56 | github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= 57 | github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= 58 | github.com/alicebob/gopher-json v0.0.0-20200520072559-a9ecdc9d1d3a h1:HbKu58rmZpUGpz5+4FfNmIU+FmZg2P3Xaj2v2bfNWmk= 59 | github.com/alicebob/gopher-json v0.0.0-20200520072559-a9ecdc9d1d3a/go.mod h1:SGnFV6hVsYE877CKEZ6tDNTjaSXYUk6QqoIK6PrAtcc= 60 | github.com/alicebob/miniredis/v2 v2.18.0 h1:EPUGD69ou4Uw4c81t9NLh0+dSou46k4tFEvf498FJ0g= 61 | github.com/alicebob/miniredis/v2 v2.18.0/go.mod h1:gquAfGbzn92jvtrSC69+6zZnwSODVXVpYDRaGhWaL6I= 62 | github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= 63 | github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= 64 | github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= 65 | github.com/armon/go-metrics v0.3.10/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= 66 | github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= 67 | github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= 68 | github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= 69 | github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= 70 | github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= 71 | github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= 72 | github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= 73 | github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= 74 | github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= 75 | github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= 76 | github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= 77 | github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= 78 | github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= 79 | github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= 80 | github.com/circonus-labs/circonus-gometrics v2.3.1+incompatible/go.mod h1:nmEj6Dob7S7YxXgwXpfOuvO54S+tGdZdw9fuRZt25Ag= 81 | github.com/circonus-labs/circonusllhist v0.1.3/go.mod h1:kMXHVDlOchFAehlya5ePtbp5jckzBHf4XRpQvBOLI+I= 82 | github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= 83 | github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= 84 | github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= 85 | github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= 86 | github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI= 87 | github.com/cncf/xds/go v0.0.0-20210312221358-fbca930ec8ed/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= 88 | github.com/cncf/xds/go v0.0.0-20210805033703-aa0b78936158/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= 89 | github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= 90 | github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= 91 | github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= 92 | github.com/cncf/xds/go v0.0.0-20211130200136-a8f946100490/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= 93 | github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= 94 | github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= 95 | github.com/cpuguy83/go-md2man/v2 v2.0.1 h1:r/myEWzV9lfsM1tFLgDyu0atFtJ1fXn261LKYj/3DxU= 96 | github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= 97 | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= 98 | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= 99 | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= 100 | github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= 101 | github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= 102 | github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= 103 | github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po= 104 | github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= 105 | github.com/envoyproxy/go-control-plane v0.9.9-0.20210217033140-668b12f5399d/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= 106 | github.com/envoyproxy/go-control-plane v0.9.9-0.20210512163311-63b5d3c536b0/go.mod h1:hliV/p42l8fGbc6Y9bQ70uLwIvmJyVE5k4iMKlh8wCQ= 107 | github.com/envoyproxy/go-control-plane v0.9.10-0.20210907150352-cf90f659a021/go.mod h1:AFq3mo9L8Lqqiid3OhADV3RfLJnjiw63cSpi+fDTRC0= 108 | github.com/envoyproxy/go-control-plane v0.10.1/go.mod h1:AY7fTTXNdv/aJ2O5jwpxAPOWUZ7hQAEvzN5Pf27BkQQ= 109 | github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= 110 | github.com/envoyproxy/protoc-gen-validate v0.6.2/go.mod h1:2t7qjJNvHPx8IjnBOzl9E9/baC+qXE/TeeyBRzgJDws= 111 | github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= 112 | github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= 113 | github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= 114 | github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= 115 | github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= 116 | github.com/fsnotify/fsnotify v1.5.1 h1:mZcQUHVQUQWoPXXtuf9yuEXKudkV2sx1E06UadKWpgI= 117 | github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= 118 | github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= 119 | github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= 120 | github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= 121 | github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= 122 | github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= 123 | github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= 124 | github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= 125 | github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= 126 | github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= 127 | github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= 128 | github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= 129 | github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= 130 | github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= 131 | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= 132 | github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= 133 | github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= 134 | github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= 135 | github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= 136 | github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= 137 | github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= 138 | github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= 139 | github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= 140 | github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= 141 | github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= 142 | github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= 143 | github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8= 144 | github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= 145 | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= 146 | github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= 147 | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= 148 | github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= 149 | github.com/golang/protobuf v1.3.4/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= 150 | github.com/golang/protobuf v1.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk= 151 | github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= 152 | github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= 153 | github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= 154 | github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= 155 | github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= 156 | github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= 157 | github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= 158 | github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= 159 | github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= 160 | github.com/golang/protobuf v1.5.1/go.mod h1:DopwsBzvsk0Fs44TXzsVbJyPhcCPeIwnvohx4u74HPM= 161 | github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= 162 | github.com/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= 163 | github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= 164 | github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= 165 | github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= 166 | github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= 167 | github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= 168 | github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 169 | github.com/google/go-cmp v0.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 170 | github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 171 | github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 172 | github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 173 | github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 174 | github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 175 | github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 176 | github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= 177 | github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= 178 | github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= 179 | github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= 180 | github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= 181 | github.com/google/martian/v3 v3.2.1/go.mod h1:oBOf6HBosgwRXnUGWUB05QECsc6uvmMiJ3+6W4l/CUk= 182 | github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= 183 | github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= 184 | github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= 185 | github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= 186 | github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= 187 | github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= 188 | github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= 189 | github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 190 | github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 191 | github.com/google/pprof v0.0.0-20210122040257-d980be63207e/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 192 | github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 193 | github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 194 | github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 195 | github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 196 | github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= 197 | github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= 198 | github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= 199 | github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= 200 | github.com/googleapis/gax-go/v2 v2.1.0/go.mod h1:Q3nei7sK6ybPYH7twZdmQpAd1MKb7pfu6SK+H1/DsU0= 201 | github.com/googleapis/gax-go/v2 v2.1.1/go.mod h1:hddJymUZASv3XPyGkUpKj8pPO47Rmb0eJc8R6ouapiM= 202 | github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= 203 | github.com/hashicorp/consul/api v1.12.0/go.mod h1:6pVBMo0ebnYdt2S3H87XhekM/HHrUoTD2XXb/VrZVy0= 204 | github.com/hashicorp/consul/sdk v0.8.0/go.mod h1:GBvyrGALthsZObzUGsfgHZQDXjg4lOjagTIwIR1vPms= 205 | github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= 206 | github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= 207 | github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= 208 | github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= 209 | github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= 210 | github.com/hashicorp/go-hclog v1.0.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= 211 | github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= 212 | github.com/hashicorp/go-immutable-radix v1.3.1/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= 213 | github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= 214 | github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= 215 | github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= 216 | github.com/hashicorp/go-retryablehttp v0.5.3/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs= 217 | github.com/hashicorp/go-rootcerts v1.0.2/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= 218 | github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= 219 | github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= 220 | github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= 221 | github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= 222 | github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= 223 | github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= 224 | github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= 225 | github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= 226 | github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= 227 | github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= 228 | github.com/hashicorp/mdns v1.0.4/go.mod h1:mtBihi+LeNXGtG8L9dX59gAEa12BDtBQSp4v/YAJqrc= 229 | github.com/hashicorp/memberlist v0.3.0/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= 230 | github.com/hashicorp/serf v0.9.6/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= 231 | github.com/hetiansu5/urlquery v1.2.7 h1:jn0h+9pIRqUziSPnRdK/gJK8S5TCnk+HZZx5fRHf8K0= 232 | github.com/hetiansu5/urlquery v1.2.7/go.mod h1:wFpZdTHRdwt7mk0EM/DdZEWtEN4xf8HJoH/BLXm/PG0= 233 | github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= 234 | github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= 235 | github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= 236 | github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= 237 | github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM= 238 | github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= 239 | github.com/jedib0t/go-pretty/v6 v6.2.7 h1:4823Lult/tJ0VI1PgW3aSKw59pMWQ6Kzv9b3Bj6MwY0= 240 | github.com/jedib0t/go-pretty/v6 v6.2.7/go.mod h1:FMkOpgGD3EZ91cW8g/96RfxoV7bdeJyzXPYgz1L1ln0= 241 | github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= 242 | github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= 243 | github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= 244 | github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= 245 | github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= 246 | github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= 247 | github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= 248 | github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= 249 | github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= 250 | github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= 251 | github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= 252 | github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= 253 | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= 254 | github.com/kr/pretty v0.2.0 h1:s5hAObm+yFO5uHYt5dYjxi2rXrsnmRpJx4OYvIWUaQs= 255 | github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= 256 | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= 257 | github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= 258 | github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= 259 | github.com/lyft/protoc-gen-star v0.5.3/go.mod h1:V0xaHgaf5oCCqmcxYcWiDfTiKsZsRc87/1qhoTACD8w= 260 | github.com/magiconair/properties v1.8.5 h1:b6kJs+EmPFMYGkow9GiUyCyOvIwYetYJ3fSaWak/Gls= 261 | github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= 262 | github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= 263 | github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= 264 | github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= 265 | github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= 266 | github.com/mattn/go-colorable v0.1.12/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= 267 | github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= 268 | github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= 269 | github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= 270 | github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= 271 | github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= 272 | github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= 273 | github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU= 274 | github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= 275 | github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= 276 | github.com/mediocregopher/radix/v4 v4.0.0 h1:BUj/kzvuppH81PTHoxQqmQhu8JpHDWRFST6JQQE0hBQ= 277 | github.com/mediocregopher/radix/v4 v4.0.0/go.mod h1:ajchozX/6ELmydxWeWM6xCFHVpZ4+67LXHOTOVR0nCE= 278 | github.com/miekg/dns v1.1.26/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= 279 | github.com/miekg/dns v1.1.41/go.mod h1:p6aan82bvRIyn+zDIv9xYNUpwa73JcSh9BKwknJysuI= 280 | github.com/mitchellh/cli v1.1.0/go.mod h1:xcISNoH86gajksDmfB23e/pu+B+GeFRMYmoHXxx3xhI= 281 | github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= 282 | github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= 283 | github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= 284 | github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= 285 | github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= 286 | github.com/mitchellh/mapstructure v1.4.3 h1:OVowDSCllw/YjdLkam3/sm7wEtOy59d8ndGgCcyj8cs= 287 | github.com/mitchellh/mapstructure v1.4.3/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= 288 | github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= 289 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= 290 | github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= 291 | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= 292 | github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= 293 | github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= 294 | github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= 295 | github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= 296 | github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= 297 | github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= 298 | github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= 299 | github.com/onsi/ginkgo v1.16.2/go.mod h1:CObGmKUOKaSC0RjmoAK7tKyn4Azo5P2IWuoMnvwxz1E= 300 | github.com/onsi/ginkgo v1.16.4 h1:29JGrr5oVBm5ulCWet69zQkzWipVXIol6ygQUe/EzNc= 301 | github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= 302 | github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= 303 | github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= 304 | github.com/onsi/gomega v1.13.0 h1:7lLHu94wT9Ij0o6EWWclhu0aOh32VxhkwEJvzuWPeak= 305 | github.com/onsi/gomega v1.13.0/go.mod h1:lRk9szgn8TxENtWd0Tp4c3wjlRfMTMH27I+3Je41yGY= 306 | github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= 307 | github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= 308 | github.com/pelletier/go-toml v1.9.4 h1:tjENF6MfZAg8e4ZmZTeWaWiT2vXtsoO6+iuOjFhECwM= 309 | github.com/pelletier/go-toml v1.9.4/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= 310 | github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= 311 | github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= 312 | github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= 313 | github.com/pkg/profile v1.6.0/go.mod h1:qBsxPvzyUincmltOk6iyRVxHYg4adc0OFOv72ZdLa18= 314 | github.com/pkg/sftp v1.10.1/go.mod h1:lYOWFsE0bwd1+KfKJaKeuokY15vzFx25BLbzYYoAxZI= 315 | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= 316 | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= 317 | github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= 318 | github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s= 319 | github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= 320 | github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= 321 | github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= 322 | github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= 323 | github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= 324 | github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= 325 | github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= 326 | github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= 327 | github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= 328 | github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= 329 | github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= 330 | github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= 331 | github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= 332 | github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= 333 | github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= 334 | github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= 335 | github.com/rs/xid v1.3.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= 336 | github.com/rs/zerolog v1.26.1 h1:/ihwxqH+4z8UxyI70wM1z9yCvkWcfz/a3mj48k/Zngc= 337 | github.com/rs/zerolog v1.26.1/go.mod h1:/wSSJWX7lVrsOwlbyTRSOJvqRlc+WjWlfes+CiJ+tmc= 338 | github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk= 339 | github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= 340 | github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= 341 | github.com/sagikazarmark/crypt v0.4.0/go.mod h1:ALv2SRj7GxYV4HO9elxH9nS6M9gW+xDNxqmyJ6RfDFM= 342 | github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= 343 | github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= 344 | github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= 345 | github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= 346 | github.com/spf13/afero v1.3.3/go.mod h1:5KUK8ByomD5Ti5Artl0RtHeI5pTF7MIDuXL3yY520V4= 347 | github.com/spf13/afero v1.6.0 h1:xoax2sJ2DT8S8xA2paPFjDCScCNeWsg75VG0DLRreiY= 348 | github.com/spf13/afero v1.6.0/go.mod h1:Ai8FlHk4v/PARR026UzYexafAt9roJ7LcLMAmO6Z93I= 349 | github.com/spf13/cast v1.4.1 h1:s0hze+J0196ZfEMTs80N7UlFt0BDuQ7Q+JDnHiMWKdA= 350 | github.com/spf13/cast v1.4.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= 351 | github.com/spf13/cobra v1.4.0 h1:y+wJpx64xcgO1V+RcnwW0LEHxTKRi2ZDPSBjWnrg88Q= 352 | github.com/spf13/cobra v1.4.0/go.mod h1:Wo4iy3BUC+X2Fybo0PDqwJIv3dNRiZLHQymsfxlB84g= 353 | github.com/spf13/jwalterweatherman v1.1.0 h1:ue6voC5bR5F8YxI5S67j9i582FU4Qvo2bmqnqMYADFk= 354 | github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0EXowPYD95IqWIGo= 355 | github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= 356 | github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= 357 | github.com/spf13/viper v1.10.1 h1:nuJZuYpG7gTj/XqiUwg8bA0cp1+M2mC3J4g5luUYBKk= 358 | github.com/spf13/viper v1.10.1/go.mod h1:IGlFPqhNAPKRxohIzWpI5QEy4kuI7tcl5WvR+8qy1rU= 359 | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= 360 | github.com/stretchr/objx v0.1.1 h1:2vfRuCMp5sSVIDSqO8oNnWJq7mPa6KVP3iPIwFBuy8A= 361 | github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= 362 | github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= 363 | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= 364 | github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= 365 | github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= 366 | github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= 367 | github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= 368 | github.com/stretchr/testify v1.7.1 h1:5TQK59W5E3v0r2duFAb7P95B6hEeOyEnHRa8MjYSMTY= 369 | github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= 370 | github.com/subosito/gotenv v1.2.0 h1:Slr1R9HxAlEKefgq5jn9U+DnETlIUa6HfgEzj0g5d7s= 371 | github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= 372 | github.com/tilinna/clock v1.0.2 h1:6BO2tyAC9JbPExKH/z9zl44FLu1lImh3nDNKA0kgrkI= 373 | github.com/tilinna/clock v1.0.2/go.mod h1:ZsP7BcY7sEEz7ktc0IVy8Us6boDrK8VradlKRUGfOao= 374 | github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= 375 | github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= 376 | github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= 377 | github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= 378 | github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= 379 | github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= 380 | github.com/yuin/goldmark v1.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= 381 | github.com/yuin/gopher-lua v0.0.0-20200816102855-ee81675732da h1:NimzV1aGyq29m5ukMK0AMWEhFaL/lrEOaephfuoiARg= 382 | github.com/yuin/gopher-lua v0.0.0-20200816102855-ee81675732da/go.mod h1:E1AXubJBdNmFERAOucpDIxNzeGfLzg0mYh+UfMWdChA= 383 | go.etcd.io/etcd/api/v3 v3.5.1/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= 384 | go.etcd.io/etcd/client/pkg/v3 v3.5.1/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= 385 | go.etcd.io/etcd/client/v2 v2.305.1/go.mod h1:pMEacxZW7o8pg4CrFE7pquyCJJzZvkvdD2RibOCCCGs= 386 | go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= 387 | go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= 388 | go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= 389 | go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= 390 | go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= 391 | go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= 392 | go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= 393 | go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= 394 | go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= 395 | go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= 396 | go.uber.org/zap v1.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo= 397 | golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= 398 | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= 399 | golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= 400 | golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= 401 | golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= 402 | golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY= 403 | golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= 404 | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= 405 | golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= 406 | golang.org/x/crypto v0.0.0-20211215165025-cf75a172585e/go.mod h1:P+XmwS30IXTQdn5tA2iutPOUgjI07+tq3H3K9MVA1s8= 407 | golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= 408 | golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= 409 | golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= 410 | golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= 411 | golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY= 412 | golang.org/x/exp v0.0.0-20191129062945-2f5052295587/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= 413 | golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= 414 | golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= 415 | golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM= 416 | golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU= 417 | golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= 418 | golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= 419 | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= 420 | golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= 421 | golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= 422 | golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= 423 | golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= 424 | golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= 425 | golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= 426 | golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs= 427 | golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= 428 | golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= 429 | golang.org/x/lint v0.0.0-20201208152925-83fdc39ff7b5/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= 430 | golang.org/x/lint v0.0.0-20210508222113-6edffad5e616/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= 431 | golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= 432 | golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= 433 | golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= 434 | golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY= 435 | golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= 436 | golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= 437 | golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= 438 | golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= 439 | golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= 440 | golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= 441 | golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= 442 | golang.org/x/mod v0.5.0/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro= 443 | golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= 444 | golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= 445 | golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= 446 | golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= 447 | golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= 448 | golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= 449 | golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= 450 | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= 451 | golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= 452 | golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= 453 | golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= 454 | golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 455 | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 456 | golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 457 | golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 458 | golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 459 | golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 460 | golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 461 | golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 462 | golang.org/x/net v0.0.0-20200222125558-5a598a2470a0/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 463 | golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 464 | golang.org/x/net v0.0.0-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= 465 | golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= 466 | golang.org/x/net v0.0.0-20200501053045-e0ff5e5a1de5/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= 467 | golang.org/x/net v0.0.0-20200506145744-7e3656a0809f/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= 468 | golang.org/x/net v0.0.0-20200513185701-a91f0712d120/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= 469 | golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= 470 | golang.org/x/net v0.0.0-20200520182314-0ba52f642ac2/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= 471 | golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= 472 | golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= 473 | golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= 474 | golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= 475 | golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= 476 | golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= 477 | golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= 478 | golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= 479 | golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= 480 | golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= 481 | golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= 482 | golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1/go.mod h1:9tjilg8BloeKEkVJvy7fQ90B1CfIiPueXVOjqfkSzI8= 483 | golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= 484 | golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= 485 | golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= 486 | golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d h1:LO7XpTYMwTqxjLcGWPijK3vRXg1aWdlNOVOHRq45d7c= 487 | golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= 488 | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= 489 | golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= 490 | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= 491 | golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= 492 | golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= 493 | golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 494 | golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 495 | golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 496 | golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 497 | golang.org/x/oauth2 v0.0.0-20210220000619-9bb904979d93/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 498 | golang.org/x/oauth2 v0.0.0-20210313182246-cd4f82c27b84/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 499 | golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 500 | golang.org/x/oauth2 v0.0.0-20210628180205-a41e5a781914/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 501 | golang.org/x/oauth2 v0.0.0-20210805134026-6f1e6394065a/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 502 | golang.org/x/oauth2 v0.0.0-20210819190943-2bc19b11175f/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 503 | golang.org/x/oauth2 v0.0.0-20211005180243-6b3c2da341f1/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 504 | golang.org/x/oauth2 v0.0.0-20211104180415-d3ed0bb246c8/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= 505 | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 506 | golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 507 | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 508 | golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 509 | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 510 | golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 511 | golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 512 | golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 513 | golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 514 | golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 515 | golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= 516 | golang.org/x/sys v0.0.0-20180816055513-1c9583448a9c/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 517 | golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 518 | golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 519 | golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 520 | golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 521 | golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 522 | golang.org/x/sys v0.0.0-20190204203706-41f3e6584952/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 523 | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 524 | golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= 525 | golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 526 | golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 527 | golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 528 | golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 529 | golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 530 | golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 531 | golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 532 | golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 533 | golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 534 | golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 535 | golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 536 | golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 537 | golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 538 | golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 539 | golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 540 | golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 541 | golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 542 | golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 543 | golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 544 | golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 545 | golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 546 | golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 547 | golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 548 | golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 549 | golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 550 | golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 551 | golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 552 | golang.org/x/sys v0.0.0-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 553 | golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 554 | golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 555 | golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 556 | golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 557 | golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 558 | golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 559 | golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 560 | golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 561 | golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 562 | golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 563 | golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 564 | golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 565 | golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 566 | golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 567 | golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 568 | golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 569 | golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 570 | golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 571 | golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 572 | golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 573 | golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 574 | golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 575 | golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 576 | golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 577 | golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 578 | golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 579 | golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 580 | golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 581 | golang.org/x/sys v0.0.0-20210816183151-1e6c022a8912/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 582 | golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 583 | golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 584 | golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 585 | golang.org/x/sys v0.0.0-20211007075335-d3039528d8ac/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 586 | golang.org/x/sys v0.0.0-20211124211545-fe61309f8881/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 587 | golang.org/x/sys v0.0.0-20211210111614-af8b64212486 h1:5hpz5aRr+W1erYCL5JRhSUBJRph7l9XkNveoExlrKYk= 588 | golang.org/x/sys v0.0.0-20211210111614-af8b64212486/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= 589 | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= 590 | golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= 591 | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= 592 | golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= 593 | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= 594 | golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= 595 | golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= 596 | golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= 597 | golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= 598 | golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= 599 | golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= 600 | golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= 601 | golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= 602 | golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= 603 | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= 604 | golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= 605 | golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= 606 | golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= 607 | golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= 608 | golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= 609 | golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= 610 | golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= 611 | golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= 612 | golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= 613 | golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= 614 | golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= 615 | golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 616 | golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 617 | golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 618 | golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 619 | golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 620 | golang.org/x/tools v0.0.0-20191115202509-3a792d9c32b2/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 621 | golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 622 | golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 623 | golang.org/x/tools v0.0.0-20191130070609-6e064ea0cf2d/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= 624 | golang.org/x/tools v0.0.0-20191216173652-a0e659d51361/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 625 | golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 626 | golang.org/x/tools v0.0.0-20200117161641-43d50277825c/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 627 | golang.org/x/tools v0.0.0-20200122220014-bf1340f18c4a/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 628 | golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 629 | golang.org/x/tools v0.0.0-20200204074204-1cc6d1ef6c74/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 630 | golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 631 | golang.org/x/tools v0.0.0-20200212150539-ea181f53ac56/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 632 | golang.org/x/tools v0.0.0-20200224181240-023911ca70b2/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 633 | golang.org/x/tools v0.0.0-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= 634 | golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= 635 | golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= 636 | golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= 637 | golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= 638 | golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= 639 | golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= 640 | golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= 641 | golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= 642 | golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= 643 | golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= 644 | golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= 645 | golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE= 646 | golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= 647 | golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= 648 | golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= 649 | golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= 650 | golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= 651 | golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= 652 | golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= 653 | golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= 654 | golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= 655 | golang.org/x/tools v0.1.3/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= 656 | golang.org/x/tools v0.1.4/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= 657 | golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= 658 | golang.org/x/tools v0.1.7/go.mod h1:LGqMHiF4EqQNHR1JncWGqT5BVaXmza+X+BDGol+dOxo= 659 | golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= 660 | golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= 661 | golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= 662 | golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= 663 | google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= 664 | google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= 665 | google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= 666 | google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= 667 | google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= 668 | google.golang.org/api v0.14.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= 669 | google.golang.org/api v0.15.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= 670 | google.golang.org/api v0.17.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= 671 | google.golang.org/api v0.18.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= 672 | google.golang.org/api v0.19.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= 673 | google.golang.org/api v0.20.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= 674 | google.golang.org/api v0.22.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= 675 | google.golang.org/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= 676 | google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= 677 | google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM= 678 | google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc= 679 | google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg= 680 | google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= 681 | google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8= 682 | google.golang.org/api v0.41.0/go.mod h1:RkxM5lITDfTzmyKFPt+wGrCJbVfniCr2ool8kTBzRTU= 683 | google.golang.org/api v0.43.0/go.mod h1:nQsDGjRXMo4lvh5hP0TKqF244gqhGcr/YSIykhUk/94= 684 | google.golang.org/api v0.47.0/go.mod h1:Wbvgpq1HddcWVtzsVLyfLp8lDg6AA241LmgIL59tHXo= 685 | google.golang.org/api v0.48.0/go.mod h1:71Pr1vy+TAZRPkPs/xlCf5SsU8WjuAWv1Pfjbtukyy4= 686 | google.golang.org/api v0.50.0/go.mod h1:4bNT5pAuq5ji4SRZm+5QIkjny9JAyVD/3gaSihNefaw= 687 | google.golang.org/api v0.51.0/go.mod h1:t4HdrdoNgyN5cbEfm7Lum0lcLDLiise1F8qDKX00sOU= 688 | google.golang.org/api v0.54.0/go.mod h1:7C4bFFOvVDGXjfDTAsgGwDgAxRDeQ4X8NvUedIt6z3k= 689 | google.golang.org/api v0.55.0/go.mod h1:38yMfeP1kfjsl8isn0tliTjIb1rJXcQi4UXlbqivdVE= 690 | google.golang.org/api v0.56.0/go.mod h1:38yMfeP1kfjsl8isn0tliTjIb1rJXcQi4UXlbqivdVE= 691 | google.golang.org/api v0.57.0/go.mod h1:dVPlbZyBo2/OjBpmvNdpn2GRm6rPy75jyU7bmhdrMgI= 692 | google.golang.org/api v0.59.0/go.mod h1:sT2boj7M9YJxZzgeZqXogmhfmRWDtPzT31xkieUbuZU= 693 | google.golang.org/api v0.61.0/go.mod h1:xQRti5UdCmoCEqFxcz93fTl338AVqDgyaDRuOZ3hg9I= 694 | google.golang.org/api v0.63.0/go.mod h1:gs4ij2ffTRXwuzzgJl/56BdwJaA194ijkfn++9tDuPo= 695 | google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= 696 | google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= 697 | google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= 698 | google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= 699 | google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= 700 | google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= 701 | google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= 702 | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= 703 | google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= 704 | google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= 705 | google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= 706 | google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= 707 | google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= 708 | google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= 709 | google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= 710 | google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= 711 | google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= 712 | google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= 713 | google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= 714 | google.golang.org/genproto v0.0.0-20200115191322-ca5a22157cba/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= 715 | google.golang.org/genproto v0.0.0-20200122232147-0452cf42e150/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= 716 | google.golang.org/genproto v0.0.0-20200204135345-fa8e72b47b90/go.mod h1:GmwEX6Z4W5gMy59cAlVYjN9JhxgbQH6Gn+gFDQe2lzA= 717 | google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 718 | google.golang.org/genproto v0.0.0-20200224152610-e50cd9704f63/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 719 | google.golang.org/genproto v0.0.0-20200228133532-8c2c7df3a383/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 720 | google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 721 | google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 722 | google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 723 | google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 724 | google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 725 | google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 726 | google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U= 727 | google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= 728 | google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA= 729 | google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 730 | google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 731 | google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 732 | google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 733 | google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 734 | google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 735 | google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 736 | google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 737 | google.golang.org/genproto v0.0.0-20210222152913-aa3ee6e6a81c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 738 | google.golang.org/genproto v0.0.0-20210303154014-9728d6b83eeb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 739 | google.golang.org/genproto v0.0.0-20210310155132-4ce2db91004e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 740 | google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= 741 | google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= 742 | google.golang.org/genproto v0.0.0-20210513213006-bf773b8c8384/go.mod h1:P3QM42oQyzQSnHPnZ/vqoCdDmzH28fzWByN9asMeM8A= 743 | google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= 744 | google.golang.org/genproto v0.0.0-20210604141403-392c879c8b08/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= 745 | google.golang.org/genproto v0.0.0-20210608205507-b6d2f5bf0d7d/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= 746 | google.golang.org/genproto v0.0.0-20210624195500-8bfb893ecb84/go.mod h1:SzzZ/N+nwJDaO1kznhnlzqS8ocJICar6hYhVyhi++24= 747 | google.golang.org/genproto v0.0.0-20210713002101-d411969a0d9a/go.mod h1:AxrInvYm1dci+enl5hChSFPOmmUF1+uAa/UsgNRWd7k= 748 | google.golang.org/genproto v0.0.0-20210716133855-ce7ef5c701ea/go.mod h1:AxrInvYm1dci+enl5hChSFPOmmUF1+uAa/UsgNRWd7k= 749 | google.golang.org/genproto v0.0.0-20210728212813-7823e685a01f/go.mod h1:ob2IJxKrgPT52GcgX759i1sleT07tiKowYBGbczaW48= 750 | google.golang.org/genproto v0.0.0-20210805201207-89edb61ffb67/go.mod h1:ob2IJxKrgPT52GcgX759i1sleT07tiKowYBGbczaW48= 751 | google.golang.org/genproto v0.0.0-20210813162853-db860fec028c/go.mod h1:cFeNkxwySK631ADgubI+/XFU/xp8FD5KIVV4rj8UC5w= 752 | google.golang.org/genproto v0.0.0-20210821163610-241b8fcbd6c8/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= 753 | google.golang.org/genproto v0.0.0-20210828152312-66f60bf46e71/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= 754 | google.golang.org/genproto v0.0.0-20210831024726-fe130286e0e2/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= 755 | google.golang.org/genproto v0.0.0-20210903162649-d08c68adba83/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= 756 | google.golang.org/genproto v0.0.0-20210909211513-a8c4777a87af/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= 757 | google.golang.org/genproto v0.0.0-20210924002016-3dee208752a0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= 758 | google.golang.org/genproto v0.0.0-20211008145708-270636b82663/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= 759 | google.golang.org/genproto v0.0.0-20211028162531-8db9c33dc351/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= 760 | google.golang.org/genproto v0.0.0-20211118181313-81c1377c94b1/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= 761 | google.golang.org/genproto v0.0.0-20211206160659-862468c7d6e0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= 762 | google.golang.org/genproto v0.0.0-20211208223120-3a66f561d7aa/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= 763 | google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= 764 | google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= 765 | google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= 766 | google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= 767 | google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= 768 | google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= 769 | google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= 770 | google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= 771 | google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKal+60= 772 | google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= 773 | google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= 774 | google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= 775 | google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= 776 | google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= 777 | google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= 778 | google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= 779 | google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= 780 | google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= 781 | google.golang.org/grpc v1.36.1/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= 782 | google.golang.org/grpc v1.37.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= 783 | google.golang.org/grpc v1.37.1/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= 784 | google.golang.org/grpc v1.38.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= 785 | google.golang.org/grpc v1.39.0/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= 786 | google.golang.org/grpc v1.39.1/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= 787 | google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= 788 | google.golang.org/grpc v1.40.1/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= 789 | google.golang.org/grpc v1.43.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= 790 | google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= 791 | google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= 792 | google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= 793 | google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= 794 | google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= 795 | google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= 796 | google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= 797 | google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= 798 | google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= 799 | google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= 800 | google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= 801 | google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= 802 | google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= 803 | google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= 804 | gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= 805 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= 806 | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= 807 | gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= 808 | gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= 809 | gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= 810 | gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= 811 | gopkg.in/ini.v1 v1.66.2 h1:XfR1dOYubytKy4Shzc2LHrrGhU0lDCfDGG1yLPmpgsI= 812 | gopkg.in/ini.v1 v1.66.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= 813 | gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= 814 | gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= 815 | gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 816 | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 817 | gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 818 | gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 819 | gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 820 | gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 821 | gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 822 | gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= 823 | gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= 824 | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= 825 | gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= 826 | gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= 827 | honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= 828 | honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= 829 | honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= 830 | honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= 831 | honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= 832 | honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= 833 | honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= 834 | rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= 835 | rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= 836 | rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= 837 | --------------------------------------------------------------------------------