forked from WebOfTrust/signify-ts
-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add integration test for aid to join multisig (WebOfTrust#258)
* add waitoperation * move waitOperation of join to before asserts for multisig join * update to latest --------- Co-authored-by: lenkan <[email protected]>
- Loading branch information
Showing
1 changed file
with
385 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,385 @@ | ||
import signify, { Serder, SignifyClient } from 'signify-ts'; | ||
import { getOrCreateClient, getOrCreateIdentifier } from './utils/test-setup'; | ||
import { | ||
markNotification, | ||
waitForNotifications, | ||
waitOperation, | ||
} from './utils/test-util'; | ||
import assert from 'assert'; | ||
|
||
describe('multisig-join', () => { | ||
const nameMember1 = 'member1'; | ||
const nameMember2 = 'member2'; | ||
const nameMember3 = 'member3'; | ||
const nameMultisig = 'multisigGroup'; | ||
|
||
let client1: SignifyClient; | ||
let client2: SignifyClient; | ||
let client3: SignifyClient; | ||
|
||
beforeAll(async () => { | ||
await signify.ready(); | ||
|
||
[client1, client2] = await Promise.all([ | ||
getOrCreateClient(), | ||
getOrCreateClient(), | ||
]); | ||
|
||
await Promise.all([ | ||
createAID(client1, nameMember1, []), | ||
createAID(client2, nameMember2, []), | ||
]); | ||
|
||
const [oobi1, oobi2] = await Promise.all([ | ||
client1.oobis().get(nameMember1, 'agent'), | ||
client2.oobis().get(nameMember2, 'agent'), | ||
]); | ||
|
||
const opOobi1 = await client1 | ||
.oobis() | ||
.resolve(oobi2.oobis[0], nameMember2); | ||
const opOobi2 = await client2 | ||
.oobis() | ||
.resolve(oobi1.oobis[0], nameMember1); | ||
await Promise.all([ | ||
waitOperation(client1, opOobi1), | ||
waitOperation(client2, opOobi2), | ||
]); | ||
}); | ||
|
||
test('should create multisig', async () => { | ||
const [aid1, aid2] = await Promise.all([ | ||
client1.identifiers().get(nameMember1), | ||
client2.identifiers().get(nameMember2), | ||
]); | ||
const states = [aid1.state, aid2.state]; | ||
const icpResult = await client1.identifiers().create(nameMultisig, { | ||
algo: signify.Algos.group, | ||
mhab: aid1, | ||
isith: 1, | ||
nsith: 1, | ||
toad: aid1.state.b.length, | ||
wits: aid1.state.b, | ||
states: states, | ||
rstates: states, | ||
}); | ||
|
||
const createMultisig1 = await icpResult.op(); | ||
|
||
const serder = icpResult.serder; | ||
|
||
const sigs = icpResult.sigs; | ||
const sigers = sigs.map((sig) => new signify.Siger({ qb64: sig })); | ||
|
||
const ims = signify.d(signify.messagize(serder, sigers)); | ||
const atc = ims.substring(serder.size); | ||
const embeds = { | ||
icp: [serder, atc], | ||
}; | ||
|
||
const smids = [aid2.state.i]; | ||
const recipients = [aid2.state.i]; | ||
|
||
await client1 | ||
.exchanges() | ||
.send( | ||
nameMember1, | ||
nameMultisig, | ||
aid1, | ||
'/multisig/icp', | ||
{ gid: serder.pre, smids: smids, rmids: smids }, | ||
embeds, | ||
recipients | ||
); | ||
|
||
const msgSaid = await waitAndMarkNotification(client2, '/multisig/icp'); | ||
|
||
const response = await client2.groups().getRequest(msgSaid); | ||
const icp = response[0].exn.e.icp; | ||
|
||
const icpResult2 = await client2.identifiers().create(nameMultisig, { | ||
algo: signify.Algos.group, | ||
mhab: aid2, | ||
isith: icp.kt, | ||
nsith: icp.nt, | ||
toad: parseInt(icp.bt), | ||
wits: icp.b, | ||
states, | ||
rstates: states, | ||
}); | ||
const createMultisig2 = await icpResult2.op(); | ||
|
||
const [createResult1, createResult2] = await Promise.all([ | ||
waitOperation(client1, createMultisig1), | ||
waitOperation(client2, createMultisig2), | ||
]); | ||
|
||
assert.equal(createResult1.response.k[0], aid1.state.k[0]); | ||
assert.equal(createResult1.response.k[1], aid2.state.k[0]); | ||
assert.equal(createResult2.response.k[0], aid1.state.k[0]); | ||
assert.equal(createResult2.response.k[1], aid2.state.k[0]); | ||
|
||
const members1 = await client1.identifiers().members(nameMultisig); | ||
const members2 = await client2.identifiers().members(nameMultisig); | ||
const eid1 = Object.keys(members1.signing[0].ends.agent)[0]; | ||
const eid2 = Object.keys(members2.signing[1].ends.agent)[0]; | ||
|
||
const [endRoleOperation1, endRoleOperation2] = await Promise.all([ | ||
client1.identifiers().addEndRole(nameMultisig, 'agent', eid1), | ||
client2.identifiers().addEndRole(nameMultisig, 'agent', eid2), | ||
]); | ||
|
||
await waitOperation(client1, await endRoleOperation1.op()); | ||
await waitOperation(client2, await endRoleOperation2.op()); | ||
}); | ||
|
||
test('should add member3 to multisig', async () => { | ||
client3 = await getOrCreateClient(); | ||
|
||
const aid3 = await createAID(client3, nameMember3, []); | ||
|
||
const [oobi1, oobi2, oobi3, oobi4] = await Promise.all([ | ||
client1.oobis().get(nameMember1, 'agent'), | ||
client2.oobis().get(nameMember2, 'agent'), | ||
client3.oobis().get(nameMember3, 'agent'), | ||
client1.oobis().get(nameMultisig, 'agent'), | ||
]); | ||
|
||
const oobiMultisig = oobi4.oobis[0].split('/agent/')[0]; | ||
|
||
const [opOobi1, opOobi2, opOobi3, opOobi4, opOobi5] = await Promise.all( | ||
[ | ||
client1.oobis().resolve(oobi3.oobis[0], nameMember3), | ||
client2.oobis().resolve(oobi3.oobis[0], nameMember3), | ||
client3.oobis().resolve(oobi1.oobis[0], nameMember1), | ||
client3.oobis().resolve(oobi2.oobis[0], nameMember2), | ||
client3.oobis().resolve(oobiMultisig, nameMultisig), | ||
] | ||
); | ||
await Promise.all([ | ||
waitOperation(client1, opOobi1), | ||
waitOperation(client2, opOobi2), | ||
waitOperation(client3, opOobi3), | ||
waitOperation(client3, opOobi4), | ||
waitOperation(client3, opOobi5), | ||
]); | ||
|
||
const [rotateResult1, rotateResult2] = await Promise.all([ | ||
client1.identifiers().rotate(nameMember1), | ||
client2.identifiers().rotate(nameMember2), | ||
]); | ||
|
||
await Promise.all([ | ||
waitOperation(client1, await rotateResult1.op()), | ||
waitOperation(client2, await rotateResult2.op()), | ||
]); | ||
|
||
const [aid1, aid2] = await Promise.all([ | ||
client1.identifiers().get(nameMember1), | ||
client2.identifiers().get(nameMember2), | ||
]); | ||
|
||
const updates = await Promise.all([ | ||
await client1.keyStates().query(aid2.prefix, '1'), | ||
await client1.keyStates().query(aid3.prefix, '0'), | ||
await client2.keyStates().query(aid1.prefix, '1'), | ||
await client2.keyStates().query(aid3.prefix, '0'), | ||
await client3.keyStates().query(aid1.prefix, '1'), | ||
await client3.keyStates().query(aid2.prefix, '1'), | ||
]); | ||
|
||
const [aid2State, aid3State, aid1State] = await Promise.all([ | ||
waitOperation(client1, updates[0]), | ||
waitOperation(client1, updates[1]), | ||
waitOperation(client2, updates[2]), | ||
waitOperation(client2, updates[3]), | ||
waitOperation(client3, updates[4]), | ||
waitOperation(client3, updates[5]), | ||
]); | ||
|
||
const states = [aid1State.response, aid2State.response]; | ||
const rstates = [...states, aid3State.response]; | ||
const rotateOperation1 = await client1 | ||
.identifiers() | ||
.rotate(nameMultisig, { states, rstates }); | ||
|
||
const serder1 = rotateOperation1.serder; | ||
const sigers = rotateOperation1.sigs.map( | ||
(sig) => new signify.Siger({ qb64: sig }) | ||
); | ||
const ims = signify.d(signify.messagize(serder1, sigers)); | ||
const atc = ims.substring(serder1.size); | ||
const rembeds = { | ||
rot: [serder1, atc], | ||
}; | ||
const smids = states.map((state) => state['i']); | ||
const rmids = rstates.map((state) => state['i']); | ||
const recp = [aid2.state, aid3.state].map((state) => state['i']); | ||
|
||
await client1 | ||
.exchanges() | ||
.send( | ||
nameMember1, | ||
nameMultisig, | ||
aid1, | ||
'/multisig/rot', | ||
{ gid: serder1.pre, smids, rmids }, | ||
rembeds, | ||
recp | ||
); | ||
|
||
await Promise.all([ | ||
waitAndMarkNotification(client2, '/multisig/rot'), | ||
waitAndMarkNotification(client3, '/multisig/rot'), | ||
]); | ||
|
||
const multisigAid = await client1.identifiers().get(nameMultisig); | ||
|
||
assert.equal(multisigAid.state.k.length, 2); | ||
assert.equal(multisigAid.state.k[0], aid1.state.k[0]); | ||
assert.equal(multisigAid.state.k[1], aid2.state.k[0]); | ||
|
||
assert.equal(multisigAid.state.n.length, 3); | ||
assert.equal(multisigAid.state.n[0], aid1.state.n[0]); | ||
assert.equal(multisigAid.state.n[1], aid2.state.n[0]); | ||
assert.equal(multisigAid.state.n[2], aid3.state.n[0]); | ||
}); | ||
test('Rotate again to get aid3 to current signing keys and join', async () => { | ||
const [rotateResult1, rotateResult2, rotateResult3] = await Promise.all( | ||
[ | ||
client1.identifiers().rotate(nameMember1), | ||
client2.identifiers().rotate(nameMember2), | ||
client3.identifiers().rotate(nameMember3), | ||
] | ||
); | ||
|
||
await Promise.all([ | ||
waitOperation(client1, await rotateResult1.op()), | ||
waitOperation(client2, await rotateResult2.op()), | ||
waitOperation(client3, await rotateResult3.op()), | ||
]); | ||
|
||
const [aid1, aid2, aid3] = await Promise.all([ | ||
client1.identifiers().get(nameMember1), | ||
client2.identifiers().get(nameMember2), | ||
client3.identifiers().get(nameMember3), | ||
]); | ||
|
||
const updates = await Promise.all([ | ||
await client1.keyStates().query(aid2.prefix, '2'), | ||
await client1.keyStates().query(aid3.prefix, '1'), | ||
await client2.keyStates().query(aid1.prefix, '2'), | ||
await client2.keyStates().query(aid3.prefix, '1'), | ||
await client3.keyStates().query(aid1.prefix, '2'), | ||
await client3.keyStates().query(aid2.prefix, '2'), | ||
]); | ||
|
||
const [aid2State, aid3State, aid1State] = await Promise.all([ | ||
waitOperation(client1, updates[0]), | ||
waitOperation(client1, updates[1]), | ||
waitOperation(client2, updates[2]), | ||
waitOperation(client2, updates[3]), | ||
waitOperation(client3, updates[4]), | ||
waitOperation(client3, updates[5]), | ||
]); | ||
|
||
const states = [ | ||
aid1State.response, | ||
aid2State.response, | ||
aid3State.response, | ||
]; | ||
const rotateOperation1 = await client1 | ||
.identifiers() | ||
.rotate(nameMultisig, { states, rstates: states }); | ||
|
||
const serder1 = rotateOperation1.serder; | ||
const sigers = rotateOperation1.sigs.map( | ||
(sig) => new signify.Siger({ qb64: sig }) | ||
); | ||
const ims = signify.d(signify.messagize(serder1, sigers)); | ||
const atc = ims.substring(serder1.size); | ||
const rembeds = { | ||
rot: [serder1, atc], | ||
}; | ||
const smids = states.map((state) => state['i']); | ||
const rmids = states.map((state) => state['i']); | ||
const recp = [aid2.state, aid3.state].map((state) => state['i']); | ||
|
||
await client1 | ||
.exchanges() | ||
.send( | ||
nameMember1, | ||
nameMultisig, | ||
aid1, | ||
'/multisig/rot', | ||
{ gid: serder1.pre, smids, rmids }, | ||
rembeds, | ||
recp | ||
); | ||
|
||
const rotationNotification3 = await waitAndMarkNotification( | ||
client3, | ||
'/multisig/rot' | ||
); | ||
|
||
const response = await client3 | ||
.groups() | ||
.getRequest(rotationNotification3); | ||
|
||
const exn3 = response[0].exn; | ||
const serder3 = new Serder(exn3.e.rot); | ||
const keeper3 = await client3.manager!.get(aid3); | ||
const sigs3 = keeper3.sign(signify.b(serder3.raw)); | ||
|
||
const joinOperation = await client3 | ||
.groups() | ||
.join(nameMultisig, serder3, sigs3, exn3.a.gid, smids, rmids); | ||
|
||
await waitOperation(client3, joinOperation); | ||
|
||
const multisigAid = await client3.identifiers().get(nameMultisig); | ||
|
||
assert.equal(multisigAid.state.k.length, 3); | ||
assert.equal(multisigAid.state.k[0], aid1.state.k[0]); | ||
assert.equal(multisigAid.state.k[1], aid2.state.k[0]); | ||
assert.equal(multisigAid.state.k[2], aid3.state.k[0]); | ||
|
||
assert.equal(multisigAid.state.n.length, 3); | ||
assert.equal(multisigAid.state.n[0], aid1.state.n[0]); | ||
assert.equal(multisigAid.state.n[1], aid2.state.n[0]); | ||
assert.equal(multisigAid.state.n[2], aid3.state.n[0]); | ||
|
||
const members = await client3.identifiers().members(nameMultisig); | ||
const eid = Object.keys(members.signing[2].ends.agent)[0]; | ||
const endRoleOperation = await client3 | ||
.identifiers() | ||
.addEndRole(nameMultisig, 'agent', eid); | ||
const endRoleResult = await waitOperation( | ||
client3, | ||
await endRoleOperation.op() | ||
); | ||
|
||
assert.equal(endRoleResult.done, true); | ||
assert.equal(endRoleResult.error, null); | ||
}); | ||
}); | ||
|
||
async function createAID(client: SignifyClient, name: string, wits: string[]) { | ||
await getOrCreateIdentifier(client, name, { | ||
wits: wits, | ||
toad: wits.length, | ||
}); | ||
return await client.identifiers().get(name); | ||
} | ||
|
||
async function waitAndMarkNotification(client: SignifyClient, route: string) { | ||
const notes = await waitForNotifications(client, route); | ||
|
||
await Promise.all( | ||
notes.map(async (note) => { | ||
await markNotification(client, note); | ||
}) | ||
); | ||
|
||
return notes[notes.length - 1]?.a.d ?? ''; | ||
} |