@bcts/frost-hubert 1.0.0-beta.0 → 1.0.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (149) hide show
  1. package/README.md +1 -1
  2. package/dist/bin/frost.cjs +17 -19
  3. package/dist/bin/frost.cjs.map +1 -1
  4. package/dist/bin/frost.mjs +16 -17
  5. package/dist/bin/frost.mjs.map +1 -1
  6. package/dist/{busy-EZU7EKr6.cjs → busy-B_h0bNAJ.cjs} +1 -1
  7. package/dist/{busy-DkM2jAIZ.mjs.map → busy-B_h0bNAJ.cjs.map} +1 -1
  8. package/dist/{busy-DkM2jAIZ.mjs → busy-BlU8_pS2.mjs} +1 -1
  9. package/dist/{busy-EZU7EKr6.cjs.map → busy-BlU8_pS2.mjs.map} +1 -1
  10. package/dist/cmd/index.cjs +6 -7
  11. package/dist/cmd/index.d.cts +1 -1
  12. package/dist/cmd/index.d.mts +1 -1
  13. package/dist/cmd/index.mjs +5 -6
  14. package/dist/{cmd-Bw9_i2_f.cjs → cmd-CCVhHzG7.cjs} +20 -21
  15. package/dist/{cmd-Bw9_i2_f.cjs.map → cmd-CCVhHzG7.cjs.map} +1 -1
  16. package/dist/{cmd-CS1uJtuD.mjs → cmd-DNsHd19v.mjs} +20 -21
  17. package/dist/{cmd-CS1uJtuD.mjs.map → cmd-DNsHd19v.mjs.map} +1 -1
  18. package/dist/{common-lThIvJmZ.cjs → common-7-BOgaTt.cjs} +2 -3
  19. package/dist/{common-lThIvJmZ.cjs.map → common-7-BOgaTt.cjs.map} +1 -1
  20. package/dist/{common-CvH6dFvQ.mjs → common-Cf1UvJaP.mjs} +3 -3
  21. package/dist/{common-CvH6dFvQ.mjs.map → common-Cf1UvJaP.mjs.map} +1 -1
  22. package/dist/{common-lKP5EzHy.cjs → common-CnvAUC2b.cjs} +3 -3
  23. package/dist/{common-lKP5EzHy.cjs.map → common-CnvAUC2b.cjs.map} +1 -1
  24. package/dist/{common-DUWvtc08.mjs → common-DNrD_-EI.mjs} +2 -2
  25. package/dist/{common-DUWvtc08.mjs.map → common-DNrD_-EI.mjs.map} +1 -1
  26. package/dist/dkg/index.cjs +1 -2
  27. package/dist/dkg/index.cjs.map +1 -1
  28. package/dist/dkg/index.d.cts +1 -1
  29. package/dist/dkg/index.d.mts +1 -1
  30. package/dist/dkg/index.mjs +1 -1
  31. package/dist/dkg/index.mjs.map +1 -1
  32. package/dist/{finalize-CNTDj6aS.mjs → finalize-BpC0rz93.mjs} +6 -6
  33. package/dist/{finalize-CNTDj6aS.mjs.map → finalize-BpC0rz93.mjs.map} +1 -1
  34. package/dist/{finalize-BRgJK-Xv.cjs → finalize-Cb0obTSo.cjs} +6 -6
  35. package/dist/{finalize-BRgJK-Xv.cjs.map → finalize-Cb0obTSo.cjs.map} +1 -1
  36. package/dist/{finalize-BfLgzn8f.cjs → finalize-DHEnKobp.cjs} +5 -5
  37. package/dist/{finalize-BfLgzn8f.cjs.map → finalize-DHEnKobp.cjs.map} +1 -1
  38. package/dist/{finalize-UPyI1yb1.cjs → finalize-DQ0VGUHO.cjs} +7 -7
  39. package/dist/{finalize-UPyI1yb1.cjs.map → finalize-DQ0VGUHO.cjs.map} +1 -1
  40. package/dist/{finalize-IA01t_Qq.mjs → finalize-DtRxHZ7H.mjs} +5 -5
  41. package/dist/{finalize-IA01t_Qq.mjs.map → finalize-DtRxHZ7H.mjs.map} +1 -1
  42. package/dist/{finalize-EC3ikHQq.mjs → finalize-T83Ko8nG.mjs} +6 -6
  43. package/dist/{finalize-EC3ikHQq.mjs.map → finalize-T83Ko8nG.mjs.map} +1 -1
  44. package/dist/frost/index.cjs +1 -1
  45. package/dist/frost/index.cjs.map +1 -1
  46. package/dist/frost/index.d.cts.map +1 -1
  47. package/dist/frost/index.d.mts.map +1 -1
  48. package/dist/frost/index.mjs +1 -1
  49. package/dist/frost/index.mjs.map +1 -1
  50. package/dist/{index-F1iNEAJR.d.cts → index-BErX9AZF.d.cts} +3 -3
  51. package/dist/index-BErX9AZF.d.cts.map +1 -0
  52. package/dist/{index-B3c-80VS.d.cts → index-BaUVw4b1.d.mts} +2 -2
  53. package/dist/index-BaUVw4b1.d.mts.map +1 -0
  54. package/dist/{index-C8QeHNwa.d.cts → index-CD50Qtgw.d.cts} +1 -1
  55. package/dist/index-CD50Qtgw.d.cts.map +1 -0
  56. package/dist/{index-DVbWyOs7.d.mts → index-CD50Qtgw.d.mts} +1 -1
  57. package/dist/index-CD50Qtgw.d.mts.map +1 -0
  58. package/dist/{index-BgbSGpxn.d.mts → index-Drklne-Y.d.mts} +3 -3
  59. package/dist/index-Drklne-Y.d.mts.map +1 -0
  60. package/dist/{index-D3QTWkEm.d.mts → index-gkmZzEuD.d.cts} +2 -2
  61. package/dist/index-gkmZzEuD.d.cts.map +1 -0
  62. package/dist/index.cjs +7 -8
  63. package/dist/index.cjs.map +1 -1
  64. package/dist/index.d.cts +3 -3
  65. package/dist/index.d.cts.map +1 -1
  66. package/dist/index.d.mts +3 -3
  67. package/dist/index.d.mts.map +1 -1
  68. package/dist/index.mjs +6 -7
  69. package/dist/index.mjs.map +1 -1
  70. package/dist/{invite-5277FQVT.cjs → invite-1tzg0B0P.cjs} +5 -5
  71. package/dist/{invite-5277FQVT.cjs.map → invite-1tzg0B0P.cjs.map} +1 -1
  72. package/dist/{invite-DUTcfTgX.cjs → invite-BLwtexAu.cjs} +4 -4
  73. package/dist/{invite-DUTcfTgX.cjs.map → invite-BLwtexAu.cjs.map} +1 -1
  74. package/dist/{invite-IU4n0dq2.mjs → invite-Be2v2SVc.mjs} +4 -4
  75. package/dist/{invite-IU4n0dq2.mjs.map → invite-Be2v2SVc.mjs.map} +1 -1
  76. package/dist/{invite-RU-OXTNS.mjs → invite-D8mQSnFz.mjs} +5 -5
  77. package/dist/{invite-RU-OXTNS.mjs.map → invite-D8mQSnFz.mjs.map} +1 -1
  78. package/dist/{parallel-D6zc6VW4.mjs → parallel-PZiwHZT8.mjs} +1 -1
  79. package/dist/{parallel-D1R6ZGlY.cjs.map → parallel-PZiwHZT8.mjs.map} +1 -1
  80. package/dist/{parallel-D1R6ZGlY.cjs → parallel-szwYx-bi.cjs} +1 -1
  81. package/dist/{parallel-D6zc6VW4.mjs.map → parallel-szwYx-bi.cjs.map} +1 -1
  82. package/dist/{proposed-participant-Dm1Eq6mX.cjs → proposed-participant-BvHNnpcZ.cjs} +1 -2
  83. package/dist/{proposed-participant-Dm1Eq6mX.cjs.map → proposed-participant-BvHNnpcZ.cjs.map} +1 -1
  84. package/dist/{proposed-participant-cWM7iUrO.mjs → proposed-participant-Detb823_.mjs} +1 -1
  85. package/dist/{proposed-participant-cWM7iUrO.mjs.map → proposed-participant-Detb823_.mjs.map} +1 -1
  86. package/dist/{receive-CAI-x4II.cjs → receive-BR-knnGv.cjs} +6 -6
  87. package/dist/{receive-CAI-x4II.cjs.map → receive-BR-knnGv.cjs.map} +1 -1
  88. package/dist/{receive-kZMsXhbK.cjs → receive-D_r4Mryr.cjs} +6 -6
  89. package/dist/{receive-kZMsXhbK.cjs.map → receive-D_r4Mryr.cjs.map} +1 -1
  90. package/dist/{receive-D2Nn68L7.mjs → receive-dkSCSGpl.mjs} +5 -5
  91. package/dist/{receive-D2Nn68L7.mjs.map → receive-dkSCSGpl.mjs.map} +1 -1
  92. package/dist/{receive-DA_KQEgk.mjs → receive-g8EhZF2Y.mjs} +6 -6
  93. package/dist/{receive-DA_KQEgk.mjs.map → receive-g8EhZF2Y.mjs.map} +1 -1
  94. package/dist/registry/index.cjs +1 -1
  95. package/dist/registry/index.cjs.map +1 -1
  96. package/dist/registry/index.d.cts +1 -1
  97. package/dist/registry/index.d.mts +1 -1
  98. package/dist/registry/index.mjs.map +1 -1
  99. package/dist/{registry-9puTaRrD.cjs → registry-CkIbA7nt.cjs} +79 -2
  100. package/dist/registry-CkIbA7nt.cjs.map +1 -0
  101. package/dist/{registry-BpCwtrRt.mjs → registry-DGjs4qDK.mjs} +74 -3
  102. package/dist/registry-DGjs4qDK.mjs.map +1 -0
  103. package/dist/{round1-BHBjru1m.cjs → round1-9FAqFvL5.cjs} +5 -5
  104. package/dist/{round1-BHBjru1m.cjs.map → round1-9FAqFvL5.cjs.map} +1 -1
  105. package/dist/{round1-CcQCGlIT.mjs → round1-B8haiMM8.mjs} +6 -6
  106. package/dist/{round1-CcQCGlIT.mjs.map → round1-B8haiMM8.mjs.map} +1 -1
  107. package/dist/{round1-Cgm7j1kI.mjs → round1-BOIE1E4O.mjs} +5 -5
  108. package/dist/{round1-Cgm7j1kI.mjs.map → round1-BOIE1E4O.mjs.map} +1 -1
  109. package/dist/{round1-4Hyx8w0x.cjs → round1-Bq0vweyQ.cjs} +5 -5
  110. package/dist/{round1-4Hyx8w0x.cjs.map → round1-Bq0vweyQ.cjs.map} +1 -1
  111. package/dist/{round1-CWSXZx5R.cjs → round1-CXkXoVQU.cjs} +9 -9
  112. package/dist/{round1-CWSXZx5R.cjs.map → round1-CXkXoVQU.cjs.map} +1 -1
  113. package/dist/{round1-7v9LlE11.mjs → round1-D8t7EzIo.mjs} +5 -5
  114. package/dist/{round1-7v9LlE11.mjs.map → round1-D8t7EzIo.mjs.map} +1 -1
  115. package/dist/{round1-DQ0fnc1H.cjs → round1-DriPu15x.cjs} +7 -7
  116. package/dist/{round1-DQ0fnc1H.cjs.map → round1-DriPu15x.cjs.map} +1 -1
  117. package/dist/{round1-CMLKN2RR.mjs → round1-Y2kcVwnR.mjs} +7 -7
  118. package/dist/{round1-CMLKN2RR.mjs.map → round1-Y2kcVwnR.mjs.map} +1 -1
  119. package/dist/{round2-BWz9SQIi.cjs → round2-AMDYMUIg.cjs} +5 -5
  120. package/dist/{round2-BWz9SQIi.cjs.map → round2-AMDYMUIg.cjs.map} +1 -1
  121. package/dist/{round2-o2Q-GMbX.cjs → round2-BHQKVJFo.cjs} +7 -7
  122. package/dist/{round2-o2Q-GMbX.cjs.map → round2-BHQKVJFo.cjs.map} +1 -1
  123. package/dist/{round2-Bl2uK93U.mjs → round2-BfetYacV.mjs} +5 -5
  124. package/dist/{round2-Bl2uK93U.mjs.map → round2-BfetYacV.mjs.map} +1 -1
  125. package/dist/{round2-Dg24w-TU.mjs → round2-Cf5CJc_8.mjs} +7 -7
  126. package/dist/{round2-Dg24w-TU.mjs.map → round2-Cf5CJc_8.mjs.map} +1 -1
  127. package/dist/{round2-LylCa84n.cjs → round2-CvrmylN1.cjs} +7 -7
  128. package/dist/{round2-LylCa84n.cjs.map → round2-CvrmylN1.cjs.map} +1 -1
  129. package/dist/{round2-CdUT-AhH.cjs → round2-Dk_w97nl.cjs} +5 -5
  130. package/dist/{round2-CdUT-AhH.cjs.map → round2-Dk_w97nl.cjs.map} +1 -1
  131. package/dist/{round2-BkNRCXgS.mjs → round2-Z2JhMwxc.mjs} +5 -5
  132. package/dist/{round2-BkNRCXgS.mjs.map → round2-Z2JhMwxc.mjs.map} +1 -1
  133. package/dist/{round2-DOA3rnV-.mjs → round2-mF6UlkT-.mjs} +6 -6
  134. package/dist/{round2-DOA3rnV-.mjs.map → round2-mF6UlkT-.mjs.map} +1 -1
  135. package/package.json +14 -14
  136. package/dist/index-B3c-80VS.d.cts.map +0 -1
  137. package/dist/index-BgbSGpxn.d.mts.map +0 -1
  138. package/dist/index-C8QeHNwa.d.cts.map +0 -1
  139. package/dist/index-D3QTWkEm.d.mts.map +0 -1
  140. package/dist/index-DVbWyOs7.d.mts.map +0 -1
  141. package/dist/index-F1iNEAJR.d.cts.map +0 -1
  142. package/dist/registry-9puTaRrD.cjs.map +0 -1
  143. package/dist/registry-BpCwtrRt.mjs.map +0 -1
  144. package/dist/storage-B-Gu68-O.cjs +0 -79
  145. package/dist/storage-B-Gu68-O.cjs.map +0 -1
  146. package/dist/storage-Bkkliz0K.mjs +0 -74
  147. package/dist/storage-Bkkliz0K.mjs.map +0 -1
  148. /package/dist/{chunk-CZWwpsFl.cjs → chunk-DakpK96I.cjs} +0 -0
  149. /package/dist/{chunk-CjcI7cDX.mjs → chunk-z9aeyW2b.mjs} +0 -0
@@ -1 +1 @@
1
- {"version":3,"file":"receive-D2Nn68L7.mjs","names":["EnvelopeFunction"],"sources":["../src/cmd/dkg/participant/receive.ts"],"sourcesContent":["/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * DKG participant receive command.\n *\n * Port of cmd/dkg/participant/receive.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\n\nimport { ARID, XID } from \"@bcts/components\";\nimport { CborDate } from \"@bcts/dcbor\";\nimport { type Envelope, Function as EnvelopeFunction } from \"@bcts/envelope\";\nimport { SealedRequest } from \"@bcts/gstp\";\nimport { XIDDocument, XIDVerifySignature } from \"@bcts/xid\";\n\nimport { DkgInvitation } from \"../../../dkg/index.js\";\nimport { Registry, resolveRegistryPath } from \"../../../registry/index.js\";\nimport { getWithIndicator } from \"../../busy.js\";\nimport { createStorageClient, type StorageClient, type StorageSelection } from \"../../storage.js\";\nimport {\n dkgStateDir,\n parseAridUr,\n parseEnvelopeUr,\n participantNamesFromRegistry,\n resolveSenderName,\n} from \"../common.js\";\n\n/**\n * Options for the DKG receive command.\n */\nexport interface DkgReceiveOptions {\n registryPath?: string;\n timeoutSeconds?: number;\n noEnvelope?: boolean;\n info?: boolean;\n sender?: string;\n invite: string;\n storageSelection?: StorageSelection;\n verbose?: boolean;\n}\n\n/**\n * Result of the DKG receive command.\n */\nexport interface DkgReceiveResult {\n groupId: string;\n requestId: string;\n minSigners: number;\n charter: string;\n validUntil: string;\n responseArid: string;\n envelopeUr?: string | undefined;\n coordinatorName?: string | undefined;\n participantNames?: string[] | undefined;\n}\n\n/**\n * Details extracted from a DKG invite.\n *\n * Port of `struct InviteDetails` from cmd/dkg/participant/receive.rs lines 117-120.\n */\nexport interface InviteDetails {\n invitation: DkgInvitation;\n participants: XIDDocument[];\n}\n\n/**\n * Resolve an invite envelope from either storage (ARID) or direct UR.\n *\n * Port of `resolve_invite_envelope()` from cmd/dkg/participant/receive.rs lines 122-152.\n */\nexport async function resolveInviteEnvelope(\n selection: StorageSelection | undefined,\n invite: string,\n timeout?: number,\n): Promise<Envelope> {\n if (selection !== undefined) {\n // Try to parse as ARID\n try {\n const arid = parseAridUr(invite);\n const client = await createStorageClient(selection);\n const envelope = await getWithIndicator(client, arid, \"Invite\", timeout, false);\n if (envelope === null || envelope === undefined) {\n throw new Error(\"Invite not found in Hubert storage\");\n }\n return envelope;\n } catch (e) {\n // Not an ARID, fall through to envelope parsing\n if (e instanceof Error && e.message.includes(\"Invite not found in Hubert storage\")) {\n throw e;\n }\n }\n\n if (timeout !== undefined) {\n throw new Error(\"--timeout is only valid when retrieving invites from Hubert\");\n }\n\n return parseEnvelopeUr(invite);\n }\n\n // No storage selection\n try {\n parseAridUr(invite);\n throw new Error(\"Hubert storage parameters are required to retrieve invites by ARID\");\n } catch (e) {\n // Not an ARID, parse as envelope\n if (e instanceof Error && e.message.includes(\"Hubert storage parameters are required\")) {\n throw e;\n }\n }\n\n return parseEnvelopeUr(invite);\n}\n\n/**\n * Decode and validate invite details from an envelope.\n *\n * Port of `decode_invite_details()` from cmd/dkg/participant/receive.rs lines 154-256.\n */\nexport function decodeInviteDetails(\n invite: Envelope,\n now: Date,\n registry: Registry,\n recipient: XIDDocument,\n expectedSender?: XIDDocument,\n): InviteDetails {\n const recipientPrivateKeys = recipient.inceptionPrivateKeys();\n\n if (recipientPrivateKeys === undefined) {\n throw new Error(\"Recipient XID document has no inception private keys\");\n }\n\n const sealedRequest = SealedRequest.tryFromEnvelope(invite, undefined, now, recipientPrivateKeys);\n\n const senderDocument = sealedRequest.sender();\n if (expectedSender !== undefined) {\n if (senderDocument.xid().urString() !== expectedSender.xid().urString()) {\n throw new Error(\"Invite sender does not match expected sender\");\n }\n } else {\n const senderXid = senderDocument.xid();\n const owner = registry.owner();\n const knownOwner = owner?.xidDocument().xid().urString() === senderXid.urString();\n const knownParticipant = registry.participant(senderXid) !== undefined;\n\n if (!knownOwner && !knownParticipant) {\n throw new Error(`Invite sender not found in registry: ${senderXid.urString()}`);\n }\n }\n\n if (!sealedRequest.request().function().equals(EnvelopeFunction.fromString(\"dkgInvite\"))) {\n throw new Error(\"Unexpected invite function\");\n }\n\n const validUntil = sealedRequest.extractObjectForParameter<Date>(\"validUntil\");\n if (validUntil <= now) {\n throw new Error(\"Invitation expired\");\n }\n\n const minSigners = sealedRequest.extractObjectForParameter<number>(\"minSigners\");\n sealedRequest.extractObjectForParameter<string>(\"charter\");\n sealedRequest.extractObjectForParameter<ARID>(\"group\");\n const participantObjects = sealedRequest.objectsForParameter(\"participant\");\n\n if (minSigners < 2) {\n throw new Error(\"min_signers must be at least 2\");\n }\n\n if (minSigners > participantObjects.length) {\n throw new Error(\"min_signers exceeds participant count\");\n }\n\n const participantDocs: XIDDocument[] = [];\n let responseArid: ARID | undefined;\n const recipientXid = recipient.xid();\n\n for (const participant of participantObjects) {\n const xidDocumentEnvelope = participant.tryUnwrap();\n const xidDocument = XIDDocument.fromEnvelope(\n xidDocumentEnvelope,\n undefined,\n XIDVerifySignature.Inception,\n );\n\n if (xidDocument.xid().urString() === recipientXid.urString()) {\n const encryptedResponseArid = participant.objectForPredicate(\"response_arid\");\n const responseAridEnvelope = encryptedResponseArid.decryptToRecipient(recipientPrivateKeys);\n responseArid = responseAridEnvelope.extractSubject((cbor) => ARID.fromTaggedCbor(cbor));\n }\n\n participantDocs.push(xidDocument);\n }\n\n const invitation = DkgInvitation.fromInvite(invite, now, expectedSender, recipient);\n\n if (responseArid === undefined) {\n throw new Error(\"Invite does not include a response ARID for this recipient\");\n }\n\n return { invitation, participants: participantDocs };\n}\n\n/**\n * Resolve a sender XID document from the registry by UR or pet name.\n *\n * Port of `resolve_sender()` usage in receive.rs for expected sender.\n */\nfunction resolveSenderXidDocument(registry: Registry, raw: string): XIDDocument {\n // Try parsing as XID UR first\n try {\n const xid = XID.fromURString(raw.trim());\n const record = registry.participant(xid);\n if (record) {\n return record.xidDocument();\n }\n const owner = registry.owner();\n if (owner?.xid().urString() === xid.urString()) {\n return owner.xidDocument();\n }\n throw new Error(`Sender with XID ${xid.urString()} not found in registry`);\n } catch {\n // Try looking up by pet name\n const result = registry.participantByPetName(raw.trim());\n if (result) {\n const [, record] = result;\n return record.xidDocument();\n }\n const owner = registry.owner();\n if (owner?.petName() === raw.trim()) {\n return owner.xidDocument();\n }\n throw new Error(`Sender '${raw}' not found in registry`);\n }\n}\n\n/**\n * Execute the DKG participant receive command.\n *\n * Fetches and validates a DKG invite from the coordinator.\n *\n * Port of `receive()` from cmd/dkg/participant/receive.rs.\n */\nexport async function receive(\n _client: StorageClient | undefined,\n options: DkgReceiveOptions,\n cwd: string,\n): Promise<DkgReceiveResult> {\n if (options.storageSelection === undefined && options.timeoutSeconds !== undefined) {\n throw new Error(\"--timeout requires Hubert storage parameters\");\n }\n\n const registryPath = resolveRegistryPath(options.registryPath, cwd);\n const registry = Registry.load(registryPath);\n\n const owner = registry.owner();\n if (!owner) {\n throw new Error(\"Registry owner with private keys is required\");\n }\n\n const expectedSender = options.sender\n ? resolveSenderXidDocument(registry, options.sender)\n : undefined;\n\n const inviteEnvelope = await resolveInviteEnvelope(\n options.storageSelection,\n options.invite,\n options.timeoutSeconds,\n );\n\n const now = CborDate.now().datetime();\n const details = decodeInviteDetails(\n inviteEnvelope,\n now,\n registry,\n owner.xidDocument(),\n expectedSender,\n );\n\n const participantNames = participantNamesFromRegistry(\n registry,\n details.participants,\n owner.xid(),\n owner.petName(),\n );\n\n const coordinatorName = resolveSenderName(registry, details.invitation.sender());\n\n // Save receive state\n const stateDir = dkgStateDir(registryPath, details.invitation.groupId().hex());\n fs.mkdirSync(stateDir, { recursive: true });\n\n const validUntilStr: string = (\n details.invitation.validUntil() as { toString(): string }\n ).toString();\n const receiveState = {\n group: details.invitation.groupId().urString(),\n request_id: details.invitation.requestId().urString(),\n response_arid: details.invitation.responseArid().urString(),\n valid_until: validUntilStr,\n min_signers: details.invitation.minSigners(),\n charter: details.invitation.charter(),\n sender: details.invitation.sender().xid().urString(),\n };\n\n fs.writeFileSync(path.join(stateDir, \"receive.json\"), JSON.stringify(receiveState, null, 2));\n\n // Output based on options\n let envelopeUr: string | undefined;\n if (options.noEnvelope !== true) {\n envelopeUr = inviteEnvelope.urString();\n console.log(envelopeUr);\n }\n\n if (options.info === true) {\n console.error(`Charter: ${details.invitation.charter()}`);\n console.error(`Min signers: ${details.invitation.minSigners()}`);\n if (coordinatorName !== undefined) {\n console.error(`Coordinator: ${coordinatorName}`);\n }\n console.error(`Participants: ${participantNames.join(\", \")}`);\n }\n\n if (options.verbose === true) {\n console.log(`Group ID: ${details.invitation.groupId().urString()}`);\n console.log(`Min signers: ${details.invitation.minSigners()}`);\n console.log(`Charter: ${details.invitation.charter()}`);\n console.log(`Valid until: ${String(details.invitation.validUntil())}`);\n console.log(`Response ARID: ${details.invitation.responseArid().urString()}`);\n }\n\n const resultValidUntilStr: string = (\n details.invitation.validUntil() as { toString(): string }\n ).toString();\n\n return {\n groupId: details.invitation.groupId().urString(),\n requestId: details.invitation.requestId().urString(),\n minSigners: details.invitation.minSigners(),\n charter: details.invitation.charter(),\n validUntil: resultValidUntilStr,\n responseArid: details.invitation.responseArid().urString(),\n envelopeUr,\n coordinatorName,\n participantNames,\n };\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"receive-dkSCSGpl.mjs","names":["EnvelopeFunction"],"sources":["../src/cmd/dkg/participant/receive.ts"],"sourcesContent":["/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * DKG participant receive command.\n *\n * Port of cmd/dkg/participant/receive.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\n\nimport { ARID, XID } from \"@bcts/components\";\nimport { CborDate } from \"@bcts/dcbor\";\nimport { type Envelope, Function as EnvelopeFunction } from \"@bcts/envelope\";\nimport { SealedRequest } from \"@bcts/gstp\";\nimport { XIDDocument, XIDVerifySignature } from \"@bcts/xid\";\n\nimport { DkgInvitation } from \"../../../dkg/index.js\";\nimport { Registry, resolveRegistryPath } from \"../../../registry/index.js\";\nimport { getWithIndicator } from \"../../busy.js\";\nimport { createStorageClient, type StorageClient, type StorageSelection } from \"../../storage.js\";\nimport {\n dkgStateDir,\n parseAridUr,\n parseEnvelopeUr,\n participantNamesFromRegistry,\n resolveSenderName,\n} from \"../common.js\";\n\n/**\n * Options for the DKG receive command.\n */\nexport interface DkgReceiveOptions {\n registryPath?: string;\n timeoutSeconds?: number;\n noEnvelope?: boolean;\n info?: boolean;\n sender?: string;\n invite: string;\n storageSelection?: StorageSelection;\n verbose?: boolean;\n}\n\n/**\n * Result of the DKG receive command.\n */\nexport interface DkgReceiveResult {\n groupId: string;\n requestId: string;\n minSigners: number;\n charter: string;\n validUntil: string;\n responseArid: string;\n envelopeUr?: string | undefined;\n coordinatorName?: string | undefined;\n participantNames?: string[] | undefined;\n}\n\n/**\n * Details extracted from a DKG invite.\n *\n * Port of `struct InviteDetails` from cmd/dkg/participant/receive.rs lines 117-120.\n */\nexport interface InviteDetails {\n invitation: DkgInvitation;\n participants: XIDDocument[];\n}\n\n/**\n * Resolve an invite envelope from either storage (ARID) or direct UR.\n *\n * Port of `resolve_invite_envelope()` from cmd/dkg/participant/receive.rs lines 122-152.\n */\nexport async function resolveInviteEnvelope(\n selection: StorageSelection | undefined,\n invite: string,\n timeout?: number,\n): Promise<Envelope> {\n if (selection !== undefined) {\n // Try to parse as ARID\n try {\n const arid = parseAridUr(invite);\n const client = await createStorageClient(selection);\n const envelope = await getWithIndicator(client, arid, \"Invite\", timeout, false);\n if (envelope === null || envelope === undefined) {\n throw new Error(\"Invite not found in Hubert storage\");\n }\n return envelope;\n } catch (e) {\n // Not an ARID, fall through to envelope parsing\n if (e instanceof Error && e.message.includes(\"Invite not found in Hubert storage\")) {\n throw e;\n }\n }\n\n if (timeout !== undefined) {\n throw new Error(\"--timeout is only valid when retrieving invites from Hubert\");\n }\n\n return parseEnvelopeUr(invite);\n }\n\n // No storage selection\n try {\n parseAridUr(invite);\n throw new Error(\"Hubert storage parameters are required to retrieve invites by ARID\");\n } catch (e) {\n // Not an ARID, parse as envelope\n if (e instanceof Error && e.message.includes(\"Hubert storage parameters are required\")) {\n throw e;\n }\n }\n\n return parseEnvelopeUr(invite);\n}\n\n/**\n * Decode and validate invite details from an envelope.\n *\n * Port of `decode_invite_details()` from cmd/dkg/participant/receive.rs lines 154-256.\n */\nexport function decodeInviteDetails(\n invite: Envelope,\n now: Date,\n registry: Registry,\n recipient: XIDDocument,\n expectedSender?: XIDDocument,\n): InviteDetails {\n const recipientPrivateKeys = recipient.inceptionPrivateKeys();\n\n if (recipientPrivateKeys === undefined) {\n throw new Error(\"Recipient XID document has no inception private keys\");\n }\n\n const sealedRequest = SealedRequest.tryFromEnvelope(invite, undefined, now, recipientPrivateKeys);\n\n const senderDocument = sealedRequest.sender();\n if (expectedSender !== undefined) {\n if (senderDocument.xid().urString() !== expectedSender.xid().urString()) {\n throw new Error(\"Invite sender does not match expected sender\");\n }\n } else {\n const senderXid = senderDocument.xid();\n const owner = registry.owner();\n const knownOwner = owner?.xidDocument().xid().urString() === senderXid.urString();\n const knownParticipant = registry.participant(senderXid) !== undefined;\n\n if (!knownOwner && !knownParticipant) {\n throw new Error(`Invite sender not found in registry: ${senderXid.urString()}`);\n }\n }\n\n if (!sealedRequest.request().function().equals(EnvelopeFunction.fromString(\"dkgInvite\"))) {\n throw new Error(\"Unexpected invite function\");\n }\n\n const validUntil = sealedRequest.extractObjectForParameter<Date>(\"validUntil\");\n if (validUntil <= now) {\n throw new Error(\"Invitation expired\");\n }\n\n const minSigners = sealedRequest.extractObjectForParameter<number>(\"minSigners\");\n sealedRequest.extractObjectForParameter<string>(\"charter\");\n sealedRequest.extractObjectForParameter<ARID>(\"group\");\n const participantObjects = sealedRequest.objectsForParameter(\"participant\");\n\n if (minSigners < 2) {\n throw new Error(\"min_signers must be at least 2\");\n }\n\n if (minSigners > participantObjects.length) {\n throw new Error(\"min_signers exceeds participant count\");\n }\n\n const participantDocs: XIDDocument[] = [];\n let responseArid: ARID | undefined;\n const recipientXid = recipient.xid();\n\n for (const participant of participantObjects) {\n const xidDocumentEnvelope = participant.tryUnwrap();\n const xidDocument = XIDDocument.fromEnvelope(\n xidDocumentEnvelope,\n undefined,\n XIDVerifySignature.Inception,\n );\n\n if (xidDocument.xid().urString() === recipientXid.urString()) {\n const encryptedResponseArid = participant.objectForPredicate(\"response_arid\");\n const responseAridEnvelope = encryptedResponseArid.decryptToRecipient(recipientPrivateKeys);\n responseArid = responseAridEnvelope.extractSubject((cbor) => ARID.fromTaggedCbor(cbor));\n }\n\n participantDocs.push(xidDocument);\n }\n\n const invitation = DkgInvitation.fromInvite(invite, now, expectedSender, recipient);\n\n if (responseArid === undefined) {\n throw new Error(\"Invite does not include a response ARID for this recipient\");\n }\n\n return { invitation, participants: participantDocs };\n}\n\n/**\n * Resolve a sender XID document from the registry by UR or pet name.\n *\n * Port of `resolve_sender()` usage in receive.rs for expected sender.\n */\nfunction resolveSenderXidDocument(registry: Registry, raw: string): XIDDocument {\n // Try parsing as XID UR first\n try {\n const xid = XID.fromURString(raw.trim());\n const record = registry.participant(xid);\n if (record) {\n return record.xidDocument();\n }\n const owner = registry.owner();\n if (owner?.xid().urString() === xid.urString()) {\n return owner.xidDocument();\n }\n throw new Error(`Sender with XID ${xid.urString()} not found in registry`);\n } catch {\n // Try looking up by pet name\n const result = registry.participantByPetName(raw.trim());\n if (result) {\n const [, record] = result;\n return record.xidDocument();\n }\n const owner = registry.owner();\n if (owner?.petName() === raw.trim()) {\n return owner.xidDocument();\n }\n throw new Error(`Sender '${raw}' not found in registry`);\n }\n}\n\n/**\n * Execute the DKG participant receive command.\n *\n * Fetches and validates a DKG invite from the coordinator.\n *\n * Port of `receive()` from cmd/dkg/participant/receive.rs.\n */\nexport async function receive(\n _client: StorageClient | undefined,\n options: DkgReceiveOptions,\n cwd: string,\n): Promise<DkgReceiveResult> {\n if (options.storageSelection === undefined && options.timeoutSeconds !== undefined) {\n throw new Error(\"--timeout requires Hubert storage parameters\");\n }\n\n const registryPath = resolveRegistryPath(options.registryPath, cwd);\n const registry = Registry.load(registryPath);\n\n const owner = registry.owner();\n if (!owner) {\n throw new Error(\"Registry owner with private keys is required\");\n }\n\n const expectedSender = options.sender\n ? resolveSenderXidDocument(registry, options.sender)\n : undefined;\n\n const inviteEnvelope = await resolveInviteEnvelope(\n options.storageSelection,\n options.invite,\n options.timeoutSeconds,\n );\n\n const now = CborDate.now().datetime();\n const details = decodeInviteDetails(\n inviteEnvelope,\n now,\n registry,\n owner.xidDocument(),\n expectedSender,\n );\n\n const participantNames = participantNamesFromRegistry(\n registry,\n details.participants,\n owner.xid(),\n owner.petName(),\n );\n\n const coordinatorName = resolveSenderName(registry, details.invitation.sender());\n\n // Save receive state\n const stateDir = dkgStateDir(registryPath, details.invitation.groupId().hex());\n fs.mkdirSync(stateDir, { recursive: true });\n\n const validUntilStr: string = (\n details.invitation.validUntil() as { toString(): string }\n ).toString();\n const receiveState = {\n group: details.invitation.groupId().urString(),\n request_id: details.invitation.requestId().urString(),\n response_arid: details.invitation.responseArid().urString(),\n valid_until: validUntilStr,\n min_signers: details.invitation.minSigners(),\n charter: details.invitation.charter(),\n sender: details.invitation.sender().xid().urString(),\n };\n\n fs.writeFileSync(path.join(stateDir, \"receive.json\"), JSON.stringify(receiveState, null, 2));\n\n // Output based on options\n let envelopeUr: string | undefined;\n if (options.noEnvelope !== true) {\n envelopeUr = inviteEnvelope.urString();\n console.log(envelopeUr);\n }\n\n if (options.info === true) {\n console.error(`Charter: ${details.invitation.charter()}`);\n console.error(`Min signers: ${details.invitation.minSigners()}`);\n if (coordinatorName !== undefined) {\n console.error(`Coordinator: ${coordinatorName}`);\n }\n console.error(`Participants: ${participantNames.join(\", \")}`);\n }\n\n if (options.verbose === true) {\n console.log(`Group ID: ${details.invitation.groupId().urString()}`);\n console.log(`Min signers: ${details.invitation.minSigners()}`);\n console.log(`Charter: ${details.invitation.charter()}`);\n console.log(`Valid until: ${String(details.invitation.validUntil())}`);\n console.log(`Response ARID: ${details.invitation.responseArid().urString()}`);\n }\n\n const resultValidUntilStr: string = (\n details.invitation.validUntil() as { toString(): string }\n ).toString();\n\n return {\n groupId: details.invitation.groupId().urString(),\n requestId: details.invitation.requestId().urString(),\n minSigners: details.invitation.minSigners(),\n charter: details.invitation.charter(),\n validUntil: resultValidUntilStr,\n responseArid: details.invitation.responseArid().urString(),\n envelopeUr,\n coordinatorName,\n participantNames,\n };\n}\n"],"mappings":"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"}
@@ -1,9 +1,9 @@
1
- import { n as __require, t as __exportAll } from "./chunk-CjcI7cDX.mjs";
2
- import { n as compareXidBytes } from "./proposed-participant-cWM7iUrO.mjs";
1
+ import { n as __require, t as __exportAll } from "./chunk-z9aeyW2b.mjs";
2
+ import { n as compareXidBytes } from "./proposed-participant-Detb823_.mjs";
3
3
  import { Registry, resolveRegistryPath } from "./registry/index.mjs";
4
- import { c as parseAridUr, l as parseEnvelopeUr, o as formatNameWithOwnerMarker } from "./common-CvH6dFvQ.mjs";
5
- import { t as getWithIndicator } from "./busy-DkM2jAIZ.mjs";
6
- import { n as signingStateDir } from "./common-DUWvtc08.mjs";
4
+ import { c as parseAridUr, l as parseEnvelopeUr, o as formatNameWithOwnerMarker } from "./common-Cf1UvJaP.mjs";
5
+ import { t as getWithIndicator } from "./busy-BlU8_pS2.mjs";
6
+ import { n as signingStateDir } from "./common-DNrD_-EI.mjs";
7
7
  import { CborDate } from "@bcts/dcbor";
8
8
  import * as fs from "node:fs";
9
9
  import * as path from "node:path";
@@ -174,4 +174,4 @@ async function receive(client, selection, options, cwd) {
174
174
  //#endregion
175
175
  export { receive_exports as n, receive as t };
176
176
 
177
- //# sourceMappingURL=receive-DA_KQEgk.mjs.map
177
+ //# sourceMappingURL=receive-g8EhZF2Y.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"receive-DA_KQEgk.mjs","names":[],"sources":["../src/cmd/sign/participant/receive.ts"],"sourcesContent":["/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Sign participant receive command.\n *\n * Port of cmd/sign/participant/receive.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\n\nimport { type ARID, type XID } from \"@bcts/components\";\nimport { compareXidBytes } from \"../../../dkg/proposed-participant.js\";\nimport { CborDate } from \"@bcts/dcbor\";\nimport type { Envelope } from \"@bcts/envelope\";\n\nimport { Registry, resolveRegistryPath, type OwnerRecord } from \"../../../registry/index.js\";\nimport { getWithIndicator } from \"../../busy.js\";\nimport { type StorageClient, type StorageSelection } from \"../../storage.js\";\nimport { parseAridUr, parseEnvelopeUr, formatNameWithOwnerMarker } from \"../../dkg/common.js\";\nimport { signingStateDir } from \"../common.js\";\n\n/**\n * Options for the sign receive command.\n */\nexport interface SignReceiveOptions {\n registryPath?: string;\n /** ARID or envelope UR string */\n request: string;\n timeoutSeconds?: number;\n /** Show request details only (info mode) */\n info?: boolean;\n /** Expected sender (XID UR or pet name) */\n sender?: string;\n}\n\n/**\n * Result of the sign receive command.\n */\nexport interface SignReceiveResult {\n sessionId: string;\n groupId: string;\n targetUr: string;\n coordinatorName: string;\n minSigners: number;\n participantNames: string[];\n}\n\n/**\n * Resolve sender from XID UR or pet name in registry.\n *\n * Port of `resolve_sender()` from cmd/dkg/common.rs lines 76-94.\n */\nfunction resolveSenderFromInput(registry: Registry, input: string): { xid: () => XID } {\n const trimmed = input.trim();\n if (trimmed === \"\") {\n throw new Error(\"Sender is required\");\n }\n\n // Try parsing as XID UR first\n try {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-require-imports, no-undef\n const { XID: XIDClass } = require(\"@bcts/components\");\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access\n const xid = XIDClass.fromURString(trimmed) as XID;\n const record = registry.participant(xid);\n if (!record) {\n throw new Error(`Sender with XID ${xid.urString()} not found`);\n }\n return record.xidDocument();\n } catch {\n // Try looking up by pet name\n const result = registry.participantByPetName(trimmed);\n if (!result) {\n throw new Error(`Sender with pet name '${trimmed}' not found`);\n }\n return result[1].xidDocument();\n }\n}\n\n/**\n * Resolve sign invite from ARID or envelope UR.\n *\n * Port of `resolve_sign_request()` from cmd/sign/participant/receive.rs lines 250-284.\n */\nasync function resolveSignInviteEnvelope(\n client: StorageClient | undefined,\n selection: StorageSelection | undefined,\n request: string,\n timeout: number | undefined,\n): Promise<Envelope> {\n if (selection !== undefined && client !== undefined) {\n // Try to parse as ARID\n try {\n const arid = parseAridUr(request);\n const envelope = await getWithIndicator(client, arid, \"Sign invite\", timeout, false);\n if (envelope === undefined || envelope === null) {\n throw new Error(\"signInvite request not found in Hubert storage\");\n }\n return envelope;\n } catch {\n // Not an ARID, try as envelope\n }\n\n if (timeout !== undefined) {\n throw new Error(\"--timeout is only valid when retrieving requests from Hubert\");\n }\n return parseEnvelopeUr(request);\n }\n\n // No storage selection\n try {\n parseAridUr(request);\n throw new Error(\"Hubert storage parameters are required to retrieve requests by ARID\");\n } catch (e) {\n if (e instanceof Error && e.message.includes(\"Hubert storage parameters\")) {\n throw e;\n }\n // Not an ARID, parse as envelope\n }\n return parseEnvelopeUr(request);\n}\n\n/**\n * Get display name for sender from registry.\n *\n * Port of `resolve_sender_name()` from cmd/dkg/common.rs lines 96-116.\n */\nfunction resolveSenderName(registry: Registry, senderXid: XID): string | undefined {\n const owner = registry.owner();\n\n // Check if sender is the owner\n if (owner?.xid().urString() === senderXid.urString()) {\n const name = owner.petName() ?? senderXid.urString();\n return formatNameWithOwnerMarker(name, true);\n }\n\n // Look up in participants\n const record = registry.participant(senderXid);\n if (record) {\n const name = record.petName() ?? record.xid().urString();\n return formatNameWithOwnerMarker(name, false);\n }\n\n return undefined;\n}\n\n/**\n * Format participant names with owner marker.\n *\n * Port of `format_participant_names()` from cmd/sign/participant/receive.rs lines 286-309.\n */\nfunction formatParticipantNames(\n registry: Registry,\n participants: XID[],\n owner: OwnerRecord,\n): string[] {\n return participants.map((xid) => {\n const isOwner = xid.urString() === owner.xid().urString();\n let name: string;\n\n if (isOwner) {\n name = owner.petName() ?? xid.urString();\n } else {\n const record = registry.participant(xid);\n name = record?.petName() ?? xid.urString();\n }\n\n return formatNameWithOwnerMarker(name, isOwner);\n });\n}\n\n/**\n * Execute the sign participant receive command.\n *\n * Fetches and validates a sign invite from the coordinator.\n *\n * Port of `CommandArgs::exec()` from cmd/sign/participant/receive.rs lines 56-247.\n */\nexport async function receive(\n client: StorageClient | undefined,\n selection: StorageSelection | undefined,\n options: SignReceiveOptions,\n cwd: string,\n): Promise<SignReceiveResult> {\n // Validate timeout requires storage\n if (selection === undefined && options.timeoutSeconds !== undefined) {\n throw new Error(\"--timeout requires Hubert storage parameters\");\n }\n\n const registryPath = resolveRegistryPath(options.registryPath, cwd);\n const registry = Registry.load(registryPath);\n const owner = registry.owner();\n\n if (!owner) {\n throw new Error(\"Registry owner with private keys is required\");\n }\n\n // Resolve expected sender if provided\n let expectedSender: { xid: () => XID } | undefined;\n if (options.sender !== undefined && options.sender !== \"\") {\n expectedSender = resolveSenderFromInput(registry, options.sender);\n }\n\n // Resolve the invite envelope\n const envelope = await resolveSignInviteEnvelope(\n client,\n selection,\n options.request,\n options.timeoutSeconds,\n );\n\n const now: CborDate = CborDate.now();\n const recipientKeys = owner.xidDocument().inceptionPrivateKeys();\n\n if (recipientKeys === null || recipientKeys === undefined) {\n throw new Error(\"Owner XID document has no inception private keys\");\n }\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef\n const { SealedRequest: SealedRequestClass } = require(\"@bcts/gstp\") as {\n SealedRequest: {\n tryFromEnvelope: (\n envelope: Envelope,\n expectedSender: XID | undefined,\n now: CborDate,\n recipientPrivateKeys: unknown,\n ) => SealedRequestInstance;\n };\n };\n\n interface SealedRequestInstance {\n sender: () => { xid: () => XID };\n function: () => { equals?: (other: unknown) => boolean; toString?: () => string };\n extractObjectForParameter: <T>(name: string) => T;\n objectForParameter: (name: string) => Envelope;\n objectsForParameter: (name: string) => ParticipantEntry[];\n }\n\n interface ParticipantEntry {\n extractSubject: () => XID;\n objectForPredicate: (name: string) => {\n decryptToRecipient: (keys: unknown) => {\n extractSubject: () => ARID;\n };\n };\n }\n\n const sealedRequest: SealedRequestInstance = SealedRequestClass.tryFromEnvelope(\n envelope,\n undefined,\n now,\n recipientKeys,\n );\n\n // Validate sender\n const senderXid = sealedRequest.sender().xid();\n\n if (expectedSender !== undefined) {\n if (senderXid.urString() !== expectedSender.xid().urString()) {\n throw new Error(\n `Request sender does not match expected sender (got ${senderXid.urString()}, expected ${expectedSender.xid().urString()})`,\n );\n }\n } else {\n const knownOwner = owner.xid().urString() === senderXid.urString();\n const knownParticipant = registry.participant(senderXid) !== undefined;\n if (!knownOwner && !knownParticipant) {\n throw new Error(`Request sender not found in registry: ${senderXid.urString()}`);\n }\n }\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef\n const { Function: FunctionClass } = require(\"@bcts/envelope\") as {\n Function: { from: (name: string) => unknown };\n };\n\n // Validate function\n const requestFunction = sealedRequest.function();\n const expectedFunction = FunctionClass.from(\"signInvite\");\n const functionMatches =\n requestFunction.equals !== undefined\n ? requestFunction.equals(expectedFunction)\n : String(requestFunction) === String(expectedFunction);\n\n if (!functionMatches) {\n throw new Error(`Unexpected request function: ${String(requestFunction)}`);\n }\n\n // Extract parameters\n const validUntil = sealedRequest.extractObjectForParameter<CborDate>(\"validUntil\");\n if (validUntil <= now) {\n throw new Error(\"signInvite request has expired\");\n }\n\n const groupId = sealedRequest.extractObjectForParameter<ARID>(\"group\");\n const sessionId = sealedRequest.extractObjectForParameter<ARID>(\"session\");\n const minSigners = Number(sealedRequest.extractObjectForParameter<bigint | number>(\"minSigners\"));\n\n // Extract participants and find our response ARID\n const participantEntries = sealedRequest.objectsForParameter(\"participant\");\n const participants: XID[] = [];\n let responseArid: ARID | undefined;\n\n for (const entry of participantEntries) {\n const xid: XID = entry.extractSubject();\n if (xid.urString() === owner.xid().urString()) {\n const encryptedArid = entry.objectForPredicate(\"response_arid\");\n const aridEnv = encryptedArid.decryptToRecipient(recipientKeys);\n responseArid = aridEnv.extractSubject();\n }\n participants.push(xid);\n }\n\n // Validations\n if (participants.length === 0) {\n throw new Error(\"signInvite request contains no participants\");\n }\n if (minSigners < 2) {\n throw new Error(\"minSigners must be at least 2\");\n }\n if (minSigners > participants.length) {\n throw new Error(\"minSigners exceeds participant count\");\n }\n\n const ownerInParticipants = participants.some((p) => p.urString() === owner.xid().urString());\n if (!ownerInParticipants) {\n throw new Error(\"signInvite request does not include this participant\");\n }\n\n if (responseArid === undefined) {\n throw new Error(\"signInvite request missing response ARID\");\n }\n\n // Sort participants by XID byte order — mirrors Rust `XID::cmp`.\n // The earlier port used `urString().localeCompare(...)`, which\n // diverges from byte order for bytes ≥ 0x80 and is locale-aware.\n participants.sort((a, b) => compareXidBytes(a.toData(), b.toData()));\n\n const targetEnvelope = sealedRequest.objectForParameter(\"target\");\n\n const coordinatorName = resolveSenderName(registry, senderXid) ?? senderXid.urString();\n const participantNames = formatParticipantNames(registry, participants, owner);\n\n // Output\n console.log(`Group: ${groupId.urString()}`);\n console.log(`Coordinator: ${coordinatorName}`);\n console.log(`Min signers: ${minSigners}`);\n console.log(`Participants: ${participantNames.join(\", \")}`);\n console.log(\"Target:\");\n console.log(targetEnvelope.format());\n\n // Primary output for scripting: session ID on its own line (no header)\n console.log(sessionId.urString());\n\n // Persist request details for follow-up commands\n const stateDir = signingStateDir(registryPath, groupId.hex(), sessionId.hex());\n fs.mkdirSync(stateDir, { recursive: true });\n\n const root: Record<string, unknown> = {\n request_envelope: envelope.urString(),\n group: groupId.urString(),\n session: sessionId.urString(),\n coordinator: senderXid.urString(),\n min_signers: minSigners,\n response_arid: responseArid.urString(),\n participants: participants.map((xid) => xid.urString()),\n target: targetEnvelope.urString(),\n };\n\n fs.writeFileSync(path.join(stateDir, \"sign_receive.json\"), JSON.stringify(root, null, 2));\n\n return {\n sessionId: sessionId.urString(),\n groupId: groupId.urString(),\n targetUr: targetEnvelope.urString(),\n coordinatorName,\n minSigners,\n participantNames,\n };\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"receive-g8EhZF2Y.mjs","names":[],"sources":["../src/cmd/sign/participant/receive.ts"],"sourcesContent":["/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Sign participant receive command.\n *\n * Port of cmd/sign/participant/receive.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\n\nimport { type ARID, type XID } from \"@bcts/components\";\nimport { compareXidBytes } from \"../../../dkg/proposed-participant.js\";\nimport { CborDate } from \"@bcts/dcbor\";\nimport type { Envelope } from \"@bcts/envelope\";\n\nimport { Registry, resolveRegistryPath, type OwnerRecord } from \"../../../registry/index.js\";\nimport { getWithIndicator } from \"../../busy.js\";\nimport { type StorageClient, type StorageSelection } from \"../../storage.js\";\nimport { parseAridUr, parseEnvelopeUr, formatNameWithOwnerMarker } from \"../../dkg/common.js\";\nimport { signingStateDir } from \"../common.js\";\n\n/**\n * Options for the sign receive command.\n */\nexport interface SignReceiveOptions {\n registryPath?: string;\n /** ARID or envelope UR string */\n request: string;\n timeoutSeconds?: number;\n /** Show request details only (info mode) */\n info?: boolean;\n /** Expected sender (XID UR or pet name) */\n sender?: string;\n}\n\n/**\n * Result of the sign receive command.\n */\nexport interface SignReceiveResult {\n sessionId: string;\n groupId: string;\n targetUr: string;\n coordinatorName: string;\n minSigners: number;\n participantNames: string[];\n}\n\n/**\n * Resolve sender from XID UR or pet name in registry.\n *\n * Port of `resolve_sender()` from cmd/dkg/common.rs lines 76-94.\n */\nfunction resolveSenderFromInput(registry: Registry, input: string): { xid: () => XID } {\n const trimmed = input.trim();\n if (trimmed === \"\") {\n throw new Error(\"Sender is required\");\n }\n\n // Try parsing as XID UR first\n try {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-require-imports, no-undef\n const { XID: XIDClass } = require(\"@bcts/components\");\n // eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access\n const xid = XIDClass.fromURString(trimmed) as XID;\n const record = registry.participant(xid);\n if (!record) {\n throw new Error(`Sender with XID ${xid.urString()} not found`);\n }\n return record.xidDocument();\n } catch {\n // Try looking up by pet name\n const result = registry.participantByPetName(trimmed);\n if (!result) {\n throw new Error(`Sender with pet name '${trimmed}' not found`);\n }\n return result[1].xidDocument();\n }\n}\n\n/**\n * Resolve sign invite from ARID or envelope UR.\n *\n * Port of `resolve_sign_request()` from cmd/sign/participant/receive.rs lines 250-284.\n */\nasync function resolveSignInviteEnvelope(\n client: StorageClient | undefined,\n selection: StorageSelection | undefined,\n request: string,\n timeout: number | undefined,\n): Promise<Envelope> {\n if (selection !== undefined && client !== undefined) {\n // Try to parse as ARID\n try {\n const arid = parseAridUr(request);\n const envelope = await getWithIndicator(client, arid, \"Sign invite\", timeout, false);\n if (envelope === undefined || envelope === null) {\n throw new Error(\"signInvite request not found in Hubert storage\");\n }\n return envelope;\n } catch {\n // Not an ARID, try as envelope\n }\n\n if (timeout !== undefined) {\n throw new Error(\"--timeout is only valid when retrieving requests from Hubert\");\n }\n return parseEnvelopeUr(request);\n }\n\n // No storage selection\n try {\n parseAridUr(request);\n throw new Error(\"Hubert storage parameters are required to retrieve requests by ARID\");\n } catch (e) {\n if (e instanceof Error && e.message.includes(\"Hubert storage parameters\")) {\n throw e;\n }\n // Not an ARID, parse as envelope\n }\n return parseEnvelopeUr(request);\n}\n\n/**\n * Get display name for sender from registry.\n *\n * Port of `resolve_sender_name()` from cmd/dkg/common.rs lines 96-116.\n */\nfunction resolveSenderName(registry: Registry, senderXid: XID): string | undefined {\n const owner = registry.owner();\n\n // Check if sender is the owner\n if (owner?.xid().urString() === senderXid.urString()) {\n const name = owner.petName() ?? senderXid.urString();\n return formatNameWithOwnerMarker(name, true);\n }\n\n // Look up in participants\n const record = registry.participant(senderXid);\n if (record) {\n const name = record.petName() ?? record.xid().urString();\n return formatNameWithOwnerMarker(name, false);\n }\n\n return undefined;\n}\n\n/**\n * Format participant names with owner marker.\n *\n * Port of `format_participant_names()` from cmd/sign/participant/receive.rs lines 286-309.\n */\nfunction formatParticipantNames(\n registry: Registry,\n participants: XID[],\n owner: OwnerRecord,\n): string[] {\n return participants.map((xid) => {\n const isOwner = xid.urString() === owner.xid().urString();\n let name: string;\n\n if (isOwner) {\n name = owner.petName() ?? xid.urString();\n } else {\n const record = registry.participant(xid);\n name = record?.petName() ?? xid.urString();\n }\n\n return formatNameWithOwnerMarker(name, isOwner);\n });\n}\n\n/**\n * Execute the sign participant receive command.\n *\n * Fetches and validates a sign invite from the coordinator.\n *\n * Port of `CommandArgs::exec()` from cmd/sign/participant/receive.rs lines 56-247.\n */\nexport async function receive(\n client: StorageClient | undefined,\n selection: StorageSelection | undefined,\n options: SignReceiveOptions,\n cwd: string,\n): Promise<SignReceiveResult> {\n // Validate timeout requires storage\n if (selection === undefined && options.timeoutSeconds !== undefined) {\n throw new Error(\"--timeout requires Hubert storage parameters\");\n }\n\n const registryPath = resolveRegistryPath(options.registryPath, cwd);\n const registry = Registry.load(registryPath);\n const owner = registry.owner();\n\n if (!owner) {\n throw new Error(\"Registry owner with private keys is required\");\n }\n\n // Resolve expected sender if provided\n let expectedSender: { xid: () => XID } | undefined;\n if (options.sender !== undefined && options.sender !== \"\") {\n expectedSender = resolveSenderFromInput(registry, options.sender);\n }\n\n // Resolve the invite envelope\n const envelope = await resolveSignInviteEnvelope(\n client,\n selection,\n options.request,\n options.timeoutSeconds,\n );\n\n const now: CborDate = CborDate.now();\n const recipientKeys = owner.xidDocument().inceptionPrivateKeys();\n\n if (recipientKeys === null || recipientKeys === undefined) {\n throw new Error(\"Owner XID document has no inception private keys\");\n }\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef\n const { SealedRequest: SealedRequestClass } = require(\"@bcts/gstp\") as {\n SealedRequest: {\n tryFromEnvelope: (\n envelope: Envelope,\n expectedSender: XID | undefined,\n now: CborDate,\n recipientPrivateKeys: unknown,\n ) => SealedRequestInstance;\n };\n };\n\n interface SealedRequestInstance {\n sender: () => { xid: () => XID };\n function: () => { equals?: (other: unknown) => boolean; toString?: () => string };\n extractObjectForParameter: <T>(name: string) => T;\n objectForParameter: (name: string) => Envelope;\n objectsForParameter: (name: string) => ParticipantEntry[];\n }\n\n interface ParticipantEntry {\n extractSubject: () => XID;\n objectForPredicate: (name: string) => {\n decryptToRecipient: (keys: unknown) => {\n extractSubject: () => ARID;\n };\n };\n }\n\n const sealedRequest: SealedRequestInstance = SealedRequestClass.tryFromEnvelope(\n envelope,\n undefined,\n now,\n recipientKeys,\n );\n\n // Validate sender\n const senderXid = sealedRequest.sender().xid();\n\n if (expectedSender !== undefined) {\n if (senderXid.urString() !== expectedSender.xid().urString()) {\n throw new Error(\n `Request sender does not match expected sender (got ${senderXid.urString()}, expected ${expectedSender.xid().urString()})`,\n );\n }\n } else {\n const knownOwner = owner.xid().urString() === senderXid.urString();\n const knownParticipant = registry.participant(senderXid) !== undefined;\n if (!knownOwner && !knownParticipant) {\n throw new Error(`Request sender not found in registry: ${senderXid.urString()}`);\n }\n }\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef\n const { Function: FunctionClass } = require(\"@bcts/envelope\") as {\n Function: { from: (name: string) => unknown };\n };\n\n // Validate function\n const requestFunction = sealedRequest.function();\n const expectedFunction = FunctionClass.from(\"signInvite\");\n const functionMatches =\n requestFunction.equals !== undefined\n ? requestFunction.equals(expectedFunction)\n : String(requestFunction) === String(expectedFunction);\n\n if (!functionMatches) {\n throw new Error(`Unexpected request function: ${String(requestFunction)}`);\n }\n\n // Extract parameters\n const validUntil = sealedRequest.extractObjectForParameter<CborDate>(\"validUntil\");\n if (validUntil <= now) {\n throw new Error(\"signInvite request has expired\");\n }\n\n const groupId = sealedRequest.extractObjectForParameter<ARID>(\"group\");\n const sessionId = sealedRequest.extractObjectForParameter<ARID>(\"session\");\n const minSigners = Number(sealedRequest.extractObjectForParameter<bigint | number>(\"minSigners\"));\n\n // Extract participants and find our response ARID\n const participantEntries = sealedRequest.objectsForParameter(\"participant\");\n const participants: XID[] = [];\n let responseArid: ARID | undefined;\n\n for (const entry of participantEntries) {\n const xid: XID = entry.extractSubject();\n if (xid.urString() === owner.xid().urString()) {\n const encryptedArid = entry.objectForPredicate(\"response_arid\");\n const aridEnv = encryptedArid.decryptToRecipient(recipientKeys);\n responseArid = aridEnv.extractSubject();\n }\n participants.push(xid);\n }\n\n // Validations\n if (participants.length === 0) {\n throw new Error(\"signInvite request contains no participants\");\n }\n if (minSigners < 2) {\n throw new Error(\"minSigners must be at least 2\");\n }\n if (minSigners > participants.length) {\n throw new Error(\"minSigners exceeds participant count\");\n }\n\n const ownerInParticipants = participants.some((p) => p.urString() === owner.xid().urString());\n if (!ownerInParticipants) {\n throw new Error(\"signInvite request does not include this participant\");\n }\n\n if (responseArid === undefined) {\n throw new Error(\"signInvite request missing response ARID\");\n }\n\n // Sort participants by XID byte order — mirrors Rust `XID::cmp`.\n // The earlier port used `urString().localeCompare(...)`, which\n // diverges from byte order for bytes ≥ 0x80 and is locale-aware.\n participants.sort((a, b) => compareXidBytes(a.toData(), b.toData()));\n\n const targetEnvelope = sealedRequest.objectForParameter(\"target\");\n\n const coordinatorName = resolveSenderName(registry, senderXid) ?? senderXid.urString();\n const participantNames = formatParticipantNames(registry, participants, owner);\n\n // Output\n console.log(`Group: ${groupId.urString()}`);\n console.log(`Coordinator: ${coordinatorName}`);\n console.log(`Min signers: ${minSigners}`);\n console.log(`Participants: ${participantNames.join(\", \")}`);\n console.log(\"Target:\");\n console.log(targetEnvelope.format());\n\n // Primary output for scripting: session ID on its own line (no header)\n console.log(sessionId.urString());\n\n // Persist request details for follow-up commands\n const stateDir = signingStateDir(registryPath, groupId.hex(), sessionId.hex());\n fs.mkdirSync(stateDir, { recursive: true });\n\n const root: Record<string, unknown> = {\n request_envelope: envelope.urString(),\n group: groupId.urString(),\n session: sessionId.urString(),\n coordinator: senderXid.urString(),\n min_signers: minSigners,\n response_arid: responseArid.urString(),\n participants: participants.map((xid) => xid.urString()),\n target: targetEnvelope.urString(),\n };\n\n fs.writeFileSync(path.join(stateDir, \"sign_receive.json\"), JSON.stringify(root, null, 2));\n\n return {\n sessionId: sessionId.urString(),\n groupId: groupId.urString(),\n targetUr: targetEnvelope.urString(),\n coordinatorName,\n minSigners,\n participantNames,\n };\n}\n"],"mappings":"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"}
@@ -1,5 +1,5 @@
1
1
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
- const require_chunk = require("../chunk-CZWwpsFl.cjs");
2
+ const require_chunk = require("../chunk-DakpK96I.cjs");
3
3
  let _bcts_components = require("@bcts/components");
4
4
  let _bcts_envelope = require("@bcts/envelope");
5
5
  let _bcts_xid = require("@bcts/xid");
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","names":["XID","ARID","SigningPublicKey","sanitizeXidUr","UR","Envelope","XIDDocument","XIDVerifySignature","UR","Envelope","XIDDocument","XIDVerifySignature","fs","path"],"sources":["../../src/registry/group-record.ts","../../src/registry/owner-record.ts","../../src/registry/participant-record.ts","../../src/registry/registry-impl.ts"],"sourcesContent":["/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Group record for the registry.\n *\n * Port of registry/group_record.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport { ARID, SigningPublicKey, XID } from \"@bcts/components\";\n\n/**\n * A participant in a group.\n *\n * Port of `struct GroupParticipant` from group_record.rs lines 9-13.\n */\nexport class GroupParticipant {\n private readonly _xid: XID;\n\n constructor(xid: XID) {\n this._xid = xid;\n }\n\n xid(): XID {\n return this._xid;\n }\n\n toJSON(): string {\n return this._xid.urString();\n }\n\n static fromJSON(value: string): GroupParticipant {\n const xid = XID.fromURString(value);\n return new GroupParticipant(xid);\n }\n}\n\n/**\n * Contribution paths for DKG state files.\n *\n * Port of `struct ContributionPaths` from group_record.rs lines 21-29.\n */\nexport class ContributionPaths {\n round1Secret?: string | undefined;\n round1Package?: string | undefined;\n round2Secret?: string | undefined;\n keyPackage?: string | undefined;\n\n constructor(init?: Partial<ContributionPaths>) {\n if (init !== undefined) {\n this.round1Secret = init.round1Secret;\n this.round1Package = init.round1Package;\n this.round2Secret = init.round2Secret;\n this.keyPackage = init.keyPackage;\n }\n }\n\n /**\n * Merge missing fields from another ContributionPaths.\n *\n * Port of `ContributionPaths::merge_missing()` from group_record.rs lines 32-45.\n */\n mergeMissing(other: ContributionPaths): void {\n this.round1Secret ??= other.round1Secret;\n this.round1Package ??= other.round1Package;\n this.round2Secret ??= other.round2Secret;\n this.keyPackage ??= other.keyPackage;\n }\n\n /**\n * Check if all fields are empty.\n *\n * Port of `ContributionPaths::is_empty()` from group_record.rs lines 47-53.\n */\n isEmpty(): boolean {\n return (\n this.round1Secret === undefined &&\n this.round1Package === undefined &&\n this.round2Secret === undefined &&\n this.keyPackage === undefined\n );\n }\n\n toJSON(): Record<string, string> {\n const obj: Record<string, string> = {};\n if (this.round1Secret !== undefined) obj[\"round1_secret\"] = this.round1Secret;\n if (this.round1Package !== undefined) obj[\"round1_package\"] = this.round1Package;\n if (this.round2Secret !== undefined) obj[\"round2_secret\"] = this.round2Secret;\n if (this.keyPackage !== undefined) obj[\"key_package\"] = this.keyPackage;\n return obj;\n }\n\n static fromJSON(json: Record<string, string>): ContributionPaths {\n return new ContributionPaths({\n round1Secret: json[\"round1_secret\"],\n round1Package: json[\"round1_package\"],\n round2Secret: json[\"round2_secret\"],\n keyPackage: json[\"key_package\"],\n });\n }\n}\n\n/**\n * A pending request entry.\n */\ninterface PendingRequestEntry {\n participant: XID;\n sendToArid?: ARID | undefined;\n collectFromArid: ARID;\n}\n\n/**\n * Tracks pending communication with participants (coordinator-side).\n *\n * Port of `struct PendingRequests` from group_record.rs lines 71-75.\n */\nexport class PendingRequests {\n private readonly requests: PendingRequestEntry[] = [];\n\n /**\n * Add a pending request where we only know where to collect from.\n *\n * Port of `PendingRequests::add_collect_only()` from group_record.rs lines 90-99.\n */\n addCollectOnly(participant: XID, collectFromArid: ARID): void {\n this.requests.push({\n participant,\n sendToArid: undefined,\n collectFromArid,\n });\n }\n\n /**\n * Add a pending request where we know where to send AND where to collect.\n *\n * Port of `PendingRequests::add_send_and_collect()` from group_record.rs lines 103-115.\n */\n addSendAndCollect(participant: XID, sendToArid: ARID, collectFromArid: ARID): void {\n this.requests.push({\n participant,\n sendToArid,\n collectFromArid,\n });\n }\n\n /**\n * Add a pending request where we only know where to send.\n *\n * Port of `PendingRequests::add_send_only()` from group_record.rs lines 118-127.\n */\n addSendOnly(participant: XID, sendToArid: ARID): void {\n this.requests.push({\n participant,\n sendToArid,\n collectFromArid: sendToArid, // Placeholder\n });\n }\n\n /**\n * Check if there are no pending requests.\n *\n * Port of `PendingRequests::is_empty()` from group_record.rs line 129.\n */\n isEmpty(): boolean {\n return this.requests.length === 0;\n }\n\n /**\n * Get the number of pending requests.\n *\n * Port of `PendingRequests::len()` from group_record.rs line 165.\n */\n len(): number {\n return this.requests.length;\n }\n\n /**\n * Iterate over (participant, collectFromArid) pairs.\n *\n * Port of `PendingRequests::iter_collect()` from group_record.rs lines 132-138.\n */\n *iterCollect(): Generator<[XID, ARID]> {\n for (const r of this.requests) {\n yield [r.participant, r.collectFromArid];\n }\n }\n\n /**\n * Iterate over (participant, sendToArid) pairs.\n *\n * Port of `PendingRequests::iter_send()` from group_record.rs lines 141-150.\n */\n *iterSend(): Generator<[XID, ARID]> {\n for (const r of this.requests) {\n if (r.sendToArid === undefined) {\n throw new Error(\"send_to_arid not set for this request\");\n }\n yield [r.participant, r.sendToArid];\n }\n }\n\n /**\n * Iterate over full (participant, sendToArid, collectFromArid) tuples.\n *\n * Port of `PendingRequests::iter_full()` from group_record.rs lines 153-163.\n */\n *iterFull(): Generator<[XID, ARID | undefined, ARID]> {\n for (const r of this.requests) {\n yield [r.participant, r.sendToArid, r.collectFromArid];\n }\n }\n\n toJSON(): unknown[] {\n return this.requests.map((r) => ({\n participant: r.participant.urString(),\n send_to_arid: r.sendToArid?.urString(),\n collect_from_arid: r.collectFromArid.urString(),\n }));\n }\n\n static fromJSON(json: unknown[]): PendingRequests {\n const pr = new PendingRequests();\n for (const entry of json as Record<string, string>[]) {\n const participant = XID.fromURString(entry[\"participant\"]);\n const sendToArid =\n entry[\"send_to_arid\"] !== undefined && entry[\"send_to_arid\"] !== \"\"\n ? ARID.fromURString(entry[\"send_to_arid\"])\n : undefined;\n const collectFromArid = ARID.fromURString(entry[\"collect_from_arid\"]);\n pr.requests.push({ participant, sendToArid, collectFromArid });\n }\n return pr;\n }\n}\n\n/**\n * Record of a DKG group.\n *\n * Port of `struct GroupRecord` from group_record.rs lines 168-186.\n */\nexport class GroupRecord {\n private readonly _charter: string;\n private readonly _minSigners: number;\n private readonly _coordinator: GroupParticipant;\n private readonly _participants: GroupParticipant[];\n private _contributions: ContributionPaths;\n private _listeningAtArid?: ARID | undefined;\n private _pendingRequests: PendingRequests;\n private _verifyingKey?: SigningPublicKey | undefined;\n\n constructor(\n charter: string,\n minSigners: number,\n coordinator: GroupParticipant,\n participants: GroupParticipant[],\n ) {\n this._charter = charter;\n this._minSigners = minSigners;\n this._coordinator = coordinator;\n this._participants = participants;\n this._contributions = new ContributionPaths();\n this._listeningAtArid = undefined;\n this._pendingRequests = new PendingRequests();\n this._verifyingKey = undefined;\n }\n\n coordinator(): GroupParticipant {\n return this._coordinator;\n }\n\n participants(): GroupParticipant[] {\n return this._participants;\n }\n\n minSigners(): number {\n return this._minSigners;\n }\n\n charter(): string {\n return this._charter;\n }\n\n contributions(): ContributionPaths {\n return this._contributions;\n }\n\n setContributions(contributions: ContributionPaths): void {\n this._contributions = contributions;\n }\n\n mergeContributions(other: ContributionPaths): void {\n this._contributions.mergeMissing(other);\n }\n\n listeningAtArid(): ARID | undefined {\n return this._listeningAtArid;\n }\n\n setListeningAtArid(arid: ARID): void {\n this._listeningAtArid = arid;\n }\n\n clearListeningAtArid(): void {\n this._listeningAtArid = undefined;\n }\n\n pendingRequests(): PendingRequests {\n return this._pendingRequests;\n }\n\n setPendingRequests(requests: PendingRequests): void {\n this._pendingRequests = requests;\n }\n\n clearPendingRequests(): void {\n this._pendingRequests = new PendingRequests();\n }\n\n /**\n * Check if the config matches another group record.\n *\n * Port of `GroupRecord::config_matches()` from group_record.rs lines 247-253.\n */\n configMatches(other: GroupRecord): boolean {\n return (\n this._charter === other._charter &&\n this._minSigners === other._minSigners &&\n this._coordinator.xid().toString() === other._coordinator.xid().toString() &&\n this._participants.length === other._participants.length &&\n this._participants.every(\n (p, i) => p.xid().toString() === other._participants[i].xid().toString(),\n )\n );\n }\n\n verifyingKey(): SigningPublicKey | undefined {\n return this._verifyingKey;\n }\n\n setVerifyingKey(key: SigningPublicKey): void {\n this._verifyingKey = key;\n }\n\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {\n charter: this._charter,\n min_signers: this._minSigners,\n coordinator: this._coordinator.toJSON(),\n participants: this._participants.map((p) => p.toJSON()),\n };\n if (!this._contributions.isEmpty()) {\n obj[\"contributions\"] = this._contributions.toJSON();\n }\n if (this._listeningAtArid !== undefined) {\n obj[\"listening_at_arid\"] = this._listeningAtArid.urString();\n }\n if (!this._pendingRequests.isEmpty()) {\n obj[\"pending_requests\"] = this._pendingRequests.toJSON();\n }\n if (this._verifyingKey !== undefined) {\n obj[\"verifying_key\"] = this._verifyingKey.urString();\n }\n return obj;\n }\n\n static fromJSON(json: Record<string, unknown>): GroupRecord {\n const charter = json[\"charter\"] as string;\n const minSigners = json[\"min_signers\"] as number;\n const coordinator = GroupParticipant.fromJSON(json[\"coordinator\"] as string);\n const participants = (json[\"participants\"] as string[]).map((p) =>\n GroupParticipant.fromJSON(p),\n );\n\n const record = new GroupRecord(charter, minSigners, coordinator, participants);\n\n if (json[\"contributions\"] !== undefined) {\n record._contributions = ContributionPaths.fromJSON(\n json[\"contributions\"] as Record<string, string>,\n );\n }\n if (json[\"listening_at_arid\"] !== undefined) {\n record._listeningAtArid = ARID.fromURString(json[\"listening_at_arid\"] as string);\n }\n if (json[\"pending_requests\"] !== undefined) {\n record._pendingRequests = PendingRequests.fromJSON(json[\"pending_requests\"] as unknown[]);\n }\n if (json[\"verifying_key\"] !== undefined) {\n record._verifyingKey = SigningPublicKey.fromURString(json[\"verifying_key\"] as string);\n }\n\n return record;\n }\n}\n","/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Owner record for the registry.\n *\n * Port of registry/owner_record.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport { type XID } from \"@bcts/components\";\nimport { Envelope } from \"@bcts/envelope\";\nimport { UR } from \"@bcts/uniform-resources\";\nimport { XIDDocument, XIDVerifySignature } from \"@bcts/xid\";\n\n/**\n * Record of the registry owner (coordinator).\n *\n * Port of `struct OwnerRecord` from owner_record.rs lines 13-17.\n */\nexport class OwnerRecord {\n private readonly _xidDocumentUr: string;\n private readonly _xidDocument: XIDDocument;\n private readonly _petName: string | undefined;\n\n private constructor(\n xidDocumentUr: string,\n xidDocument: XIDDocument,\n petName: string | undefined,\n ) {\n this._xidDocumentUr = xidDocumentUr;\n this._xidDocument = xidDocument;\n this._petName = petName;\n }\n\n /**\n * Create an owner record from a signed XID UR string.\n *\n * Port of `OwnerRecord::from_signed_xid_ur()` from owner_record.rs lines 20-32.\n */\n static fromSignedXidUr(xidDocumentUr: string, petName?: string): OwnerRecord {\n const [raw, document] = parseRelaxedXidDocument(xidDocumentUr);\n\n if (document.inceptionPrivateKeys() === undefined) {\n throw new Error(\"Owner XID document must include private keys\");\n }\n\n return new OwnerRecord(raw, document, petName);\n }\n\n /**\n * Get the XID of the owner.\n *\n * Port of `OwnerRecord::xid()` from owner_record.rs line 34.\n */\n xid(): XID {\n return this._xidDocument.xid();\n }\n\n /**\n * Get the XID document of the owner.\n *\n * Port of `OwnerRecord::xid_document()` from owner_record.rs line 36.\n */\n xidDocument(): XIDDocument {\n return this._xidDocument;\n }\n\n /**\n * Get the UR string of the XID document.\n *\n * Port of `OwnerRecord::xid_document_ur()` from owner_record.rs line 38.\n */\n xidDocumentUr(): string {\n return this._xidDocumentUr;\n }\n\n /**\n * Get the pet name of the owner.\n *\n * Port of `OwnerRecord::pet_name()` from owner_record.rs line 40.\n */\n petName(): string | undefined {\n return this._petName;\n }\n\n /**\n * Serialize to JSON object.\n */\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {\n xid_document: this._xidDocumentUr,\n };\n if (this._petName !== undefined) {\n obj[\"pet_name\"] = this._petName;\n }\n return obj;\n }\n\n /**\n * Deserialize from JSON object.\n *\n * Mirrors Rust's `#[serde(deny_unknown_fields)]` derive on\n * `OwnerRecord` (`owner_record.rs:13-17`) — any field not in\n * `{xid_document, pet_name}` causes Rust's `serde_json::from_str`\n * to error with `unknown field`, and we mirror that here so a\n * registry file produced by a future Rust version with extra\n * fields is rejected explicitly rather than silently lossy.\n */\n static fromJSON(json: Record<string, unknown>): OwnerRecord {\n for (const key of Object.keys(json)) {\n if (key !== \"xid_document\" && key !== \"pet_name\") {\n throw new Error(`unknown field \\`${key}\\`, expected \\`xid_document\\` or \\`pet_name\\``);\n }\n }\n const xidDocumentUr = json[\"xid_document\"] as string;\n const petName = json[\"pet_name\"] as string | undefined;\n return OwnerRecord.fromSignedXidUr(xidDocumentUr, petName);\n }\n}\n\n/**\n * Parse a XID document with relaxed validation (no signature verification).\n *\n * Port of `parse_relaxed_xid_document()` from owner_record.rs lines 144-165.\n */\nfunction parseRelaxedXidDocument(xidDocumentUr: string): [string, XIDDocument] {\n const sanitized = sanitizeXidUr(xidDocumentUr);\n const ur = UR.fromURString(sanitized);\n\n if (ur.urTypeStr() !== \"xid\" && ur.urTypeStr() !== \"envelope\") {\n throw new Error(`Expected a ur:xid document, found ur:${ur.urTypeStr()}`);\n }\n\n const envelopeCbor = ur.cbor();\n let envelope: Envelope;\n try {\n envelope = Envelope.fromTaggedCbor(envelopeCbor);\n } catch {\n envelope = Envelope.fromUntaggedCbor(envelopeCbor);\n }\n\n const document = XIDDocument.fromEnvelope(envelope, undefined, XIDVerifySignature.None);\n\n return [sanitized, document];\n}\n\n/**\n * Sanitize XID UR input.\n *\n * Port of `sanitize_xid_ur()` from owner_record.rs lines 167-174.\n */\nfunction sanitizeXidUr(input: string): string {\n const trimmed = input.trim();\n if (trimmed.length === 0) {\n throw new Error(\"XID document is required\");\n }\n return trimmed;\n}\n","/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Participant record for the registry.\n *\n * Port of registry/participant_record.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport { type PublicKeys, type XID } from \"@bcts/components\";\nimport { Envelope } from \"@bcts/envelope\";\nimport { UR } from \"@bcts/uniform-resources\";\nimport { XIDDocument, XIDVerifySignature } from \"@bcts/xid\";\n\n/**\n * Record of a participant in the registry.\n *\n * Port of `struct ParticipantRecord` from participant_record.rs lines 12-17.\n */\nexport class ParticipantRecord {\n private readonly _xidDocumentUr: string;\n private readonly _xidDocument: XIDDocument;\n private readonly _publicKeys: PublicKeys;\n private readonly _petName: string | undefined;\n\n private constructor(\n xidDocumentUr: string,\n xidDocument: XIDDocument,\n publicKeys: PublicKeys,\n petName: string | undefined,\n ) {\n this._xidDocumentUr = xidDocumentUr;\n this._xidDocument = xidDocument;\n this._publicKeys = publicKeys;\n this._petName = petName;\n }\n\n /**\n * Create a participant record from a signed XID UR string.\n *\n * Port of `ParticipantRecord::from_signed_xid_ur()` from participant_record.rs lines 20-26.\n */\n static fromSignedXidUr(xidDocumentUr: string, petName?: string): ParticipantRecord {\n const [raw, document] = parseSignedXidDocument(xidDocumentUr);\n return ParticipantRecord.buildFromParts(document, raw, petName);\n }\n\n /**\n * Get the pet name of the participant.\n *\n * Port of `ParticipantRecord::pet_name()` from participant_record.rs line 28.\n */\n petName(): string | undefined {\n return this._petName;\n }\n\n /**\n * Get the public keys of the participant.\n *\n * Port of `ParticipantRecord::public_keys()` from participant_record.rs line 29.\n */\n publicKeys(): PublicKeys {\n return this._publicKeys;\n }\n\n /**\n * Get the XID of the participant.\n *\n * Port of `ParticipantRecord::xid()` from participant_record.rs line 30.\n */\n xid(): XID {\n return this._xidDocument.xid();\n }\n\n /**\n * Get the XID document of the participant.\n *\n * Port of `ParticipantRecord::xid_document()` from participant_record.rs line 31.\n */\n xidDocument(): XIDDocument {\n return this._xidDocument;\n }\n\n /**\n * Get the UR string of the XID document.\n *\n * Port of `ParticipantRecord::xid_document_ur()` from participant_record.rs line 32.\n */\n xidDocumentUr(): string {\n return this._xidDocumentUr;\n }\n\n /**\n * Build from constituent parts.\n *\n * Port of `ParticipantRecord::build_from_parts()` from participant_record.rs lines 34-49.\n */\n private static buildFromParts(\n document: XIDDocument,\n xidDocumentUr: string,\n petName: string | undefined,\n ): ParticipantRecord {\n const inceptionKey = document.inceptionKey();\n if (inceptionKey === undefined) {\n throw new Error(\"XID document missing inception key\");\n }\n\n const publicKeys = inceptionKey.publicKeys();\n\n return new ParticipantRecord(xidDocumentUr, document, publicKeys, petName);\n }\n\n /**\n * Recreate from serialized data.\n *\n * Port of `ParticipantRecord::recreate_from_serialized()` from participant_record.rs lines 51-57.\n */\n private static recreateFromSerialized(\n xidDocumentUr: string,\n petName: string | undefined,\n ): ParticipantRecord {\n const [raw, document] = parseSignedXidDocument(xidDocumentUr);\n return ParticipantRecord.buildFromParts(document, raw, petName);\n }\n\n /**\n * Serialize to JSON object.\n */\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {\n xid_document: this._xidDocumentUr,\n };\n if (this._petName !== undefined) {\n obj[\"pet_name\"] = this._petName;\n }\n return obj;\n }\n\n /**\n * Deserialize from JSON object.\n *\n * Mirrors Rust's `#[serde(deny_unknown_fields)]` derive on\n * `ParticipantRecord` (`participant_record.rs:12-17`) — Rust's\n * `serde_json::from_str` errors with `unknown field` for any\n * field outside `{xid_document, pet_name}`. We mirror that\n * behaviour so a registry file produced by a future Rust\n * version with extra fields is rejected explicitly rather than\n * silently lossy.\n */\n static fromJSON(json: Record<string, unknown>): ParticipantRecord {\n for (const key of Object.keys(json)) {\n if (key !== \"xid_document\" && key !== \"pet_name\") {\n throw new Error(`unknown field \\`${key}\\`, expected \\`xid_document\\` or \\`pet_name\\``);\n }\n }\n const xidDocumentUr = json[\"xid_document\"] as string;\n const petName = json[\"pet_name\"] as string | undefined;\n return ParticipantRecord.recreateFromSerialized(xidDocumentUr, petName);\n }\n}\n\n/**\n * Parse a signed XID document from a UR string.\n *\n * Port of `parse_signed_xid_document()` from participant_record.rs lines 170-194.\n */\nfunction parseSignedXidDocument(xidDocumentUr: string): [string, XIDDocument] {\n const sanitized = sanitizeXidUr(xidDocumentUr);\n const ur = UR.fromURString(sanitized);\n\n if (ur.urTypeStr() !== \"xid\" && ur.urTypeStr() !== \"envelope\") {\n throw new Error(`Expected a ur:xid document, found ur:${ur.urTypeStr()}`);\n }\n\n const envelopeCbor = ur.cbor();\n let envelope: Envelope;\n try {\n envelope = Envelope.fromTaggedCbor(envelopeCbor);\n } catch {\n try {\n envelope = Envelope.fromUntaggedCbor(envelopeCbor);\n } catch (e) {\n throw new Error(\n `Unable to decode XID document envelope: ${(e as Error).message ?? String(e)}`,\n {\n cause: e,\n },\n );\n }\n }\n\n // Mirror Rust `participant_record.rs:198-203`'s `.context(...)` wrap:\n // any failure from `XIDDocument::from_envelope(..., XIDVerifySignature::Inception)`\n // is surfaced as \"XID document must be signed by its inception key: <cause>\".\n let document: XIDDocument;\n try {\n document = XIDDocument.fromEnvelope(envelope, undefined, XIDVerifySignature.Inception);\n } catch (e) {\n throw new Error(\n `XID document must be signed by its inception key: ${(e as Error).message ?? String(e)}`,\n { cause: e },\n );\n }\n\n return [sanitized, document];\n}\n\n/**\n * Sanitize XID UR input.\n *\n * Port of `sanitize_xid_ur()` from participant_record.rs lines 196-203.\n */\nfunction sanitizeXidUr(input: string): string {\n const trimmed = input.trim();\n if (trimmed.length === 0) {\n throw new Error(\"XID document is required\");\n }\n return trimmed;\n}\n","/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Registry implementation for managing participants and groups.\n *\n * Port of registry/registry_impl.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\n\nimport { type ARID, type PublicKeys, type XID } from \"@bcts/components\";\n\nimport { GroupRecord } from \"./group-record.js\";\nimport { OwnerRecord } from \"./owner-record.js\";\nimport { ParticipantRecord } from \"./participant-record.js\";\n\n/**\n * Outcome of adding a participant to the registry.\n *\n * Port of `enum AddOutcome` from registry_impl.rs.\n */\nexport enum AddOutcome {\n /** Participant was already present in the registry */\n AlreadyPresent = \"already_present\",\n /** Participant was successfully inserted */\n Inserted = \"inserted\",\n}\n\n/**\n * Outcome of setting the owner in the registry.\n *\n * Port of `enum OwnerOutcome` from registry_impl.rs.\n */\nexport enum OwnerOutcome {\n /** Owner was already present in the registry */\n AlreadyPresent = \"already_present\",\n /** Owner was successfully inserted */\n Inserted = \"inserted\",\n}\n\n/**\n * Outcome of recording a group in the registry.\n *\n * Port of `enum GroupOutcome` from registry_impl.rs.\n */\nexport enum GroupOutcome {\n /** Group was successfully inserted as new */\n Inserted = \"inserted\",\n /** Group already existed and was updated (contributions merged) */\n Updated = \"updated\",\n}\n\n/**\n * Registry for managing participants and groups.\n *\n * Port of `struct Registry` from registry_impl.rs lines 22-26.\n */\nexport class Registry {\n private _owner?: OwnerRecord | undefined;\n private readonly _participants: Map<string, ParticipantRecord>; // Map by XID UR string\n private readonly _groups: Map<string, GroupRecord>; // Map by ARID hex\n\n constructor() {\n this._owner = undefined;\n this._participants = new Map();\n this._groups = new Map();\n }\n\n /**\n * Get the owner record.\n */\n owner(): OwnerRecord | undefined {\n return this._owner;\n }\n\n /**\n * Set the owner record.\n *\n * Returns the outcome indicating whether the owner was already present or newly inserted.\n *\n * Port of `Registry::set_owner()` from registry_impl.rs.\n */\n setOwner(owner: OwnerRecord): OwnerOutcome {\n // Check for pet name conflicts with participants\n const petName = owner.petName();\n if (petName !== undefined) {\n const conflicting = this.participantByPetName(petName);\n if (conflicting?.[1].petName() === petName) {\n throw new Error(`Pet name '${petName}' already used by a participant`);\n }\n }\n\n if (this._owner === undefined) {\n this._owner = owner;\n return OwnerOutcome.Inserted;\n }\n\n const existing = this._owner;\n const existingXidUr = existing.xid().urString();\n const ownerXidUr = owner.xid().urString();\n\n if (\n existingXidUr === ownerXidUr &&\n existing.xidDocumentUr() === owner.xidDocumentUr() &&\n existing.petName() === owner.petName()\n ) {\n return OwnerOutcome.AlreadyPresent;\n }\n\n if (existingXidUr === ownerXidUr) {\n if (existing.xidDocumentUr() !== owner.xidDocumentUr()) {\n throw new Error(\"Owner already exists with different keys\");\n }\n this._owner = owner;\n return OwnerOutcome.Inserted;\n }\n\n throw new Error(`Owner already recorded for ${existing.xid().toString()}`);\n }\n\n /**\n * Get all participants.\n */\n participants(): Map<string, ParticipantRecord> {\n return this._participants;\n }\n\n /**\n * Get a participant by XID.\n */\n participant(xid: XID): ParticipantRecord | undefined {\n return this._participants.get(xid.urString());\n }\n\n /**\n * Add a participant.\n *\n * Mirrors Rust `Registry::add_participant`\n * (`registry_impl.rs:83-124`):\n *\n * 1. If `record.pet_name()` is already used by some other XID,\n * bail with `\"Pet name '{name}' already used by another\n * participant\"`.\n * 2. If `record.pet_name()` matches an existing record under the\n * *same* XID and the public keys also match, return\n * `AlreadyPresent`.\n * 3. If the pet name matches the same XID but public keys don't,\n * bail with `\"Participant already exists with a different pet\n * name\"`.\n * 4. Otherwise look up by XID. If present and public-keys + pet-name\n * both match, return `AlreadyPresent`; if XID is present but\n * anything differs, bail. If XID is new, insert and return\n * `Inserted`.\n *\n * The earlier port short-circuited on `participants.has(xidUr)` and\n * always returned `AlreadyPresent` — silently allowing re-adds with\n * a different pet name or different public keys, which Rust\n * correctly forbids.\n */\n addParticipant(xid: XID, record: ParticipantRecord): AddOutcome {\n const xidUr = xid.urString();\n const petName = record.petName();\n\n // Steps 1–3: pet-name conflict resolution.\n if (petName !== undefined) {\n for (const [existingXidUr, existingRecord] of this._participants) {\n if (existingRecord.petName() === petName) {\n if (existingXidUr !== xidUr) {\n throw new Error(`Pet name '${petName}' already used by another participant`);\n }\n if (publicKeysEqual(existingRecord.publicKeys(), record.publicKeys())) {\n return AddOutcome.AlreadyPresent;\n }\n throw new Error(\"Participant already exists with a different pet name\");\n }\n }\n }\n\n // Step 4: XID lookup.\n const existing = this._participants.get(xidUr);\n if (existing !== undefined) {\n if (\n publicKeysEqual(existing.publicKeys(), record.publicKeys()) &&\n existing.petName() === record.petName()\n ) {\n return AddOutcome.AlreadyPresent;\n }\n throw new Error(\"Participant already exists with a different pet name\");\n }\n\n this._participants.set(xidUr, record);\n return AddOutcome.Inserted;\n }\n\n /**\n * Check if a pet name is already used.\n */\n petNameExists(petName: string): boolean {\n for (const record of this._participants.values()) {\n if (record.petName() === petName) {\n return true;\n }\n }\n return false;\n }\n\n /**\n * Find a participant by pet name.\n *\n * Port of `Registry::participant_by_pet_name()` from registry_impl.rs.\n */\n participantByPetName(petName: string): [XID, ParticipantRecord] | undefined {\n for (const record of this._participants.values()) {\n if (record.petName() === petName) {\n return [record.xid(), record];\n }\n }\n return undefined;\n }\n\n /**\n * Get all groups.\n */\n groups(): Map<string, GroupRecord> {\n return this._groups;\n }\n\n /**\n * Get a group by ARID.\n */\n group(arid: ARID): GroupRecord | undefined {\n return this._groups.get(arid.hex());\n }\n\n /**\n * Get a mutable reference to a group by ARID.\n */\n groupMut(arid: ARID): GroupRecord | undefined {\n return this._groups.get(arid.hex());\n }\n\n /**\n * Record a group in the registry.\n *\n * If the group already exists:\n * - Validates that the config matches\n * - Merges contributions from the new record\n * - Updates verifying key if not already set\n *\n * Port of `Registry::record_group()` from registry_impl.rs.\n */\n recordGroup(groupId: ARID, record: GroupRecord): GroupOutcome {\n const key = groupId.hex();\n const existing = this._groups.get(key);\n\n if (existing !== undefined) {\n // Validate config matches\n if (!existing.configMatches(record)) {\n throw new Error(`Group ${groupId.hex()} already exists with a different configuration`);\n }\n\n // Merge contributions\n existing.mergeContributions(record.contributions());\n\n // Update verifying key if not already set\n const existingKey = existing.verifyingKey();\n const recordKey = record.verifyingKey();\n if (existingKey === undefined && recordKey !== undefined) {\n existing.setVerifyingKey(recordKey);\n } else if (\n existingKey !== undefined &&\n recordKey !== undefined &&\n existingKey.urString() !== recordKey.urString()\n ) {\n throw new Error(`Group ${groupId.hex()} already exists with a different verifying key`);\n }\n\n return GroupOutcome.Updated;\n }\n\n this._groups.set(key, record);\n return GroupOutcome.Inserted;\n }\n\n /**\n * Add a group (simple variant without merge logic).\n *\n * @deprecated Use recordGroup() for proper merge behavior.\n */\n addGroup(arid: ARID, record: GroupRecord): void {\n this._groups.set(arid.hex(), record);\n }\n\n /**\n * Load a registry from a file.\n */\n static load(filePath: string): Registry {\n if (!fs.existsSync(filePath)) {\n return new Registry();\n }\n\n const content = fs.readFileSync(filePath, \"utf-8\");\n const json = JSON.parse(content) as Record<string, unknown>;\n return Registry.fromJSON(json);\n }\n\n /**\n * Save the registry to a file.\n */\n save(filePath: string): void {\n const dir = path.dirname(filePath);\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n const json = this.toJSON();\n fs.writeFileSync(filePath, JSON.stringify(json, null, 2));\n }\n\n /**\n * Serialize to JSON object.\n *\n * Mirrors Rust `Registry`'s field declaration order\n * (`registry_impl.rs:8-14` — `owner, participants, groups`). JSON\n * object member order is not semantically significant, but\n * `serde_json::to_string_pretty` emits keys in declaration order,\n * so for byte-equal `registry.json` (used by the integration tests\n * as a string assertion) the TS port must match Rust's order.\n * Empty `owner` is omitted via `Option::None` skip in Rust; we\n * reproduce that by only setting the key when the owner exists.\n */\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {};\n\n if (this._owner !== undefined) {\n obj[\"owner\"] = this._owner.toJSON();\n }\n\n const participants: Record<string, unknown> = {};\n for (const [xidUr, record] of this._participants) {\n participants[xidUr] = record.toJSON();\n }\n obj[\"participants\"] = participants;\n\n const groups: Record<string, unknown> = {};\n for (const [aridHex, record] of this._groups) {\n groups[aridHex] = record.toJSON();\n }\n obj[\"groups\"] = groups;\n\n return obj;\n }\n\n /**\n * Deserialize from JSON object.\n */\n static fromJSON(json: Record<string, unknown>): Registry {\n const registry = new Registry();\n\n if (json[\"owner\"] !== undefined) {\n registry._owner = OwnerRecord.fromJSON(json[\"owner\"] as Record<string, unknown>);\n }\n\n const participantsJson = json[\"participants\"] as\n | Record<string, Record<string, unknown>>\n | undefined;\n if (participantsJson !== undefined) {\n for (const [, recordJson] of Object.entries(participantsJson)) {\n const record = ParticipantRecord.fromJSON(recordJson);\n registry._participants.set(record.xid().urString(), record);\n }\n }\n\n const groupsJson = json[\"groups\"] as Record<string, Record<string, unknown>> | undefined;\n if (groupsJson !== undefined) {\n for (const [aridHex, recordJson] of Object.entries(groupsJson)) {\n const record = GroupRecord.fromJSON(recordJson);\n registry._groups.set(aridHex, record);\n }\n }\n\n return registry;\n }\n}\n\n/**\n * Resolve the registry file path from a given argument.\n *\n * Port of `resolve_registry_path()` from registry/mod.rs lines 49-78.\n */\nexport function resolveRegistryPath(registryArg: string | undefined, cwd: string): string {\n if (registryArg === undefined || registryArg === \"\") {\n return path.join(cwd, \"registry.json\");\n }\n\n // If it ends with / or is a directory, append registry.json\n if (registryArg.endsWith(\"/\") || registryArg.endsWith(path.sep)) {\n return path.join(cwd, registryArg, \"registry.json\");\n }\n\n // If it's just a filename, put it in cwd\n if (!registryArg.includes(\"/\") && !registryArg.includes(path.sep)) {\n return path.join(cwd, registryArg);\n }\n\n // Otherwise, treat as relative path\n return path.resolve(cwd, registryArg);\n}\n\n/**\n * Structural equality on `PublicKeys`, mirroring Rust's\n * `PartialEq::eq` derive (per-component comparison of the signing\n * and encapsulation public keys).\n *\n * @internal\n */\nfunction publicKeysEqual(a: PublicKeys, b: PublicKeys): boolean {\n return a.equals(b);\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","names":["XID","ARID","SigningPublicKey","sanitizeXidUr","UR","Envelope","XIDDocument","XIDVerifySignature","UR","Envelope","XIDDocument","XIDVerifySignature","fs","path"],"sources":["../../src/registry/group-record.ts","../../src/registry/owner-record.ts","../../src/registry/participant-record.ts","../../src/registry/registry-impl.ts"],"sourcesContent":["/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Group record for the registry.\n *\n * Port of registry/group_record.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport { ARID, SigningPublicKey, XID } from \"@bcts/components\";\n\n/**\n * A participant in a group.\n *\n * Port of `struct GroupParticipant` from group_record.rs lines 9-13.\n */\nexport class GroupParticipant {\n private readonly _xid: XID;\n\n constructor(xid: XID) {\n this._xid = xid;\n }\n\n xid(): XID {\n return this._xid;\n }\n\n toJSON(): string {\n return this._xid.urString();\n }\n\n static fromJSON(value: string): GroupParticipant {\n const xid = XID.fromURString(value);\n return new GroupParticipant(xid);\n }\n}\n\n/**\n * Contribution paths for DKG state files.\n *\n * Port of `struct ContributionPaths` from group_record.rs lines 21-29.\n */\nexport class ContributionPaths {\n round1Secret?: string | undefined;\n round1Package?: string | undefined;\n round2Secret?: string | undefined;\n keyPackage?: string | undefined;\n\n constructor(init?: Partial<ContributionPaths>) {\n if (init !== undefined) {\n this.round1Secret = init.round1Secret;\n this.round1Package = init.round1Package;\n this.round2Secret = init.round2Secret;\n this.keyPackage = init.keyPackage;\n }\n }\n\n /**\n * Merge missing fields from another ContributionPaths.\n *\n * Port of `ContributionPaths::merge_missing()` from group_record.rs lines 32-45.\n */\n mergeMissing(other: ContributionPaths): void {\n this.round1Secret ??= other.round1Secret;\n this.round1Package ??= other.round1Package;\n this.round2Secret ??= other.round2Secret;\n this.keyPackage ??= other.keyPackage;\n }\n\n /**\n * Check if all fields are empty.\n *\n * Port of `ContributionPaths::is_empty()` from group_record.rs lines 47-53.\n */\n isEmpty(): boolean {\n return (\n this.round1Secret === undefined &&\n this.round1Package === undefined &&\n this.round2Secret === undefined &&\n this.keyPackage === undefined\n );\n }\n\n toJSON(): Record<string, string> {\n const obj: Record<string, string> = {};\n if (this.round1Secret !== undefined) obj[\"round1_secret\"] = this.round1Secret;\n if (this.round1Package !== undefined) obj[\"round1_package\"] = this.round1Package;\n if (this.round2Secret !== undefined) obj[\"round2_secret\"] = this.round2Secret;\n if (this.keyPackage !== undefined) obj[\"key_package\"] = this.keyPackage;\n return obj;\n }\n\n static fromJSON(json: Record<string, string>): ContributionPaths {\n return new ContributionPaths({\n round1Secret: json[\"round1_secret\"],\n round1Package: json[\"round1_package\"],\n round2Secret: json[\"round2_secret\"],\n keyPackage: json[\"key_package\"],\n });\n }\n}\n\n/**\n * A pending request entry.\n */\ninterface PendingRequestEntry {\n participant: XID;\n sendToArid?: ARID | undefined;\n collectFromArid: ARID;\n}\n\n/**\n * Tracks pending communication with participants (coordinator-side).\n *\n * Port of `struct PendingRequests` from group_record.rs lines 71-75.\n */\nexport class PendingRequests {\n private readonly requests: PendingRequestEntry[] = [];\n\n /**\n * Add a pending request where we only know where to collect from.\n *\n * Port of `PendingRequests::add_collect_only()` from group_record.rs lines 90-99.\n */\n addCollectOnly(participant: XID, collectFromArid: ARID): void {\n this.requests.push({\n participant,\n sendToArid: undefined,\n collectFromArid,\n });\n }\n\n /**\n * Add a pending request where we know where to send AND where to collect.\n *\n * Port of `PendingRequests::add_send_and_collect()` from group_record.rs lines 103-115.\n */\n addSendAndCollect(participant: XID, sendToArid: ARID, collectFromArid: ARID): void {\n this.requests.push({\n participant,\n sendToArid,\n collectFromArid,\n });\n }\n\n /**\n * Add a pending request where we only know where to send.\n *\n * Port of `PendingRequests::add_send_only()` from group_record.rs lines 118-127.\n */\n addSendOnly(participant: XID, sendToArid: ARID): void {\n this.requests.push({\n participant,\n sendToArid,\n collectFromArid: sendToArid, // Placeholder\n });\n }\n\n /**\n * Check if there are no pending requests.\n *\n * Port of `PendingRequests::is_empty()` from group_record.rs line 129.\n */\n isEmpty(): boolean {\n return this.requests.length === 0;\n }\n\n /**\n * Get the number of pending requests.\n *\n * Port of `PendingRequests::len()` from group_record.rs line 165.\n */\n len(): number {\n return this.requests.length;\n }\n\n /**\n * Iterate over (participant, collectFromArid) pairs.\n *\n * Port of `PendingRequests::iter_collect()` from group_record.rs lines 132-138.\n */\n *iterCollect(): Generator<[XID, ARID]> {\n for (const r of this.requests) {\n yield [r.participant, r.collectFromArid];\n }\n }\n\n /**\n * Iterate over (participant, sendToArid) pairs.\n *\n * Port of `PendingRequests::iter_send()` from group_record.rs lines 141-150.\n */\n *iterSend(): Generator<[XID, ARID]> {\n for (const r of this.requests) {\n if (r.sendToArid === undefined) {\n throw new Error(\"send_to_arid not set for this request\");\n }\n yield [r.participant, r.sendToArid];\n }\n }\n\n /**\n * Iterate over full (participant, sendToArid, collectFromArid) tuples.\n *\n * Port of `PendingRequests::iter_full()` from group_record.rs lines 153-163.\n */\n *iterFull(): Generator<[XID, ARID | undefined, ARID]> {\n for (const r of this.requests) {\n yield [r.participant, r.sendToArid, r.collectFromArid];\n }\n }\n\n toJSON(): unknown[] {\n return this.requests.map((r) => ({\n participant: r.participant.urString(),\n send_to_arid: r.sendToArid?.urString(),\n collect_from_arid: r.collectFromArid.urString(),\n }));\n }\n\n static fromJSON(json: unknown[]): PendingRequests {\n const pr = new PendingRequests();\n for (const entry of json as Record<string, string>[]) {\n const participant = XID.fromURString(entry[\"participant\"]);\n const sendToArid =\n entry[\"send_to_arid\"] !== undefined && entry[\"send_to_arid\"] !== \"\"\n ? ARID.fromURString(entry[\"send_to_arid\"])\n : undefined;\n const collectFromArid = ARID.fromURString(entry[\"collect_from_arid\"]);\n pr.requests.push({ participant, sendToArid, collectFromArid });\n }\n return pr;\n }\n}\n\n/**\n * Record of a DKG group.\n *\n * Port of `struct GroupRecord` from group_record.rs lines 168-186.\n */\nexport class GroupRecord {\n private readonly _charter: string;\n private readonly _minSigners: number;\n private readonly _coordinator: GroupParticipant;\n private readonly _participants: GroupParticipant[];\n private _contributions: ContributionPaths;\n private _listeningAtArid?: ARID | undefined;\n private _pendingRequests: PendingRequests;\n private _verifyingKey?: SigningPublicKey | undefined;\n\n constructor(\n charter: string,\n minSigners: number,\n coordinator: GroupParticipant,\n participants: GroupParticipant[],\n ) {\n this._charter = charter;\n this._minSigners = minSigners;\n this._coordinator = coordinator;\n this._participants = participants;\n this._contributions = new ContributionPaths();\n this._listeningAtArid = undefined;\n this._pendingRequests = new PendingRequests();\n this._verifyingKey = undefined;\n }\n\n coordinator(): GroupParticipant {\n return this._coordinator;\n }\n\n participants(): GroupParticipant[] {\n return this._participants;\n }\n\n minSigners(): number {\n return this._minSigners;\n }\n\n charter(): string {\n return this._charter;\n }\n\n contributions(): ContributionPaths {\n return this._contributions;\n }\n\n setContributions(contributions: ContributionPaths): void {\n this._contributions = contributions;\n }\n\n mergeContributions(other: ContributionPaths): void {\n this._contributions.mergeMissing(other);\n }\n\n listeningAtArid(): ARID | undefined {\n return this._listeningAtArid;\n }\n\n setListeningAtArid(arid: ARID): void {\n this._listeningAtArid = arid;\n }\n\n clearListeningAtArid(): void {\n this._listeningAtArid = undefined;\n }\n\n pendingRequests(): PendingRequests {\n return this._pendingRequests;\n }\n\n setPendingRequests(requests: PendingRequests): void {\n this._pendingRequests = requests;\n }\n\n clearPendingRequests(): void {\n this._pendingRequests = new PendingRequests();\n }\n\n /**\n * Check if the config matches another group record.\n *\n * Port of `GroupRecord::config_matches()` from group_record.rs lines 247-253.\n */\n configMatches(other: GroupRecord): boolean {\n return (\n this._charter === other._charter &&\n this._minSigners === other._minSigners &&\n this._coordinator.xid().toString() === other._coordinator.xid().toString() &&\n this._participants.length === other._participants.length &&\n this._participants.every(\n (p, i) => p.xid().toString() === other._participants[i].xid().toString(),\n )\n );\n }\n\n verifyingKey(): SigningPublicKey | undefined {\n return this._verifyingKey;\n }\n\n setVerifyingKey(key: SigningPublicKey): void {\n this._verifyingKey = key;\n }\n\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {\n charter: this._charter,\n min_signers: this._minSigners,\n coordinator: this._coordinator.toJSON(),\n participants: this._participants.map((p) => p.toJSON()),\n };\n if (!this._contributions.isEmpty()) {\n obj[\"contributions\"] = this._contributions.toJSON();\n }\n if (this._listeningAtArid !== undefined) {\n obj[\"listening_at_arid\"] = this._listeningAtArid.urString();\n }\n if (!this._pendingRequests.isEmpty()) {\n obj[\"pending_requests\"] = this._pendingRequests.toJSON();\n }\n if (this._verifyingKey !== undefined) {\n obj[\"verifying_key\"] = this._verifyingKey.urString();\n }\n return obj;\n }\n\n static fromJSON(json: Record<string, unknown>): GroupRecord {\n const charter = json[\"charter\"] as string;\n const minSigners = json[\"min_signers\"] as number;\n const coordinator = GroupParticipant.fromJSON(json[\"coordinator\"] as string);\n const participants = (json[\"participants\"] as string[]).map((p) =>\n GroupParticipant.fromJSON(p),\n );\n\n const record = new GroupRecord(charter, minSigners, coordinator, participants);\n\n if (json[\"contributions\"] !== undefined) {\n record._contributions = ContributionPaths.fromJSON(\n json[\"contributions\"] as Record<string, string>,\n );\n }\n if (json[\"listening_at_arid\"] !== undefined) {\n record._listeningAtArid = ARID.fromURString(json[\"listening_at_arid\"] as string);\n }\n if (json[\"pending_requests\"] !== undefined) {\n record._pendingRequests = PendingRequests.fromJSON(json[\"pending_requests\"] as unknown[]);\n }\n if (json[\"verifying_key\"] !== undefined) {\n record._verifyingKey = SigningPublicKey.fromURString(json[\"verifying_key\"] as string);\n }\n\n return record;\n }\n}\n","/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Owner record for the registry.\n *\n * Port of registry/owner_record.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport { type XID } from \"@bcts/components\";\nimport { Envelope } from \"@bcts/envelope\";\nimport { UR } from \"@bcts/uniform-resources\";\nimport { XIDDocument, XIDVerifySignature } from \"@bcts/xid\";\n\n/**\n * Record of the registry owner (coordinator).\n *\n * Port of `struct OwnerRecord` from owner_record.rs lines 13-17.\n */\nexport class OwnerRecord {\n private readonly _xidDocumentUr: string;\n private readonly _xidDocument: XIDDocument;\n private readonly _petName: string | undefined;\n\n private constructor(\n xidDocumentUr: string,\n xidDocument: XIDDocument,\n petName: string | undefined,\n ) {\n this._xidDocumentUr = xidDocumentUr;\n this._xidDocument = xidDocument;\n this._petName = petName;\n }\n\n /**\n * Create an owner record from a signed XID UR string.\n *\n * Port of `OwnerRecord::from_signed_xid_ur()` from owner_record.rs lines 20-32.\n */\n static fromSignedXidUr(xidDocumentUr: string, petName?: string): OwnerRecord {\n const [raw, document] = parseRelaxedXidDocument(xidDocumentUr);\n\n if (document.inceptionPrivateKeys() === undefined) {\n throw new Error(\"Owner XID document must include private keys\");\n }\n\n return new OwnerRecord(raw, document, petName);\n }\n\n /**\n * Get the XID of the owner.\n *\n * Port of `OwnerRecord::xid()` from owner_record.rs line 34.\n */\n xid(): XID {\n return this._xidDocument.xid();\n }\n\n /**\n * Get the XID document of the owner.\n *\n * Port of `OwnerRecord::xid_document()` from owner_record.rs line 36.\n */\n xidDocument(): XIDDocument {\n return this._xidDocument;\n }\n\n /**\n * Get the UR string of the XID document.\n *\n * Port of `OwnerRecord::xid_document_ur()` from owner_record.rs line 38.\n */\n xidDocumentUr(): string {\n return this._xidDocumentUr;\n }\n\n /**\n * Get the pet name of the owner.\n *\n * Port of `OwnerRecord::pet_name()` from owner_record.rs line 40.\n */\n petName(): string | undefined {\n return this._petName;\n }\n\n /**\n * Serialize to JSON object.\n */\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {\n xid_document: this._xidDocumentUr,\n };\n if (this._petName !== undefined) {\n obj[\"pet_name\"] = this._petName;\n }\n return obj;\n }\n\n /**\n * Deserialize from JSON object.\n *\n * Mirrors Rust's `#[serde(deny_unknown_fields)]` derive on\n * `OwnerRecord` (`owner_record.rs:13-17`) — any field not in\n * `{xid_document, pet_name}` causes Rust's `serde_json::from_str`\n * to error with `unknown field`, and we mirror that here so a\n * registry file produced by a future Rust version with extra\n * fields is rejected explicitly rather than silently lossy.\n */\n static fromJSON(json: Record<string, unknown>): OwnerRecord {\n for (const key of Object.keys(json)) {\n if (key !== \"xid_document\" && key !== \"pet_name\") {\n throw new Error(`unknown field \\`${key}\\`, expected \\`xid_document\\` or \\`pet_name\\``);\n }\n }\n const xidDocumentUr = json[\"xid_document\"] as string;\n const petName = json[\"pet_name\"] as string | undefined;\n return OwnerRecord.fromSignedXidUr(xidDocumentUr, petName);\n }\n}\n\n/**\n * Parse a XID document with relaxed validation (no signature verification).\n *\n * Port of `parse_relaxed_xid_document()` from owner_record.rs lines 144-165.\n */\nfunction parseRelaxedXidDocument(xidDocumentUr: string): [string, XIDDocument] {\n const sanitized = sanitizeXidUr(xidDocumentUr);\n const ur = UR.fromURString(sanitized);\n\n if (ur.urTypeStr() !== \"xid\" && ur.urTypeStr() !== \"envelope\") {\n throw new Error(`Expected a ur:xid document, found ur:${ur.urTypeStr()}`);\n }\n\n const envelopeCbor = ur.cbor();\n let envelope: Envelope;\n try {\n envelope = Envelope.fromTaggedCbor(envelopeCbor);\n } catch {\n envelope = Envelope.fromUntaggedCbor(envelopeCbor);\n }\n\n const document = XIDDocument.fromEnvelope(envelope, undefined, XIDVerifySignature.None);\n\n return [sanitized, document];\n}\n\n/**\n * Sanitize XID UR input.\n *\n * Port of `sanitize_xid_ur()` from owner_record.rs lines 167-174.\n */\nfunction sanitizeXidUr(input: string): string {\n const trimmed = input.trim();\n if (trimmed.length === 0) {\n throw new Error(\"XID document is required\");\n }\n return trimmed;\n}\n","/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Participant record for the registry.\n *\n * Port of registry/participant_record.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport { type PublicKeys, type XID } from \"@bcts/components\";\nimport { Envelope } from \"@bcts/envelope\";\nimport { UR } from \"@bcts/uniform-resources\";\nimport { XIDDocument, XIDVerifySignature } from \"@bcts/xid\";\n\n/**\n * Record of a participant in the registry.\n *\n * Port of `struct ParticipantRecord` from participant_record.rs lines 12-17.\n */\nexport class ParticipantRecord {\n private readonly _xidDocumentUr: string;\n private readonly _xidDocument: XIDDocument;\n private readonly _publicKeys: PublicKeys;\n private readonly _petName: string | undefined;\n\n private constructor(\n xidDocumentUr: string,\n xidDocument: XIDDocument,\n publicKeys: PublicKeys,\n petName: string | undefined,\n ) {\n this._xidDocumentUr = xidDocumentUr;\n this._xidDocument = xidDocument;\n this._publicKeys = publicKeys;\n this._petName = petName;\n }\n\n /**\n * Create a participant record from a signed XID UR string.\n *\n * Port of `ParticipantRecord::from_signed_xid_ur()` from participant_record.rs lines 20-26.\n */\n static fromSignedXidUr(xidDocumentUr: string, petName?: string): ParticipantRecord {\n const [raw, document] = parseSignedXidDocument(xidDocumentUr);\n return ParticipantRecord.buildFromParts(document, raw, petName);\n }\n\n /**\n * Get the pet name of the participant.\n *\n * Port of `ParticipantRecord::pet_name()` from participant_record.rs line 28.\n */\n petName(): string | undefined {\n return this._petName;\n }\n\n /**\n * Get the public keys of the participant.\n *\n * Port of `ParticipantRecord::public_keys()` from participant_record.rs line 29.\n */\n publicKeys(): PublicKeys {\n return this._publicKeys;\n }\n\n /**\n * Get the XID of the participant.\n *\n * Port of `ParticipantRecord::xid()` from participant_record.rs line 30.\n */\n xid(): XID {\n return this._xidDocument.xid();\n }\n\n /**\n * Get the XID document of the participant.\n *\n * Port of `ParticipantRecord::xid_document()` from participant_record.rs line 31.\n */\n xidDocument(): XIDDocument {\n return this._xidDocument;\n }\n\n /**\n * Get the UR string of the XID document.\n *\n * Port of `ParticipantRecord::xid_document_ur()` from participant_record.rs line 32.\n */\n xidDocumentUr(): string {\n return this._xidDocumentUr;\n }\n\n /**\n * Build from constituent parts.\n *\n * Port of `ParticipantRecord::build_from_parts()` from participant_record.rs lines 34-49.\n */\n private static buildFromParts(\n document: XIDDocument,\n xidDocumentUr: string,\n petName: string | undefined,\n ): ParticipantRecord {\n const inceptionKey = document.inceptionKey();\n if (inceptionKey === undefined) {\n throw new Error(\"XID document missing inception key\");\n }\n\n const publicKeys = inceptionKey.publicKeys();\n\n return new ParticipantRecord(xidDocumentUr, document, publicKeys, petName);\n }\n\n /**\n * Recreate from serialized data.\n *\n * Port of `ParticipantRecord::recreate_from_serialized()` from participant_record.rs lines 51-57.\n */\n private static recreateFromSerialized(\n xidDocumentUr: string,\n petName: string | undefined,\n ): ParticipantRecord {\n const [raw, document] = parseSignedXidDocument(xidDocumentUr);\n return ParticipantRecord.buildFromParts(document, raw, petName);\n }\n\n /**\n * Serialize to JSON object.\n */\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {\n xid_document: this._xidDocumentUr,\n };\n if (this._petName !== undefined) {\n obj[\"pet_name\"] = this._petName;\n }\n return obj;\n }\n\n /**\n * Deserialize from JSON object.\n *\n * Mirrors Rust's `#[serde(deny_unknown_fields)]` derive on\n * `ParticipantRecord` (`participant_record.rs:12-17`) — Rust's\n * `serde_json::from_str` errors with `unknown field` for any\n * field outside `{xid_document, pet_name}`. We mirror that\n * behaviour so a registry file produced by a future Rust\n * version with extra fields is rejected explicitly rather than\n * silently lossy.\n */\n static fromJSON(json: Record<string, unknown>): ParticipantRecord {\n for (const key of Object.keys(json)) {\n if (key !== \"xid_document\" && key !== \"pet_name\") {\n throw new Error(`unknown field \\`${key}\\`, expected \\`xid_document\\` or \\`pet_name\\``);\n }\n }\n const xidDocumentUr = json[\"xid_document\"] as string;\n const petName = json[\"pet_name\"] as string | undefined;\n return ParticipantRecord.recreateFromSerialized(xidDocumentUr, petName);\n }\n}\n\n/**\n * Parse a signed XID document from a UR string.\n *\n * Port of `parse_signed_xid_document()` from participant_record.rs lines 170-194.\n */\nfunction parseSignedXidDocument(xidDocumentUr: string): [string, XIDDocument] {\n const sanitized = sanitizeXidUr(xidDocumentUr);\n const ur = UR.fromURString(sanitized);\n\n if (ur.urTypeStr() !== \"xid\" && ur.urTypeStr() !== \"envelope\") {\n throw new Error(`Expected a ur:xid document, found ur:${ur.urTypeStr()}`);\n }\n\n const envelopeCbor = ur.cbor();\n let envelope: Envelope;\n try {\n envelope = Envelope.fromTaggedCbor(envelopeCbor);\n } catch {\n try {\n envelope = Envelope.fromUntaggedCbor(envelopeCbor);\n } catch (e) {\n throw new Error(\n `Unable to decode XID document envelope: ${(e as Error).message ?? String(e)}`,\n {\n cause: e,\n },\n );\n }\n }\n\n // Mirror Rust `participant_record.rs:198-203`'s `.context(...)` wrap:\n // any failure from `XIDDocument::from_envelope(..., XIDVerifySignature::Inception)`\n // is surfaced as \"XID document must be signed by its inception key: <cause>\".\n let document: XIDDocument;\n try {\n document = XIDDocument.fromEnvelope(envelope, undefined, XIDVerifySignature.Inception);\n } catch (e) {\n throw new Error(\n `XID document must be signed by its inception key: ${(e as Error).message ?? String(e)}`,\n { cause: e },\n );\n }\n\n return [sanitized, document];\n}\n\n/**\n * Sanitize XID UR input.\n *\n * Port of `sanitize_xid_ur()` from participant_record.rs lines 196-203.\n */\nfunction sanitizeXidUr(input: string): string {\n const trimmed = input.trim();\n if (trimmed.length === 0) {\n throw new Error(\"XID document is required\");\n }\n return trimmed;\n}\n","/**\n * Copyright © 2023-2026 Blockchain Commons, LLC\n * Copyright © 2025-2026 Parity Technologies\n *\n *\n * Registry implementation for managing participants and groups.\n *\n * Port of registry/registry_impl.rs from frost-hubert-rust.\n *\n * @module\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\n\nimport { type ARID, type PublicKeys, type XID } from \"@bcts/components\";\n\nimport { GroupRecord } from \"./group-record.js\";\nimport { OwnerRecord } from \"./owner-record.js\";\nimport { ParticipantRecord } from \"./participant-record.js\";\n\n/**\n * Outcome of adding a participant to the registry.\n *\n * Port of `enum AddOutcome` from registry_impl.rs.\n */\nexport enum AddOutcome {\n /** Participant was already present in the registry */\n AlreadyPresent = \"already_present\",\n /** Participant was successfully inserted */\n Inserted = \"inserted\",\n}\n\n/**\n * Outcome of setting the owner in the registry.\n *\n * Port of `enum OwnerOutcome` from registry_impl.rs.\n */\nexport enum OwnerOutcome {\n /** Owner was already present in the registry */\n AlreadyPresent = \"already_present\",\n /** Owner was successfully inserted */\n Inserted = \"inserted\",\n}\n\n/**\n * Outcome of recording a group in the registry.\n *\n * Port of `enum GroupOutcome` from registry_impl.rs.\n */\nexport enum GroupOutcome {\n /** Group was successfully inserted as new */\n Inserted = \"inserted\",\n /** Group already existed and was updated (contributions merged) */\n Updated = \"updated\",\n}\n\n/**\n * Registry for managing participants and groups.\n *\n * Port of `struct Registry` from registry_impl.rs lines 22-26.\n */\nexport class Registry {\n private _owner?: OwnerRecord | undefined;\n private readonly _participants: Map<string, ParticipantRecord>; // Map by XID UR string\n private readonly _groups: Map<string, GroupRecord>; // Map by ARID hex\n\n constructor() {\n this._owner = undefined;\n this._participants = new Map();\n this._groups = new Map();\n }\n\n /**\n * Get the owner record.\n */\n owner(): OwnerRecord | undefined {\n return this._owner;\n }\n\n /**\n * Set the owner record.\n *\n * Returns the outcome indicating whether the owner was already present or newly inserted.\n *\n * Port of `Registry::set_owner()` from registry_impl.rs.\n */\n setOwner(owner: OwnerRecord): OwnerOutcome {\n // Check for pet name conflicts with participants\n const petName = owner.petName();\n if (petName !== undefined) {\n const conflicting = this.participantByPetName(petName);\n if (conflicting?.[1].petName() === petName) {\n throw new Error(`Pet name '${petName}' already used by a participant`);\n }\n }\n\n if (this._owner === undefined) {\n this._owner = owner;\n return OwnerOutcome.Inserted;\n }\n\n const existing = this._owner;\n const existingXidUr = existing.xid().urString();\n const ownerXidUr = owner.xid().urString();\n\n if (\n existingXidUr === ownerXidUr &&\n existing.xidDocumentUr() === owner.xidDocumentUr() &&\n existing.petName() === owner.petName()\n ) {\n return OwnerOutcome.AlreadyPresent;\n }\n\n if (existingXidUr === ownerXidUr) {\n if (existing.xidDocumentUr() !== owner.xidDocumentUr()) {\n throw new Error(\"Owner already exists with different keys\");\n }\n this._owner = owner;\n return OwnerOutcome.Inserted;\n }\n\n throw new Error(`Owner already recorded for ${existing.xid().toString()}`);\n }\n\n /**\n * Get all participants.\n */\n participants(): Map<string, ParticipantRecord> {\n return this._participants;\n }\n\n /**\n * Get a participant by XID.\n */\n participant(xid: XID): ParticipantRecord | undefined {\n return this._participants.get(xid.urString());\n }\n\n /**\n * Add a participant.\n *\n * Mirrors Rust `Registry::add_participant`\n * (`registry_impl.rs:83-124`):\n *\n * 1. If `record.pet_name()` is already used by some other XID,\n * bail with `\"Pet name '{name}' already used by another\n * participant\"`.\n * 2. If `record.pet_name()` matches an existing record under the\n * *same* XID and the public keys also match, return\n * `AlreadyPresent`.\n * 3. If the pet name matches the same XID but public keys don't,\n * bail with `\"Participant already exists with a different pet\n * name\"`.\n * 4. Otherwise look up by XID. If present and public-keys + pet-name\n * both match, return `AlreadyPresent`; if XID is present but\n * anything differs, bail. If XID is new, insert and return\n * `Inserted`.\n *\n * The earlier port short-circuited on `participants.has(xidUr)` and\n * always returned `AlreadyPresent` — silently allowing re-adds with\n * a different pet name or different public keys, which Rust\n * correctly forbids.\n */\n addParticipant(xid: XID, record: ParticipantRecord): AddOutcome {\n const xidUr = xid.urString();\n const petName = record.petName();\n\n // Steps 1–3: pet-name conflict resolution.\n if (petName !== undefined) {\n for (const [existingXidUr, existingRecord] of this._participants) {\n if (existingRecord.petName() === petName) {\n if (existingXidUr !== xidUr) {\n throw new Error(`Pet name '${petName}' already used by another participant`);\n }\n if (publicKeysEqual(existingRecord.publicKeys(), record.publicKeys())) {\n return AddOutcome.AlreadyPresent;\n }\n throw new Error(\"Participant already exists with a different pet name\");\n }\n }\n }\n\n // Step 4: XID lookup.\n const existing = this._participants.get(xidUr);\n if (existing !== undefined) {\n if (\n publicKeysEqual(existing.publicKeys(), record.publicKeys()) &&\n existing.petName() === record.petName()\n ) {\n return AddOutcome.AlreadyPresent;\n }\n throw new Error(\"Participant already exists with a different pet name\");\n }\n\n this._participants.set(xidUr, record);\n return AddOutcome.Inserted;\n }\n\n /**\n * Check if a pet name is already used.\n */\n petNameExists(petName: string): boolean {\n for (const record of this._participants.values()) {\n if (record.petName() === petName) {\n return true;\n }\n }\n return false;\n }\n\n /**\n * Find a participant by pet name.\n *\n * Port of `Registry::participant_by_pet_name()` from registry_impl.rs.\n */\n participantByPetName(petName: string): [XID, ParticipantRecord] | undefined {\n for (const record of this._participants.values()) {\n if (record.petName() === petName) {\n return [record.xid(), record];\n }\n }\n return undefined;\n }\n\n /**\n * Get all groups.\n */\n groups(): Map<string, GroupRecord> {\n return this._groups;\n }\n\n /**\n * Get a group by ARID.\n */\n group(arid: ARID): GroupRecord | undefined {\n return this._groups.get(arid.hex());\n }\n\n /**\n * Get a mutable reference to a group by ARID.\n */\n groupMut(arid: ARID): GroupRecord | undefined {\n return this._groups.get(arid.hex());\n }\n\n /**\n * Record a group in the registry.\n *\n * If the group already exists:\n * - Validates that the config matches\n * - Merges contributions from the new record\n * - Updates verifying key if not already set\n *\n * Port of `Registry::record_group()` from registry_impl.rs.\n */\n recordGroup(groupId: ARID, record: GroupRecord): GroupOutcome {\n const key = groupId.hex();\n const existing = this._groups.get(key);\n\n if (existing !== undefined) {\n // Validate config matches\n if (!existing.configMatches(record)) {\n throw new Error(`Group ${groupId.hex()} already exists with a different configuration`);\n }\n\n // Merge contributions\n existing.mergeContributions(record.contributions());\n\n // Update verifying key if not already set\n const existingKey = existing.verifyingKey();\n const recordKey = record.verifyingKey();\n if (existingKey === undefined && recordKey !== undefined) {\n existing.setVerifyingKey(recordKey);\n } else if (\n existingKey !== undefined &&\n recordKey !== undefined &&\n existingKey.urString() !== recordKey.urString()\n ) {\n throw new Error(`Group ${groupId.hex()} already exists with a different verifying key`);\n }\n\n return GroupOutcome.Updated;\n }\n\n this._groups.set(key, record);\n return GroupOutcome.Inserted;\n }\n\n /**\n * Add a group (simple variant without merge logic).\n *\n * @deprecated Use recordGroup() for proper merge behavior.\n */\n addGroup(arid: ARID, record: GroupRecord): void {\n this._groups.set(arid.hex(), record);\n }\n\n /**\n * Load a registry from a file.\n */\n static load(filePath: string): Registry {\n if (!fs.existsSync(filePath)) {\n return new Registry();\n }\n\n const content = fs.readFileSync(filePath, \"utf-8\");\n const json = JSON.parse(content) as Record<string, unknown>;\n return Registry.fromJSON(json);\n }\n\n /**\n * Save the registry to a file.\n */\n save(filePath: string): void {\n const dir = path.dirname(filePath);\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n const json = this.toJSON();\n fs.writeFileSync(filePath, JSON.stringify(json, null, 2));\n }\n\n /**\n * Serialize to JSON object.\n *\n * Mirrors Rust `Registry`'s field declaration order\n * (`registry_impl.rs:8-14` — `owner, participants, groups`). JSON\n * object member order is not semantically significant, but\n * `serde_json::to_string_pretty` emits keys in declaration order,\n * so for byte-equal `registry.json` (used by the integration tests\n * as a string assertion) the TS port must match Rust's order.\n * Empty `owner` is omitted via `Option::None` skip in Rust; we\n * reproduce that by only setting the key when the owner exists.\n */\n toJSON(): Record<string, unknown> {\n const obj: Record<string, unknown> = {};\n\n if (this._owner !== undefined) {\n obj[\"owner\"] = this._owner.toJSON();\n }\n\n const participants: Record<string, unknown> = {};\n for (const [xidUr, record] of this._participants) {\n participants[xidUr] = record.toJSON();\n }\n obj[\"participants\"] = participants;\n\n const groups: Record<string, unknown> = {};\n for (const [aridHex, record] of this._groups) {\n groups[aridHex] = record.toJSON();\n }\n obj[\"groups\"] = groups;\n\n return obj;\n }\n\n /**\n * Deserialize from JSON object.\n */\n static fromJSON(json: Record<string, unknown>): Registry {\n const registry = new Registry();\n\n if (json[\"owner\"] !== undefined) {\n registry._owner = OwnerRecord.fromJSON(json[\"owner\"] as Record<string, unknown>);\n }\n\n const participantsJson = json[\"participants\"] as\n | Record<string, Record<string, unknown>>\n | undefined;\n if (participantsJson !== undefined) {\n for (const [, recordJson] of Object.entries(participantsJson)) {\n const record = ParticipantRecord.fromJSON(recordJson);\n registry._participants.set(record.xid().urString(), record);\n }\n }\n\n const groupsJson = json[\"groups\"] as Record<string, Record<string, unknown>> | undefined;\n if (groupsJson !== undefined) {\n for (const [aridHex, recordJson] of Object.entries(groupsJson)) {\n const record = GroupRecord.fromJSON(recordJson);\n registry._groups.set(aridHex, record);\n }\n }\n\n return registry;\n }\n}\n\n/**\n * Resolve the registry file path from a given argument.\n *\n * Port of `resolve_registry_path()` from registry/mod.rs lines 49-78.\n */\nexport function resolveRegistryPath(registryArg: string | undefined, cwd: string): string {\n if (registryArg === undefined || registryArg === \"\") {\n return path.join(cwd, \"registry.json\");\n }\n\n // If it ends with / or is a directory, append registry.json\n if (registryArg.endsWith(\"/\") || registryArg.endsWith(path.sep)) {\n return path.join(cwd, registryArg, \"registry.json\");\n }\n\n // If it's just a filename, put it in cwd\n if (!registryArg.includes(\"/\") && !registryArg.includes(path.sep)) {\n return path.join(cwd, registryArg);\n }\n\n // Otherwise, treat as relative path\n return path.resolve(cwd, registryArg);\n}\n\n/**\n * Structural equality on `PublicKeys`, mirroring Rust's\n * `PartialEq::eq` derive (per-component comparison of the signing\n * and encapsulation public keys).\n *\n * @internal\n */\nfunction publicKeysEqual(a: PublicKeys, b: PublicKeys): boolean {\n return a.equals(b);\n}\n"],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- import { a as resolveRegistryPath, c as ContributionPaths, d as PendingRequests, i as Registry, l as GroupParticipant, n as GroupOutcome, o as ParticipantRecord, r as OwnerOutcome, s as OwnerRecord, t as AddOutcome, u as GroupRecord } from "../index-C8QeHNwa.cjs";
1
+ import { a as resolveRegistryPath, c as ContributionPaths, d as PendingRequests, i as Registry, l as GroupParticipant, n as GroupOutcome, o as ParticipantRecord, r as OwnerOutcome, s as OwnerRecord, t as AddOutcome, u as GroupRecord } from "../index-CD50Qtgw.cjs";
2
2
  export { AddOutcome, ContributionPaths, GroupOutcome, GroupParticipant, GroupRecord, OwnerOutcome, OwnerRecord, ParticipantRecord, PendingRequests, Registry, resolveRegistryPath };
@@ -1,2 +1,2 @@
1
- import { a as resolveRegistryPath, c as ContributionPaths, d as PendingRequests, i as Registry, l as GroupParticipant, n as GroupOutcome, o as ParticipantRecord, r as OwnerOutcome, s as OwnerRecord, t as AddOutcome, u as GroupRecord } from "../index-DVbWyOs7.mjs";
1
+ import { a as resolveRegistryPath, c as ContributionPaths, d as PendingRequests, i as Registry, l as GroupParticipant, n as GroupOutcome, o as ParticipantRecord, r as OwnerOutcome, s as OwnerRecord, t as AddOutcome, u as GroupRecord } from "../index-CD50Qtgw.mjs";
2
2
  export { AddOutcome, ContributionPaths, GroupOutcome, GroupParticipant, GroupRecord, OwnerOutcome, OwnerRecord, ParticipantRecord, PendingRequests, Registry, resolveRegistryPath };