-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathconnections.py
349 lines (293 loc) · 11 KB
/
connections.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
from dataclasses import dataclass
from typing import Optional
from app.models.tenants import CreateTenantResponse
from app.routes.connections import CreateInvitation
from app.routes.connections import router as conn_router
from app.routes.oob import router as oob_router
from app.routes.wallet.dids import router as did_router
from app.services.trust_registry.actors import fetch_actor_by_id
from app.tests.util.regression_testing import (
RegressionTestConfig,
TestMode,
assert_fail_on_recreating_fixtures,
)
from app.tests.util.webhooks import assert_both_webhooks_received, check_webhook_state
from app.util.string import base64_to_json
from shared import RichAsyncClient
from shared.models.connection_record import Connection
OOB_BASE_PATH = oob_router.prefix
CONNECTIONS_BASE_PATH = conn_router.prefix
DID_BASE_PATH = did_router.prefix
@dataclass
class BobAliceConnect:
alice_connection_id: str
bob_connection_id: str
@dataclass
class AcmeAliceConnect:
alice_connection_id: str
acme_connection_id: str
@dataclass
class FaberAliceConnect:
alice_connection_id: str
faber_connection_id: str
@dataclass
class MeldCoAliceConnect:
alice_connection_id: str
meld_co_connection_id: str
async def assert_both_connections_ready(
member_client_1: RichAsyncClient,
member_client_2: RichAsyncClient,
connection_id_1: str,
connection_id_2: str,
) -> None:
await assert_both_webhooks_received(
member_client_1,
member_client_2,
topic="connections",
state="completed",
field_id_1=connection_id_1,
field_id_2=connection_id_2,
)
async def create_bob_alice_connection(
alice_member_client: RichAsyncClient, bob_member_client: RichAsyncClient, alias: str
):
# Bob create invitation
bob_invitation = (
await bob_member_client.post(
f"{OOB_BASE_PATH}/create-invitation",
json={
"alias": alias,
"multi_use": False,
"use_public_did": False,
"create_connection": True,
},
)
).json()
# Alice accept invitation
alice_oob_response = (
await alice_member_client.post(
f"{OOB_BASE_PATH}/accept-invitation",
json={"alias": alias, "invitation": bob_invitation["invitation"]},
)
).json()
# Get connection details
alice_connection = await check_webhook_state(
client=alice_member_client,
topic="connections",
state="completed",
filter_map={"connection_id": alice_oob_response["connection_id"]},
)
# Use Alice's connection DID to fetch Bob's connection
their_did = alice_connection["my_did"]
bob_connection = await check_webhook_state(
client=bob_member_client,
topic="connections",
state="completed",
filter_map={"their_did": their_did},
)
bob_connection_id = bob_connection["connection_id"]
alice_connection_id = alice_connection["connection_id"]
return BobAliceConnect(
alice_connection_id=alice_connection_id, bob_connection_id=bob_connection_id
)
async def fetch_existing_connection_by_alias(
member_client: RichAsyncClient,
alias: Optional[str] = None,
their_label: Optional[str] = None,
their_did: Optional[str] = None,
) -> Optional[Connection]:
params = {"state": "completed", "limit": 10000}
if alias:
params.update({"alias": alias})
if their_did:
params.update({"their_did": their_did})
list_connections_response = await member_client.get(
CONNECTIONS_BASE_PATH, params=params
)
list_connections = list_connections_response.json()
if their_label: # to handle Trust Registry invites, where alias is null
list_connections = [
connection
for connection in list_connections
if (
connection["their_label"] == their_label # filter by their label
and connection["alias"] is None # TR OOB invite has null alias
)
]
num_connections = len(list_connections)
assert (
num_connections < 2
), f"{member_client.name} should have 1 or 0 connections, got: {num_connections}"
if list_connections:
return Connection.model_validate(list_connections[0])
async def fetch_or_create_connection(
alice_member_client: RichAsyncClient,
bob_member_client: RichAsyncClient,
connection_alias: str,
did_exchange: bool = False,
) -> BobAliceConnect:
# fetch connection with this alias for both bob and alice
alice_connection = await fetch_existing_connection_by_alias(
alice_member_client, connection_alias
)
their_did = alice_connection.my_did if alice_connection else None
bob_connection = await fetch_existing_connection_by_alias(
member_client=bob_member_client,
alias=connection_alias if not their_did else None,
their_did=their_did,
)
# Check if connections exist
if alice_connection and bob_connection:
return BobAliceConnect(
alice_connection_id=alice_connection.connection_id,
bob_connection_id=bob_connection.connection_id,
)
else:
# Create connection since they don't exist
assert_fail_on_recreating_fixtures()
if did_exchange:
return await create_did_exchange(
bob_member_client=bob_member_client,
alice_member_client=alice_member_client,
alias=connection_alias,
)
else:
return await create_bob_alice_connection(
bob_member_client=bob_member_client,
alice_member_client=alice_member_client,
alias=connection_alias,
)
async def create_connection_by_test_mode(
test_mode: str,
alice_member_client: RichAsyncClient,
bob_member_client: RichAsyncClient,
alias: str,
did_exchange: bool = False,
) -> BobAliceConnect:
if test_mode == TestMode.clean_run:
if did_exchange:
return await create_did_exchange(
bob_member_client=bob_member_client,
alice_member_client=alice_member_client,
alias=alias,
)
else:
return await create_bob_alice_connection(
bob_member_client=bob_member_client,
alice_member_client=alice_member_client,
alias=alias,
)
elif test_mode == TestMode.regression_run:
connection_alias_prefix = RegressionTestConfig.reused_connection_alias
return await fetch_or_create_connection(
alice_member_client,
bob_member_client,
connection_alias=f"{connection_alias_prefix}-{alias}",
did_exchange=did_exchange,
)
else:
assert False, f"unknown test mode: {test_mode}"
async def connect_using_trust_registry_invite(
alice_member_client: RichAsyncClient,
alice_tenant: CreateTenantResponse,
verifier_client: RichAsyncClient,
verifier: CreateTenantResponse,
connection_alias: str,
) -> AcmeAliceConnect:
acme_actor = await fetch_actor_by_id(verifier.wallet_id)
assert acme_actor.didcomm_invitation
invitation = acme_actor.didcomm_invitation
invitation_json = base64_to_json(invitation.split("?oob=")[1])
# accept invitation on alice side -- she uses here connection alias
invitation_response = (
await alice_member_client.post(
f"{OOB_BASE_PATH}/accept-invitation",
json={"alias": connection_alias, "invitation": invitation_json},
)
).json()
alice_label = alice_tenant.wallet_label
payload = await check_webhook_state(
client=verifier_client,
topic="connections",
state="completed",
filter_map={
"their_label": alice_label,
},
)
alice_connection_id = invitation_response["connection_id"]
acme_connection_id = payload["connection_id"]
# both connections should be active before continuing
await assert_both_connections_ready(
alice_member_client, verifier_client, alice_connection_id, acme_connection_id
)
return AcmeAliceConnect(
alice_connection_id=alice_connection_id, acme_connection_id=acme_connection_id
)
async def fetch_or_create_trust_registry_connection(
alice_member_client: RichAsyncClient,
alice_tenant: CreateTenantResponse,
verifier_client: RichAsyncClient,
verifier: CreateTenantResponse,
connection_alias: str,
) -> AcmeAliceConnect:
# fetch connection by alias for alice's side
alice_connection = await fetch_existing_connection_by_alias(
alice_member_client, alias=connection_alias
)
their_did = alice_connection.my_did if alice_connection else None
verifier_connection = await fetch_existing_connection_by_alias(
verifier_client,
alias=None,
their_label=alice_tenant.wallet_label,
their_did=their_did,
)
# Check if connections exist
if alice_connection and verifier_connection:
return AcmeAliceConnect(
alice_connection_id=alice_connection.connection_id,
acme_connection_id=verifier_connection.connection_id,
)
else:
assert not alice_connection, "Alice has connection, but not found for verifier"
assert not verifier_connection, "Verifier has connection, but not for Alice"
# Create connection since they don't exist
assert_fail_on_recreating_fixtures()
return await connect_using_trust_registry_invite(
alice_member_client=alice_member_client,
alice_tenant=alice_tenant,
verifier_client=verifier_client,
verifier=verifier,
connection_alias=connection_alias,
)
async def create_did_exchange(
bob_member_client: RichAsyncClient, alice_member_client: RichAsyncClient, alias: str
) -> BobAliceConnect:
# Get Bob's public DID. Bob is the issuer in this case i.e. should have public DID
did_response = (await bob_member_client.get(f"{DID_BASE_PATH}/public")).json()
bob_public_did = did_response["did"]
# Alice create invitation
alice_connection = (
await alice_member_client.post(
f"{CONNECTIONS_BASE_PATH}/did-exchange/create-request",
params={
"their_public_did": bob_public_did,
"alias": alias,
},
)
).json()
their_did = alice_connection["my_did"]
bob_connection = await check_webhook_state(
client=bob_member_client,
topic="connections",
state="request-received",
filter_map={"their_did": their_did},
)
bob_connection_id = bob_connection["connection_id"]
alice_connection_id = alice_connection["connection_id"]
# validate both connections should be active
await assert_both_connections_ready(
alice_member_client, bob_member_client, alice_connection_id, bob_connection_id
)
return BobAliceConnect(
alice_connection_id=alice_connection_id, bob_connection_id=bob_connection_id
)