├── .vscode
├── launch.json
└── tasks.json
├── README.MD
├── _img
├── broken_contract.png
├── contracts.png
├── debugging.png
├── graph.png
└── verified_contracts.png
├── _pacts
├── price-service-product-service.json
└── search-service-product-service.json
├── _postman
└── CDC_Testing.postman_collection.json
├── consumer-price-service
├── .gitignore
├── go.mod
├── go.sum
├── src
│ ├── api.go
│ ├── dto
│ │ ├── productDto.go
│ │ └── productPriceDto.go
│ ├── httpclient
│ │ └── product_http_client.go
│ └── main.go
└── test
│ └── product_contract_test.go
├── consumer-search-service
├── .gitignore
├── go.mod
├── go.sum
├── src
│ ├── api.go
│ ├── dto
│ │ ├── productDto.go
│ │ └── productSearchDto.go
│ ├── httpclient
│ │ └── product_http_client.go
│ └── main.go
└── test
│ └── product_contract_test.go
├── docker-compose.yml
└── provider-product-service
├── .gitignore
├── provider-product-service.sln
├── src
├── Controllers
│ └── ProductsController.cs
├── Data
│ ├── DbInitilializer.cs
│ ├── Entity
│ │ └── Product.cs
│ ├── Migrations
│ │ ├── 20210408200516_initial.Designer.cs
│ │ ├── 20210408200516_initial.cs
│ │ └── ProductDBContextModelSnapshot.cs
│ ├── ProductDBContext.cs
│ └── TypeConfiguration
│ │ └── ProductTypeConfiguration.cs
├── Dto
│ └── ProductDto.cs
├── Program.cs
├── Properties
│ └── launchSettings.json
├── Provider.csproj
└── appsettings.json
└── test
├── ProviderApiTests.csproj
├── ProviderContractTests.cs
├── ProviderStateMiddleware.cs
├── TestStartup.cs
└── XUnitHelpers
└── XunitOutputter.cs
/.vscode/launch.json:
--------------------------------------------------------------------------------
1 | {
2 | "version": "0.2.0",
3 | "configurations": [
4 | {
5 | "name": "provider",
6 | "type": "coreclr",
7 | "request": "launch",
8 | "preLaunchTask": "build",
9 | "program": "${workspaceFolder}/provider-product-service/src/bin/Debug/net6.0/Provider.dll",
10 | "args": [],
11 | "cwd": "${workspaceFolder}/provider-product-service/src",
12 | "stopAtEntry": false,
13 | "serverReadyAction": {
14 | "action": "openExternally",
15 | "pattern": "\\\\bNow listening on:\\\\s+(https?://\\\\S+)"
16 | },
17 | "env": {
18 | "ASPNETCORE_ENVIRONMENT": "Development"
19 | },
20 | "sourceFileMap": {
21 | "/Views": "${workspaceFolder}/Views"
22 | }
23 | },
24 | {
25 | "name": "consumer-price",
26 | "type": "go",
27 | "request": "launch",
28 | "mode": "debug",
29 | "program": "${workspaceFolder}/consumer-price-service/src",
30 | },
31 | {
32 | "name": "consumer-search",
33 | "type": "go",
34 | "request": "launch",
35 | "mode": "debug",
36 | "program": "${workspaceFolder}/consumer-search-service/src",
37 | },
38 | ],
39 | "compounds": [
40 | {
41 | "name": "All",
42 | "configurations": [
43 | "provider",
44 | "consumer-price",
45 | "consumer-search"
46 | ]
47 | }
48 | ]
49 | }
--------------------------------------------------------------------------------
/.vscode/tasks.json:
--------------------------------------------------------------------------------
1 | {
2 | "version": "2.0.0",
3 | "tasks": [
4 | {
5 | "label": "build",
6 | "command": "dotnet",
7 | "type": "process",
8 | "args": [
9 | "build",
10 | "${workspaceFolder}/provider-product-service/src/Provider.csproj",
11 | "/property:GenerateFullPaths=true",
12 | "/consoleloggerparameters:NoSummary"
13 | ],
14 | "problemMatcher": "$msCompile"
15 | },
16 | {
17 | "label": "publish",
18 | "command": "dotnet",
19 | "type": "process",
20 | "args": [
21 | "publish",
22 | "${workspaceFolder}/provider-product-service/src/Provider.csproj",
23 | "/property:GenerateFullPaths=true",
24 | "/consoleloggerparameters:NoSummary"
25 | ],
26 | "problemMatcher": "$msCompile"
27 | },
28 | {
29 | "label": "watch",
30 | "command": "dotnet",
31 | "type": "process",
32 | "args": [
33 | "watch",
34 | "run",
35 | "${workspaceFolder}/provider-product-service/src/Provider.csproj",
36 | "/property:GenerateFullPaths=true",
37 | "/consoleloggerparameters:NoSummary"
38 | ],
39 | "problemMatcher": "$msCompile"
40 | }
41 | ]
42 | }
--------------------------------------------------------------------------------
/README.MD:
--------------------------------------------------------------------------------
1 | ## Why?
2 |
3 | E2E testing is one of the biggest challenges in distributed systems. Consumer-Driven Contract Testing is an excellent alternative to deal with this challenge.
4 |
5 |
6 | This workshop shows how to implement Consumer-Driven Contract Testing between the microservices which are developed in different tech stacks.
7 |
8 | Pact framework is used to create and verify contracts between the microservices.
9 |
10 | There are 3 microservices that communicate syncronously each other.
11 |
12 | **Porovider Service** -> Product Service (.Net5)
13 |
14 | **Consumer Service(1)** -> Search Service (Go 1.16)
15 |
16 | **Consumer Service(2)** -> Price Service (Go 1.16)
17 |
18 | ## Run in Debug Mode
19 |
20 | You need Pact Broker and Postgres container to play with the. workshop. First, run ```docker-compose up``` in the root path.
21 |
22 | Then, Select 'All' Debug Configuration and start debugging. (for Vs Code users) All microservices will be running in debug mode.
23 |
24 |
25 |
26 | You may want to play with the services before creating and verifying contracts. See sample postman requests here.
27 |
28 |
29 | ## Creating Contracts
30 |
31 | After running all services, you need to publish your consumer-driven contracts. Run contract tests for Search and Price services;
32 |
33 | ```
34 | cd consumer-price-service/test
35 | go test
36 | ```
37 |
38 | ```
39 | cd consumer-search-service/test
40 | go test
41 | ```
42 |
43 | After running these tests successfully, you will see the unverified contracts on Pact Broker home page which is hosting on http://localhost:9292
44 |
45 |
46 |
47 | And, you can also see the network graph between the services by clicking on any of them.
48 |
49 |
50 |
51 |
52 | ## Verifying Contracts
53 |
54 | You have 2 contracts that were published by 2 consumer services. To verify these contracts, just run the provider's contract test like below.
55 |
56 | ```
57 | cd provider-product-service/test
58 | dotnet test
59 | ```
60 |
61 | If the test is passed, you will see verified contracts like below. Pay attention to the "Last Verified" column;
62 |
63 |
64 |
65 | ## Next?
66 |
67 | Try to break contracts between the services. Pact Broker shows the broken contracts with red background.
68 |
69 | Following Pact list means;
70 |
71 | 'Product Service just made a change that breaks the contract with Price Service. Search service is not affected by this change.'
72 |
73 |
74 |
75 |
--------------------------------------------------------------------------------
/_img/broken_contract.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suadev/consumer-driven-contract-testing-microservices/2061ac67dc34e0e762bd845364acf3f75a429cde/_img/broken_contract.png
--------------------------------------------------------------------------------
/_img/contracts.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suadev/consumer-driven-contract-testing-microservices/2061ac67dc34e0e762bd845364acf3f75a429cde/_img/contracts.png
--------------------------------------------------------------------------------
/_img/debugging.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suadev/consumer-driven-contract-testing-microservices/2061ac67dc34e0e762bd845364acf3f75a429cde/_img/debugging.png
--------------------------------------------------------------------------------
/_img/graph.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suadev/consumer-driven-contract-testing-microservices/2061ac67dc34e0e762bd845364acf3f75a429cde/_img/graph.png
--------------------------------------------------------------------------------
/_img/verified_contracts.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suadev/consumer-driven-contract-testing-microservices/2061ac67dc34e0e762bd845364acf3f75a429cde/_img/verified_contracts.png
--------------------------------------------------------------------------------
/_pacts/price-service-product-service.json:
--------------------------------------------------------------------------------
1 | {
2 | "consumer": {
3 | "name": "price-service"
4 | },
5 | "provider": {
6 | "name": "product-service"
7 | },
8 | "interactions": [
9 | {
10 | "description": "A GET request to retrieve products.",
11 | "providerState": "There are available products",
12 | "request": {
13 | "method": "GET",
14 | "path": "/products",
15 | "matchingRules": {
16 | "$.path": {
17 | "match": "regex",
18 | "regex": "\\/products"
19 | }
20 | }
21 | },
22 | "response": {
23 | "status": 200,
24 | "headers": {
25 | "Content-Type": "application/json; charset=utf-8"
26 | },
27 | "body": [
28 | {
29 | "ID": 1,
30 | "Name": "Lorem Ipsum",
31 | "IsActive": true
32 | }
33 | ],
34 | "matchingRules": {
35 | "$.headers.Content-Type": {
36 | "match": "regex",
37 | "regex": "application\\/json"
38 | },
39 | "$.body": {
40 | "match": "type"
41 | }
42 | }
43 | }
44 | },
45 | {
46 | "description": "A GET request to retrieve single product.",
47 | "providerState": "There is a single product for a given product id",
48 | "request": {
49 | "method": "GET",
50 | "path": "/products/1",
51 | "matchingRules": {
52 | "$.path": {
53 | "match": "regex",
54 | "regex": "\\/products\\/[0-9]+"
55 | }
56 | }
57 | },
58 | "response": {
59 | "status": 200,
60 | "headers": {
61 | "Content-Type": "application/json; charset=utf-8"
62 | },
63 | "body": {
64 | "ID": 1,
65 | "Name": "Lorem Ipsum",
66 | "IsActive": true
67 | },
68 | "matchingRules": {
69 | "$.headers.Content-Type": {
70 | "match": "regex",
71 | "regex": "application\\/json"
72 | },
73 | "$.body": {
74 | "match": "type"
75 | }
76 | }
77 | }
78 | }
79 | ],
80 | "metadata": {
81 | "pactSpecification": {
82 | "version": "2.0.0"
83 | }
84 | }
85 | }
--------------------------------------------------------------------------------
/_pacts/search-service-product-service.json:
--------------------------------------------------------------------------------
1 | {
2 | "consumer": {
3 | "name": "search-service"
4 | },
5 | "provider": {
6 | "name": "product-service"
7 | },
8 | "interactions": [
9 | {
10 | "description": "A GET request to search products.",
11 | "providerState": "There are available products for a given keyword.",
12 | "request": {
13 | "method": "GET",
14 | "path": "/products/search",
15 | "query": "keyword=foo",
16 | "matchingRules": {
17 | "$.path": {
18 | "match": "regex",
19 | "regex": "\\/products\\/search"
20 | },
21 | "$.query.keyword[0]": {
22 | "match": "regex",
23 | "regex": "[a-zA-Z]+"
24 | }
25 | }
26 | },
27 | "response": {
28 | "status": 200,
29 | "headers": {
30 | "Content-Type": "application/json; charset=utf-8"
31 | },
32 | "body": [
33 | {
34 | "ID": 1,
35 | "Name": "Lorem Ipsum"
36 | }
37 | ],
38 | "matchingRules": {
39 | "$.headers.Content-Type": {
40 | "match": "regex",
41 | "regex": "application\\/json"
42 | },
43 | "$.body": {
44 | "match": "type"
45 | }
46 | }
47 | }
48 | }
49 | ],
50 | "metadata": {
51 | "pactSpecification": {
52 | "version": "2.0.0"
53 | }
54 | }
55 | }
--------------------------------------------------------------------------------
/_postman/CDC_Testing.postman_collection.json:
--------------------------------------------------------------------------------
1 | {
2 | "info": {
3 | "_postman_id": "2d4e3217-3933-4f07-b696-97f202029281",
4 | "name": "CDC_Testing",
5 | "schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json"
6 | },
7 | "item": [
8 | {
9 | "name": "Provider",
10 | "item": [
11 | {
12 | "name": "provider-get-products",
13 | "request": {
14 | "auth": {
15 | "type": "bearer",
16 | "bearer": [
17 | {
18 | "key": "token",
19 | "value": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1bmlxdWVfbmFtZSI6InN1YWRldkBnbWFpbC5jb20iLCJ1c2VyX2lkIjoiMjJkMWQwYWUtYWEyNi00Y2JiLTlhZWMtNzY0MDE4MGZmNDViIiwibmJmIjoxNjE1NTc0NzY1LCJleHAiOjE2MTU2NjExNjUsImlhdCI6MTYxNTU3NDc2NSwiaXNzIjoibG9jYWxob3N0IiwiYXVkIjoiU3VhdCBLw5ZTRSJ9.qLV4qNYZM_ouyWulEPBEaNy2Bg3aK7lLnLLoKkQ5fBM",
20 | "type": "string"
21 | }
22 | ]
23 | },
24 | "method": "GET",
25 | "header": [],
26 | "url": {
27 | "raw": "http://localhost:3001/api/products",
28 | "protocol": "http",
29 | "host": [
30 | "localhost"
31 | ],
32 | "port": "3001",
33 | "path": [
34 | "api",
35 | "products"
36 | ]
37 | }
38 | },
39 | "response": []
40 | },
41 | {
42 | "name": "provider-get-single-product",
43 | "request": {
44 | "auth": {
45 | "type": "bearer",
46 | "bearer": [
47 | {
48 | "key": "token",
49 | "value": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1bmlxdWVfbmFtZSI6InN1YWRldkBnbWFpbC5jb20iLCJ1c2VyX2lkIjoiMjJkMWQwYWUtYWEyNi00Y2JiLTlhZWMtNzY0MDE4MGZmNDViIiwibmJmIjoxNjE1NTc0NzY1LCJleHAiOjE2MTU2NjExNjUsImlhdCI6MTYxNTU3NDc2NSwiaXNzIjoibG9jYWxob3N0IiwiYXVkIjoiU3VhdCBLw5ZTRSJ9.qLV4qNYZM_ouyWulEPBEaNy2Bg3aK7lLnLLoKkQ5fBM",
50 | "type": "string"
51 | }
52 | ]
53 | },
54 | "method": "GET",
55 | "header": [],
56 | "url": {
57 | "raw": "http://localhost:3001/api/products/1",
58 | "protocol": "http",
59 | "host": [
60 | "localhost"
61 | ],
62 | "port": "3001",
63 | "path": [
64 | "api",
65 | "products",
66 | "1"
67 | ]
68 | }
69 | },
70 | "response": []
71 | },
72 | {
73 | "name": "provider-search-products",
74 | "request": {
75 | "auth": {
76 | "type": "bearer",
77 | "bearer": [
78 | {
79 | "key": "token",
80 | "value": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1bmlxdWVfbmFtZSI6InN1YWRldkBnbWFpbC5jb20iLCJ1c2VyX2lkIjoiMjJkMWQwYWUtYWEyNi00Y2JiLTlhZWMtNzY0MDE4MGZmNDViIiwibmJmIjoxNjE1NTc0NzY1LCJleHAiOjE2MTU2NjExNjUsImlhdCI6MTYxNTU3NDc2NSwiaXNzIjoibG9jYWxob3N0IiwiYXVkIjoiU3VhdCBLw5ZTRSJ9.qLV4qNYZM_ouyWulEPBEaNy2Bg3aK7lLnLLoKkQ5fBM",
81 | "type": "string"
82 | }
83 | ]
84 | },
85 | "method": "GET",
86 | "header": [],
87 | "url": {
88 | "raw": "http://localhost:3001/api/products/search?keyword=mo",
89 | "protocol": "http",
90 | "host": [
91 | "localhost"
92 | ],
93 | "port": "3001",
94 | "path": [
95 | "api",
96 | "products",
97 | "search"
98 | ],
99 | "query": [
100 | {
101 | "key": "keyword",
102 | "value": "mo"
103 | }
104 | ]
105 | }
106 | },
107 | "response": []
108 | }
109 | ]
110 | },
111 | {
112 | "name": "Consumer",
113 | "item": [
114 | {
115 | "name": "consumer-price_get-product-price",
116 | "request": {
117 | "auth": {
118 | "type": "bearer",
119 | "bearer": [
120 | {
121 | "key": "token",
122 | "value": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1bmlxdWVfbmFtZSI6InN1YWRldkBnbWFpbC5jb20iLCJ1c2VyX2lkIjoiMjJkMWQwYWUtYWEyNi00Y2JiLTlhZWMtNzY0MDE4MGZmNDViIiwibmJmIjoxNjE1NTc0NzY1LCJleHAiOjE2MTU2NjExNjUsImlhdCI6MTYxNTU3NDc2NSwiaXNzIjoibG9jYWxob3N0IiwiYXVkIjoiU3VhdCBLw5ZTRSJ9.qLV4qNYZM_ouyWulEPBEaNy2Bg3aK7lLnLLoKkQ5fBM",
123 | "type": "string"
124 | }
125 | ]
126 | },
127 | "method": "GET",
128 | "header": [],
129 | "url": {
130 | "raw": "http://localhost:3002/api/product-prices",
131 | "protocol": "http",
132 | "host": [
133 | "localhost"
134 | ],
135 | "port": "3002",
136 | "path": [
137 | "api",
138 | "product-prices"
139 | ]
140 | }
141 | },
142 | "response": []
143 | },
144 | {
145 | "name": "consumer-price_get-single-product-price",
146 | "request": {
147 | "auth": {
148 | "type": "bearer",
149 | "bearer": [
150 | {
151 | "key": "token",
152 | "value": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1bmlxdWVfbmFtZSI6InN1YWRldkBnbWFpbC5jb20iLCJ1c2VyX2lkIjoiMjJkMWQwYWUtYWEyNi00Y2JiLTlhZWMtNzY0MDE4MGZmNDViIiwibmJmIjoxNjE1NTc0NzY1LCJleHAiOjE2MTU2NjExNjUsImlhdCI6MTYxNTU3NDc2NSwiaXNzIjoibG9jYWxob3N0IiwiYXVkIjoiU3VhdCBLw5ZTRSJ9.qLV4qNYZM_ouyWulEPBEaNy2Bg3aK7lLnLLoKkQ5fBM",
153 | "type": "string"
154 | }
155 | ]
156 | },
157 | "method": "GET",
158 | "header": [],
159 | "url": {
160 | "raw": "http://localhost:3002/api/product-price/1",
161 | "protocol": "http",
162 | "host": [
163 | "localhost"
164 | ],
165 | "port": "3002",
166 | "path": [
167 | "api",
168 | "product-price",
169 | "1"
170 | ]
171 | }
172 | },
173 | "response": []
174 | },
175 | {
176 | "name": "consumer-search_search-products",
177 | "request": {
178 | "auth": {
179 | "type": "bearer",
180 | "bearer": [
181 | {
182 | "key": "token",
183 | "value": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1bmlxdWVfbmFtZSI6InN1YWRldkBnbWFpbC5jb20iLCJ1c2VyX2lkIjoiMjJkMWQwYWUtYWEyNi00Y2JiLTlhZWMtNzY0MDE4MGZmNDViIiwibmJmIjoxNjE1NTc0NzY1LCJleHAiOjE2MTU2NjExNjUsImlhdCI6MTYxNTU3NDc2NSwiaXNzIjoibG9jYWxob3N0IiwiYXVkIjoiU3VhdCBLw5ZTRSJ9.qLV4qNYZM_ouyWulEPBEaNy2Bg3aK7lLnLLoKkQ5fBM",
184 | "type": "string"
185 | }
186 | ]
187 | },
188 | "method": "GET",
189 | "header": [],
190 | "url": {
191 | "raw": "http://localhost:3003/api/products/search?keyword=mo",
192 | "protocol": "http",
193 | "host": [
194 | "localhost"
195 | ],
196 | "port": "3003",
197 | "path": [
198 | "api",
199 | "products",
200 | "search"
201 | ],
202 | "query": [
203 | {
204 | "key": "keyword",
205 | "value": "mo"
206 | }
207 | ]
208 | }
209 | },
210 | "response": []
211 | }
212 | ]
213 | }
214 | ]
215 | }
--------------------------------------------------------------------------------
/consumer-price-service/.gitignore:
--------------------------------------------------------------------------------
1 | # Binaries for programs and plugins
2 | *.exe
3 | *.exe~
4 | *.dll
5 | *.so
6 | *.dylib
7 |
8 | # Test binary, built with `go test -c`
9 | *.test
10 |
11 | # Output of the go coverage tool, specifically when used with LiteIDE
12 | *.out
13 |
14 | # Dependency directories (remove the comment below to include it)
15 | # vendor/
16 |
17 | __debug_bin
18 |
19 |
20 | _pact_consumer_logs/
21 |
22 | .DS_Store
--------------------------------------------------------------------------------
/consumer-price-service/go.mod:
--------------------------------------------------------------------------------
1 | module cdc-testing-workshop/consumer-price-service
2 |
3 | go 1.16
4 |
5 | require (
6 | github.com/gin-gonic/gin v1.7.7
7 | github.com/golang/protobuf v1.5.3 // indirect
8 | github.com/google/go-cmp v0.5.9 // indirect
9 | github.com/hashicorp/go-version v1.6.0 // indirect
10 | github.com/kr/pretty v0.3.0 // indirect
11 | github.com/mattn/go-isatty v0.0.17 // indirect
12 | github.com/pact-foundation/pact-go v1.7.0
13 | github.com/rogpeppe/go-internal v1.9.0 // indirect
14 | github.com/stretchr/testify v1.8.3 // indirect
15 | golang.org/x/crypto v0.11.0 // indirect
16 | google.golang.org/protobuf v1.31.0 // indirect
17 | gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect
18 | )
19 |
--------------------------------------------------------------------------------
/consumer-price-service/go.sum:
--------------------------------------------------------------------------------
1 | cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
2 | cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
3 | cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
4 | cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU=
5 | cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY=
6 | cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc=
7 | cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0=
8 | cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o=
9 | cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE=
10 | cloud.google.com/go/firestore v1.1.0/go.mod h1:ulACoGHTpvq5r8rxGJ4ddJZBZqakUQqClKRT5SZwBmk=
11 | cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I=
12 | cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw=
13 | dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
14 | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
15 | github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
16 | github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
17 | github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
18 | github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
19 | github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o=
20 | github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY=
21 | github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8=
22 | github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
23 | github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
24 | github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs=
25 | github.com/bketelsen/crypt v0.0.3-0.20200106085610-5cbc8cc4026c/go.mod h1:MKsuJmJgSg28kpZDP6UIiPt0e0Oz0kqKNGyRaWEPv84=
26 | github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
27 | github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
28 | github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
29 | github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk=
30 | github.com/coreos/etcd v3.3.13+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
31 | github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
32 | github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
33 | github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
34 | github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
35 | github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
36 | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
37 | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
38 | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
39 | github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
40 | github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no=
41 | github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
42 | github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
43 | github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
44 | github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
45 | github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
46 | github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
47 | github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
48 | github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M=
49 | github.com/gin-gonic/gin v1.7.7 h1:3DoBmSbJbZAWqXJC3SLjAPfutPJJRN1U5pALB7EeTTs=
50 | github.com/gin-gonic/gin v1.7.7/go.mod h1:axIBovoeJpVj8S3BwE0uPMTeReE4+AfFtqpqaZ1qq1U=
51 | github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU=
52 | github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
53 | github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
54 | github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
55 | github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A=
56 | github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
57 | github.com/go-playground/locales v0.13.0 h1:HyWk6mgj5qFqCT5fjGBuRArbVDfE4hi8+e8ceBS/t7Q=
58 | github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8=
59 | github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD876Lmtgy7VtROAbHHXk8no=
60 | github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA=
61 | github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI=
62 | github.com/go-playground/validator/v10 v10.4.1 h1:pH2c5ADXtd66mxoE0Zm9SUhxE20r7aM3F26W0hOn+GE=
63 | github.com/go-playground/validator/v10 v10.4.1/go.mod h1:nlOn6nFhuKACm19sB/8EGNn9GlaMV7XkbRSipzJ0Ii4=
64 | github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
65 | github.com/gobwas/httphead v0.0.0-20180130184737-2c6c146eadee/go.mod h1:L0fX3K22YWvt/FAX9NnzrNzcI4wNYi9Yku4O0LKYflo=
66 | github.com/gobwas/pool v0.2.0/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw=
67 | github.com/gobwas/ws v1.0.2/go.mod h1:szmBTxLgaFppYjEmNtny/v3w89xOydFnnZMcgRRu/EM=
68 | github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
69 | github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4=
70 | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
71 | github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
72 | github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
73 | github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
74 | github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y=
75 | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
76 | github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
77 | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
78 | github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
79 | github.com/golang/protobuf v1.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk=
80 | github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
81 | github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
82 | github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
83 | github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
84 | github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
85 | github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
86 | github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
87 | github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
88 | github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg=
89 | github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
90 | github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
91 | github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
92 | github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
93 | github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
94 | github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
95 | github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
96 | github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
97 | github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
98 | github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
99 | github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
100 | github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
101 | github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
102 | github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
103 | github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
104 | github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
105 | github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
106 | github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
107 | github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
108 | github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY=
109 | github.com/gorilla/websocket v1.4.1/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
110 | github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
111 | github.com/graph-gophers/graphql-go v1.2.0/go.mod h1:9CQHMSxwO4MprSdzoIEobiHpoLtHm77vfxsvsIN5Vuc=
112 | github.com/graph-gophers/graphql-go v1.3.0/go.mod h1:9CQHMSxwO4MprSdzoIEobiHpoLtHm77vfxsvsIN5Vuc=
113 | github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
114 | github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
115 | github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
116 | github.com/hashicorp/consul/api v1.1.0/go.mod h1:VmuI/Lkw1nC05EYQWNKwWGbkg+FbDBtguAZLlVdkD9Q=
117 | github.com/hashicorp/consul/sdk v0.1.1/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8=
118 | github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
119 | github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80=
120 | github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60=
121 | github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM=
122 | github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk=
123 | github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU=
124 | github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU=
125 | github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4=
126 | github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
127 | github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
128 | github.com/hashicorp/go-version v1.5.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
129 | github.com/hashicorp/go-version v1.6.0 h1:feTTfFNnjP967rlCxM/I9g701jU+RN74YKx2mOkIeek=
130 | github.com/hashicorp/go-version v1.6.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
131 | github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90=
132 | github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
133 | github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
134 | github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
135 | github.com/hashicorp/logutils v1.0.0 h1:dLEQVugN8vlakKOUE3ihGLTZJRB4j+M2cdTm/ORI65Y=
136 | github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64=
137 | github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ=
138 | github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I=
139 | github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc=
140 | github.com/hasura/go-graphql-client v0.6.3/go.mod h1:kvaJsDhxGbkIJ1jgebkrnt9EDIELZHpsAMint56v+2I=
141 | github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
142 | github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo=
143 | github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
144 | github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
145 | github.com/json-iterator/go v1.1.10 h1:Kz6Cvnvv2wGdaG/V8yMvfkmNiXq9Ya2KUv4rouJJr68=
146 | github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
147 | github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
148 | github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU=
149 | github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
150 | github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q=
151 | github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
152 | github.com/klauspost/compress v1.10.3/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs=
153 | github.com/klauspost/compress v1.14.2/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
154 | github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
155 | github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc=
156 | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
157 | github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
158 | github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0=
159 | github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk=
160 | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
161 | github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
162 | github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
163 | github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
164 | github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
165 | github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
166 | github.com/magiconair/properties v1.8.1/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ=
167 | github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
168 | github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
169 | github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
170 | github.com/mattn/go-isatty v0.0.17 h1:BTarxUcIeDqL27Mc+vyvdWYSL28zpIhv3RoTdsLMPng=
171 | github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
172 | github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
173 | github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg=
174 | github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc=
175 | github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
176 | github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
177 | github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
178 | github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg=
179 | github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY=
180 | github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
181 | github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
182 | github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
183 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
184 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
185 | github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
186 | github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
187 | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
188 | github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
189 | github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U=
190 | github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o=
191 | github.com/pact-foundation/pact-go v1.7.0 h1:5iyVyg+avkWz9Jn7cefRmlPbXu+KMZvWblIe15v4fc8=
192 | github.com/pact-foundation/pact-go v1.7.0/go.mod h1:NcAbRqIE0cjRF+JKl2vcLlzjvrgcZrnq4SwQu2o4PeA=
193 | github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc=
194 | github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
195 | github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
196 | github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
197 | github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
198 | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
199 | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
200 | github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI=
201 | github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
202 | github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso=
203 | github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
204 | github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
205 | github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
206 | github.com/prometheus/common v0.0.0-20181113130724-41aa239b4cce/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
207 | github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
208 | github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
209 | github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
210 | github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
211 | github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
212 | github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
213 | github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
214 | github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8=
215 | github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs=
216 | github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
217 | github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts=
218 | github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc=
219 | github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
220 | github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
221 | github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc=
222 | github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA=
223 | github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM=
224 | github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA=
225 | github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ=
226 | github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
227 | github.com/spf13/cobra v1.1.3/go.mod h1:pGADOWyqRD/YMrPZigI/zbliZ2wVD/23d+is3pSWzOo=
228 | github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo=
229 | github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
230 | github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
231 | github.com/spf13/viper v1.7.0/go.mod h1:8WkrPz2fc9jxqZNCJI/76HCieCp4Q8HaLFoCha5qpdg=
232 | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
233 | github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
234 | github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
235 | github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
236 | github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
237 | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
238 | github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
239 | github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
240 | github.com/stretchr/testify v1.7.4/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
241 | github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
242 | github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY=
243 | github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
244 | github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw=
245 | github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
246 | github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo=
247 | github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
248 | github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
249 | github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
250 | github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
251 | github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
252 | go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
253 | go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
254 | go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
255 | go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
256 | go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
257 | go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
258 | golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
259 | golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
260 | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
261 | golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
262 | golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
263 | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
264 | golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
265 | golang.org/x/crypto v0.11.0 h1:6Ewdq3tDic1mg5xRO4milcWCfMVQhI4NkqWWvqejpuA=
266 | golang.org/x/crypto v0.11.0/go.mod h1:xgJhtzW8F9jGdVFWZESrid1U1bjeNy4zgy5cRr/CIio=
267 | golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
268 | golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
269 | golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
270 | golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek=
271 | golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY=
272 | golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
273 | golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
274 | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
275 | golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
276 | golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
277 | golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
278 | golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
279 | golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
280 | golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
281 | golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE=
282 | golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o=
283 | golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
284 | golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY=
285 | golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
286 | golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
287 | golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
288 | golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
289 | golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
290 | golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
291 | golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
292 | golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
293 | golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
294 | golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
295 | golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
296 | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
297 | golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
298 | golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
299 | golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
300 | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
301 | golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
302 | golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
303 | golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
304 | golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg=
305 | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
306 | golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
307 | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
308 | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
309 | golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
310 | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
311 | golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
312 | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
313 | golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
314 | golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
315 | golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
316 | golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
317 | golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
318 | golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
319 | golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
320 | golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
321 | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
322 | golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
323 | golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
324 | golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
325 | golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
326 | golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
327 | golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
328 | golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
329 | golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
330 | golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
331 | golang.org/x/sys v0.0.0-20211019181941-9d821ace8654/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
332 | golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
333 | golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
334 | golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
335 | golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
336 | golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
337 | golang.org/x/sys v0.10.0 h1:SqMFp9UcQJZa+pmYuAKjd9xq1f0j5rLcDIk0mj4qAsA=
338 | golang.org/x/sys v0.10.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
339 | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
340 | golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
341 | golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
342 | golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo=
343 | golang.org/x/term v0.10.0/go.mod h1:lpqdcUyK/oCiQxvxVrppt5ggO2KCZ5QblwqPnfZ6d5o=
344 | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
345 | golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
346 | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
347 | golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
348 | golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
349 | golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
350 | golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
351 | golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
352 | golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
353 | golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
354 | golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
355 | golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
356 | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
357 | golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
358 | golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
359 | golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
360 | golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
361 | golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
362 | golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
363 | golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
364 | golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
365 | golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
366 | golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
367 | golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
368 | golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
369 | golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
370 | golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
371 | golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
372 | golang.org/x/tools v0.0.0-20191112195655-aa38f8e97acc/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
373 | golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
374 | golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
375 | golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
376 | golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
377 | golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
378 | golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
379 | google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
380 | google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M=
381 | google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
382 | google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
383 | google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI=
384 | google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
385 | google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
386 | google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
387 | google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0=
388 | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
389 | google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
390 | google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
391 | google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
392 | google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
393 | google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
394 | google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
395 | google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8=
396 | google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
397 | google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
398 | google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
399 | google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
400 | google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
401 | google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
402 | google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
403 | google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
404 | google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
405 | google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
406 | google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
407 | google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
408 | google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
409 | google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
410 | google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
411 | google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
412 | google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
413 | google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
414 | google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8=
415 | google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
416 | gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
417 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
418 | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
419 | gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
420 | gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
421 | gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
422 | gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
423 | gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo=
424 | gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
425 | gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
426 | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
427 | gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
428 | gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
429 | gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
430 | gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
431 | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
432 | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
433 | gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
434 | honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
435 | honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
436 | honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
437 | honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
438 | honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
439 | nhooyr.io/websocket v1.8.7/go.mod h1:B70DZP8IakI65RVQ51MsWP/8jndNma26DVA/nFSCgW0=
440 | rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
441 |
--------------------------------------------------------------------------------
/consumer-price-service/src/api.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "math/rand"
5 | "net/http"
6 | "strconv"
7 |
8 | "cdc-testing-workshop/consumer-price-service/src/dto"
9 | "cdc-testing-workshop/consumer-price-service/src/httpclient"
10 |
11 | "github.com/gin-gonic/gin"
12 | )
13 |
14 | func getProductPrices(context *gin.Context) {
15 | client := httpclient.NewProductServiceClient()
16 | products, err := client.GetProducts()
17 | if err != nil {
18 | context.JSON(http.StatusBadRequest, err.Error())
19 | return
20 | }
21 |
22 | result := make([]dto.ProductPriceDto, 0, len(products))
23 | for _, product := range products {
24 | result = append(result, dto.ProductPriceDto{
25 | ID: product.ID,
26 | Name: product.Name,
27 | IsActive: product.IsActive,
28 | Price: 100 + rand.Float64()*(500-100), // random price between 100-500
29 | })
30 | }
31 | context.JSON(http.StatusOK, result)
32 | }
33 |
34 | func getProductPrice(context *gin.Context) {
35 | productID, _ := strconv.Atoi(context.Param("id"))
36 |
37 | client := httpclient.NewProductServiceClient()
38 | product, err := client.GetProduct(productID)
39 | if err != nil {
40 | context.JSON(http.StatusBadRequest, err.Error())
41 | return
42 | }
43 |
44 | result := dto.ProductPriceDto{
45 | ID: product.ID,
46 | Name: product.Name,
47 | IsActive: product.IsActive,
48 | Price: 100 + rand.Float64()*(500-100), // random price between 100-500
49 | }
50 | context.JSON(http.StatusOK, result)
51 | }
52 |
--------------------------------------------------------------------------------
/consumer-price-service/src/dto/productDto.go:
--------------------------------------------------------------------------------
1 | package dto
2 |
3 | type ProductDto struct {
4 | ID int
5 | Name string
6 | IsActive bool
7 | }
8 |
--------------------------------------------------------------------------------
/consumer-price-service/src/dto/productPriceDto.go:
--------------------------------------------------------------------------------
1 | package dto
2 |
3 | type ProductPriceDto struct {
4 | ID int
5 | Name string
6 | IsActive bool
7 | Price float64
8 | }
9 |
--------------------------------------------------------------------------------
/consumer-price-service/src/httpclient/product_http_client.go:
--------------------------------------------------------------------------------
1 | package httpclient
2 |
3 | import (
4 | "cdc-testing-workshop/consumer-price-service/src/dto"
5 | "encoding/json"
6 | "errors"
7 | "fmt"
8 | "io/ioutil"
9 | "net/http"
10 | )
11 |
12 | type ProductServiceClient struct {
13 | HostUrl string
14 | }
15 |
16 | func NewProductServiceClient() *ProductServiceClient {
17 | return &ProductServiceClient{
18 | HostUrl: "http://localhost:3001/api",
19 | }
20 | }
21 |
22 | func (c *ProductServiceClient) GetProducts() ([]dto.ProductDto, error) {
23 | resp, err := http.Get(c.HostUrl + "/products")
24 | if err != nil {
25 | return nil, err
26 | }
27 |
28 | defer resp.Body.Close()
29 | if resp.StatusCode != 200 {
30 | return nil, errors.New("failed getting products")
31 | }
32 |
33 | body, err := ioutil.ReadAll(resp.Body)
34 | if err != nil {
35 | return nil, err
36 | }
37 |
38 | var products []dto.ProductDto
39 | json.Unmarshal(body, &products)
40 | return products, nil
41 | }
42 |
43 | func (c *ProductServiceClient) GetProduct(id int) (dto.ProductDto, error) {
44 | resp, err := http.Get(fmt.Sprintf("%s/products/%d", c.HostUrl, id))
45 | if err != nil {
46 | return dto.ProductDto{}, err
47 | }
48 |
49 | defer resp.Body.Close()
50 | if resp.StatusCode != 200 {
51 | return dto.ProductDto{}, errors.New("failed getting single product")
52 | }
53 |
54 | body, err := ioutil.ReadAll(resp.Body)
55 | if err != nil {
56 | return dto.ProductDto{}, err
57 | }
58 |
59 | var product dto.ProductDto
60 | json.Unmarshal(body, &product)
61 | return product, nil
62 | }
63 |
--------------------------------------------------------------------------------
/consumer-price-service/src/main.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "net/http"
5 |
6 | "github.com/gin-gonic/gin"
7 | )
8 |
9 | func main() {
10 | handler := initRoutes()
11 | err := newHttpServer(handler).ListenAndServe()
12 | if err != nil {
13 | panic("Couldn't start the HTTP server.")
14 | }
15 | }
16 |
17 | func newHttpServer(handler http.Handler) *http.Server {
18 | return &http.Server{
19 | Addr: ":3002",
20 | Handler: handler,
21 | }
22 | }
23 |
24 | func initRoutes() *gin.Engine {
25 | router := gin.Default()
26 | routerGroup := router.Group("/api")
27 | routerGroup.GET("/product-prices", getProductPrices)
28 | routerGroup.GET("/product-price/:id", getProductPrice)
29 | return router
30 | }
31 |
--------------------------------------------------------------------------------
/consumer-price-service/test/product_contract_test.go:
--------------------------------------------------------------------------------
1 | package test
2 |
3 | import (
4 | "cdc-testing-workshop/consumer-price-service/src/dto"
5 | "cdc-testing-workshop/consumer-price-service/src/httpclient"
6 | "fmt"
7 | "log"
8 | "os"
9 | "path/filepath"
10 | "testing"
11 | "time"
12 |
13 | "github.com/pact-foundation/pact-go/dsl"
14 | "github.com/pact-foundation/pact-go/types"
15 | )
16 |
17 | var pact dsl.Pact
18 |
19 | func TestMain(m *testing.M) {
20 |
21 | setupMockServer()
22 |
23 | var exitCode = m.Run()
24 |
25 | pact.WritePact()
26 | pact.Teardown()
27 |
28 | err := publishPacts()
29 | if err != nil {
30 | log.Println("ERROR: ", err)
31 | os.Exit(1)
32 | }
33 | os.Exit(exitCode)
34 | }
35 |
36 | func publishPacts() error {
37 |
38 | publisher := dsl.Publisher{}
39 |
40 | var dir, _ = os.Getwd()
41 | var pactDir = fmt.Sprintf("%s/../../_pacts", dir)
42 |
43 | now := time.Now()
44 | consumerVersion := fmt.Sprintf("1.0.%02d.%02d.%02d.%02d", now.Day(), now.Month(), now.Hour(), now.Minute())
45 |
46 | return publisher.Publish(types.PublishRequest{
47 | PactURLs: []string{filepath.FromSlash(fmt.Sprintf("%s/price-service-product-service.json", pactDir))},
48 | ConsumerVersion: consumerVersion,
49 | PactBroker: "http://localhost:9292",
50 | BrokerUsername: "admin",
51 | BrokerPassword: "admin",
52 | })
53 | }
54 |
55 | var term = dsl.Term
56 |
57 | type request = dsl.Request
58 |
59 | func Test_Create_Get_Products_Contract(t *testing.T) {
60 | t.Run("Get Products...", func(t *testing.T) {
61 | productsResponse := [1]dto.ProductDto{
62 | {
63 | ID: 1,
64 | Name: "Lorem Ipsum",
65 | IsActive: true,
66 | }}
67 |
68 | pact.
69 | AddInteraction().
70 | Given("There are available products").
71 | UponReceiving("A GET request to retrieve products.").
72 | WithRequest(request{
73 | Method: "GET",
74 | Path: term("/products", "/products"),
75 | }).
76 | WillRespondWith(dsl.Response{
77 | Status: 200,
78 | Body: dsl.Like(productsResponse),
79 | Headers: dsl.MapMatcher{
80 | "Content-Type": term("application/json; charset=utf-8", `application\/json`),
81 | },
82 | })
83 |
84 | err := pact.Verify(func() error {
85 | _, err := productServiceClient.GetProducts()
86 | return err
87 | })
88 | if err != nil {
89 | t.Fatalf("Error on Verify: %v", err)
90 | }
91 | })
92 | }
93 |
94 | func Test_Create_Get_Single_Product_Contract(t *testing.T) {
95 | t.Run("Get Product...", func(t *testing.T) {
96 | productResponse := dto.ProductDto{
97 | ID: 1,
98 | Name: "Lorem Ipsum",
99 | IsActive: true,
100 | }
101 |
102 | pact.
103 | AddInteraction().
104 | Given("There is a single product for a given product id").
105 | UponReceiving("A GET request to retrieve single product.").
106 | WithRequest(request{
107 | Method: "GET",
108 | Path: term("/products/1", "/products/[0-9]+"),
109 | }).
110 | WillRespondWith(dsl.Response{
111 | Status: 200,
112 | Body: dsl.Like(productResponse),
113 | Headers: dsl.MapMatcher{
114 | "Content-Type": term("application/json; charset=utf-8", `application\/json`),
115 | },
116 | })
117 |
118 | err := pact.Verify(func() error {
119 | _, err := productServiceClient.GetProduct(1)
120 | return err
121 | })
122 | if err != nil {
123 | t.Fatalf("Error on Verify: %v", err)
124 | }
125 | })
126 | }
127 |
128 | var productServiceClient *httpclient.ProductServiceClient
129 |
130 | func setupMockServer() {
131 | pact = createPact()
132 | // Start service to get access to the port
133 | pact.Setup(true)
134 | productServiceClient = &httpclient.ProductServiceClient{
135 | HostUrl: fmt.Sprintf("http://localhost:%d", pact.Server.Port),
136 | }
137 | }
138 |
139 | func createPact() dsl.Pact {
140 | return dsl.Pact{
141 | Consumer: "price-service",
142 | Provider: "product-service",
143 | LogDir: "../_pact_consumer_logs",
144 | PactDir: "../../_pacts",
145 | LogLevel: "DEBUG",
146 | DisableToolValidityCheck: true,
147 | }
148 | }
149 |
--------------------------------------------------------------------------------
/consumer-search-service/.gitignore:
--------------------------------------------------------------------------------
1 | # Binaries for programs and plugins
2 | *.exe
3 | *.exe~
4 | *.dll
5 | *.so
6 | *.dylib
7 |
8 | # Test binary, built with `go test -c`
9 | *.test
10 |
11 | # Output of the go coverage tool, specifically when used with LiteIDE
12 | *.out
13 |
14 | # Dependency directories (remove the comment below to include it)
15 | # vendor/
16 |
17 | __debug_bin
18 |
19 |
20 | _pact_consumer_logs/
21 |
22 | .DS_Store
--------------------------------------------------------------------------------
/consumer-search-service/go.mod:
--------------------------------------------------------------------------------
1 | module cdc-testing-workshop/consumer-search-service
2 |
3 | go 1.16
4 |
5 | require (
6 | github.com/gin-gonic/gin v1.7.0
7 | github.com/pact-foundation/pact-go v1.5.2
8 | )
9 |
--------------------------------------------------------------------------------
/consumer-search-service/go.sum:
--------------------------------------------------------------------------------
1 | cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
2 | github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
3 | github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
4 | github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
5 | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
6 | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
7 | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
8 | github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
9 | github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
10 | github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
11 | github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
12 | github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M=
13 | github.com/gin-gonic/gin v1.7.0 h1:jGB9xAJQ12AIGNB4HguylppmDK1Am9ppF7XnGXXJuoU=
14 | github.com/gin-gonic/gin v1.7.0/go.mod h1:jD2toBW3GZUr5UMcdrwQA10I7RuaFOl/SGeDjXkfUtY=
15 | github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A=
16 | github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
17 | github.com/go-playground/locales v0.13.0 h1:HyWk6mgj5qFqCT5fjGBuRArbVDfE4hi8+e8ceBS/t7Q=
18 | github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8=
19 | github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD876Lmtgy7VtROAbHHXk8no=
20 | github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA=
21 | github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI=
22 | github.com/go-playground/validator/v10 v10.3.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI=
23 | github.com/go-playground/validator/v10 v10.4.1 h1:pH2c5ADXtd66mxoE0Zm9SUhxE20r7aM3F26W0hOn+GE=
24 | github.com/go-playground/validator/v10 v10.4.1/go.mod h1:nlOn6nFhuKACm19sB/8EGNn9GlaMV7XkbRSipzJ0Ii4=
25 | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
26 | github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
27 | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
28 | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
29 | github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
30 | github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
31 | github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
32 | github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
33 | github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
34 | github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
35 | github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
36 | github.com/golang/protobuf v1.4.2 h1:+Z5KGCizgyZCbGh1KZqA0fcLLkwbsjIzS4aV2v7wJX0=
37 | github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
38 | github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
39 | github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
40 | github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
41 | github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
42 | github.com/google/go-cmp v0.5.0 h1:/QaMHBdZ26BB3SSst0Iwl10Epc+xhTquomWX0oZEB6w=
43 | github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
44 | github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
45 | github.com/hashicorp/go-version v1.0.0 h1:21MVWPKDphxa7ineQQTrCU5brh7OuVVAzGOCnnCPtE8=
46 | github.com/hashicorp/go-version v1.0.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
47 | github.com/hashicorp/logutils v0.0.0-20150609070431-0dc08b1671f3 h1:oD64EFjELI9RY9yoWlfua58r+etdnoIC871z+rr6lkA=
48 | github.com/hashicorp/logutils v0.0.0-20150609070431-0dc08b1671f3/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64=
49 | github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
50 | github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
51 | github.com/json-iterator/go v1.1.10 h1:Kz6Cvnvv2wGdaG/V8yMvfkmNiXq9Ya2KUv4rouJJr68=
52 | github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
53 | github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
54 | github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
55 | github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY=
56 | github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
57 | github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
58 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
59 | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
60 | github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
61 | github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
62 | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
63 | github.com/pact-foundation/pact-go v1.5.2 h1:3dpMKDcvowCmSsJscU/3D9qB4vIZH89NDxoXZcv73sA=
64 | github.com/pact-foundation/pact-go v1.5.2/go.mod h1:dN3I+UHNkIk5ZMxSt2+lqavCqfTu+xsx56ZME8BNWlU=
65 | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
66 | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
67 | github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
68 | github.com/spf13/cobra v0.0.0-20160604044732-f447048345b6/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ=
69 | github.com/spf13/pflag v0.0.0-20160427162146-cb88ea77998c/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
70 | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
71 | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
72 | github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk=
73 | github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
74 | github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo=
75 | github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
76 | github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
77 | github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
78 | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
79 | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9 h1:psW17arqaxU48Z5kZ0CQnkZWQJsqcURM6tKiBApRjXI=
80 | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
81 | golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
82 | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
83 | golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
84 | golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
85 | golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
86 | golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
87 | golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
88 | golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
89 | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
90 | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
91 | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
92 | golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
93 | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
94 | golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
95 | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
96 | golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
97 | golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
98 | golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae h1:Ih9Yo4hSPImZOpfGuA4bR/ORKTAbhZo2AbWNRCnevdo=
99 | golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
100 | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
101 | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
102 | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
103 | golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
104 | golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
105 | golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
106 | golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
107 | golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
108 | golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
109 | google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
110 | google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
111 | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
112 | google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
113 | google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
114 | google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
115 | google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
116 | google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
117 | google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
118 | google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
119 | google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
120 | google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
121 | google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
122 | google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
123 | google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
124 | google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
125 | google.golang.org/protobuf v1.25.0 h1:Ejskq+SyPohKW+1uil0JJMtmHCgJPJ/qWTxr8qp+R4c=
126 | google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
127 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
128 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
129 | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
130 | gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
131 | gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
132 | gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
133 | honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
134 | honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
135 |
--------------------------------------------------------------------------------
/consumer-search-service/src/api.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "net/http"
5 |
6 | "cdc-testing-workshop/consumer-search-service/src/dto"
7 | "cdc-testing-workshop/consumer-search-service/src/httpclient"
8 |
9 | "github.com/gin-gonic/gin"
10 | )
11 |
12 | func searchProducts(context *gin.Context) {
13 | client := httpclient.NewProductServiceClient()
14 | keyword,_ := context.GetQuery("keyword")
15 |
16 | products, err := client.SearchProducts(keyword)
17 | if err != nil {
18 | context.JSON(http.StatusBadRequest, err.Error())
19 | return
20 | }
21 |
22 | result := make([]dto.ProductSearchDto, 0, len(products))
23 | for _, product := range products {
24 | result = append(result, dto.ProductSearchDto{
25 | ID: product.ID,
26 | Name: product.Name,
27 | })
28 | }
29 | context.JSON(http.StatusOK, result)
30 | }
31 |
--------------------------------------------------------------------------------
/consumer-search-service/src/dto/productDto.go:
--------------------------------------------------------------------------------
1 | package dto
2 |
3 | type ProductDto struct {
4 | ID int
5 | Name string
6 | }
7 |
--------------------------------------------------------------------------------
/consumer-search-service/src/dto/productSearchDto.go:
--------------------------------------------------------------------------------
1 | package dto
2 |
3 | type ProductSearchDto struct {
4 | ID int
5 | Name string
6 | }
7 |
--------------------------------------------------------------------------------
/consumer-search-service/src/httpclient/product_http_client.go:
--------------------------------------------------------------------------------
1 | package httpclient
2 |
3 | import (
4 | "cdc-testing-workshop/consumer-search-service/src/dto"
5 | "encoding/json"
6 | "errors"
7 | "io/ioutil"
8 | "net/http"
9 | )
10 |
11 | type ProductServiceClient struct {
12 | HostUrl string
13 | }
14 |
15 | func NewProductServiceClient() *ProductServiceClient {
16 | return &ProductServiceClient{
17 | HostUrl: "http://localhost:3001/api",
18 | }
19 | }
20 |
21 | func (c *ProductServiceClient) SearchProducts(keyword string) ([]dto.ProductDto, error) {
22 | resp, err := http.Get(c.HostUrl + "/products/search?keyword=" + keyword)
23 | if err != nil {
24 | return nil, err
25 | }
26 |
27 | defer resp.Body.Close()
28 | if resp.StatusCode != 200 {
29 | return nil, errors.New("failed getting products")
30 | }
31 |
32 | body, err := ioutil.ReadAll(resp.Body)
33 | if err != nil {
34 | return nil, err
35 | }
36 |
37 | var products []dto.ProductDto
38 | json.Unmarshal(body, &products)
39 | return products, nil
40 | }
41 |
--------------------------------------------------------------------------------
/consumer-search-service/src/main.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "net/http"
5 |
6 | "github.com/gin-gonic/gin"
7 | )
8 |
9 | func main() {
10 | handler := initRoutes()
11 | err := newHttpServer(handler).ListenAndServe()
12 | if err != nil {
13 | panic("Couldn't start the HTTP server.")
14 | }
15 | }
16 |
17 | func newHttpServer(handler http.Handler) *http.Server {
18 | return &http.Server{
19 | Addr: ":3003",
20 | Handler: handler,
21 | }
22 | }
23 |
24 | func initRoutes() *gin.Engine {
25 | router := gin.Default()
26 | routerGroup := router.Group("/api")
27 | routerGroup.GET("/products/search", searchProducts)
28 | return router
29 | }
30 |
--------------------------------------------------------------------------------
/consumer-search-service/test/product_contract_test.go:
--------------------------------------------------------------------------------
1 | package test
2 |
3 | import (
4 | "cdc-testing-workshop/consumer-search-service/src/dto"
5 | "cdc-testing-workshop/consumer-search-service/src/httpclient"
6 | "fmt"
7 | "log"
8 | "os"
9 | "path/filepath"
10 | "testing"
11 | "time"
12 |
13 | "github.com/pact-foundation/pact-go/dsl"
14 | "github.com/pact-foundation/pact-go/types"
15 | )
16 |
17 | var pact dsl.Pact
18 |
19 | func TestMain(m *testing.M) {
20 |
21 | setupMockServer()
22 |
23 | var exitCode = m.Run()
24 |
25 | pact.WritePact()
26 | pact.Teardown()
27 |
28 | err := publishPacts()
29 | if err != nil {
30 | log.Println("ERROR: ", err)
31 | os.Exit(1)
32 | }
33 | os.Exit(exitCode)
34 | }
35 |
36 | func publishPacts() error {
37 |
38 | publisher := dsl.Publisher{}
39 |
40 | var dir, _ = os.Getwd()
41 | var pactDir = fmt.Sprintf("%s/../../_pacts", dir)
42 |
43 | now := time.Now()
44 | consumerVersion := fmt.Sprintf("1.0.%02d.%02d.%02d.%02d", now.Day(), now.Month(), now.Hour(), now.Minute())
45 |
46 | return publisher.Publish(types.PublishRequest{
47 | PactURLs: []string{filepath.FromSlash(fmt.Sprintf("%s/search-service-product-service.json", pactDir))},
48 | ConsumerVersion: consumerVersion,
49 | PactBroker: "http://localhost:9292",
50 | BrokerUsername: "admin",
51 | BrokerPassword: "admin",
52 | })
53 | }
54 |
55 | var term = dsl.Term
56 |
57 | type request = dsl.Request
58 |
59 | func Test_Create_Search_Products_Contract(t *testing.T) {
60 | t.Run("Search Products...", func(t *testing.T) {
61 | keyword := "loremipsum"
62 | productsResponse := [1]dto.ProductDto{
63 | {
64 | ID: 1,
65 | Name: "Lorem Ipsum",
66 | }}
67 |
68 | pact.
69 | AddInteraction().
70 | Given("There are available products for a given keyword.").
71 | UponReceiving("A GET request to search products.").
72 | WithRequest(request{
73 | Method: "GET",
74 | Path: term("/products/search", "/products/search"),
75 | Query: dsl.MapMatcher{
76 | "keyword": term("foo", "[a-zA-Z]+"),
77 | },
78 | }).
79 | WillRespondWith(dsl.Response{
80 | Status: 200,
81 | Body: dsl.Like(productsResponse),
82 | Headers: dsl.MapMatcher{
83 | "Content-Type": term("application/json; charset=utf-8", `application\/json`),
84 | },
85 | })
86 |
87 | err := pact.Verify(func() error {
88 | _, err := productServiceClient.SearchProducts(keyword)
89 | return err
90 | })
91 | if err != nil {
92 | t.Fatalf("Error on Verify: %v", err)
93 | }
94 | })
95 | }
96 |
97 | var productServiceClient *httpclient.ProductServiceClient
98 |
99 | func setupMockServer() {
100 | pact = createPact()
101 | // Start service to get access to the port
102 | pact.Setup(true)
103 | productServiceClient = &httpclient.ProductServiceClient{
104 | HostUrl: fmt.Sprintf("http://localhost:%d", pact.Server.Port),
105 | }
106 | }
107 |
108 | func createPact() dsl.Pact {
109 | return dsl.Pact{
110 | Consumer: "search-service",
111 | Provider: "product-service",
112 | LogDir: "../_pact_consumer_logs",
113 | PactDir: "../../_pacts",
114 | LogLevel: "DEBUG",
115 | DisableToolValidityCheck: true,
116 | }
117 | }
118 |
--------------------------------------------------------------------------------
/docker-compose.yml:
--------------------------------------------------------------------------------
1 | version: '3.7'
2 |
3 | services:
4 | postgres:
5 | image: postgres:10.6-alpine
6 | container_name: cdc_postgres
7 | ports:
8 | - "5492:5432"
9 | environment:
10 | - POSTGRES_USER=dbadmin
11 | - POSTGRES_PASSWORD=dbadmin
12 |
13 | broker:
14 | container_name: cdc_pact_broker
15 | image: pactfoundation/pact-broker:latest
16 | ports:
17 | - 9292:9292
18 | environment:
19 | - PACT_BROKER_DATABASE_URL=postgresql://dbadmin:dbadmin@postgres:5432/postgres
20 | - PACT_BROKER_BASIC_AUTH_USERNAME=admin
21 | - PACT_BROKER_BASIC_AUTH_PASSWORD=admin
22 | - PACT_BROKER_LOG_LEVEL=INFO
23 | - PACT_BROKER_PORT=9292
24 | depends_on:
25 | - postgres
--------------------------------------------------------------------------------
/provider-product-service/.gitignore:
--------------------------------------------------------------------------------
1 | ## Ignore Visual Studio temporary files, build results, and
2 | ## files generated by popular Visual Studio add-ons.
3 |
4 | # User-specific files
5 | *.suo
6 | *.user
7 | *.userosscache
8 | *.sln.docstates
9 |
10 | # User-specific files (MonoDevelop/Xamarin Studio)
11 | *.userprefs
12 |
13 | # Build results
14 | [Dd]ebug/
15 | [Dd]ebugPublic/
16 | [Rr]elease/
17 | [Rr]eleases/
18 | x64/
19 | x86/
20 | bld/
21 | [Bb]in/
22 | [Oo]bj/
23 | [Ll]og/
24 |
25 | # Visual Studio 2015 cache/options directory
26 | .vs/
27 | # Uncomment if you have tasks that create the project's static files in wwwroot
28 | #wwwroot/
29 |
30 | # MSTest test Results
31 | [Tt]est[Rr]esult*/
32 | [Bb]uild[Ll]og.*
33 |
34 | # NUNIT
35 | *.VisualState.xml
36 | TestResult.xml
37 |
38 | # Build Results of an ATL Project
39 | [Dd]ebugPS/
40 | [Rr]eleasePS/
41 | dlldata.c
42 |
43 | # DNX
44 | project.lock.json
45 | project.fragment.lock.json
46 | artifacts/
47 |
48 | *_i.c
49 | *_p.c
50 | *_i.h
51 | *.ilk
52 | *.meta
53 | *.obj
54 | *.pch
55 | *.pdb
56 | *.pgc
57 | *.pgd
58 | *.rsp
59 | *.sbr
60 | *.tlb
61 | *.tli
62 | *.tlh
63 | *.tmp
64 | *.tmp_proj
65 | *.log
66 | *.vspscc
67 | *.vssscc
68 | .builds
69 | *.pidb
70 | *.svclog
71 | *.scc
72 |
73 | # Chutzpah Test files
74 | _Chutzpah*
75 |
76 | # Visual C++ cache files
77 | ipch/
78 | *.aps
79 | *.ncb
80 | *.opendb
81 | *.opensdf
82 | *.sdf
83 | *.cachefile
84 | *.VC.db
85 | *.VC.VC.opendb
86 |
87 | # Visual Studio profiler
88 | *.psess
89 | *.vsp
90 | *.vspx
91 | *.sap
92 |
93 | # TFS 2012 Local Workspace
94 | $tf/
95 |
96 | # Guidance Automation Toolkit
97 | *.gpState
98 |
99 | # ReSharper is a .NET coding add-in
100 | _ReSharper*/
101 | *.[Rr]e[Ss]harper
102 | *.DotSettings.user
103 |
104 | # JustCode is a .NET coding add-in
105 | .JustCode
106 |
107 | # TeamCity is a build add-in
108 | _TeamCity*
109 |
110 | # DotCover is a Code Coverage Tool
111 | *.dotCover
112 |
113 | # Visual Studio code coverage results
114 | *.coverage
115 | *.coveragexml
116 |
117 | # NCrunch
118 | _NCrunch_*
119 | .*crunch*.local.xml
120 | nCrunchTemp_*
121 |
122 | # MightyMoose
123 | *.mm.*
124 | AutoTest.Net/
125 |
126 | # Web workbench (sass)
127 | .sass-cache/
128 |
129 | # Installshield output folder
130 | [Ee]xpress/
131 |
132 | # DocProject is a documentation generator add-in
133 | DocProject/buildhelp/
134 | DocProject/Help/*.HxT
135 | DocProject/Help/*.HxC
136 | DocProject/Help/*.hhc
137 | DocProject/Help/*.hhk
138 | DocProject/Help/*.hhp
139 | DocProject/Help/Html2
140 | DocProject/Help/html
141 |
142 | # Click-Once directory
143 | publish/
144 |
145 | # Publish Web Output
146 | *.[Pp]ublish.xml
147 | *.azurePubxml
148 | # TODO: Comment the next line if you want to checkin your web deploy settings
149 | # but database connection strings (with potential passwords) will be unencrypted
150 | *.pubxml
151 | *.publishproj
152 |
153 | # Microsoft Azure Web App publish settings. Comment the next line if you want to
154 | # checkin your Azure Web App publish settings, but sensitive information contained
155 | # in these scripts will be unencrypted
156 | PublishScripts/
157 |
158 | # NuGet Packages
159 | *.nupkg
160 | # The packages folder can be ignored because of Package Restore
161 | **/packages/*
162 | # except build/, which is used as an MSBuild target.
163 | !**/packages/build/
164 | # Uncomment if necessary however generally it will be regenerated when needed
165 | #!**/packages/repositories.config
166 | # NuGet v3's project.json files produces more ignoreable files
167 | *.nuget.props
168 | *.nuget.targets
169 |
170 | # Microsoft Azure Build Output
171 | csx/
172 | *.build.csdef
173 |
174 | # Microsoft Azure Emulator
175 | ecf/
176 | rcf/
177 |
178 | # Windows Store app package directories and files
179 | AppPackages/
180 | BundleArtifacts/
181 | Package.StoreAssociation.xml
182 | _pkginfo.txt
183 |
184 | # Visual Studio cache files
185 | # files ending in .cache can be ignored
186 | *.[Cc]ache
187 | # but keep track of directories ending in .cache
188 | !*.[Cc]ache/
189 |
190 | # Others
191 | ClientBin/
192 | ~$*
193 | *~
194 | *.dbmdl
195 | *.dbproj.schemaview
196 | *.jfm
197 | *.pfx
198 | *.publishsettings
199 | node_modules/
200 | orleans.codegen.cs
201 |
202 | # Since there are multiple workflows, uncomment next line to ignore bower_components
203 | # (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
204 | #bower_components/
205 |
206 | # RIA/Silverlight projects
207 | Generated_Code/
208 |
209 | # Backup & report files from converting an old project file
210 | # to a newer Visual Studio version. Backup files are not needed,
211 | # because we have git ;-)
212 | _UpgradeReport_Files/
213 | Backup*/
214 | UpgradeLog*.XML
215 | UpgradeLog*.htm
216 |
217 | # SQL Server files
218 | *.mdf
219 | *.ldf
220 |
221 | # Grafana
222 | *.db
223 |
224 |
225 | # Business Intelligence projects
226 | *.rdl.data
227 | *.bim.layout
228 | *.bim_*.settings
229 |
230 | # Microsoft Fakes
231 | FakesAssemblies/
232 |
233 | # GhostDoc plugin setting file
234 | *.GhostDoc.xml
235 |
236 | # Node.js Tools for Visual Studio
237 | .ntvs_analysis.dat
238 |
239 | # Visual Studio 6 build log
240 | *.plg
241 |
242 | # Visual Studio 6 workspace options file
243 | *.opt
244 |
245 | # Visual Studio LightSwitch build output
246 | **/*.HTMLClient/GeneratedArtifacts
247 | **/*.DesktopClient/GeneratedArtifacts
248 | **/*.DesktopClient/ModelManifest.xml
249 | **/*.Server/GeneratedArtifacts
250 | **/*.Server/ModelManifest.xml
251 | _Pvt_Extensions
252 |
253 | # Paket dependency manager
254 | .paket/paket.exe
255 | paket-files/
256 |
257 | # FAKE - F# Make
258 | .fake/
259 |
260 | # JetBrains Rider
261 | .idea/
262 | *.sln.iml
263 |
264 | # CodeRush
265 | .cr/
266 |
267 | # Python Tools for Visual Studio (PTVS)
268 | __pycache__/
269 | *.pyc
270 |
271 | .vscode/*
272 | !.vscode/settings.json
273 | !.vscode/tasks.json
274 | !.vscode/launch.json
275 |
276 | NuGet.config
277 | appsettings.local.json
--------------------------------------------------------------------------------
/provider-product-service/provider-product-service.sln:
--------------------------------------------------------------------------------
1 |
2 | Microsoft Visual Studio Solution File, Format Version 12.00
3 | # Visual Studio Version 17
4 | VisualStudioVersion = 17.5.002.0
5 | MinimumVisualStudioVersion = 10.0.40219.1
6 | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ProviderApiTests", "test\ProviderApiTests.csproj", "{2635B849-4540-4861-9AB5-8681356E8A77}"
7 | EndProject
8 | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Provider", "src\Provider.csproj", "{5AC87D36-A460-44CA-ACC6-B2D1FBB83ECD}"
9 | EndProject
10 | Global
11 | GlobalSection(SolutionConfigurationPlatforms) = preSolution
12 | Debug|Any CPU = Debug|Any CPU
13 | Release|Any CPU = Release|Any CPU
14 | EndGlobalSection
15 | GlobalSection(ProjectConfigurationPlatforms) = postSolution
16 | {2635B849-4540-4861-9AB5-8681356E8A77}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
17 | {2635B849-4540-4861-9AB5-8681356E8A77}.Debug|Any CPU.Build.0 = Debug|Any CPU
18 | {2635B849-4540-4861-9AB5-8681356E8A77}.Release|Any CPU.ActiveCfg = Release|Any CPU
19 | {2635B849-4540-4861-9AB5-8681356E8A77}.Release|Any CPU.Build.0 = Release|Any CPU
20 | {5AC87D36-A460-44CA-ACC6-B2D1FBB83ECD}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
21 | {5AC87D36-A460-44CA-ACC6-B2D1FBB83ECD}.Debug|Any CPU.Build.0 = Debug|Any CPU
22 | {5AC87D36-A460-44CA-ACC6-B2D1FBB83ECD}.Release|Any CPU.ActiveCfg = Release|Any CPU
23 | {5AC87D36-A460-44CA-ACC6-B2D1FBB83ECD}.Release|Any CPU.Build.0 = Release|Any CPU
24 | EndGlobalSection
25 | GlobalSection(SolutionProperties) = preSolution
26 | HideSolutionNode = FALSE
27 | EndGlobalSection
28 | GlobalSection(ExtensibilityGlobals) = postSolution
29 | SolutionGuid = {4F2C902C-ABE4-4CDD-A93F-5508BBEFAB13}
30 | EndGlobalSection
31 | EndGlobal
32 |
--------------------------------------------------------------------------------
/provider-product-service/src/Controllers/ProductsController.cs:
--------------------------------------------------------------------------------
1 | using System.Linq;
2 | using System.Threading.Tasks;
3 | using Microsoft.AspNetCore.Mvc;
4 | using Microsoft.EntityFrameworkCore;
5 | using Data;
6 | using Dto;
7 |
8 | namespace Controllers
9 | {
10 | [Route("api/products")]
11 | [ApiController]
12 | public class ProductsController : ControllerBase
13 | {
14 | private readonly ProductDBContext _dbContext;
15 | public ProductsController(ProductDBContext dbContext)
16 | {
17 | _dbContext = dbContext;
18 | }
19 |
20 | [HttpGet]
21 | public async Task GetAllProducts()
22 | {
23 | var products = await _dbContext.Products
24 | .Select(s => new ProductDto
25 | {
26 | ID = s.ID,
27 | IsActive = s.IsActive,
28 | Name = s.Name,
29 | StockCount = s.StockCount
30 | }).ToListAsync();
31 | return Ok(products);
32 | }
33 |
34 | [HttpGet("{id}")]
35 | public async Task GetProduct(int id)
36 | {
37 | var product = await _dbContext.Products
38 | .Select(s => new ProductDto
39 | {
40 | ID = s.ID,
41 | IsActive = s.IsActive,
42 | Name = s.Name,
43 | StockCount = s.StockCount
44 | }).FirstOrDefaultAsync(s => s.ID == id);
45 | return Ok(product);
46 | }
47 |
48 | [HttpGet("search")]
49 | public async Task Search([FromQuery] string keyword)
50 | {
51 | var normalizedKeyword = keyword.ToUpperInvariant();
52 |
53 | var products = await _dbContext.Products
54 | .Where(s => s.NormalizedName.Contains(keyword.ToUpperInvariant()))
55 | .Select(s => new ProductDto
56 | {
57 | ID = s.ID,
58 | IsActive = s.IsActive,
59 | Name = s.Name,
60 | StockCount = s.StockCount
61 | }).ToListAsync();
62 | return Ok(products);
63 | }
64 | }
65 | }
66 |
--------------------------------------------------------------------------------
/provider-product-service/src/Data/DbInitilializer.cs:
--------------------------------------------------------------------------------
1 | using System;
2 | using Microsoft.EntityFrameworkCore;
3 | using Microsoft.Extensions.DependencyInjection;
4 |
5 | namespace Data
6 | {
7 | public static class DbInitilializer
8 | {
9 | public static void Migrate(IServiceProvider serviceProvider)
10 | {
11 | using (var scope = serviceProvider.CreateScope())
12 | {
13 | var context = scope.ServiceProvider.GetRequiredService();
14 | context.Database.Migrate();
15 | }
16 | }
17 | }
18 | }
--------------------------------------------------------------------------------
/provider-product-service/src/Data/Entity/Product.cs:
--------------------------------------------------------------------------------
1 | namespace Data
2 | {
3 | public class Product
4 | {
5 | public Product(int id, string name, int stockCount, bool isActive = true)
6 | {
7 | ID = id;
8 | Name = name;
9 | NormalizedName = name.ToUpperInvariant();
10 | StockCount = stockCount;
11 | IsActive = isActive;
12 | }
13 |
14 | public Product()
15 | { }
16 |
17 | public int ID { get; private set; }
18 | public string Name { get; private set; }
19 | public string NormalizedName { get; private set; }
20 | public int StockCount { get; private set; }
21 | public bool IsActive { get; private set; }
22 | }
23 | }
--------------------------------------------------------------------------------
/provider-product-service/src/Data/Migrations/20210408200516_initial.Designer.cs:
--------------------------------------------------------------------------------
1 | //
2 | using Data;
3 | using Microsoft.EntityFrameworkCore;
4 | using Microsoft.EntityFrameworkCore.Infrastructure;
5 | using Microsoft.EntityFrameworkCore.Migrations;
6 | using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
7 | using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
8 |
9 | namespace Provider.Data.Migrations
10 | {
11 | [DbContext(typeof(ProductDBContext))]
12 | [Migration("20210408200516_initial")]
13 | partial class initial
14 | {
15 | protected override void BuildTargetModel(ModelBuilder modelBuilder)
16 | {
17 | #pragma warning disable 612, 618
18 | modelBuilder
19 | .HasDefaultSchema("product")
20 | .HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn)
21 | .HasAnnotation("ProductVersion", "3.1.2")
22 | .HasAnnotation("Relational:MaxIdentifierLength", 63);
23 |
24 | modelBuilder.Entity("Data.Product", b =>
25 | {
26 | b.Property("ID")
27 | .ValueGeneratedOnAdd()
28 | .HasColumnName("id")
29 | .HasColumnType("integer")
30 | .HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn);
31 |
32 | b.Property("IsActive")
33 | .HasColumnName("is_active")
34 | .HasColumnType("boolean");
35 |
36 | b.Property("Name")
37 | .HasColumnName("name")
38 | .HasColumnType("text");
39 |
40 | b.Property("NormalizedName")
41 | .HasColumnName("normalized_name")
42 | .HasColumnType("text");
43 |
44 | b.Property("StockCount")
45 | .HasColumnName("stock_count")
46 | .HasColumnType("integer");
47 |
48 | b.HasKey("ID")
49 | .HasName("pk_products");
50 |
51 | b.ToTable("products");
52 |
53 | b.HasData(
54 | new
55 | {
56 | ID = 1,
57 | IsActive = true,
58 | Name = "Keyboard",
59 | NormalizedName = "KEYBOARD",
60 | StockCount = 40
61 | },
62 | new
63 | {
64 | ID = 2,
65 | IsActive = true,
66 | Name = "Mouse",
67 | NormalizedName = "MOUSE",
68 | StockCount = 20
69 | },
70 | new
71 | {
72 | ID = 3,
73 | IsActive = true,
74 | Name = "Monitor",
75 | NormalizedName = "MONITOR",
76 | StockCount = 50
77 | },
78 | new
79 | {
80 | ID = 4,
81 | IsActive = true,
82 | Name = "Printer",
83 | NormalizedName = "PRINTER",
84 | StockCount = 50
85 | },
86 | new
87 | {
88 | ID = 5,
89 | IsActive = true,
90 | Name = "Earphone",
91 | NormalizedName = "EARPHONE",
92 | StockCount = 20
93 | });
94 | });
95 | #pragma warning restore 612, 618
96 | }
97 | }
98 | }
99 |
--------------------------------------------------------------------------------
/provider-product-service/src/Data/Migrations/20210408200516_initial.cs:
--------------------------------------------------------------------------------
1 | using Microsoft.EntityFrameworkCore.Migrations;
2 | using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
3 |
4 | namespace Provider.Data.Migrations
5 | {
6 | public partial class initial : Migration
7 | {
8 | protected override void Up(MigrationBuilder migrationBuilder)
9 | {
10 | migrationBuilder.EnsureSchema(
11 | name: "product");
12 |
13 | migrationBuilder.CreateTable(
14 | name: "products",
15 | schema: "product",
16 | columns: table => new
17 | {
18 | id = table.Column(nullable: false)
19 | .Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
20 | name = table.Column(nullable: true),
21 | normalized_name = table.Column(nullable: true),
22 | stock_count = table.Column(nullable: false),
23 | is_active = table.Column(nullable: false)
24 | },
25 | constraints: table =>
26 | {
27 | table.PrimaryKey("pk_products", x => x.id);
28 | });
29 |
30 | migrationBuilder.InsertData(
31 | schema: "product",
32 | table: "products",
33 | columns: new[] { "id", "is_active", "name", "normalized_name", "stock_count" },
34 | values: new object[,]
35 | {
36 | { 1, true, "Keyboard", "KEYBOARD", 40 },
37 | { 2, true, "Mouse", "MOUSE", 20 },
38 | { 3, true, "Monitor", "MONITOR", 50 },
39 | { 4, true, "Printer", "PRINTER", 50 },
40 | { 5, true, "Earphone", "EARPHONE", 20 }
41 | });
42 | }
43 |
44 | protected override void Down(MigrationBuilder migrationBuilder)
45 | {
46 | migrationBuilder.DropTable(
47 | name: "products",
48 | schema: "product");
49 | }
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/provider-product-service/src/Data/Migrations/ProductDBContextModelSnapshot.cs:
--------------------------------------------------------------------------------
1 | //
2 | using Data;
3 | using Microsoft.EntityFrameworkCore;
4 | using Microsoft.EntityFrameworkCore.Infrastructure;
5 | using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
6 | using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
7 |
8 | namespace Provider.Data.Migrations
9 | {
10 | [DbContext(typeof(ProductDBContext))]
11 | partial class ProductDBContextModelSnapshot : ModelSnapshot
12 | {
13 | protected override void BuildModel(ModelBuilder modelBuilder)
14 | {
15 | #pragma warning disable 612, 618
16 | modelBuilder
17 | .HasDefaultSchema("product")
18 | .HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn)
19 | .HasAnnotation("ProductVersion", "3.1.2")
20 | .HasAnnotation("Relational:MaxIdentifierLength", 63);
21 |
22 | modelBuilder.Entity("Data.Product", b =>
23 | {
24 | b.Property("ID")
25 | .ValueGeneratedOnAdd()
26 | .HasColumnName("id")
27 | .HasColumnType("integer")
28 | .HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn);
29 |
30 | b.Property("IsActive")
31 | .HasColumnName("is_active")
32 | .HasColumnType("boolean");
33 |
34 | b.Property("Name")
35 | .HasColumnName("name")
36 | .HasColumnType("text");
37 |
38 | b.Property("NormalizedName")
39 | .HasColumnName("normalized_name")
40 | .HasColumnType("text");
41 |
42 | b.Property("StockCount")
43 | .HasColumnName("stock_count")
44 | .HasColumnType("integer");
45 |
46 | b.HasKey("ID")
47 | .HasName("pk_products");
48 |
49 | b.ToTable("products");
50 |
51 | b.HasData(
52 | new
53 | {
54 | ID = 1,
55 | IsActive = true,
56 | Name = "Keyboard",
57 | NormalizedName = "KEYBOARD",
58 | StockCount = 40
59 | },
60 | new
61 | {
62 | ID = 2,
63 | IsActive = true,
64 | Name = "Mouse",
65 | NormalizedName = "MOUSE",
66 | StockCount = 20
67 | },
68 | new
69 | {
70 | ID = 3,
71 | IsActive = true,
72 | Name = "Monitor",
73 | NormalizedName = "MONITOR",
74 | StockCount = 50
75 | },
76 | new
77 | {
78 | ID = 4,
79 | IsActive = true,
80 | Name = "Printer",
81 | NormalizedName = "PRINTER",
82 | StockCount = 50
83 | },
84 | new
85 | {
86 | ID = 5,
87 | IsActive = true,
88 | Name = "Earphone",
89 | NormalizedName = "EARPHONE",
90 | StockCount = 20
91 | });
92 | });
93 | #pragma warning restore 612, 618
94 | }
95 | }
96 | }
97 |
--------------------------------------------------------------------------------
/provider-product-service/src/Data/ProductDBContext.cs:
--------------------------------------------------------------------------------
1 | using Microsoft.EntityFrameworkCore;
2 |
3 | namespace Data
4 | {
5 | public class ProductDBContext : DbContext
6 | {
7 | public ProductDBContext(DbContextOptions options)
8 | : base(options)
9 | {
10 | }
11 | public DbSet Products { get; set; }
12 |
13 | protected override void OnModelCreating(ModelBuilder modelBuilder)
14 | {
15 | modelBuilder.HasDefaultSchema("product");
16 | modelBuilder.ApplyConfiguration(new ProductTypeConfiguration());
17 | }
18 | }
19 | }
--------------------------------------------------------------------------------
/provider-product-service/src/Data/TypeConfiguration/ProductTypeConfiguration.cs:
--------------------------------------------------------------------------------
1 |
2 |
3 | using Microsoft.EntityFrameworkCore;
4 | using Microsoft.EntityFrameworkCore.Metadata.Builders;
5 | namespace Data
6 | {
7 | public class ProductTypeConfiguration : IEntityTypeConfiguration
8 | {
9 | public void Configure(EntityTypeBuilder builder)
10 | {
11 | builder.HasData(
12 | new Product(1, "Keyboard", 40),
13 | new Product(2, "Mouse", 20),
14 | new Product(3, "Monitor", 50),
15 | new Product(4, "Printer", 50),
16 | new Product(5, "Earphone", 20));
17 | }
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/provider-product-service/src/Dto/ProductDto.cs:
--------------------------------------------------------------------------------
1 | namespace Dto
2 | {
3 | public class ProductDto
4 | {
5 | public int ID { get; set; }
6 | public string Name { get; set; }
7 | public int StockCount { get; set; }
8 | public bool IsActive { get; set; }
9 | }
10 | }
--------------------------------------------------------------------------------
/provider-product-service/src/Program.cs:
--------------------------------------------------------------------------------
1 | using Data;
2 | using Microsoft.AspNetCore.Builder;
3 | using Microsoft.AspNetCore.Hosting;
4 | using Microsoft.Extensions.Hosting;
5 | using Microsoft.EntityFrameworkCore;
6 | using Microsoft.Extensions.Configuration;
7 | using Microsoft.Extensions.DependencyInjection;
8 |
9 | namespace Provider
10 | {
11 | public class Program
12 | {
13 | public static void Main(string[] args)
14 | {
15 | CreateHostBuilder(args).Build().Run();
16 | }
17 |
18 | public static IHostBuilder CreateHostBuilder(string[] args) =>
19 | Host.CreateDefaultBuilder(args)
20 | .ConfigureWebHostDefaults(webBuilder => webBuilder
21 | .ConfigureServices((hostContext, services) =>
22 | {
23 | services.AddDbContext(options => options
24 | .UseNpgsql(hostContext.Configuration.GetConnectionString("ProductDBConnStr"))
25 | .UseSnakeCaseNamingConvention())
26 | .AddControllers();
27 | })
28 | .Configure(app =>
29 | {
30 | app.UseDeveloperExceptionPage()
31 | .UseRouting()
32 | .UseEndpoints(endpoints =>
33 | {
34 | endpoints.MapControllers();
35 | });
36 |
37 | DbInitilializer.Migrate(app.ApplicationServices);
38 | }));
39 | }
40 | }
41 |
--------------------------------------------------------------------------------
/provider-product-service/src/Properties/launchSettings.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "http://json.schemastore.org/launchsettings.json",
3 | "iisSettings": {
4 | "windowsAuthentication": false,
5 | "anonymousAuthentication": true,
6 | "iisExpress": {
7 | "applicationUrl": "http://localhost:23696",
8 | "sslPort": 44357
9 | }
10 | },
11 | "profiles": {
12 | "IIS Express": {
13 | "commandName": "IISExpress",
14 | "launchBrowser": true,
15 | "launchUrl": "swagger",
16 | "environmentVariables": {
17 | "ASPNETCORE_ENVIRONMENT": "Development"
18 | }
19 | },
20 | "src": {
21 | "commandName": "Project",
22 | "dotnetRunMessages": "true",
23 | "launchBrowser": true,
24 | "launchUrl": "swagger",
25 | "applicationUrl": "http://localhost:3001",
26 | "environmentVariables": {
27 | "ASPNETCORE_ENVIRONMENT": "Development"
28 | }
29 | }
30 | }
31 | }
--------------------------------------------------------------------------------
/provider-product-service/src/Provider.csproj:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | net6.0
5 |
6 |
7 |
8 |
9 |
10 |
11 | runtime; build; native; contentfiles; analyzers; buildtransitive
12 | all
13 |
14 |
15 |
16 |
17 |
--------------------------------------------------------------------------------
/provider-product-service/src/appsettings.json:
--------------------------------------------------------------------------------
1 | {
2 | "ConnectionStrings": {
3 | "ProductDBConnStr": "Host=localhost;Port=5492;Username=dbadmin;Password=dbadmin;Database=Products;"
4 | },
5 | "Logging": {
6 | "LogLevel": {
7 | "Default": "Information",
8 | "Microsoft": "Warning",
9 | "Microsoft.Hosting.Lifetime": "Information"
10 | }
11 | },
12 | "AllowedHosts": "*"
13 | }
--------------------------------------------------------------------------------
/provider-product-service/test/ProviderApiTests.csproj:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | net6.0
5 | false
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 | runtime; build; native; contentfiles; analyzers; buildtransitive
15 | all
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
--------------------------------------------------------------------------------
/provider-product-service/test/ProviderContractTests.cs:
--------------------------------------------------------------------------------
1 | using System;
2 | using System.Collections.Generic;
3 | using PactNet;
4 | using PactNet.Infrastructure.Outputters;
5 | using Xunit;
6 | using Xunit.Abstractions;
7 | using Microsoft.AspNetCore.Hosting;
8 | using Microsoft.AspNetCore;
9 | using ContractTests.XUnitHelpers;
10 |
11 | namespace ContractTests
12 | {
13 | public class ProviderContractTests : IDisposable
14 | {
15 | private string _providerUri { get; }
16 | private IWebHost _webHost { get; }
17 | private ITestOutputHelper _output { get; }
18 |
19 | public ProviderContractTests(ITestOutputHelper output)
20 | {
21 | _output = output;
22 | _providerUri = "http://localhost:3001";
23 |
24 | _webHost = WebHost.CreateDefaultBuilder()
25 | .UseUrls(_providerUri)
26 | .UseStartup()
27 | .Build();
28 |
29 | _webHost.Start();
30 | }
31 |
32 | [Fact]
33 | public void should_verify_pacts()
34 | {
35 | var verifierConfig = new PactVerifierConfig
36 | {
37 | Outputters = new List
38 | {
39 | new XUnitOutputter(_output)
40 | },
41 | Verbose = true,
42 | PublishVerificationResults = true,
43 | ProviderVersion = $"1.0.{DateTime.Now.ToString("dd.MM.HH.mm")}"
44 | };
45 |
46 | new PactVerifier(verifierConfig)
47 | // .ProviderState($"{_providerUri}/provider-states")
48 | .ServiceProvider("product-service", _providerUri)
49 | .HonoursPactWith("price-service")
50 | // .PactUri(@"..\..\..\..\..\_pacts\price-service-product-service.json")
51 | .PactBroker("http://localhost:9292", new PactUriOptions("admin", "admin"))
52 | .Verify();
53 |
54 | new PactVerifier(verifierConfig)
55 | .ServiceProvider("product-service", _providerUri)
56 | .HonoursPactWith("search-service")
57 | .PactBroker("http://localhost:9292", new PactUriOptions("admin", "admin"))
58 | .Verify();
59 | }
60 |
61 | private bool _disposed = false;
62 |
63 | protected virtual void Dispose(bool disposing)
64 | {
65 | if (!_disposed)
66 | {
67 | _disposed = true;
68 | }
69 | }
70 |
71 | public void Dispose()
72 | {
73 | Dispose(true);
74 | }
75 | }
76 | }
--------------------------------------------------------------------------------
/provider-product-service/test/ProviderStateMiddleware.cs:
--------------------------------------------------------------------------------
1 | using System.Collections.Generic;
2 | using System.IO;
3 | using System.Threading.Tasks;
4 | using Dto;
5 | using Microsoft.AspNetCore.Http;
6 | using Newtonsoft.Json;
7 |
8 | namespace ContractTests
9 | {
10 | public class ProviderStateMiddleware
11 | {
12 | private readonly RequestDelegate _next;
13 | public ProviderStateMiddleware(RequestDelegate next)
14 | {
15 | _next = next;
16 | }
17 |
18 | public async Task Invoke(HttpContext context)
19 | {
20 | context.Response.ContentType = "application/json; charset=utf-8";
21 |
22 |
23 | if (context.Request.Path.Value == "/products/1") // get product
24 | {
25 | var product = new ProductDto
26 | {
27 | ID = 1,
28 | Name = "Sample Product",
29 | StockCount = 20,
30 | IsActive = true
31 | };
32 |
33 | var serializedProduct = JsonConvert.SerializeObject(product);
34 | await context.Response.WriteAsync(serializedProduct);
35 | }
36 | else // get products and search products
37 | {
38 | var products = new List();
39 | products.Add(new ProductDto
40 | {
41 | ID = 1,
42 | Name = "Sample Product",
43 | StockCount = 20,
44 | IsActive = true
45 | });
46 |
47 | var serializedProducts = JsonConvert.SerializeObject(products);
48 | await context.Response.WriteAsync(serializedProducts);
49 | }
50 |
51 | // if (context.Request.Path.Value == "/provider-states")
52 | // {
53 | // var response = await HandleProviderStatesRequest(context);
54 | // context.Response.ContentType = "application/json; charset=utf-8";
55 | // await context.Response.WriteAsync(response);
56 | // }
57 | // else
58 | // {
59 | // await _next(context);
60 | // }
61 | }
62 |
63 | // private async Task HandleProviderStatesRequest(HttpContext context)
64 | // {
65 | // context.Response.StatusCode = (int)HttpStatusCode.OK;
66 |
67 | // if (context.Request.Method.ToUpper() == HttpMethod.Post.ToString().ToUpper() &&
68 | // context.Request.Body != null)
69 | // {
70 | // var jsonRequestBody = String.Empty;
71 | // using (var reader = new StreamReader(context.Request.Body, Encoding.UTF8))
72 | // {
73 | // jsonRequestBody = await reader.ReadToEndAsync();
74 | // }
75 |
76 | // var providerState = JsonConvert.DeserializeObject(jsonRequestBody);
77 |
78 | // if (providerState.State == "There are available products")
79 | // {
80 | // // todo: insert products into the db.
81 | // var products = new List();
82 | // products.Add(new ProductDto
83 | // {
84 | // ID = 1,
85 | // Name = "Sample Product",
86 | // StockCount = 20,
87 | // IsActive = true
88 | // });
89 | // return JsonConvert.SerializeObject(products);
90 | // }
91 | // else if (providerState.State == "There is a single product for a given product id")
92 | // {
93 | // // todo: insert single product with ID 1 into the db.
94 | // var product = new ProductDto
95 | // {
96 | // ID = 1,
97 | // Name = "Sample Product",
98 | // StockCount = 20,
99 | // IsActive = true
100 | // };
101 | // return JsonConvert.SerializeObject(product);
102 | // }
103 | // }
104 | // return string.Empty;
105 | // }
106 |
107 | // public class ProviderState
108 | // {
109 | // public string Consumer { get; set; }
110 | // public string State { get; set; }
111 | // }
112 | }
113 | }
--------------------------------------------------------------------------------
/provider-product-service/test/TestStartup.cs:
--------------------------------------------------------------------------------
1 | using Microsoft.AspNetCore.Builder;
2 | using Microsoft.Extensions.Configuration;
3 | using Microsoft.Extensions.DependencyInjection;
4 | using Microsoft.AspNetCore.Hosting;
5 |
6 | namespace ContractTests
7 | {
8 | public class TestStartup
9 | {
10 | public TestStartup(IConfiguration configuration)
11 | {
12 | Configuration = configuration;
13 | }
14 |
15 | public IConfiguration Configuration { get; }
16 |
17 | public void ConfigureServices(IServiceCollection services)
18 | {
19 | services.AddControllers();
20 | }
21 |
22 | public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
23 | {
24 | app.UseMiddleware();
25 | app.UseRouting();
26 | }
27 | }
28 | }
--------------------------------------------------------------------------------
/provider-product-service/test/XUnitHelpers/XunitOutputter.cs:
--------------------------------------------------------------------------------
1 | using PactNet.Infrastructure.Outputters;
2 | using Xunit.Abstractions;
3 |
4 | namespace ContractTests.XUnitHelpers
5 | {
6 | public class XUnitOutputter : IOutput
7 | {
8 | private readonly ITestOutputHelper _output;
9 |
10 | public XUnitOutputter(ITestOutputHelper output)
11 | => _output = output;
12 |
13 | public void WriteLine(string line)
14 | {
15 | _output.WriteLine(line);
16 | }
17 | }
18 | }
--------------------------------------------------------------------------------