├── .assets
└── logo.png
├── .github
└── workflows
│ ├── ci.yaml
│ └── release.yaml
├── .gitignore
├── LICENSE
├── Makefile
├── README.md
├── cmd
└── secretize
│ └── main.go
├── examples
└── secret-generator.yaml
├── go.mod
├── go.sum
├── internal
├── k8s
│ ├── secret.go
│ └── secret_test.go
└── providers
│ ├── aws_sm.go
│ ├── aws_sm_test.go
│ ├── azure_vault.go
│ ├── azure_vault_test.go
│ ├── env.go
│ ├── env_test.go
│ ├── hashicorp_vault.go
│ ├── hashicorp_vault_test.go
│ ├── k8s_secret.go
│ ├── k8s_secret_test.go
│ └── secrets_provider.go
└── pkg
├── generator
├── generator.go
└── generator_test.go
└── utils
├── utils.go
└── utils_test.go
/.assets/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/bbl/secretize/b1fb284bd4535ee042c482d7473a3e09f7405418/.assets/logo.png
--------------------------------------------------------------------------------
/.github/workflows/ci.yaml:
--------------------------------------------------------------------------------
1 | on: push
2 | name: CI
3 | jobs:
4 | checks:
5 | name: run
6 | runs-on: ubuntu-latest
7 | steps:
8 | - uses: actions/checkout@master
9 |
10 | - name: build
11 | uses: cedrickring/golang-action@1.6.0
12 | with:
13 | args: make build
14 |
15 | - name: test
16 | uses: cedrickring/golang-action@1.6.0
17 | with:
18 | args: make test
19 |
20 | - name: publish code coverage results
21 | run: |
22 | go test -race -coverprofile=coverage.txt -covermode=atomic ./...
23 | bash <(curl -s https://codecov.io/bash)
24 | env:
25 | CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
--------------------------------------------------------------------------------
/.github/workflows/release.yaml:
--------------------------------------------------------------------------------
1 | on:
2 | release:
3 | types: [created]
4 |
5 | jobs:
6 | release-linux-amd64:
7 | name: release linux/amd64
8 | runs-on: ubuntu-latest
9 | steps:
10 | - uses: actions/checkout@v2
11 |
12 | - uses: wangyoucao577/go-release-action@master
13 | with:
14 | github_token: ${{ secrets.GITHUB_TOKEN }}
15 | goos: linux
16 | goarch: amd64
17 | project_path: ./cmd/secretize
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .idea
2 | ./secretize
3 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2020 Bogdan
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 |
--------------------------------------------------------------------------------
/Makefile:
--------------------------------------------------------------------------------
1 |
2 |
3 | build:
4 | go build ./cmd/secretize
5 |
6 | test:
7 | @go test -v -cover ./...
8 | @echo "All tests passed"
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Secretize is a kustomize plugin that helps generating kubernetes secrets from various sources.
5 | It's like a swiss army knife, but for kubernetes secrets.
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 | ---
17 |
18 | ## Sources
19 |
20 | Secretize is able to generate secrets using the following providers:
21 |
22 | - [AWS Secret Manager](https://docs.aws.amazon.com/secretsmanager/latest/userguide/intro.html)
23 | - [Azure Vault](https://docs.microsoft.com/en-us/azure/key-vault/)
24 | - [Hashicorp Vault](https://www.vaultproject.io/)
25 | - [Other K8S secrets](https://kubernetes.io/docs/concepts/configuration/secret/)
26 | - [Environment variables](https://en.wikipedia.org/wiki/Environment_variable)
27 |
28 | It is possible to use multiple providers at once.
29 |
30 |
31 | ## Installation
32 |
33 | Install secretize to your `$XDG_CONFIG_HOME/kustomize/plugin` folder:
34 |
35 | 1. Export the `XDG_CONFIG_HOME` variable if it's not already set:
36 |
37 | ```bash
38 | export XDG_CONFIG_HOME=~/.config
39 | ```
40 |
41 | 2. Download the release binary into the kustomize plugin folder:
42 |
43 | ```bash
44 | export SECRETIZE_DIR="$XDG_CONFIG_HOME/kustomize/plugin/secretize/v1/secretgenerator"
45 | mkdir -p "$SECRETIZE_DIR"
46 | curl -L https://github.com/bbl/secretize/releases/download/v0.0.1/secretize-v0.0.1-linux-amd64.tar.gz | tar -xz -C $SECRETIZE_DIR
47 | ```
48 |
49 | ## Usage
50 |
51 | All providers can generate two types of secrets: `literals` and `kv` (Key-Value secrets).
52 | Literal secrets simply generate a single string output, while KV secrets will output with a dictionary of the key-value pairs.
53 |
54 | The full configuration API could be found in the [examples/secret-generator.yaml](./examples/secret-generator.yaml) file.
55 |
56 | ### AWS Secrets Manager
57 |
58 | Fetching literal secrets is as simple, as using a default kustomize `secretGenerator` plugin:
59 |
60 | ```yaml
61 | apiVersion: secretize/v1
62 | kind: SecretGenerator
63 | metadata:
64 | name: aws-sm-secrets
65 | sources:
66 | - provider: aws-sm
67 | literals:
68 | - mySecret
69 | - newName=mySecret
70 | ```
71 |
72 | The above config would query AWS Secrets Manager provider to get the `mySecret` string value. As a result, the following manifest will be generated:
73 |
74 | ```yaml
75 | apiVersion: v1
76 | kind: Secret
77 | metadata:
78 | name: aws-sm-secrets
79 | data:
80 | mySecret: c2VjcmV0X3ZhbHVlXzE= # a sample base64 encoded data
81 | newName: c2VjcmV0X3ZhbHVlXzE=
82 | ```
83 |
84 | Now let's assume that value of `mySecret` is a json string:
85 | ```json
86 | {
87 | "secret_key_1":"secret_value_1",
88 | "secret_key_2": "secret_value_2"
89 | }
90 | ```
91 |
92 | The generator config can be slightly modified, to generate a `kv` secret:
93 |
94 | ```yaml
95 | apiVersion: secretize/v1
96 | kind: SecretGenerator
97 | metadata:
98 | name: aws-sm-secrets
99 | sources:
100 | - provider: aws-sm
101 | kv:
102 | - mySecret
103 | ```
104 |
105 | As a result, the following secret is generated:
106 |
107 | ```yaml
108 | apiVersion: v1
109 | kind: Secret
110 | metadata:
111 | name: aws-sm-secrets
112 | data:
113 | secret_key_1: c2VjcmV0X3ZhbHVlXzE=
114 | secret_key_2: c2VjcmV0X3ZhbHVlXzI=
115 | ```
116 |
117 | ### Azure Vault
118 |
119 | Azure Vault configuration is pretty similar to the above examples. However, there's additional `params` field, which is used to specify the Vault Name:
120 |
121 |
122 | ```yaml
123 | apiVersion: secretize/v1
124 | kind: SecretGenerator
125 | metadata:
126 | name: aws-sm-secrets
127 | sources:
128 | - provider: azure-vault
129 | params:
130 | name: vault-name
131 | kv:
132 | - kv-secrets # will treat this as JSON, the same way as in the AWS example
133 | literals:
134 | - literal-secret-1
135 | - new_name=literal-secret-1
136 | ```
137 |
138 |
139 | ### Hashicorp Vault
140 |
141 | Some providers only support key-value output, e.g. Hashicorp Vault and K8S Secret.
142 | For instance, the `mySecret` in Hashicorp Vault might look like the following:
143 | ```bash
144 | vault kv get secret/mySecret
145 | ====== Data ======
146 | Key Value
147 | --- -----
148 | secret_key_1 secret_value_1
149 | secret_key_2 secret_value_2
150 | ```
151 |
152 | Querying provider's `kv` secrets will generate the corresponding key-value data:
153 |
154 | ```yaml
155 | apiVersion: secretize/v1
156 | kind: SecretGenerator
157 | metadata:
158 | name: hashicorp-vault-secrets
159 | sources:
160 | - provider: hashicorp-vault
161 | kv:
162 | - secret/data/mySecret # you need to specify the full path in hashicorp vault provider
163 | ```
164 | ```yaml
165 | apiVersion: v1
166 | kind: Secret
167 | metadata:
168 | name: hashicorp-vault-secrets
169 | data:
170 | secret_key_1: c2VjcmV0X3ZhbHVlXzE=
171 | secret_key_2: c2VjcmV0X3ZhbHVlXzI=
172 | ```
173 |
174 | However you're able to query a certain literal in the key-value output using the following syntax: `secret-name:key`, e.g.:
175 |
176 | ```yaml
177 | apiVersion: secretize/v1
178 | kind: SecretGenerator
179 | metadata:
180 | name: hashicorp-vault-secrets
181 | sources:
182 | - provider: hashicorp-vault
183 | literals:
184 | - secret/data/mySecret-1:secret_key_1
185 | ```
186 |
187 | As a result, the following manifest will be generated:
188 |
189 | ```yaml
190 | apiVersion: v1
191 | kind: Secret
192 | metadata:
193 | name: hashicorp-vault-secrets
194 | data:
195 | secret_key_1: c2VjcmV0X3ZhbHVlXzE=
196 | ```
197 |
198 | ## Kubernetes Secret
199 |
200 | Kubernetes secret provider is similar to the Hashicorp Vault. Additionally, this provider expects the `params` field with the `namespace` specification.
201 | You're able to get the entire secret data using the `kv` query, or get a particular key using the `literals` query with the `:` delimiter syntax:
202 |
203 | ```yaml
204 | # The original secret in a default namespace
205 | #
206 | apiVersion: v1
207 | kind: Secret
208 | metadata:
209 | name: original-secret
210 | namespace: default
211 | data:
212 | secret_key_1: c2VjcmV0X3ZhbHVlXzE=
213 | secret_key_2: c2VjcmV0X3ZhbHVlXzI=
214 | ---
215 | # Secret generator configuration
216 | #
217 | apiVersion: secretize/v1
218 | kind: SecretGenerator
219 | metadata:
220 | name: kubernetes-secrets
221 | sources:
222 | - provider: k8s-secret
223 | params:
224 | namespace: default
225 | kv:
226 | - original-secret
227 | literals:
228 | - new_name=original-secret:secret_key_1
229 | ---
230 | # Generated secret
231 | #
232 | apiVersion: v1
233 | kind: Secret
234 | metadata:
235 | name: kubernetes-secrets
236 | data:
237 | secret_key_1: c2VjcmV0X3ZhbHVlXzE=
238 | secret_key_2: c2VjcmV0X3ZhbHVlXzI=
239 | new_name: c2VjcmV0X3ZhbHVlXzE=
240 |
241 | ```
242 |
243 |
244 | ## Env
245 |
246 | The environment variables plugin is similar to the AWS and Azure plugins. The `literals` would simply fetch corresponding environment variables, while `kv` would treat each variable as JSON and try to parse it:
247 |
248 | ```yaml
249 | apiVersion: secretize/v1
250 | kind: SecretGenerator
251 | metadata:
252 | name: env-secrets
253 | sources:
254 | - provider: env
255 | kv:
256 | - MY_KV_SECRET
257 | literals:
258 | - MY_LITERAL_SECRET
259 | ```
260 |
261 | Secretize will fetch the corresponding environment variables during the `kustomize build` command:
262 |
263 | ```bash
264 | export MY_KV_SECRET='{"secret_key_1":"secret_value_1", "secret_key_2": "secret_value_2"}'
265 | export MY_LITERAL_SECRET=super_secret
266 |
267 | kustomize build
268 | ```
269 |
270 | The following secret is generated:
271 |
272 | ```yaml
273 | apiVersion: v1
274 | kind: Secret
275 | metadata:
276 | name: env-kv-secrets
277 | data:
278 | MY_LITERAL_SECRET: c3VwZXJfc2VjcmV0
279 | secret_key_1: c2VjcmV0X3ZhbHVlXzE=
280 | secret_key_2: c2VjcmV0X3ZhbHVlXzI=
281 | ```
282 |
--------------------------------------------------------------------------------
/cmd/secretize/main.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "fmt"
5 | "github.com/bbl/secretize/pkg/generator"
6 | "github.com/bbl/secretize/pkg/utils"
7 | log "github.com/sirupsen/logrus"
8 | "io/ioutil"
9 |
10 | "os"
11 | "path/filepath"
12 | )
13 |
14 | func main() {
15 |
16 | if len(os.Args) < 2 {
17 | log.Fatal(
18 | "No argument passed, use `secretize /path/to/generator-config.yaml`")
19 | }
20 |
21 | filename, _ := filepath.Abs(os.Args[1])
22 | yamlFile, err := ioutil.ReadFile(filename)
23 | utils.FatalErrCheck(err)
24 |
25 | secretGenerator, err := generator.ParseConfig(yamlFile)
26 | utils.FatalErrCheck(err)
27 |
28 | secrets, err := secretGenerator.FetchSecrets(generator.ProviderRegistry)
29 | utils.FatalErrCheck(err)
30 |
31 | s := secretGenerator.Generate(secrets)
32 | out, err := s.ToYamlStr()
33 | utils.FatalErrCheck(err)
34 | fmt.Println(out)
35 | }
36 |
--------------------------------------------------------------------------------
/examples/secret-generator.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: secretize/v1
2 | kind: SecretGenerator
3 | metadata:
4 | name: my-app-secrets # this will be the generated secret name
5 | labels:
6 | app: my-app # labels are added to the generated secret
7 | sources:
8 |
9 | - provider: aws-sm
10 | literals:
11 | # Fetches a single secret as string
12 | - mySecret
13 | # Set the generated key name explicitly
14 | - explicitName=anotherSecret
15 | kv:
16 | # Tries to parse a secret as JSON. Example:
17 | # aws secretsmanager create-secret --name myJsonSecret \
18 | # --secret-string '{"secret_key_1":"secret_value_1", "secret_key_2": "secret_value_2"}'
19 | - myJsonSecret
20 |
21 | - provider: env
22 | literals:
23 | # Fetches a single variable as a string
24 | - MY_LITERAL_SECRET
25 | # Set the generated key name explicitly
26 | - newName=MY_LITERAL_SECRET
27 | kv:
28 | # Tries to parse a variable as JSON. Example:
29 | # export MY_KV_SECRET='{"secret_key_1":"secret_value_1", "secret_key_2": "secret_value_2"}'
30 | - MY_KV_SECRET
31 |
32 | - provider: azure-vault
33 | params:
34 | name: my-vault # Azure Vault name
35 | literals:
36 | # Fetches a single secret as string
37 | - mySecret
38 | # Set the generated key name explicitly
39 | - explicitName=anotherSecret
40 | kv:
41 | # Tries to parse a secret as JSON. Example:
42 | # az keyvault secret set --vault-name my-vault --name myJsonSecret \
43 | # --value '{"secret_key_1":"secret_value_1", "secret_key_2": "secret_value_2"}'
44 | - myJsonSecret
45 |
46 | - provider: hashicorp-vault
47 | literals:
48 | # Fetches a single secret as string.
49 | # Note the `:` separator - this is crucial to query the particular key in the Hashicorp Vault secret data.
50 | - secret/data/mySecret:secret_key_1
51 | # Set the generated key name explicitly
52 | - explicitName=secret/data/mySecret:secret_key_1
53 | kv:
54 | # Fetch all secret data. Example:
55 | # vault kv put secret/mySecret secret_key_1=secret_value_1 secret_key_2=secret_value_2
56 | - secret/data/mySecret
57 |
58 | - provider: k8s-secret
59 | params:
60 | namespace: default # Kubernetes namespace name of the original secret
61 | literals:
62 | # Fetches a single secret as string.
63 | # Note the `:` separator - this is crucial to query the particular key in the K8S secret data.
64 | - original-secret:secret_key_1
65 | # Set the generated key name explicitly
66 | - explicitName=original-secret:secret_key_1
67 | kv:
68 | # Fetch all secret data. Example:
69 | # kubectl create secret generic original-secret \
70 | # --from-literal=secret_key_1=secret_value_1 --from-literal=secret_key_2=secret_value_2
71 | - original-secret
72 |
73 |
74 |
75 |
76 |
--------------------------------------------------------------------------------
/go.mod:
--------------------------------------------------------------------------------
1 | module github.com/bbl/secretize
2 |
3 | go 1.13
4 |
5 | require (
6 | github.com/Azure/azure-sdk-for-go v46.4.0+incompatible
7 | github.com/Azure/go-autorest/autorest v0.11.6
8 | github.com/Azure/go-autorest/autorest/azure/auth v0.5.2
9 | github.com/Azure/go-autorest/autorest/to v0.4.0 // indirect
10 | github.com/Azure/go-autorest/autorest/validation v0.3.0 // indirect
11 | github.com/aws/aws-sdk-go v1.35.1
12 | github.com/hashicorp/vault/api v1.0.4
13 | github.com/kr/pretty v0.2.0 // indirect
14 | github.com/magiconair/properties v1.8.0
15 | github.com/sirupsen/logrus v1.6.0
16 | github.com/stretchr/testify v1.4.0
17 | golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4 // indirect
18 | gopkg.in/yaml.v2 v2.3.0
19 | k8s.io/api v0.17.0
20 | k8s.io/apimachinery v0.17.0
21 | k8s.io/client-go v0.17.0
22 | sigs.k8s.io/kustomize/api v0.6.2
23 | )
24 |
--------------------------------------------------------------------------------
/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 h1:ROfEUZz+Gh5pa62DJWXSaonyu3StP6EA6lPEXPI6mCo=
4 | cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
5 | github.com/360EntSecGroup-Skylar/excelize v1.4.1/go.mod h1:vnax29X2usfl7HHkBrX5EvSCJcmH3dT9luvxzu8iGAE=
6 | github.com/Azure/azure-sdk-for-go v46.4.0+incompatible h1:fCN6Pi+tEiEwFa8RSmtVlFHRXEZ+DJm9gfx/MKqYWw4=
7 | github.com/Azure/azure-sdk-for-go v46.4.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc=
8 | github.com/Azure/go-autorest v14.2.0+incompatible h1:V5VMDjClD3GiElqLWO7mz2MxNAK/vTfRHdAubSIPRgs=
9 | github.com/Azure/go-autorest v14.2.0+incompatible/go.mod h1:r+4oMnoxhatjLLJ6zxSWATqVooLgysK6ZNox3g/xq24=
10 | github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI=
11 | github.com/Azure/go-autorest/autorest v0.11.6 h1:LIzfhNo9I3+il0KO2JY1/lgJmjig7lY0wFulQNZkbtg=
12 | github.com/Azure/go-autorest/autorest v0.11.6/go.mod h1:V6p3pKZx1KKkJubbxnDWrzNhEIfOy/pTGasLqzHIPHs=
13 | github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0=
14 | github.com/Azure/go-autorest/autorest/adal v0.9.4 h1:1/DtH4Szusk4psLBrJn/gocMRIf1ji30WAz3GfyULRQ=
15 | github.com/Azure/go-autorest/autorest/adal v0.9.4/go.mod h1:/3SMAM86bP6wC9Ev35peQDUeqFZBMH07vvUOmg4z/fE=
16 | github.com/Azure/go-autorest/autorest/azure/auth v0.5.2 h1:R1pgoZkhXuv4+0ky9r3e5pcnRXWcXGIuPXpC/xkc7uI=
17 | github.com/Azure/go-autorest/autorest/azure/auth v0.5.2/go.mod h1:q98IH4qgc3eWM4/WOeR5+YPmBuy8Lq0jNRDwSM0CuFk=
18 | github.com/Azure/go-autorest/autorest/azure/cli v0.4.1 h1:jwcD1wURu0+hKceV04MubZmKLzwEYOCz6q4aOtVZ+Ng=
19 | github.com/Azure/go-autorest/autorest/azure/cli v0.4.1/go.mod h1:JfDgiIO1/RPu6z42AdQTyjOoCM2MFhLqSBDvMEkDgcg=
20 | github.com/Azure/go-autorest/autorest/date v0.1.0/go.mod h1:plvfp3oPSKwf2DNjlBjWF/7vwR+cUD/ELuzDCXwHUVA=
21 | github.com/Azure/go-autorest/autorest/date v0.3.0 h1:7gUk1U5M/CQbp9WoqinNzJar+8KY+LPI6wiWrP/myHw=
22 | github.com/Azure/go-autorest/autorest/date v0.3.0/go.mod h1:BI0uouVdmngYNUzGWeSYnokU+TrmwEsOqdt8Y6sso74=
23 | github.com/Azure/go-autorest/autorest/mocks v0.1.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0=
24 | github.com/Azure/go-autorest/autorest/mocks v0.2.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0=
25 | github.com/Azure/go-autorest/autorest/mocks v0.4.1 h1:K0laFcLE6VLTOwNgSxaGbUcLPuGXlNkbVvq4cW4nIHk=
26 | github.com/Azure/go-autorest/autorest/mocks v0.4.1/go.mod h1:LTp+uSrOhSkaKrUy935gNZuuIPPVsHlr9DSOxSayd+k=
27 | github.com/Azure/go-autorest/autorest/to v0.4.0 h1:oXVqrxakqqV1UZdSazDOPOLvOIz+XA683u8EctwboHk=
28 | github.com/Azure/go-autorest/autorest/to v0.4.0/go.mod h1:fE8iZBn7LQR7zH/9XU2NcPR4o9jEImooCeWJcYV/zLE=
29 | github.com/Azure/go-autorest/autorest/validation v0.3.0 h1:3I9AAI63HfcLtphd9g39ruUwRI+Ca+z/f36KHPFRUss=
30 | github.com/Azure/go-autorest/autorest/validation v0.3.0/go.mod h1:yhLgjC0Wda5DYXl6JAsWyUe4KVNffhoDhG0zVzUMo3E=
31 | github.com/Azure/go-autorest/logger v0.1.0/go.mod h1:oExouG+K6PryycPJfVSxi/koC6LSNgds39diKLz7Vrc=
32 | github.com/Azure/go-autorest/logger v0.2.0 h1:e4RVHVZKC5p6UANLJHkM4OfR1UKZPj8Wt8Pcx+3oqrE=
33 | github.com/Azure/go-autorest/logger v0.2.0/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8=
34 | github.com/Azure/go-autorest/tracing v0.5.0/go.mod h1:r/s2XiOKccPW3HrqB+W0TQzfbtp2fGCgRFtBroKn4Dk=
35 | github.com/Azure/go-autorest/tracing v0.6.0 h1:TYi4+3m5t6K48TGI9AUdb+IzbnSxvnvUMfuitfgcfuo=
36 | github.com/Azure/go-autorest/tracing v0.6.0/go.mod h1:+vhtPC754Xsa23ID7GlGsrdKBpUA79WCAKPPZVC2DeU=
37 | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
38 | github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ=
39 | github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
40 | github.com/OpenPeeDeeP/depguard v1.0.1/go.mod h1:xsIw86fROiiwelg+jB2uM9PiKihMMmUx/1V+TNhjQvM=
41 | github.com/PuerkitoBio/goquery v1.5.0/go.mod h1:qD2PgZ9lccMbQlc7eEOjaeRlFQON7xY8kdmcsrnKqMg=
42 | github.com/PuerkitoBio/purell v1.0.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
43 | github.com/PuerkitoBio/purell v1.1.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
44 | github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
45 | github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
46 | github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
47 | github.com/StackExchange/wmi v0.0.0-20180116203802-5d049714c4a6/go.mod h1:3eOhrUMpNV+6aFIbp5/iudMxNCF27Vw2OZgy4xEx0Fg=
48 | github.com/agnivade/levenshtein v1.0.1/go.mod h1:CURSv5d9Uaml+FovSIICkLbAUZ9S4RqaHDIsdSBg7lM=
49 | github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
50 | github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
51 | github.com/andreyvit/diff v0.0.0-20170406064948-c7f18ee00883/go.mod h1:rCTlJbsFo29Kk6CurOXKm700vrz8f0KW0JNfpkRJY/8=
52 | github.com/andybalholm/cascadia v1.0.0/go.mod h1:GsXiBklL0woXo1j/WYWtSYYC4ouU9PqHO0sqidkEA4Y=
53 | github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8=
54 | github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY=
55 | github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8=
56 | github.com/asaskevich/govalidator v0.0.0-20180720115003-f9ffefc3facf/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY=
57 | github.com/asaskevich/govalidator v0.0.0-20190424111038-f61b66f89f4a/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY=
58 | github.com/aws/aws-sdk-go v1.35.1 h1:dGBUiVpdG6Zho3taAqGJKxuhn+qIrP3OdjfrtqowDyc=
59 | github.com/aws/aws-sdk-go v1.35.1/go.mod h1:H7NKnBqNVzoTJpGfLrQkkD+ytBA93eiDYi/+8rV9s48=
60 | github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
61 | github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
62 | github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d/go.mod h1:6QX/PXZ00z/TKoufEY6K/a0k6AhaJrQKdFe6OfVXsa4=
63 | github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs=
64 | github.com/bombsimon/wsl v1.2.5/go.mod h1:43lEF/i0kpXbLCeDXL9LMT8c92HyBywXb0AsgMHYngM=
65 | github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
66 | github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI=
67 | github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI=
68 | github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU=
69 | github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
70 | github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk=
71 | github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
72 | github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk=
73 | github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
74 | github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
75 | github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
76 | github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE=
77 | github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
78 | github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY=
79 | github.com/davecgh/go-spew v0.0.0-20151105211317-5215b55f46b2/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
80 | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
81 | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
82 | github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
83 | github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no=
84 | github.com/dimchansky/utfbom v1.1.0 h1:FcM3g+nofKgUteL8dm/UpdRXNC9KmADgTpLKsu0TRo4=
85 | github.com/dimchansky/utfbom v1.1.0/go.mod h1:rO41eb7gLfo8SF1jd9F8HplJm1Fewwi4mQvIirEdv+8=
86 | github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
87 | github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
88 | github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM=
89 | github.com/dustmop/soup v1.1.2-0.20190516214245-38228baa104e/go.mod h1:CgNC6SGbT+Xb8wGGvzilttZL1mc5sQ/5KkcxsZttMIk=
90 | github.com/elazarl/goproxy v0.0.0-20170405201442-c4fc26588b6e/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
91 | github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
92 | github.com/evanphx/json-patch v4.2.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
93 | github.com/evanphx/json-patch v4.5.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
94 | github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
95 | github.com/fatih/structs v1.1.0/go.mod h1:9NiDSp5zOcgEDl+j00MP/WkGVPOlPRLejGD8Ga6PJ7M=
96 | github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
97 | github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
98 | github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
99 | github.com/globalsign/mgo v0.0.0-20180905125535-1ca0a4f7cbcb/go.mod h1:xkRDCp4j0OGD1HRkm4kmhM+pmpv3AKq5SU7GMg4oO/Q=
100 | github.com/globalsign/mgo v0.0.0-20181015135952-eeefdecb41b8/go.mod h1:xkRDCp4j0OGD1HRkm4kmhM+pmpv3AKq5SU7GMg4oO/Q=
101 | github.com/go-critic/go-critic v0.3.5-0.20190904082202-d79a9f0c64db/go.mod h1:+sE8vrLDS2M0pZkBk0wy6+nLdKexVDrl/jBqQOTDThA=
102 | github.com/go-errors/errors v1.0.1/go.mod h1:f4zRHt4oKfwPJE5k8C9vpYG+aDHdBFUsgrm6/TyX73Q=
103 | github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
104 | github.com/go-ldap/ldap v3.0.2+incompatible/go.mod h1:qfd9rJvER9Q0/D/Sqn1DfHRoBp40uXYvFoEVrNEPqRc=
105 | github.com/go-lintpack/lintpack v0.5.2/go.mod h1:NwZuYi2nUHho8XEIZ6SIxihrnPoqBTDqfpXvXAN0sXM=
106 | github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
107 | github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
108 | github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas=
109 | github.com/go-ole/go-ole v1.2.1/go.mod h1:7FAglXiTm7HKlQRDeOQ6ZNUHidzCWXuZWq/1dTyBNF8=
110 | github.com/go-openapi/analysis v0.0.0-20180825180245-b006789cd277/go.mod h1:k70tL6pCuVxPJOHXQ+wIac1FUrvNkHolPie/cLEU6hI=
111 | github.com/go-openapi/analysis v0.17.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
112 | github.com/go-openapi/analysis v0.18.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
113 | github.com/go-openapi/analysis v0.19.2/go.mod h1:3P1osvZa9jKjb8ed2TPng3f0i/UY9snX6gxi44djMjk=
114 | github.com/go-openapi/analysis v0.19.5/go.mod h1:hkEAkxagaIvIP7VTn8ygJNkd4kAYON2rCu0v0ObL0AU=
115 | github.com/go-openapi/errors v0.17.0/go.mod h1:LcZQpmvG4wyF5j4IhA73wkLFQg+QJXOQHVjmcZxhka0=
116 | github.com/go-openapi/errors v0.18.0/go.mod h1:LcZQpmvG4wyF5j4IhA73wkLFQg+QJXOQHVjmcZxhka0=
117 | github.com/go-openapi/errors v0.19.2/go.mod h1:qX0BLWsyaKfvhluLejVpVNwNRdXZhEbTA4kxxpKBC94=
118 | github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0=
119 | github.com/go-openapi/jsonpointer v0.17.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M=
120 | github.com/go-openapi/jsonpointer v0.18.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M=
121 | github.com/go-openapi/jsonpointer v0.19.2/go.mod h1:3akKfEdA7DF1sugOqz1dVQHBcuDBPKZGEoHC/NkiQRg=
122 | github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg=
123 | github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg=
124 | github.com/go-openapi/jsonreference v0.17.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I=
125 | github.com/go-openapi/jsonreference v0.18.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I=
126 | github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwohSTlpa0o73RUL1owJc=
127 | github.com/go-openapi/jsonreference v0.19.3/go.mod h1:rjx6GuL8TTa9VaixXglHmQmIL98+wF9xc8zWvFonSJ8=
128 | github.com/go-openapi/loads v0.17.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU=
129 | github.com/go-openapi/loads v0.18.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU=
130 | github.com/go-openapi/loads v0.19.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU=
131 | github.com/go-openapi/loads v0.19.2/go.mod h1:QAskZPMX5V0C2gvfkGZzJlINuP7Hx/4+ix5jWFxsNPs=
132 | github.com/go-openapi/loads v0.19.4/go.mod h1:zZVHonKd8DXyxyw4yfnVjPzBjIQcLt0CCsn0N0ZrQsk=
133 | github.com/go-openapi/runtime v0.0.0-20180920151709-4f900dc2ade9/go.mod h1:6v9a6LTXWQCdL8k1AO3cvqx5OtZY/Y9wKTgaoP6YRfA=
134 | github.com/go-openapi/runtime v0.19.0/go.mod h1:OwNfisksmmaZse4+gpV3Ne9AyMOlP1lt4sK4FXt0O64=
135 | github.com/go-openapi/runtime v0.19.4/go.mod h1:X277bwSUBxVlCYR3r7xgZZGKVvBd/29gLDlFGtJ8NL4=
136 | github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc=
137 | github.com/go-openapi/spec v0.17.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI=
138 | github.com/go-openapi/spec v0.18.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI=
139 | github.com/go-openapi/spec v0.19.2/go.mod h1:sCxk3jxKgioEJikev4fgkNmwS+3kuYdJtcsZsD5zxMY=
140 | github.com/go-openapi/spec v0.19.3/go.mod h1:FpwSN1ksY1eteniUU7X0N/BgJ7a4WvBFVA8Lj9mJglo=
141 | github.com/go-openapi/spec v0.19.5/go.mod h1:Hm2Jr4jv8G1ciIAo+frC/Ft+rR2kQDh8JHKHb3gWUSk=
142 | github.com/go-openapi/strfmt v0.17.0/go.mod h1:P82hnJI0CXkErkXi8IKjPbNBM6lV6+5pLP5l494TcyU=
143 | github.com/go-openapi/strfmt v0.18.0/go.mod h1:P82hnJI0CXkErkXi8IKjPbNBM6lV6+5pLP5l494TcyU=
144 | github.com/go-openapi/strfmt v0.19.0/go.mod h1:+uW+93UVvGGq2qGaZxdDeJqSAqBqBdl+ZPMF/cC8nDY=
145 | github.com/go-openapi/strfmt v0.19.3/go.mod h1:0yX7dbo8mKIvc3XSKp7MNfxw4JytCfCD6+bY1AVL9LU=
146 | github.com/go-openapi/strfmt v0.19.5/go.mod h1:eftuHTlB/dI8Uq8JJOyRlieZf+WkkxUuk0dgdHXr2Qk=
147 | github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I=
148 | github.com/go-openapi/swag v0.17.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg=
149 | github.com/go-openapi/swag v0.18.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg=
150 | github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
151 | github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
152 | github.com/go-openapi/validate v0.18.0/go.mod h1:Uh4HdOzKt19xGIGm1qHf/ofbX1YQ4Y+MYsct2VUrAJ4=
153 | github.com/go-openapi/validate v0.19.2/go.mod h1:1tRCw7m3jtI8eNWEEliiAqUIcBztB2KDnRCRMUi7GTA=
154 | github.com/go-openapi/validate v0.19.8/go.mod h1:8DJv2CVJQ6kGNpFW6eV9N3JviE1C85nY1c2z52x1Gk4=
155 | github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg=
156 | github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
157 | github.com/go-test/deep v1.0.2-0.20181118220953-042da051cf31/go.mod h1:wGDj63lr65AM2AQyKZd/NYHGb0R+1RLqB8NKt3aSFNA=
158 | github.com/go-toolsmith/astcast v1.0.0/go.mod h1:mt2OdQTeAQcY4DQgPSArJjHCcOwlX+Wl/kwN+LbLGQ4=
159 | github.com/go-toolsmith/astcopy v1.0.0/go.mod h1:vrgyG+5Bxrnz4MZWPF+pI4R8h3qKRjjyvV/DSez4WVQ=
160 | github.com/go-toolsmith/astequal v0.0.0-20180903214952-dcb477bfacd6/go.mod h1:H+xSiq0+LtiDC11+h1G32h7Of5O3CYFJ99GVbS5lDKY=
161 | github.com/go-toolsmith/astequal v1.0.0/go.mod h1:H+xSiq0+LtiDC11+h1G32h7Of5O3CYFJ99GVbS5lDKY=
162 | github.com/go-toolsmith/astfmt v0.0.0-20180903215011-8f8ee99c3086/go.mod h1:mP93XdblcopXwlyN4X4uodxXQhldPGZbcEJIimQHrkg=
163 | github.com/go-toolsmith/astfmt v1.0.0/go.mod h1:cnWmsOAuq4jJY6Ct5YWlVLmcmLMn1JUPuQIHCY7CJDw=
164 | github.com/go-toolsmith/astinfo v0.0.0-20180906194353-9809ff7efb21/go.mod h1:dDStQCHtmZpYOmjRP/8gHHnCCch3Zz3oEgCdZVdtweU=
165 | github.com/go-toolsmith/astp v0.0.0-20180903215135-0af7e3c24f30/go.mod h1:SV2ur98SGypH1UjcPpCatrV5hPazG6+IfNHbkDXBRrk=
166 | github.com/go-toolsmith/astp v1.0.0/go.mod h1:RSyrtpVlfTFGDYRbrjyWP1pYu//tSFcvdYrA8meBmLI=
167 | github.com/go-toolsmith/pkgload v0.0.0-20181119091011-e9e65178eee8/go.mod h1:WoMrjiy4zvdS+Bg6z9jZH82QXwkcgCBX6nOfnmdaHks=
168 | github.com/go-toolsmith/pkgload v1.0.0/go.mod h1:5eFArkbO80v7Z0kdngIxsRXRMTaX4Ilcwuh3clNrQJc=
169 | github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
170 | github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
171 | github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
172 | github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b/go.mod h1:F1TvTiK9OcQqauNUHlbJvyl9Qa1QvF/gOUDKA14jxHU=
173 | github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
174 | github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4=
175 | github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d h1:3PaI8p3seN09VjbTYC/QWlUZdZ1qS1zGjy7LH2Wt07I=
176 | github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
177 | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
178 | github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
179 | github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
180 | github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
181 | github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
182 | github.com/golang/protobuf v0.0.0-20161109072736-4bd1920723d7/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
183 | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
184 | github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
185 | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
186 | github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
187 | github.com/golangci/check v0.0.0-20180506172741-cfe4005ccda2/go.mod h1:k9Qvh+8juN+UKMCS/3jFtGICgW8O96FVaZsaxdzDkR4=
188 | github.com/golangci/dupl v0.0.0-20180902072040-3e9179ac440a/go.mod h1:ryS0uhF+x9jgbj/N71xsEqODy9BN81/GonCZiOzirOk=
189 | github.com/golangci/errcheck v0.0.0-20181223084120-ef45e06d44b6/go.mod h1:DbHgvLiFKX1Sh2T1w8Q/h4NAI8MHIpzCdnBUDTXU3I0=
190 | github.com/golangci/go-misc v0.0.0-20180628070357-927a3d87b613/go.mod h1:SyvUF2NxV+sN8upjjeVYr5W7tyxaT1JVtvhKhOn2ii8=
191 | github.com/golangci/goconst v0.0.0-20180610141641-041c5f2b40f3/go.mod h1:JXrF4TWy4tXYn62/9x8Wm/K/dm06p8tCKwFRDPZG/1o=
192 | github.com/golangci/gocyclo v0.0.0-20180528134321-2becd97e67ee/go.mod h1:ozx7R9SIwqmqf5pRP90DhR2Oay2UIjGuKheCBCNwAYU=
193 | github.com/golangci/gofmt v0.0.0-20190930125516-244bba706f1a/go.mod h1:9qCChq59u/eW8im404Q2WWTrnBUQKjpNYKMbU4M7EFU=
194 | github.com/golangci/golangci-lint v1.21.0/go.mod h1:phxpHK52q7SE+5KpPnti4oZTdFCEsn/tKN+nFvCKXfk=
195 | github.com/golangci/ineffassign v0.0.0-20190609212857-42439a7714cc/go.mod h1:e5tpTHCfVze+7EpLEozzMB3eafxo2KT5veNg1k6byQU=
196 | github.com/golangci/lint-1 v0.0.0-20191013205115-297bf364a8e0/go.mod h1:66R6K6P6VWk9I95jvqGxkqJxVWGFy9XlDwLwVz1RCFg=
197 | github.com/golangci/maligned v0.0.0-20180506175553-b1d89398deca/go.mod h1:tvlJhZqDe4LMs4ZHD0oMUlt9G2LWuDGoisJTBzLMV9o=
198 | github.com/golangci/misspell v0.0.0-20180809174111-950f5d19e770/go.mod h1:dEbvlSfYbMQDtrpRMQU675gSDLDNa8sCPPChZ7PhiVA=
199 | github.com/golangci/prealloc v0.0.0-20180630174525-215b22d4de21/go.mod h1:tf5+bzsHdTM0bsB7+8mt0GUMvjCgwLpTapNZHU8AajI=
200 | github.com/golangci/revgrep v0.0.0-20180526074752-d9c87f5ffaf0/go.mod h1:qOQCunEYvmd/TLamH+7LlVccLvUH5kZNhbCgTHoBbp4=
201 | github.com/golangci/unconvert v0.0.0-20180507085042-28b1c447d1f4/go.mod h1:Izgrg8RkN3rCIMLGE9CyYmU9pY2Jer6DgANEnZ/L/cQ=
202 | github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
203 | github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
204 | github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
205 | github.com/google/go-cmp v0.3.0 h1:crn/baboCvb5fXaQ0IJ1SGTsTVrWpDsCWC8EGETZijY=
206 | github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
207 | github.com/google/gofuzz v0.0.0-20161122191042-44d81051d367/go.mod h1:HP5RmnzzSNb993RKQDq4+1A4ia9nllfqcQFTQJedwGI=
208 | github.com/google/gofuzz v1.0.0 h1:A8PeW59pxE9IoFRqBp37U+mSNaQoZ46F1f0f863XSXw=
209 | github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
210 | github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
211 | github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
212 | github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
213 | github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ=
214 | github.com/google/uuid v1.0.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
215 | github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
216 | github.com/googleapis/gax-go/v2 v2.0.4 h1:hU4mGcQI4DaAYW+IbTun+2qEZVFxK0ySjQLTbS0VQKc=
217 | github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
218 | github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d h1:7XGaL1e6bYS1yIonGp9761ExpPPV1ui0SAC59Yube9k=
219 | github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY=
220 | github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8=
221 | github.com/gordonklaus/ineffassign v0.0.0-20200809085317-e36bfde3bb78 h1:U/zHjaVG/sECz5xhnh7kPH+Fv/maPbhZPcaTquo5sPg=
222 | github.com/gordonklaus/ineffassign v0.0.0-20200809085317-e36bfde3bb78/go.mod h1:cuNKsD1zp2v6XfE/orVX2QE1LC+i254ceGcVeDT3pTU=
223 | github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
224 | github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3/go.mod h1:eEOZF4jCKGi+aprrirO9e7WKB3beBRtWgqGunKl6pKE=
225 | github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA=
226 | github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
227 | github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
228 | github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
229 | github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
230 | github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80=
231 | github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80=
232 | github.com/hashicorp/go-hclog v0.0.0-20180709165350-ff2cf002a8dd/go.mod h1:9bjs9uLqI8l75knNv3lV1kA55veR+WUPSiKIWcQHudI=
233 | github.com/hashicorp/go-hclog v0.8.0/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ=
234 | github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60=
235 | github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk=
236 | github.com/hashicorp/go-plugin v1.0.1/go.mod h1:++UyYGoz3o5w9ZzAdZxtQKrWWP+iqPBn3cQptSMzBuY=
237 | github.com/hashicorp/go-retryablehttp v0.5.4/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs=
238 | github.com/hashicorp/go-rootcerts v1.0.1/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8=
239 | github.com/hashicorp/go-safetemp v1.0.0/go.mod h1:oaerMy3BhqiTbVye6QuFhFtIceqFoDHxNAB65b+Rj1I=
240 | github.com/hashicorp/go-sockaddr v1.0.2/go.mod h1:rB4wwRAUzs07qva3c5SdrY/NEtAUjGlgmH/UkBUC97A=
241 | github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
242 | github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
243 | github.com/hashicorp/go-version v1.1.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
244 | github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
245 | github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
246 | github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
247 | github.com/hashicorp/vault/api v1.0.4/go.mod h1:gDcqh3WGcR1cpF5AJz/B1UFheUEneMoIospckxBxk6Q=
248 | github.com/hashicorp/vault/sdk v0.1.13/go.mod h1:B+hVj7TpuQY1Y/GPbCpffmgd+tSEwvhkWnjtSYCaS2M=
249 | github.com/hashicorp/yamux v0.0.0-20180604194846-3520598351bb/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM=
250 | github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM=
251 | github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
252 | github.com/imdario/mergo v0.3.5 h1:JboBksRwiiAJWvIYJVo46AfV+IAIKZpfrSzVKj42R4Q=
253 | github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
254 | github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
255 | github.com/jmespath/go-jmespath v0.4.0 h1:BEgLn5cpjn8UN1mAw4NjwDrS35OdebyEtFe+9YPoQUg=
256 | github.com/jmespath/go-jmespath v0.4.0/go.mod h1:T8mJZnbsbmF+m6zOOFylbeCJqk5+pHWvzYPziyZiYoo=
257 | github.com/jmespath/go-jmespath/internal/testify v1.5.1 h1:shLQSRRSCCPj3f2gpwzGwWFoC7ycTf1rcQZHOlsJ6N8=
258 | github.com/jmespath/go-jmespath/internal/testify v1.5.1/go.mod h1:L3OGu8Wl2/fWfCI6z80xFu9LTZmf1ZRjMHUOPmWr69U=
259 | github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo=
260 | github.com/json-iterator/go v0.0.0-20180612202835-f2b4162afba3/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
261 | github.com/json-iterator/go v1.1.8 h1:QiWkFLKq0T7mpzwOTu6BzNDbfTE8OLrYhVKYMLF46Ok=
262 | github.com/json-iterator/go v1.1.8/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
263 | github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
264 | github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
265 | github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q=
266 | github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00=
267 | github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
268 | github.com/klauspost/compress v1.4.0/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A=
269 | github.com/klauspost/compress v1.4.1/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A=
270 | github.com/klauspost/cpuid v0.0.0-20180405133222-e7e905edc00e/go.mod h1:Pj4uuM528wm8OyEC2QMXAi2YiTZ96dNQPGgoMS4s3ek=
271 | github.com/klauspost/cpuid v1.2.0/go.mod h1:Pj4uuM528wm8OyEC2QMXAi2YiTZ96dNQPGgoMS4s3ek=
272 | github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
273 | github.com/konsorten/go-windows-terminal-sequences v1.0.3 h1:CE8S1cTafDpPvMhIxNJKvHsGVBgn1xWYf1NbHQhywc8=
274 | github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
275 | github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc=
276 | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
277 | github.com/kr/pretty v0.2.0 h1:s5hAObm+yFO5uHYt5dYjxi2rXrsnmRpJx4OYvIWUaQs=
278 | github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
279 | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
280 | github.com/kr/pty v1.1.5/go.mod h1:9r2w37qlBe7rQ6e1fg1S/9xpWHSnaqNdHD3WcMdbPDA=
281 | github.com/kr/pty v1.1.8/go.mod h1:O1sed60cT9XZ5uDucP5qwvh+TE3NnUj51EiZO/lmSfw=
282 | github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
283 | github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
284 | github.com/lib/pq v1.2.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo=
285 | github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e/go.mod h1:7rIyQOR62GCctdiQpZ/zOJlFyk6y+94wXzv6RNZgaR4=
286 | github.com/magiconair/properties v1.8.0 h1:LLgXmsheXeRoUOBOjtwPQCWIYqM/LU1ayDtDePerRcY=
287 | github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ=
288 | github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
289 | github.com/mailru/easyjson v0.0.0-20180823135443-60711f1a8329/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
290 | github.com/mailru/easyjson v0.0.0-20190312143242-1de009706dbe/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
291 | github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
292 | github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
293 | github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs=
294 | github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
295 | github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
296 | github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
297 | github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
298 | github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
299 | github.com/mattn/goveralls v0.0.2/go.mod h1:8d1ZMHsd7fW6IRPKQh46F2WRpyib5/X4FOpevwGNQEw=
300 | github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
301 | github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc=
302 | github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw=
303 | github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
304 | github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
305 | github.com/mitchellh/go-ps v0.0.0-20190716172923-621e5597135b/go.mod h1:r1VsdOzOPt1ZSrGZWFoNhsAedKnEd6r9Np1+5blZCWk=
306 | github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
307 | github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
308 | github.com/mitchellh/go-wordwrap v1.0.0/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo=
309 | github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
310 | github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw=
311 | github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
312 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
313 | github.com/modern-go/reflect2 v0.0.0-20180320133207-05fbef0ca5da/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
314 | github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
315 | github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
316 | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
317 | github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826/go.mod h1:TaXosZuwdSHYgviHp1DAtfrULt5eUgsSMsZf+YrPgl8=
318 | github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00/go.mod h1:Pm3mSP3c5uWn86xMLZ5Sa7JB9GsEZySvHYXCTK4E9q4=
319 | github.com/mozilla/tls-observatory v0.0.0-20190404164649-a3c1b6cfecfd/go.mod h1:SrKMQvPiws7F7iqYp8/TX+IhxCYhzr6N/1yb8cwHsGk=
320 | github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
321 | github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
322 | github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw=
323 | github.com/nbutton23/zxcvbn-go v0.0.0-20180912185939-ae427f1e4c1d/go.mod h1:o96djdrsSGy3AWPyBgZMAGfxZNfgntdJG+11KU4QvbU=
324 | github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA=
325 | github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U=
326 | github.com/onsi/ginkgo v0.0.0-20170829012221-11459a886d9c/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
327 | github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
328 | github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
329 | github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
330 | github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
331 | github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc=
332 | github.com/paulmach/orb v0.1.3/go.mod h1:VFlX/8C+IQ1p6FTRRKzKoOPJnvEtA5G0Veuqwbu//Vk=
333 | github.com/pborman/uuid v1.2.0/go.mod h1:X/NO0urCmaxf9VXbdlT7C2Yzkj2IKimNn4k+gtPdI/k=
334 | github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
335 | github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU=
336 | github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY=
337 | github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
338 | github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
339 | github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
340 | github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
341 | github.com/pmezard/go-difflib v0.0.0-20151028094244-d8ed2627bdf0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
342 | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
343 | github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI=
344 | github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
345 | github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso=
346 | github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
347 | github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
348 | github.com/prometheus/common v0.0.0-20181113130724-41aa239b4cce/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
349 | github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
350 | github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
351 | github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
352 | github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
353 | github.com/qri-io/starlib v0.4.2-0.20200213133954-ff2e8cd5ef8d/go.mod h1:7DPO4domFU579Ga6E61sB9VFNaniPVwJP5C4bBCu3wA=
354 | github.com/quasilyte/go-consistent v0.0.0-20190521200055-c6f3937de18c/go.mod h1:5STLWrekHfjyYwxBRVRXNOSewLJ3PWfDJd1VyTS21fI=
355 | github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
356 | github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
357 | github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
358 | github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
359 | github.com/ryanuber/columnize v2.1.0+incompatible/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts=
360 | github.com/ryanuber/go-glob v1.0.0/go.mod h1:807d1WSdnB0XRJzKNil9Om6lcp/3a0v4qIHxIXzX/Yc=
361 | github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d/go.mod h1:w5+eXa0mYznDkHaMCXA4XYffjlH+cy1oyKbfzJXa2Do=
362 | github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo=
363 | github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM=
364 | github.com/shirou/gopsutil v0.0.0-20190901111213-e4ec7b275ada/go.mod h1:WWnYX4lzhCH5h/3YBfyVA3VbLYjlMZZAQcW9ojMexNc=
365 | github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4/go.mod h1:qsXQc7+bwAM3Q1u/4XEfrquwF8Lw7D7y5cD8CuHnfIc=
366 | github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk=
367 | github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ=
368 | github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
369 | github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
370 | github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
371 | github.com/sirupsen/logrus v1.6.0 h1:UBcNElsrwanuuMsnGSlYmtmgbb23qDR5dG+6X6Oo89I=
372 | github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
373 | github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM=
374 | github.com/sourcegraph/go-diff v0.5.1/go.mod h1:j2dHj3m8aZgQO8lMTcTnBcXkRRRqi34cd2MNlA9u1mE=
375 | github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA=
376 | github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ=
377 | github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk=
378 | github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
379 | github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU=
380 | github.com/spf13/cobra v1.0.0/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE=
381 | github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo=
382 | github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
383 | github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
384 | github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
385 | github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
386 | github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s=
387 | github.com/spf13/viper v1.4.0/go.mod h1:PTJ7Z/lr49W6bUbkmS1V3by4uWynFiR9p7+dSq/yZzE=
388 | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
389 | github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
390 | github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE=
391 | github.com/stretchr/testify v0.0.0-20151208002404-e3a8ff8ce365/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
392 | github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
393 | github.com/stretchr/testify v1.2.3-0.20181224173747-660f15d67dbb/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
394 | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
395 | github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
396 | github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
397 | github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e/go.mod h1:Qimiffbc6q9tBWlVV6x0P9sat/ao1xEkREYPPj9hphk=
398 | github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
399 | github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
400 | github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
401 | github.com/ulikunitz/xz v0.5.5/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8=
402 | github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
403 | github.com/ultraware/whitespace v0.0.4/go.mod h1:aVMh/gQve5Maj9hQ/hg+F75lr/X5A89uZnzAmWSineA=
404 | github.com/uudashr/gocognit v0.0.0-20190926065955-1655d0de0517/go.mod h1:j44Ayx2KW4+oB6SWMv8KsmHzZrOInQav7D3cQMJ5JUM=
405 | github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
406 | github.com/valyala/fasthttp v1.2.0/go.mod h1:4vX61m6KN+xDduDNwXrhIAVZaZaZiQ1luJk8LWSxF3s=
407 | github.com/valyala/quicktemplate v1.2.0/go.mod h1:EH+4AkTd43SvgIbQHYu59/cJyxDoOVRUAfrukLPuGJ4=
408 | github.com/valyala/tcplisten v0.0.0-20161114210144-ceec8f93295a/go.mod h1:v3UYOV9WzVtRmSR+PDvWpU/qWl4Wa5LApYYX4ZtKbio=
409 | github.com/vektah/gqlparser v1.1.2/go.mod h1:1ycwN7Ij5njmMkPPAOaRFY4rET2Enx7IkVv3vaXspKw=
410 | github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
411 | github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg=
412 | github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
413 | github.com/yujunz/go-getter v1.4.1-lite/go.mod h1:sbmqxXjyLunH1PkF3n7zSlnVeMvmYUuIl9ZVs/7NyCc=
414 | go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
415 | go.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
416 | go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
417 | go.mongodb.org/mongo-driver v1.1.2/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
418 | go.opencensus.io v0.21.0 h1:mU6zScU4U1YAFPHEHYk+3JC4SY7JxgkqS10ZOSyksNg=
419 | go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
420 | go.starlark.net v0.0.0-20190528202925-30ae18b8564f/go.mod h1:c1/X6cHgvdXj6pUlmWKMkuqRnW4K8x2vwt6JAaaircg=
421 | go.starlark.net v0.0.0-20200306205701-8dd3e2ee1dd5/go.mod h1:nmDLcffg48OtT/PSW0Hg7FvpRQsQh5OSqIylirxKC7o=
422 | go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
423 | go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
424 | go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
425 | golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
426 | golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
427 | golang.org/x/crypto v0.0.0-20190211182817-74369b46fc67/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
428 | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
429 | golang.org/x/crypto v0.0.0-20190320223903-b7391e95e576/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
430 | golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
431 | golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
432 | golang.org/x/crypto v0.0.0-20190617133340-57b3e21c3d56/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
433 | golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
434 | golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY=
435 | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9 h1:psW17arqaxU48Z5kZ0CQnkZWQJsqcURM6tKiBApRjXI=
436 | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
437 | golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
438 | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
439 | golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
440 | golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
441 | golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
442 | golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
443 | golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
444 | golang.org/x/net v0.0.0-20180218175443-cbe0f9307d01/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
445 | golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
446 | golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
447 | golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
448 | golang.org/x/net v0.0.0-20180911220305-26e67e76b6c3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
449 | golang.org/x/net v0.0.0-20181005035420-146acd28ed58/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
450 | golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
451 | golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
452 | golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
453 | golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
454 | golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
455 | golang.org/x/net v0.0.0-20190320064053-1272bf9dcd53/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
456 | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
457 | golang.org/x/net v0.0.0-20190522155817-f3200d17e092/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
458 | golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
459 | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
460 | golang.org/x/net v0.0.0-20190827160401-ba9fcec4b297/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
461 | golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
462 | golang.org/x/net v0.0.0-20191004110552-13f9640d40b9/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
463 | golang.org/x/net v0.0.0-20200202094626-16171245cfb2 h1:CCH4IOTTfewWjGOlSp+zGcjutRKlBEZQ6wTn8ozI/nI=
464 | golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
465 | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
466 | golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
467 | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45 h1:SVwTIAaPC2U/AvvLNZ2a7OVsmBpC8L5BlwK1whH3hm0=
468 | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
469 | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
470 | golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
471 | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
472 | golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
473 | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
474 | golang.org/x/sys v0.0.0-20170830134202-bb24a47a89ea/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
475 | golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
476 | golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
477 | golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
478 | golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
479 | golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
480 | golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
481 | golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
482 | golang.org/x/sys v0.0.0-20190129075346-302c3dd5f1cc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
483 | golang.org/x/sys v0.0.0-20190209173611-3b5209105503/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
484 | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
485 | golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
486 | golang.org/x/sys v0.0.0-20190321052220-f7bb7a8bee54/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
487 | golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
488 | golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
489 | golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
490 | golang.org/x/sys v0.0.0-20190616124812-15dcb6c0061f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
491 | golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
492 | golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
493 | golang.org/x/sys v0.0.0-20191002063906-3421d5a6bb1c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
494 | golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4 h1:5/PjkGUjvEU5Gl6BxmvKRPpqo2uNMv4rcHBMwzk/st8=
495 | golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
496 | golang.org/x/text v0.0.0-20160726164857-2910a502d2bf/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
497 | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
498 | golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
499 | golang.org/x/text v0.3.1-0.20181227161524-e6919f6577db/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
500 | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
501 | golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
502 | golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
503 | golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
504 | golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
505 | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
506 | golang.org/x/tools v0.0.0-20181011042414-1f849cf54d09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
507 | golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
508 | golang.org/x/tools v0.0.0-20181117154741-2ddaf7f79a09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
509 | golang.org/x/tools v0.0.0-20190110163146-51295c7ec13a/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
510 | golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
511 | golang.org/x/tools v0.0.0-20190125232054-d66bd3c5d5a6/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
512 | golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
513 | golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
514 | golang.org/x/tools v0.0.0-20190311215038-5c2858a9cfe5/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
515 | golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
516 | golang.org/x/tools v0.0.0-20190322203728-c1a832b0ad89/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
517 | golang.org/x/tools v0.0.0-20190521203540-521d6ed310dd/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
518 | golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
519 | golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
520 | golang.org/x/tools v0.0.0-20190617190820-da514acc4774/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
521 | golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
522 | golang.org/x/tools v0.0.0-20190719005602-e377ae9d6386/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI=
523 | golang.org/x/tools v0.0.0-20190910044552-dd2b5c81c578/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
524 | golang.org/x/tools v0.0.0-20190930201159-7c411dea38b0/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
525 | golang.org/x/tools v0.0.0-20191010075000-0337d82405ff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
526 | golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
527 | golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
528 | google.golang.org/api v0.4.0 h1:KKgc1aqhV8wDPbDzlDtpvyjZFY3vjz85FP7p4wcQUyI=
529 | google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
530 | google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
531 | google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
532 | google.golang.org/appengine v1.5.0 h1:KxkO13IPW4Lslp2bz+KHP2E3gtFlrIGNThxkZQ3g+4c=
533 | google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
534 | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
535 | google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
536 | google.golang.org/genproto v0.0.0-20190404172233-64821d5d2107/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
537 | google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7 h1:ZUjXAXmrAyrmmCPHgCA/vChHcpsX27MZ3yBonD/z1KE=
538 | google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
539 | google.golang.org/grpc v1.14.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw=
540 | google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
541 | google.golang.org/grpc v1.21.0/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
542 | google.golang.org/grpc v1.22.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
543 | gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
544 | gopkg.in/asn1-ber.v1 v1.0.0-20181015200546-f715ec2f112d/go.mod h1:cuepJuh7vyXfUyUwEgHQXw849cJrilpS5NeIjOWESAw=
545 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
546 | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
547 | gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo=
548 | gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
549 | gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
550 | gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
551 | gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw=
552 | gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo=
553 | gopkg.in/square/go-jose.v2 v2.3.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
554 | gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
555 | gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
556 | gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
557 | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
558 | gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
559 | gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
560 | gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
561 | gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
562 | gopkg.in/yaml.v3 v3.0.0-20200121175148-a6ecf24a6d71/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
563 | honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
564 | honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
565 | honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
566 | honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
567 | k8s.io/api v0.17.0 h1:H9d/lw+VkZKEVIUc8F3wgiQ+FUXTTr21M87jXLU7yqM=
568 | k8s.io/api v0.17.0/go.mod h1:npsyOePkeP0CPwyGfXDHxvypiYMJxBWAMpQxCaJ4ZxI=
569 | k8s.io/apimachinery v0.17.0 h1:xRBnuie9rXcPxUkDizUsGvPf1cnlZCFu210op7J7LJo=
570 | k8s.io/apimachinery v0.17.0/go.mod h1:b9qmWdKlLuU9EBh+06BtLcSf/Mu89rWL33naRxs1uZg=
571 | k8s.io/client-go v0.17.0 h1:8QOGvUGdqDMFrm9sD6IUFl256BcffynGoe80sxgTEDg=
572 | k8s.io/client-go v0.17.0/go.mod h1:TYgR6EUHs6k45hb6KWjVD6jFZvJV4gHDikv/It0xz+k=
573 | k8s.io/gengo v0.0.0-20190128074634-0689ccc1d7d6/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0=
574 | k8s.io/klog v0.0.0-20181102134211-b9b56d5dfc92/go.mod h1:Gq+BEi5rUBO/HRz0bTSXDUcqjScdoY3a9IHpCEIOOfk=
575 | k8s.io/klog v0.3.0/go.mod h1:Gq+BEi5rUBO/HRz0bTSXDUcqjScdoY3a9IHpCEIOOfk=
576 | k8s.io/klog v1.0.0 h1:Pt+yjF5aB1xDSVbau4VsWe+dQNzA0qv1LlXdC2dF6Q8=
577 | k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I=
578 | k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a/go.mod h1:1TqjTSzOxsLGIKfj0lK8EeCP7K1iUG65v09OM0/WG5E=
579 | k8s.io/utils v0.0.0-20191114184206-e782cd3c129f h1:GiPwtSzdP43eI1hpPCbROQCCIgCuiMMNF8YUVLF3vJo=
580 | k8s.io/utils v0.0.0-20191114184206-e782cd3c129f/go.mod h1:sZAwmy6armz5eXlNoLmJcl4F1QuKu7sr+mFQ0byX7Ew=
581 | mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed/go.mod h1:Xkxe497xwlCKkIaQYRfC7CSLworTXY9RMqwhhCm+8Nc=
582 | mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
583 | mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
584 | sigs.k8s.io/kustomize/api v0.6.2 h1:qZzMiyllvwBv6KQ8V3VsF452INpDW4eWEHOfyeKvkHw=
585 | sigs.k8s.io/kustomize/api v0.6.2/go.mod h1:OL467fU5FuolXnIPUqhBLSXUUD00/IBjHs+dBdAS75E=
586 | sigs.k8s.io/kustomize/kyaml v0.8.1/go.mod h1:UTm64bSWVdBUA8EQoYCxVOaBQxUdIOr5LKWxA4GNbkw=
587 | sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
588 | sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
589 | sigs.k8s.io/yaml v1.2.0 h1:kr/MCeFWJWTwyaHoR9c8EjH9OumOmoF9YGiZd7lFm/Q=
590 | sigs.k8s.io/yaml v1.2.0/go.mod h1:yfXDCHCao9+ENCvLSE62v9VSji2MKu5jeNfTrofGhJc=
591 | sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4/go.mod h1:ketZ/q3QxT9HOBeFhu6RdvsftgpsbFHBF5Cas6cDKZ0=
592 |
--------------------------------------------------------------------------------
/internal/k8s/secret.go:
--------------------------------------------------------------------------------
1 | package k8s
2 |
3 | import (
4 | "gopkg.in/yaml.v2"
5 | "sigs.k8s.io/kustomize/api/types"
6 | )
7 |
8 | const (
9 | ApiV1 = "v1"
10 | SecretKind = "Secret"
11 | )
12 |
13 | type Secret struct {
14 | ApiVersion string `yaml:"apiVersion"`
15 | Kind string `yaml:"kind"`
16 | Meta types.ObjectMeta `yaml:"metadata"`
17 | Data map[string]string `yaml:"data"`
18 | Type string `yaml:"type,omitempty"`
19 | }
20 |
21 | func NewSecret(meta types.ObjectMeta, secretType string, data map[string]string) *Secret {
22 | return &Secret{
23 | ApiVersion: ApiV1,
24 | Kind: SecretKind,
25 | Meta: meta,
26 | Type: secretType,
27 | Data: data,
28 | }
29 | }
30 |
31 | func (s *Secret) ToYamlStr() (string, error) {
32 | out, err := yaml.Marshal(s)
33 | return string(out), err
34 | }
35 |
--------------------------------------------------------------------------------
/internal/k8s/secret_test.go:
--------------------------------------------------------------------------------
1 | package k8s
2 |
3 | import (
4 | "github.com/stretchr/testify/assert"
5 | "sigs.k8s.io/kustomize/api/types"
6 | "testing"
7 | )
8 |
9 | const (
10 | Name = "mysecret"
11 | Namespace = "test"
12 | Type = "opaque"
13 | )
14 |
15 | func TestNewSecret(t *testing.T) {
16 | s := NewSecret(types.ObjectMeta{Namespace: Namespace, Name: Name}, Type, nil)
17 | assert.Equal(t, s.Type, Type)
18 | assert.Equal(t, s.Meta.Name, Name)
19 | assert.Equal(t, s.Meta.Namespace, Namespace)
20 | }
21 |
22 | func TestSecret_ToYamlStr(t *testing.T) {
23 | s := NewSecret(types.ObjectMeta{Namespace: Namespace, Name: Name}, Type, nil)
24 | _, err := s.ToYamlStr()
25 | assert.NoError(t, err)
26 | }
27 |
--------------------------------------------------------------------------------
/internal/providers/aws_sm.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "github.com/aws/aws-sdk-go/aws/session"
5 | "github.com/aws/aws-sdk-go/service/secretsmanager"
6 | "github.com/aws/aws-sdk-go/service/secretsmanager/secretsmanageriface"
7 | )
8 |
9 | type AwsSMProvider struct {
10 | Client secretsmanageriface.SecretsManagerAPI
11 | }
12 |
13 | func (p *AwsSMProvider) GetSecret(name string) (string, error) {
14 | resp, err := p.Client.GetSecretValue(&secretsmanager.GetSecretValueInput{SecretId: &name})
15 | if err != nil {
16 | return "", err
17 | }
18 | return *resp.SecretString, err
19 | }
20 |
21 | func (p *AwsSMProvider) GetKVSecrets(name string) (map[string]string, error) {
22 | return jsonKVSecrets(p, name)
23 | }
24 |
25 | func NewAwsSMProvider() (SecretsProvider, error) {
26 | s, err := session.NewSession()
27 | if err != nil {
28 | return nil, err
29 | }
30 | return &AwsSMProvider{Client: secretsmanager.New(s)}, nil
31 |
32 | }
33 |
--------------------------------------------------------------------------------
/internal/providers/aws_sm_test.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "fmt"
5 | "github.com/aws/aws-sdk-go/service/secretsmanager"
6 | "github.com/aws/aws-sdk-go/service/secretsmanager/secretsmanageriface"
7 | "github.com/stretchr/testify/assert"
8 | "testing"
9 | )
10 |
11 | type fakeSMClient struct {
12 | secretsmanageriface.SecretsManagerAPI
13 | data map[string]string
14 | }
15 |
16 | var (
17 | EmptySecretError = fmt.Errorf("secret value is missing")
18 | TestKey = "key"
19 | TestValue = "value"
20 | TestSecretKey = "secret-key"
21 | TestSecretValue = "secret-value"
22 | TestJsonStr = fmt.Sprintf(`{"%s":"%s"}`, TestSecretKey, TestSecretValue)
23 | TestMap = map[string]string{
24 | TestKey: TestValue,
25 | }
26 | TestMapJson = map[string]string{
27 | TestKey: TestJsonStr,
28 | }
29 | )
30 |
31 | func (m *fakeSMClient) setData(data map[string]string) {
32 | m.data = data
33 | }
34 |
35 | func newFakeSMClient(data map[string]string) *fakeSMClient {
36 | return &fakeSMClient{data: data}
37 | }
38 |
39 | func (m *fakeSMClient) GetSecretValue(in *secretsmanager.GetSecretValueInput) (*secretsmanager.GetSecretValueOutput, error) {
40 | if val, found := m.data[*in.SecretId]; found {
41 | return &secretsmanager.GetSecretValueOutput{SecretString: &val}, nil
42 |
43 | }
44 | return nil, EmptySecretError
45 | }
46 |
47 | func TestAwsSMProvider_GetSecret(t *testing.T) {
48 |
49 | p := AwsSMProvider{
50 | Client: newFakeSMClient(TestMap),
51 | }
52 | res, err := p.GetSecret(TestKey)
53 | assert.NoError(t, err)
54 | assert.Equal(t, TestValue, res)
55 | }
56 |
57 | func TestAwsSMProvider_GetSecretErr(t *testing.T) {
58 | p := AwsSMProvider{
59 | Client: newFakeSMClient(nil),
60 | }
61 | _, err := p.GetSecret(TestKey)
62 | assert.Error(t, err)
63 | }
64 |
65 | func TestAwsSMProvider_GetKVSecrets(t *testing.T) {
66 | p := AwsSMProvider{
67 | Client: newFakeSMClient(TestMapJson),
68 | }
69 | res, err := p.GetKVSecrets(TestKey)
70 | assert.NoError(t, err)
71 | assert.Equal(t, TestSecretValue, res[TestSecretKey])
72 | }
73 |
74 | func TestAwsSMProvider_GetKVSecretsErr(t *testing.T) {
75 | mockClient := newFakeSMClient(map[string]string{})
76 | p := AwsSMProvider{
77 | Client: mockClient,
78 | }
79 | _, err := p.GetKVSecrets(TestKey)
80 | assert.Error(t, err)
81 |
82 | // non-json value
83 | mockClient.setData(TestMap)
84 | _, err = p.GetKVSecrets(TestKey)
85 | assert.Error(t, err)
86 |
87 | }
88 |
89 | func TestNewAwsSMProvider(t *testing.T) {
90 | _, err := NewAwsSMProvider()
91 | assert.NoError(t, err)
92 | }
93 |
--------------------------------------------------------------------------------
/internal/providers/azure_vault.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "context"
5 | "fmt"
6 | "github.com/Azure/azure-sdk-for-go/profiles/latest/keyvault/keyvault"
7 | "github.com/Azure/azure-sdk-for-go/profiles/latest/keyvault/keyvault/keyvaultapi"
8 | "github.com/Azure/go-autorest/autorest"
9 | "github.com/Azure/go-autorest/autorest/azure/auth"
10 | "os"
11 | "strings"
12 | )
13 |
14 | func AzureVaultUrl(vaultName string) string {
15 | return fmt.Sprintf("https://%s.vault.azure.net", vaultName)
16 | }
17 |
18 | type AzureVaultProvider struct {
19 | vaultUrl string
20 | Client keyvaultapi.BaseClientAPI
21 | }
22 |
23 | func NewAzureVaultProvider(vaultName string) (SecretsProvider, error) {
24 | authorizer, err := getAuthorizer()
25 | if err != nil {
26 | return nil, err
27 | }
28 | client := keyvault.New()
29 | client.Authorizer = authorizer
30 | return &AzureVaultProvider{vaultUrl: AzureVaultUrl(vaultName), Client: client}, nil
31 | }
32 |
33 | func (p *AzureVaultProvider) GetSecret(name string) (string, error) {
34 | resp, err := p.Client.GetSecret(context.Background(), p.vaultUrl, name, "")
35 | if err != nil {
36 | return "", err
37 | }
38 | return *resp.Value, err
39 | }
40 |
41 | func (p *AzureVaultProvider) GetKVSecrets(name string) (map[string]string, error) {
42 | return jsonKVSecrets(p, name)
43 | }
44 |
45 | func getAuthorizer() (autorest.Authorizer, error) {
46 | settings, err := auth.GetSettingsFromEnvironment()
47 | if err != nil {
48 | return nil, err
49 | }
50 | settings.Values[auth.Resource] = strings.TrimSuffix(settings.Environment.KeyVaultEndpoint, "/")
51 |
52 | // based on Azure SDK EnvironmentSettings.GetAuthorizer()
53 | //1.Client Credentials
54 | if c, e := settings.GetClientCredentials(); e == nil {
55 | return c.Authorizer()
56 | }
57 |
58 | //2. Client Certificate
59 | if c, e := settings.GetClientCertificate(); e == nil {
60 | return c.Authorizer()
61 | }
62 |
63 | //3. Username Password
64 | if c, e := settings.GetUsernamePassword(); e == nil {
65 | return c.Authorizer()
66 | }
67 |
68 | // 4. MSI
69 | if _, present := os.LookupEnv("AZURE_USE_MSI"); present {
70 | return settings.GetMSI().Authorizer()
71 | }
72 |
73 | // 5. CLI
74 | return auth.NewAuthorizerFromCLIWithResource(settings.Values[auth.Resource])
75 | }
76 |
--------------------------------------------------------------------------------
/internal/providers/azure_vault_test.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "context"
5 | "github.com/Azure/azure-sdk-for-go/profiles/latest/keyvault/keyvault"
6 | "github.com/Azure/azure-sdk-for-go/profiles/latest/keyvault/keyvault/keyvaultapi"
7 | "github.com/stretchr/testify/assert"
8 | "os"
9 | "testing"
10 | )
11 |
12 | type fakeAzureClient struct {
13 | keyvaultapi.BaseClientAPI
14 | data map[string]string
15 | }
16 |
17 | func newFakeAzureClient(data map[string]string) *fakeAzureClient {
18 | return &fakeAzureClient{data: data}
19 | }
20 |
21 | func (f *fakeAzureClient) GetSecret(ctx context.Context, vaultBaseURL string, secretName string, secretVersion string) (keyvault.SecretBundle, error) {
22 | if val, found := f.data[secretName]; found {
23 | return keyvault.SecretBundle{Value: &val}, nil
24 | }
25 | return keyvault.SecretBundle{}, EmptySecretError
26 | }
27 |
28 | func TestAzureVaultUrl(t *testing.T) {
29 | res := AzureVaultUrl("test")
30 | assert.Equal(t, "https://test.vault.azure.net", res)
31 | }
32 |
33 | func TestAzureVaultProvider_GetSecret(t *testing.T) {
34 | p := AzureVaultProvider{Client: newFakeAzureClient(TestMap)}
35 | res, err := p.GetSecret(TestKey)
36 | assert.NoError(t, err)
37 | assert.Equal(t, TestValue, res)
38 | }
39 |
40 | func TestAzureVaultProvider_GetSecretErr(t *testing.T) {
41 | p := AzureVaultProvider{Client: newFakeAzureClient(map[string]string{})}
42 | _, err := p.GetSecret(TestKey)
43 | assert.Error(t, err)
44 | }
45 |
46 | func TestAzureVaultProvider_GetKVSecrets(t *testing.T) {
47 | p := AzureVaultProvider{Client: newFakeAzureClient(TestMapJson)}
48 | res, err := p.GetKVSecrets(TestKey)
49 | assert.NoError(t, err)
50 | assert.Equal(t, TestSecretValue, res[TestSecretKey])
51 | }
52 |
53 | func TestAzureVaultProvider_GetKVSecretsErr(t *testing.T) {
54 | client := newFakeAzureClient(map[string]string{})
55 | p := AzureVaultProvider{
56 | Client: client,
57 | }
58 | _, err := p.GetKVSecrets(TestKey)
59 | assert.Error(t, err)
60 |
61 | // non-json value
62 | client.data = TestMap
63 | _, err = p.GetKVSecrets(TestKey)
64 | assert.Error(t, err)
65 | }
66 |
67 | func TestNewAzureVaultProvider(t *testing.T) {
68 | assert.NoError(t, os.Setenv("AZURE_CLIENT_ID", "test"))
69 | assert.NoError(t, os.Setenv("AZURE_CLIENT_SECRET", "test"))
70 |
71 | _, err := NewAzureVaultProvider("")
72 | assert.NoError(t, err)
73 | }
74 |
75 | func TestNewAzureVaultProviderErr(t *testing.T) {
76 | assert.NoError(t, os.Unsetenv("AZURE_CLIENT_ID"))
77 | assert.NoError(t, os.Unsetenv("AZURE_CLIENT_SECRET"))
78 |
79 | _, err := NewAzureVaultProvider("")
80 | assert.Error(t, err)
81 | }
82 |
--------------------------------------------------------------------------------
/internal/providers/env.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "fmt"
5 | "os"
6 | )
7 |
8 | type EnvProvider struct {
9 | }
10 |
11 | func (p *EnvProvider) GetSecret(name string) (string, error) {
12 | if res, found := os.LookupEnv(name); found {
13 | return res, nil
14 | }
15 | return "", fmt.Errorf("couldn't find env variable: %s", name)
16 | }
17 |
18 | func (p *EnvProvider) GetKVSecrets(name string) (map[string]string, error) {
19 | return jsonKVSecrets(p, name)
20 | }
21 |
22 | func NewEnvProvider() SecretsProvider {
23 | return &EnvProvider{}
24 |
25 | }
26 |
--------------------------------------------------------------------------------
/internal/providers/env_test.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "github.com/stretchr/testify/assert"
5 | "os"
6 | "testing"
7 | )
8 |
9 | func TestNewEnvProvider(t *testing.T) {
10 | p := NewEnvProvider()
11 | assert.NotNil(t, p)
12 | }
13 |
14 | func TestEnvProvider_GetSecret(t *testing.T) {
15 | p := NewEnvProvider()
16 | err := os.Setenv(TestKey, TestValue)
17 | assert.NoError(t, err)
18 |
19 | res, err := p.GetSecret(TestKey)
20 | assert.NoError(t, err)
21 | assert.Equal(t, TestValue, res)
22 | }
23 |
24 | func TestEnvProvider_GetSecretErr(t *testing.T) {
25 | p := NewEnvProvider()
26 | os.Unsetenv(TestKey)
27 | _, err := p.GetSecret(TestKey)
28 | assert.Error(t, err)
29 | }
30 |
31 | func TestEnvProvider_GetKVSecrets(t *testing.T) {
32 | p := NewEnvProvider()
33 | err := os.Setenv(TestKey, TestJsonStr)
34 | assert.NoError(t, err)
35 |
36 | res, err := p.GetKVSecrets(TestKey)
37 | assert.NoError(t, err)
38 | assert.Equal(t, TestSecretValue, res[TestSecretKey])
39 | }
40 |
41 | func TestEnvProvider_GetKVSecretsErr(t *testing.T) {
42 | p := NewEnvProvider()
43 |
44 | os.Unsetenv(TestKey)
45 | _, err := p.GetKVSecrets(TestKey)
46 | assert.Error(t, err)
47 |
48 | err = os.Setenv(TestKey, TestValue)
49 | assert.NoError(t, err)
50 |
51 | _, err = p.GetKVSecrets(TestKey)
52 | assert.Error(t, err)
53 | }
54 |
--------------------------------------------------------------------------------
/internal/providers/hashicorp_vault.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "fmt"
5 | "github.com/hashicorp/vault/api"
6 | )
7 |
8 | type VaultReader interface {
9 | Read(string) (*api.Secret, error)
10 | }
11 |
12 | type HashicorpVaultProvider struct {
13 | Client VaultReader
14 | }
15 |
16 | func (p *HashicorpVaultProvider) GetKVSecrets(path string) (map[string]string, error) {
17 | s, err := getHashicorpVaultSecret(p.Client, path)
18 | if err != nil {
19 | return nil, err
20 | }
21 | kvSecrets := make(map[string]string)
22 | for k, v := range s.Data["data"].(map[string]interface{}) {
23 | kvSecrets[k] = v.(string)
24 | }
25 | return kvSecrets, err
26 | }
27 |
28 | func NewHashicorpVaultProvider() (SecretsProvider, error) {
29 | client, err := api.NewClient(nil)
30 | if err != nil {
31 | return nil, err
32 | }
33 |
34 | return &HashicorpVaultProvider{Client: client.Logical()}, err
35 | }
36 |
37 | func getKVSecret(name string, f func(path string) (map[string]string, error)) (string, error) {
38 | path, field := parseSecretLiteral(name)
39 | if field == "" {
40 | return "", fmt.Errorf("vault field is empty")
41 | }
42 | data, err := f(path)
43 | if err != nil {
44 | return "", err
45 | }
46 | if val, exists := data[field]; exists {
47 | return val, err
48 | }
49 | return "", fmt.Errorf("field \"%s\" not present in secret %s", field, path)
50 | }
51 |
52 | func (p *HashicorpVaultProvider) GetSecret(name string) (string, error) {
53 | return getKVSecret(name, func(path string) (map[string]string, error) {
54 | s, err := getHashicorpVaultSecret(p.Client, path)
55 | if err != nil {
56 | return nil, err
57 | }
58 | kvSecrets := make(map[string]string)
59 | for k, v := range s.Data["data"].(map[string]interface{}) {
60 | kvSecrets[k] = v.(string)
61 | }
62 | return kvSecrets, nil
63 | })
64 | }
65 |
66 | func getHashicorpVaultSecret(client VaultReader, path string) (*api.Secret, error) {
67 | s, err := client.Read(path)
68 | if err != nil {
69 | return nil, err
70 | }
71 | if s == nil {
72 | return nil, fmt.Errorf("couldn't find the spicified secret mount: %s", path)
73 | }
74 | if s.Data == nil {
75 | return nil, fmt.Errorf("no value found at %s", path)
76 | }
77 | return s, err
78 | }
79 |
--------------------------------------------------------------------------------
/internal/providers/hashicorp_vault_test.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "fmt"
5 | "github.com/hashicorp/vault/api"
6 | "github.com/stretchr/testify/assert"
7 | "testing"
8 | )
9 |
10 | const (
11 | TestSecretPath = "test"
12 | )
13 |
14 | type fakeVaultClient struct {
15 | secrets map[string]*api.Secret
16 | }
17 |
18 | func NewFakeVaultClient() *fakeVaultClient {
19 | return &fakeVaultClient{
20 | secrets: map[string]*api.Secret{},
21 | }
22 | }
23 |
24 | func (f *fakeVaultClient) SetSecret(path string, s *api.Secret) {
25 | f.secrets[path] = s
26 | }
27 |
28 | func (f *fakeVaultClient) Read(path string) (*api.Secret, error) {
29 | if val, found := f.secrets[path]; found {
30 | return val, nil
31 | }
32 | return nil, EmptySecretError
33 | }
34 |
35 | func TestNewHashicorpVaultProvider(t *testing.T) {
36 | p, err := NewHashicorpVaultProvider()
37 | assert.NoError(t, err)
38 | assert.NotNil(t, p)
39 |
40 | }
41 |
42 | func TestHashicorpVaultProvider_GetSecret(t *testing.T) {
43 |
44 | fakeClient := NewFakeVaultClient()
45 |
46 | fakeClient.SetSecret(TestSecretPath, &api.Secret{
47 | Data: map[string]interface{}{
48 | "data": map[string]interface{}{
49 | TestSecretKey: TestSecretValue,
50 | },
51 | },
52 | })
53 |
54 | p := HashicorpVaultProvider{
55 | Client: fakeClient,
56 | }
57 | res, err := p.GetSecret(fmt.Sprintf("%s:%s", TestSecretPath, TestSecretKey))
58 | assert.NoError(t, err)
59 | assert.Equal(t, TestSecretValue, res)
60 |
61 | }
62 |
63 | func TestHashicorpVaultProvider_GetSecretErr(t *testing.T) {
64 | p := HashicorpVaultProvider{
65 | Client: NewFakeVaultClient(),
66 | }
67 | _, err := p.GetSecret(fmt.Sprintf("%s:%s", TestSecretPath, TestSecretKey))
68 | assert.Error(t, err)
69 | }
70 |
71 | func TestHashicorpVaultProvider_GetKVSecrets(t *testing.T) {
72 | fakeClient := NewFakeVaultClient()
73 |
74 | fakeClient.SetSecret(TestSecretPath, &api.Secret{
75 | Data: map[string]interface{}{
76 | "data": map[string]interface{}{
77 | TestSecretKey: TestSecretValue,
78 | },
79 | },
80 | })
81 |
82 | p := HashicorpVaultProvider{
83 | Client: fakeClient,
84 | }
85 |
86 | res, err := p.GetKVSecrets(TestSecretPath)
87 | assert.NoError(t, err)
88 | assert.Equal(t,
89 | map[string]string{
90 | TestSecretKey: TestSecretValue,
91 | },
92 | res)
93 | }
94 |
95 | func TestHashicorpVaultProvider_GetKVSecretsErr(t *testing.T) {
96 | p := HashicorpVaultProvider{
97 | Client: NewFakeVaultClient(),
98 | }
99 | _, err := p.GetKVSecrets(TestSecretKey)
100 | assert.Error(t, err)
101 | }
102 |
--------------------------------------------------------------------------------
/internal/providers/k8s_secret.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "encoding/base64"
5 | v12 "k8s.io/apimachinery/pkg/apis/meta/v1"
6 | v1 "k8s.io/client-go/kubernetes/typed/core/v1"
7 | "k8s.io/client-go/tools/clientcmd"
8 | "os"
9 | )
10 |
11 | type K8SSecretProvider struct {
12 | Client v1.SecretInterface
13 | }
14 |
15 | func Base64Decode(src []byte) ([]byte, error) {
16 | decoded := make([]byte, base64.StdEncoding.DecodedLen(len(src)))
17 | _, err := base64.StdEncoding.Decode(decoded, src)
18 | return decoded, err
19 | }
20 |
21 | func (p *K8SSecretProvider) GetSecret(name string) (string, error) {
22 | return getKVSecret(name, func(path string) (map[string]string, error) {
23 | s, err := p.Client.Get(path, v12.GetOptions{})
24 | if err != nil {
25 | return nil, err
26 | }
27 | res := make(map[string]string)
28 | for k, v := range s.Data {
29 | decoded, err := Base64Decode(v)
30 | if err != nil {
31 | return nil, err
32 | }
33 | res[k] = string(decoded)
34 | }
35 | return res, nil
36 |
37 | })
38 | }
39 |
40 | func (p *K8SSecretProvider) GetKVSecrets(path string) (map[string]string, error) {
41 | res, err := p.Client.Get(path, v12.GetOptions{})
42 | if err != nil {
43 | return nil, err
44 | }
45 |
46 | kvSecrets := make(map[string]string)
47 | for k, v := range res.Data {
48 | decoded, err := Base64Decode(v)
49 | if err != nil {
50 | return nil, err
51 | }
52 | kvSecrets[k] = string(decoded)
53 | }
54 | return kvSecrets, err
55 | }
56 |
57 | func NewK8SSecretProvider(namespace string) (SecretsProvider, error) {
58 | kubeConfigPath := os.Getenv("KUBECONFIG")
59 |
60 | config, err := clientcmd.BuildConfigFromFlags("", kubeConfigPath)
61 | if err != nil {
62 | return nil, err
63 | }
64 | v1Client, err := v1.NewForConfig(config)
65 | if err != nil {
66 | return nil, err
67 | }
68 | return &K8SSecretProvider{Client: v1Client.Secrets(namespace)}, nil
69 | }
70 |
--------------------------------------------------------------------------------
/internal/providers/k8s_secret_test.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "encoding/base64"
5 | "fmt"
6 | "github.com/stretchr/testify/assert"
7 | v1 "k8s.io/api/core/v1"
8 | v12 "k8s.io/apimachinery/pkg/apis/meta/v1"
9 | testclient "k8s.io/client-go/kubernetes/fake"
10 | v13 "k8s.io/client-go/kubernetes/typed/core/v1"
11 | "testing"
12 | )
13 |
14 | const (
15 | TestSecretName = "test"
16 | )
17 |
18 | func configureTestClient() (v13.SecretInterface, error) {
19 | k8sClient := testclient.NewSimpleClientset().CoreV1().Secrets("test")
20 | secretValue := make([]byte, base64.StdEncoding.EncodedLen(len([]byte(TestSecretValue))))
21 | base64.StdEncoding.Encode(secretValue, []byte(TestSecretValue))
22 | _, err := k8sClient.Create(&v1.Secret{
23 | ObjectMeta: v12.ObjectMeta{
24 | Name: TestSecretName,
25 | },
26 | Data: map[string][]byte{
27 | TestSecretKey: secretValue,
28 | },
29 | })
30 | return k8sClient, err
31 | }
32 |
33 | func TestK8SSecretProvider_GetSecret(t *testing.T) {
34 | k8sClient, err := configureTestClient()
35 | assert.NoError(t, err)
36 | p := K8SSecretProvider{
37 | Client: k8sClient,
38 | }
39 |
40 | res, err := p.GetSecret(fmt.Sprintf("%s:%s", TestSecretName, TestSecretKey))
41 | assert.NoError(t, err)
42 | assert.Equal(t, TestSecretValue, res)
43 | }
44 |
45 | func TestK8SSecretProvider_GetSecretErr(t *testing.T) {
46 | k8sClient, err := configureTestClient()
47 | assert.NoError(t, err)
48 | p := K8SSecretProvider{
49 | Client: k8sClient,
50 | }
51 | _, err = p.GetSecret("")
52 | assert.Error(t, err)
53 | }
54 |
55 | func TestK8SSecretProvider_GetKVSecrets(t *testing.T) {
56 | k8sClient, err := configureTestClient()
57 | assert.NoError(t, err)
58 | p := K8SSecretProvider{
59 | Client: k8sClient,
60 | }
61 | res, err := p.GetKVSecrets(TestSecretName)
62 | assert.NoError(t, err)
63 | assert.Equal(t,
64 | map[string]string{
65 | TestSecretKey: TestSecretValue,
66 | },
67 | res)
68 | }
69 |
70 | func TestK8SSecretProvider_GetKVSecretsErr(t *testing.T) {
71 | k8sClient, err := configureTestClient()
72 | assert.NoError(t, err)
73 | p := K8SSecretProvider{
74 | Client: k8sClient,
75 | }
76 | _, err = p.GetKVSecrets("")
77 | assert.Error(t, err)
78 | }
79 |
--------------------------------------------------------------------------------
/internal/providers/secrets_provider.go:
--------------------------------------------------------------------------------
1 | package providers
2 |
3 | import (
4 | "encoding/json"
5 | "strings"
6 | )
7 |
8 | const (
9 | secretNameSeparator = ":"
10 | )
11 |
12 | func parseSecretLiteral(literal string) (path string, field string) {
13 | res := strings.Split(literal, secretNameSeparator)
14 | path = res[0]
15 | if len(res) > 1 {
16 | field = res[1]
17 | }
18 | return path, field
19 | }
20 |
21 | func jsonKVSecrets(p SecretsProvider, name string) (map[string]string, error) {
22 | secret, err := p.GetSecret(name)
23 | if err != nil {
24 | return nil, err
25 | }
26 | kvSecrets := make(map[string]string)
27 | err = json.Unmarshal([]byte(secret), &kvSecrets)
28 | return kvSecrets, err
29 | }
30 |
31 | type SecretsProvider interface {
32 | GetSecret(name string) (string, error)
33 | GetKVSecrets(path string) (map[string]string, error)
34 | }
35 |
--------------------------------------------------------------------------------
/pkg/generator/generator.go:
--------------------------------------------------------------------------------
1 | package generator
2 |
3 | import (
4 | "encoding/base64"
5 | "github.com/bbl/secretize/internal/k8s"
6 | "github.com/bbl/secretize/internal/providers"
7 | "github.com/bbl/secretize/pkg/utils"
8 | "gopkg.in/yaml.v2"
9 | "sigs.k8s.io/kustomize/api/types"
10 | "strings"
11 | )
12 |
13 | type RegistryFunc func(params map[string]string) map[string]func() (providers.SecretsProvider, error)
14 |
15 | func ProviderRegistry(params map[string]string) map[string]func() (providers.SecretsProvider, error) {
16 | return map[string]func() (providers.SecretsProvider, error){
17 | "aws-sm": providers.NewAwsSMProvider,
18 | "hashicorp-vault": providers.NewHashicorpVaultProvider,
19 | "azure-vault": func() (providers.SecretsProvider, error) {
20 | return providers.NewAzureVaultProvider(params["name"])
21 | },
22 | "k8s-secret": func() (providers.SecretsProvider, error) {
23 | return providers.NewK8SSecretProvider(params["namespace"])
24 | },
25 | "env": func() (providers.SecretsProvider, error) {
26 | return providers.NewEnvProvider(), nil
27 | },
28 | }
29 | }
30 |
31 | type Literal struct {
32 | Key string
33 | Value string
34 | }
35 |
36 | type SecretsSpec struct {
37 | KVLiterals []string `yaml:"kv"`
38 | Literals []Literal `yaml:"literals"`
39 | }
40 |
41 | type SecretSource struct {
42 | Provider string `yaml:"provider"`
43 | SecretsSpec `yaml:",inline"`
44 | Params map[string]string `yaml:"params"`
45 | }
46 |
47 | type SecretGenerator struct {
48 | Meta types.ObjectMeta `yaml:"metadata"`
49 | Type string `yaml:"type"`
50 | Sources []SecretSource `yaml:"sources"`
51 | Literals []Literal `yaml:"literals"`
52 | }
53 |
54 | func (l *Literal) UnmarshalYAML(unmarshal func(interface{}) error) error {
55 | stringLiteral := ""
56 | err := unmarshal(&stringLiteral)
57 | if err != nil {
58 | return err
59 | }
60 | l.Key = stringLiteral
61 | l.Value = stringLiteral
62 |
63 | if !strings.Contains(stringLiteral, "=") {
64 | return nil
65 | }
66 |
67 | res := strings.Split(stringLiteral, "=")
68 | l.Key = res[0]
69 | l.Value = res[1]
70 |
71 | return nil
72 | }
73 |
74 | func ParseConfig(data []byte) (*SecretGenerator, error) {
75 | conf := SecretGenerator{}
76 | err := yaml.Unmarshal(data, &conf)
77 | return &conf, err
78 | }
79 |
80 | func (sg *SecretGenerator) FetchSecrets(registry RegistryFunc) (map[string]string, error) {
81 | secrets := make(map[string]string)
82 |
83 | for _, s := range sg.Sources {
84 | provider, err := registry(s.Params)[s.Provider]()
85 | if err != nil {
86 | return nil, err
87 | }
88 | providerSecrets, err := FetchProviderSecrets(provider, s.SecretsSpec)
89 | if err != nil {
90 | return nil, err
91 | }
92 | secrets = utils.Merge(secrets, providerSecrets)
93 | }
94 | return secrets, nil
95 | }
96 |
97 | func FetchProviderSecrets(p providers.SecretsProvider, spec SecretsSpec) (map[string]string, error) {
98 | res := make(map[string]string)
99 |
100 | for _, l := range spec.Literals {
101 | resp, err := p.GetSecret(l.Value)
102 | if err != nil {
103 | return nil, err
104 | }
105 | res[l.Key] = resp
106 | }
107 |
108 | for _, v := range spec.KVLiterals {
109 | resp, err := p.GetKVSecrets(v)
110 | if err != nil {
111 | return nil, err
112 | }
113 | res = utils.Merge(res, resp)
114 | }
115 |
116 | return res, nil
117 | }
118 |
119 | func (sg *SecretGenerator) Generate(secrets map[string]string) *k8s.Secret {
120 | secrets = utils.Map(secrets, func(v string) string {
121 | return base64.StdEncoding.EncodeToString([]byte(v))
122 | })
123 | return k8s.NewSecret(sg.Meta, sg.Type, secrets)
124 | }
125 |
--------------------------------------------------------------------------------
/pkg/generator/generator_test.go:
--------------------------------------------------------------------------------
1 | package generator
2 |
3 | import (
4 | "encoding/base64"
5 | "fmt"
6 | "github.com/bbl/secretize/internal/providers"
7 | "github.com/stretchr/testify/assert"
8 | "sigs.k8s.io/kustomize/api/types"
9 | "testing"
10 | )
11 |
12 | var (
13 | testSG = SecretGenerator{
14 | Meta: types.ObjectMeta{},
15 | Type: "",
16 | Sources: nil,
17 | Literals: nil,
18 | }
19 | testSecretsSpec = SecretsSpec{
20 | Literals: []Literal{
21 | {Key: "secret-key-name", Value: "provider-key-name"},
22 | },
23 | KVLiterals: []string{
24 | "test-kv",
25 | }}
26 | testProvider = &fakeProvider{
27 | literals: map[string]string{
28 | "provider-key-name": "test-value",
29 | },
30 | kv: map[string]map[string]string{
31 | "test-kv": {},
32 | },
33 | }
34 | )
35 |
36 | func TestParseConfig(t *testing.T) {
37 | confStr := `metadata:
38 | name: infrabot-secrets
39 | annotations:
40 | kustomize.config.k8s.io/behavior: "merge"
41 | sources:
42 | - provider: hashicorp-vault
43 | literals:
44 | - NAME=example:example
45 | `
46 | conf, err := ParseConfig([]byte(confStr))
47 | assert.NoError(t, err)
48 | assert.Equal(t, "hashicorp-vault", conf.Sources[0].Provider)
49 | }
50 |
51 | type fakeProvider struct {
52 | literals map[string]string
53 | kv map[string]map[string]string
54 | }
55 |
56 | func (f *fakeProvider) GetSecret(name string) (string, error) {
57 | if val, ok := f.literals[name]; ok {
58 | return val, nil
59 | }
60 | return "", fmt.Errorf("literal key not found error: %s", name)
61 | }
62 | func (f *fakeProvider) GetKVSecrets(path string) (map[string]string, error) {
63 | if val, ok := f.kv[path]; ok {
64 | return val, nil
65 | }
66 | return nil, fmt.Errorf("kv path not found error, %s", path)
67 | }
68 |
69 | func TestFetchProviderSecrets(t *testing.T) {
70 | res, err := FetchProviderSecrets(testProvider, testSecretsSpec)
71 |
72 | assert.NoError(t, err)
73 | assert.Equal(t, "test-value", res["secret-key-name"])
74 |
75 | }
76 |
77 | func TestFetchProviderSecretsErr(t *testing.T) {
78 | _, err := FetchProviderSecrets(&fakeProvider{}, testSecretsSpec)
79 | assert.Error(t, err)
80 |
81 | _, err = FetchProviderSecrets(&fakeProvider{
82 | literals: map[string]string{
83 | "provider-key-name": "test-value",
84 | },
85 | }, testSecretsSpec)
86 | assert.Error(t, err)
87 |
88 | }
89 |
90 | func TestSecretGenerator_FetchSecrets(t *testing.T) {
91 | testSG.Sources = append(testSG.Sources, SecretSource{
92 | Provider: "fake",
93 | SecretsSpec: testSecretsSpec,
94 | Params: nil,
95 | })
96 | _, err := testSG.FetchSecrets(func(params map[string]string) map[string]func() (providers.SecretsProvider, error) {
97 | return map[string]func() (providers.SecretsProvider, error){
98 | "fake": func() (providers.SecretsProvider, error) {
99 | return testProvider, nil
100 | },
101 | }
102 | })
103 | assert.NoError(t, err)
104 | }
105 |
106 | func TestSecretGenerator_FetchSecretsErr(t *testing.T) {
107 | testSG.Sources = append(testSG.Sources, SecretSource{
108 | Provider: "fake",
109 | SecretsSpec: testSecretsSpec,
110 | Params: nil,
111 | })
112 |
113 | _, err := testSG.FetchSecrets(func(params map[string]string) map[string]func() (providers.SecretsProvider, error) {
114 | return map[string]func() (providers.SecretsProvider, error){
115 | "fake": func() (providers.SecretsProvider, error) {
116 | return nil, fmt.Errorf("error")
117 | },
118 | }
119 | })
120 | assert.Error(t, err)
121 | assert.Equal(t, "error", err.Error())
122 |
123 | _, err = testSG.FetchSecrets(func(params map[string]string) map[string]func() (providers.SecretsProvider, error) {
124 | return map[string]func() (providers.SecretsProvider, error){
125 | "fake": func() (providers.SecretsProvider, error) {
126 | return &fakeProvider{}, nil
127 | },
128 | }
129 | })
130 | assert.Error(t, err)
131 | }
132 |
133 | func TestProviderRegistry(t *testing.T) {
134 | registry := ProviderRegistry(map[string]string{})
135 | assert.NotEmpty(t, registry)
136 | }
137 |
138 | func TestSecretGenerator_Generate(t *testing.T) {
139 | s := testSG.Generate(map[string]string{
140 | "test-key": "test-value",
141 | })
142 | assert.Equal(t, base64.StdEncoding.EncodeToString([]byte("test-value")), s.Data["test-key"])
143 | }
144 |
--------------------------------------------------------------------------------
/pkg/utils/utils.go:
--------------------------------------------------------------------------------
1 | package utils
2 |
3 | import log "github.com/sirupsen/logrus"
4 |
5 | func Merge(m1, m2 map[string]string) map[string]string {
6 | for k, v := range m2 {
7 | m1[k] = v
8 | }
9 | return m1
10 | }
11 |
12 | func Map(m map[string]string, f func(string) string) map[string]string {
13 | res := make(map[string]string)
14 | for k, v := range m {
15 | res[k] = f(v)
16 | }
17 | return res
18 | }
19 |
20 | func FatalErrCheck(err error) {
21 | if err != nil {
22 | log.Fatal(err)
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/pkg/utils/utils_test.go:
--------------------------------------------------------------------------------
1 | package utils
2 |
3 | import (
4 | "errors"
5 | "github.com/sirupsen/logrus"
6 | "github.com/stretchr/testify/assert"
7 | "testing"
8 | )
9 |
10 | func TestMap(t *testing.T) {
11 | m := map[string]string{
12 | "k": "v",
13 | }
14 | res := Map(m, func(s string) string {
15 | return s
16 | })
17 | assert.Equal(t, res, m)
18 | }
19 |
20 | func TestMerge(t *testing.T) {
21 | m1 := map[string]string{
22 | "k": "v",
23 | }
24 | m2 := map[string]string{
25 | "k2": "v2",
26 | }
27 | res := Merge(m1, m2)
28 | assert.Equal(t, res, map[string]string{
29 | "k": "v",
30 | "k2": "v2",
31 | })
32 | }
33 |
34 | func TestFatalErrCheck(t *testing.T) {
35 | exited := false
36 | logrus.StandardLogger().ExitFunc = func(i int) {
37 | exited = true
38 | }
39 | FatalErrCheck(errors.New("fatal error"))
40 | assert.True(t, exited)
41 |
42 | exited = false
43 | FatalErrCheck(nil)
44 | assert.False(t, exited)
45 | }
46 |
--------------------------------------------------------------------------------