@bitgo-beta/sdk-coin-canton 1.0.1-beta.24 → 1.0.1-beta.241
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.
- package/dist/resources/hash/hash.js +3 -2
- package/dist/resources/proto/damlTransaction.js +4 -4
- package/dist/resources/proto/damlTransactionNode.js +3 -3
- package/dist/resources/proto/damlTransactionNodeSeed.js +2 -2
- package/dist/resources/proto/metadata/metadataGlobalKeyMappingEntry.js +4 -4
- package/dist/resources/proto/metadata/metadataInputContract.js +3 -3
- package/dist/resources/proto/metadata/metadataSubmitterInfo.js +2 -2
- package/dist/resources/proto/metadata.js +5 -5
- package/dist/resources/proto/node/empty.js +2 -2
- package/dist/resources/proto/node/globalKey.js +4 -4
- package/dist/resources/proto/node/identifier.js +2 -2
- package/dist/resources/proto/node/node.js +16 -8
- package/dist/resources/proto/node/timestamp.js +3 -3
- package/dist/resources/proto/node/value.js +28 -14
- package/dist/resources/proto/preparedTransaction.js +4 -4
- package/dist/src/canton.d.ts +18 -2
- package/dist/src/canton.d.ts.map +1 -1
- package/dist/src/canton.js +107 -6
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +2 -1
- package/dist/src/lib/constant.d.ts +4 -0
- package/dist/src/lib/constant.d.ts.map +1 -1
- package/dist/src/lib/constant.js +6 -2
- package/dist/src/lib/iface.d.ts +77 -20
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +1 -1
- package/dist/src/lib/index.d.ts +7 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +16 -2
- package/dist/src/lib/oneStepPreApprovalBuilder.d.ts +7 -41
- package/dist/src/lib/oneStepPreApprovalBuilder.d.ts.map +1 -1
- package/dist/src/lib/oneStepPreApprovalBuilder.js +18 -82
- package/dist/src/lib/transaction/transaction.d.ts +13 -2
- package/dist/src/lib/transaction/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction/transaction.js +178 -9
- package/dist/src/lib/transactionBuilder.d.ts +4 -4
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +1 -5
- package/dist/src/lib/transactionBuilderFactory.d.ts +17 -3
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +63 -6
- package/dist/src/lib/transferAcceptanceBuilder.d.ts +59 -0
- package/dist/src/lib/transferAcceptanceBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferAcceptanceBuilder.js +116 -0
- package/dist/src/lib/transferAcknowledgeBuilder.d.ts +72 -0
- package/dist/src/lib/transferAcknowledgeBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferAcknowledgeBuilder.js +133 -0
- package/dist/src/lib/transferBuilder.d.ts +79 -3
- package/dist/src/lib/transferBuilder.d.ts.map +1 -1
- package/dist/src/lib/transferBuilder.js +153 -1
- package/dist/src/lib/transferRejectionBuilder.d.ts +59 -0
- package/dist/src/lib/transferRejectionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferRejectionBuilder.js +116 -0
- package/dist/src/lib/utils.d.ts +22 -2
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +148 -28
- package/dist/src/lib/walletInitBuilder.d.ts +5 -10
- package/dist/src/lib/walletInitBuilder.d.ts.map +1 -1
- package/dist/src/lib/walletInitBuilder.js +13 -18
- package/dist/src/lib/walletInitialization/walletInitTransaction.d.ts +2 -1
- package/dist/src/lib/walletInitialization/walletInitTransaction.d.ts.map +1 -1
- package/dist/src/lib/walletInitialization/walletInitTransaction.js +46 -5
- package/dist/test/helper.d.ts +3 -0
- package/dist/test/helper.d.ts.map +1 -0
- package/dist/test/helper.js +9 -0
- package/dist/test/integration/canton.integration.d.ts +2 -0
- package/dist/test/integration/canton.integration.d.ts.map +1 -0
- package/dist/test/integration/canton.integration.js +157 -0
- package/dist/test/resources.d.ts +78 -7
- package/dist/test/resources.d.ts.map +1 -1
- package/dist/test/resources.js +83 -10
- package/dist/test/unit/builder/oneStepEnablement/oneStepEnablementBuilder.js +14 -29
- package/dist/test/unit/builder/transfer/transferBuilder.d.ts +2 -0
- package/dist/test/unit/builder/transfer/transferBuilder.d.ts.map +1 -0
- package/dist/test/unit/builder/transfer/transferBuilder.js +63 -0
- package/dist/test/unit/builder/transferAccept/transferAcceptBuilder.d.ts +2 -0
- package/dist/test/unit/builder/transferAccept/transferAcceptBuilder.d.ts.map +1 -0
- package/dist/test/unit/builder/transferAccept/transferAcceptBuilder.js +57 -0
- package/dist/test/unit/builder/transferAcknowledge/transferAcknowledgeBuilder.d.ts +2 -0
- package/dist/test/unit/builder/transferAcknowledge/transferAcknowledgeBuilder.d.ts.map +1 -0
- package/dist/test/unit/builder/transferAcknowledge/transferAcknowledgeBuilder.js +32 -0
- package/dist/test/unit/builder/transferReject/transferRejectBuilder.d.ts +2 -0
- package/dist/test/unit/builder/transferReject/transferRejectBuilder.d.ts.map +1 -0
- package/dist/test/unit/builder/transferReject/transferRejectBuilder.js +57 -0
- package/dist/test/unit/builder/walletInit/walletInitBuilder.js +10 -13
- package/dist/test/unit/utils.js +101 -3
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +24 -7
package/dist/src/lib/utils.js
CHANGED
|
@@ -4,6 +4,7 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
6
|
exports.Utils = void 0;
|
|
7
|
+
const bignumber_js_1 = __importDefault(require("bignumber.js"));
|
|
7
8
|
const crypto_1 = __importDefault(require("crypto"));
|
|
8
9
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
9
10
|
const hash_js_1 = require("../../resources/hash/hash.js");
|
|
@@ -12,11 +13,24 @@ const constant_1 = require("./constant");
|
|
|
12
13
|
class Utils {
|
|
13
14
|
/** @inheritdoc */
|
|
14
15
|
isValidAddress(address) {
|
|
15
|
-
|
|
16
|
+
if (!address || address.trim() === '')
|
|
17
|
+
return false;
|
|
18
|
+
const [partyHint, fingerprint] = address.trim().split('::');
|
|
19
|
+
if (!fingerprint)
|
|
20
|
+
return false;
|
|
21
|
+
// all memoIds are considered valid as long as strings are passed
|
|
22
|
+
const [fingerprintPart] = fingerprint.trim().split('?memoId=');
|
|
23
|
+
if (!partyHint || !fingerprintPart)
|
|
24
|
+
return false;
|
|
25
|
+
return this.isValidCantonHex(fingerprintPart);
|
|
16
26
|
}
|
|
17
27
|
/** @inheritdoc */
|
|
18
28
|
isValidBlockId(hash) {
|
|
19
|
-
|
|
29
|
+
// In canton, there is no block hash, we store the height as the _id (hash)
|
|
30
|
+
// this will be of the form, <blockHeight>_<version>
|
|
31
|
+
const [height] = hash.split('_');
|
|
32
|
+
const blockHeight = Number(height);
|
|
33
|
+
return !isNaN(blockHeight) && blockHeight > 0;
|
|
20
34
|
}
|
|
21
35
|
/** @inheritdoc */
|
|
22
36
|
isValidPrivateKey(key) {
|
|
@@ -34,6 +48,23 @@ class Utils {
|
|
|
34
48
|
isValidTransactionId(txId) {
|
|
35
49
|
throw new Error('Method not implemented.');
|
|
36
50
|
}
|
|
51
|
+
/**
|
|
52
|
+
* Method to validate the input is a valid canton hex string
|
|
53
|
+
* @param {String} value the hex string value
|
|
54
|
+
* @returns {Boolean} true if valid
|
|
55
|
+
*/
|
|
56
|
+
isValidCantonHex(value) {
|
|
57
|
+
const regex = /^[a-fA-F0-9]{68}$/;
|
|
58
|
+
return regex.test(value);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Helper method to convert hex value to base64
|
|
62
|
+
* @param {String} hexString - hex encoded string
|
|
63
|
+
* @returns {String} base64 encoded string
|
|
64
|
+
*/
|
|
65
|
+
getBase64FromHex(hexString) {
|
|
66
|
+
return Buffer.from(hexString, 'hex').toString('base64');
|
|
67
|
+
}
|
|
37
68
|
/**
|
|
38
69
|
* Method to create fingerprint (part of the canton partyId) from public key
|
|
39
70
|
* @param {String} publicKey the public key
|
|
@@ -47,14 +78,20 @@ class Utils {
|
|
|
47
78
|
/**
|
|
48
79
|
* Method to parse raw canton transaction & get required data
|
|
49
80
|
* @param {String} rawData base64 encoded string
|
|
81
|
+
* @param {TransactionType} txType the transaction type
|
|
50
82
|
* @returns {PreparedTxnParsedInfo}
|
|
51
83
|
*/
|
|
52
|
-
parseRawCantonTransactionData(rawData) {
|
|
84
|
+
parseRawCantonTransactionData(rawData, txType) {
|
|
53
85
|
const decodedData = this.decodePreparedTransaction(rawData);
|
|
54
86
|
let sender = '';
|
|
55
87
|
let receiver = '';
|
|
56
88
|
let amount = '';
|
|
57
|
-
|
|
89
|
+
let memoId;
|
|
90
|
+
let preApprovalNode = [];
|
|
91
|
+
let transferNode = [];
|
|
92
|
+
let transferAcceptRejectNode = [];
|
|
93
|
+
const nodes = decodedData.transaction?.nodes;
|
|
94
|
+
nodes?.forEach((node) => {
|
|
58
95
|
const versionedNode = node.versionedNode;
|
|
59
96
|
if (!versionedNode || versionedNode.oneofKind !== 'v1')
|
|
60
97
|
return;
|
|
@@ -63,37 +100,106 @@ class Utils {
|
|
|
63
100
|
if (nodeType.oneofKind !== 'create')
|
|
64
101
|
return;
|
|
65
102
|
const createNode = nodeType.create;
|
|
66
|
-
// Check if it's the correct template
|
|
67
103
|
const template = createNode.templateId;
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
// Now parse the 'create' argument
|
|
71
|
-
if (createNode.argument?.sum?.oneofKind !== 'record')
|
|
104
|
+
const argSum = createNode.argument?.sum;
|
|
105
|
+
if (!argSum || argSum.oneofKind !== 'record')
|
|
72
106
|
return;
|
|
73
|
-
const fields =
|
|
107
|
+
const fields = argSum.record?.fields;
|
|
74
108
|
if (!fields)
|
|
75
109
|
return;
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
110
|
+
if (template?.entityName === 'TransferPreapprovalProposal' &&
|
|
111
|
+
!preApprovalNode.length &&
|
|
112
|
+
txType === sdk_core_1.TransactionType.OneStepPreApproval) {
|
|
113
|
+
preApprovalNode = fields;
|
|
114
|
+
}
|
|
115
|
+
if (template?.entityName === 'Amulet' &&
|
|
116
|
+
!transferAcceptRejectNode.length &&
|
|
117
|
+
(txType === sdk_core_1.TransactionType.TransferAccept || txType === sdk_core_1.TransactionType.TransferReject)) {
|
|
118
|
+
transferAcceptRejectNode = fields;
|
|
119
|
+
}
|
|
120
|
+
});
|
|
121
|
+
nodes?.forEach((node) => {
|
|
122
|
+
const versionedNode = node.versionedNode;
|
|
123
|
+
if (!versionedNode || versionedNode.oneofKind !== 'v1')
|
|
79
124
|
return;
|
|
80
|
-
const
|
|
81
|
-
|
|
125
|
+
const v1Node = versionedNode.v1;
|
|
126
|
+
const nodeType = v1Node.nodeType;
|
|
127
|
+
if (nodeType.oneofKind !== 'exercise')
|
|
82
128
|
return;
|
|
83
|
-
const
|
|
84
|
-
const
|
|
85
|
-
if (!
|
|
129
|
+
const exerciseNode = nodeType.exercise;
|
|
130
|
+
const choiceId = exerciseNode.choiceId;
|
|
131
|
+
if (!choiceId || choiceId !== 'TransferFactory_Transfer')
|
|
86
132
|
return;
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
if (!receiverData || receiverData.oneofKind !== 'party')
|
|
133
|
+
const argSum = exerciseNode.chosenValue?.sum;
|
|
134
|
+
if (!argSum || argSum.oneofKind !== 'record')
|
|
90
135
|
return;
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
if (!amountData || amountData.oneofKind !== 'numeric')
|
|
136
|
+
const fields = argSum.record?.fields;
|
|
137
|
+
if (!fields)
|
|
94
138
|
return;
|
|
95
|
-
|
|
139
|
+
transferNode = fields;
|
|
96
140
|
});
|
|
141
|
+
const getField = (fields, label) => fields.find((f) => f.label === label)?.value?.sum;
|
|
142
|
+
if (preApprovalNode.length) {
|
|
143
|
+
const receiverData = getField(preApprovalNode, 'receiver');
|
|
144
|
+
if (receiverData?.oneofKind === 'party')
|
|
145
|
+
receiver = receiverData.party ?? '';
|
|
146
|
+
const providerData = getField(preApprovalNode, 'provider');
|
|
147
|
+
if (providerData?.oneofKind === 'party')
|
|
148
|
+
sender = providerData.party ?? '';
|
|
149
|
+
amount = '0';
|
|
150
|
+
}
|
|
151
|
+
else if (transferNode.length) {
|
|
152
|
+
const transferField = transferNode.find((f) => f.label === 'transfer');
|
|
153
|
+
const transferSum = transferField?.value?.sum;
|
|
154
|
+
if (transferSum && transferSum.oneofKind === 'record') {
|
|
155
|
+
const transferRecord = transferSum.record?.fields;
|
|
156
|
+
if (transferRecord?.length) {
|
|
157
|
+
const senderData = getField(transferRecord, 'sender');
|
|
158
|
+
if (senderData?.oneofKind === 'party')
|
|
159
|
+
sender = senderData.party ?? '';
|
|
160
|
+
const receiverData = getField(transferRecord, 'receiver');
|
|
161
|
+
if (receiverData?.oneofKind === 'party')
|
|
162
|
+
receiver = receiverData.party ?? '';
|
|
163
|
+
const amountData = getField(transferRecord, 'amount');
|
|
164
|
+
if (amountData?.oneofKind === 'numeric')
|
|
165
|
+
amount = amountData.numeric ?? '';
|
|
166
|
+
const metaField = getField(transferRecord, 'meta');
|
|
167
|
+
if (metaField?.oneofKind === 'record') {
|
|
168
|
+
const metaFields = metaField.record?.fields;
|
|
169
|
+
if (metaFields && metaFields.length) {
|
|
170
|
+
const valuesField = getField(metaFields, 'values');
|
|
171
|
+
if (valuesField?.oneofKind === 'textMap') {
|
|
172
|
+
const entries = valuesField.textMap?.entries ?? [];
|
|
173
|
+
const memoEntry = entries.find((e) => e.key === 'splice.lfdecentralizedtrust.org/reason');
|
|
174
|
+
if (memoEntry) {
|
|
175
|
+
const memoValue = memoEntry?.value?.sum;
|
|
176
|
+
if (memoValue?.oneofKind === 'text') {
|
|
177
|
+
memoId = memoValue.text;
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
else if (transferAcceptRejectNode.length) {
|
|
187
|
+
const dsoData = getField(transferAcceptRejectNode, 'dso');
|
|
188
|
+
if (dsoData?.oneofKind === 'party')
|
|
189
|
+
sender = dsoData.party ?? '';
|
|
190
|
+
const ownerData = getField(transferAcceptRejectNode, 'owner');
|
|
191
|
+
if (ownerData?.oneofKind === 'party')
|
|
192
|
+
receiver = ownerData.party ?? '';
|
|
193
|
+
const amountField = getField(transferAcceptRejectNode, 'amount');
|
|
194
|
+
if (amountField && amountField.oneofKind === 'record') {
|
|
195
|
+
const amountRecord = amountField.record?.fields;
|
|
196
|
+
if (amountRecord?.length) {
|
|
197
|
+
const initialAmountData = getField(amountRecord, 'initialAmount');
|
|
198
|
+
if (initialAmountData?.oneofKind === 'numeric')
|
|
199
|
+
amount = initialAmountData.numeric ?? '';
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
}
|
|
97
203
|
if (!sender || !receiver || !amount) {
|
|
98
204
|
const missingFields = [];
|
|
99
205
|
if (!sender)
|
|
@@ -104,11 +210,16 @@ class Utils {
|
|
|
104
210
|
missingFields.push('amount');
|
|
105
211
|
throw new Error(`invalid transaction data: missing ${missingFields.join(', ')}`);
|
|
106
212
|
}
|
|
107
|
-
|
|
213
|
+
const convertedAmount = this.convertAmountToLowestUnit(new bignumber_js_1.default(amount));
|
|
214
|
+
const parsedData = {
|
|
108
215
|
sender,
|
|
109
216
|
receiver,
|
|
110
|
-
amount,
|
|
217
|
+
amount: convertedAmount,
|
|
111
218
|
};
|
|
219
|
+
if (memoId) {
|
|
220
|
+
parsedData.memoId = memoId;
|
|
221
|
+
}
|
|
222
|
+
return parsedData;
|
|
112
223
|
}
|
|
113
224
|
/**
|
|
114
225
|
* Computes the topology hash from the API response of the 'create party' endpoint.
|
|
@@ -236,8 +347,17 @@ class Utils {
|
|
|
236
347
|
buf.writeInt32BE(value, 0);
|
|
237
348
|
return buf;
|
|
238
349
|
}
|
|
350
|
+
/**
|
|
351
|
+
* Convert to canton raw units
|
|
352
|
+
* @param {BigNumber} value
|
|
353
|
+
* @returns {String} the converted raw canton units
|
|
354
|
+
* @private
|
|
355
|
+
*/
|
|
356
|
+
convertAmountToLowestUnit(value) {
|
|
357
|
+
return value.multipliedBy(new bignumber_js_1.default(10).pow(10)).toFixed(0);
|
|
358
|
+
}
|
|
239
359
|
}
|
|
240
360
|
exports.Utils = Utils;
|
|
241
361
|
const utils = new Utils();
|
|
242
362
|
exports.default = utils;
|
|
243
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import crypto from 'crypto';\n\nimport { BaseUtils, isValidEd25519PublicKey } from '@bitgo-beta/sdk-core';\n\nimport { computePreparedTransaction } from '../../resources/hash/hash.js';\nimport { PreparedTransaction } from '../../resources/proto/preparedTransaction.js';\n\nimport { CryptoKeyFormat, SigningAlgorithmSpec, SigningKeySpec } from './constant';\nimport { PreparedTransaction as IPreparedTransaction, PreparedTxnParsedInfo } from './iface';\nimport { RecordField } from './resourcesInterface';\n\nexport class Utils implements BaseUtils {\n  /** @inheritdoc */\n  isValidAddress(address: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritdoc */\n  isValidBlockId(hash: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritdoc */\n  isValidPrivateKey(key: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritdoc */\n  isValidPublicKey(key: string): boolean {\n    return isValidEd25519PublicKey(key);\n  }\n\n  /** @inheritdoc */\n  isValidSignature(signature: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritdoc */\n  isValidTransactionId(txId: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Method to create fingerprint (part of the canton partyId) from public key\n   * @param {String} publicKey the public key\n   * @returns {String}\n   */\n  getAddressFromPublicKey(publicKey: string): string {\n    const key = this.signingPublicKeyFromEd25519(publicKey);\n    const hashPurpose = 12;\n    return this.computeSha256CantonHash(hashPurpose, key.publicKey);\n  }\n\n  /**\n   * Method to parse raw canton transaction & get required data\n   * @param {String} rawData base64 encoded string\n   * @returns {PreparedTxnParsedInfo}\n   */\n  parseRawCantonTransactionData(rawData: string): PreparedTxnParsedInfo {\n    const decodedData = this.decodePreparedTransaction(rawData);\n    let sender = '';\n    let receiver = '';\n    let amount = '';\n    decodedData.transaction?.nodes?.forEach((node) => {\n      const versionedNode = node.versionedNode;\n      if (!versionedNode || versionedNode.oneofKind !== 'v1') return;\n\n      const v1Node = versionedNode.v1;\n      const nodeType = v1Node.nodeType;\n\n      if (nodeType.oneofKind !== 'create') return;\n\n      const createNode = nodeType.create;\n\n      // Check if it's the correct template\n      const template = createNode.templateId;\n      if (template?.entityName !== 'AmuletTransferInstruction') return;\n\n      // Now parse the 'create' argument\n      if (createNode.argument?.sum?.oneofKind !== 'record') return;\n      const fields = createNode.argument?.sum?.record?.fields;\n      if (!fields) return;\n\n      // Find the 'transfer' field\n      const transferField = fields.find((f) => f.label === 'transfer');\n      if (transferField?.value?.sum?.oneofKind !== 'record') return;\n      const transferRecord = transferField?.value?.sum?.record?.fields;\n      if (!transferRecord) return;\n\n      const getField = (fields: RecordField[], label: string) => fields.find((f) => f.label === label)?.value?.sum;\n\n      const senderData = getField(transferRecord, 'sender');\n      if (!senderData || senderData.oneofKind !== 'party') return;\n      sender = senderData.party;\n      const receiverData = getField(transferRecord, 'receiver');\n      if (!receiverData || receiverData.oneofKind !== 'party') return;\n      receiver = receiverData.party;\n      const amountData = getField(transferRecord, 'amount');\n      if (!amountData || amountData.oneofKind !== 'numeric') return;\n      amount = amountData.numeric;\n    });\n    if (!sender || !receiver || !amount) {\n      const missingFields: string[] = [];\n      if (!sender) missingFields.push('sender');\n      if (!receiver) missingFields.push('receiver');\n      if (!amount) missingFields.push('amount');\n      throw new Error(`invalid transaction data: missing ${missingFields.join(', ')}`);\n    }\n    return {\n      sender,\n      receiver,\n      amount,\n    };\n  }\n\n  /**\n   * Computes the topology hash from the API response of the 'create party' endpoint.\n   *\n   * @param topologyTransactions - List of base64-encoded topology transactions from the Canton API.\n   * @returns The final base64-encoded topology transaction hash.\n   */\n  computeHashFromCreatePartyResponse(topologyTransactions: string[]): string {\n    const txBuffers = topologyTransactions.map((tx) => Buffer.from(tx, 'base64'));\n    return this.computeHashFromTopologyTransaction(txBuffers);\n  }\n\n  async computeHashFromPrepareSubmissionResponse(preparedTransactionBase64: string): Promise<string> {\n    const preparedTransaction = this.decodePreparedTransaction(preparedTransactionBase64);\n    const hash = await computePreparedTransaction(preparedTransaction);\n    return Buffer.from(hash).toString('base64');\n  }\n\n  /**\n   * Computes the final topology transaction hash for a list of prepared Canton transactions.\n   *\n   * Each transaction is first hashed with purpose `11`, then all hashes are combined and\n   * hashed again with purpose `55`, following the Canton topology hash rules.\n   *\n   * The resulting hash is encoded as a base64 string.\n   *\n   * @param {Buffer[]} preparedTransactions - An array of Canton transaction buffers.\n   * @returns {string} The final topology hash, base64-encoded.\n   */\n  private computeHashFromTopologyTransaction(preparedTransactions: Buffer[]): string {\n    const rawHashes = preparedTransactions.map((tx) => this.computeSha256CantonHash(11, tx));\n    const combinedHashes = this.computeMultiHashForTopology(rawHashes);\n    const computedHash = this.computeSha256CantonHash(55, combinedHashes);\n    return Buffer.from(computedHash, 'hex').toString('base64');\n  }\n\n  /**\n   * Converts a base64-encoded Ed25519 public key string into a structured signing public key object.\n   * @param {String} publicKey The base64-encoded Ed25519 public key\n   * @returns {Object} The structured signing key object formatted for use with cryptographic operations\n   * @private\n   */\n  private signingPublicKeyFromEd25519(publicKey: string): {\n    format: number;\n    publicKey: Buffer;\n    scheme: number;\n    keySpec: number;\n    usage: [];\n  } {\n    return {\n      format: CryptoKeyFormat.RAW,\n      publicKey: Buffer.from(publicKey, 'base64'),\n      scheme: SigningAlgorithmSpec.ED25519,\n      keySpec: SigningKeySpec.EC_CURVE25519,\n      usage: [],\n    };\n  }\n\n  /**\n   * Creates a buffer with a 4-byte big-endian integer prefix followed by the provided byte buffer\n   * @param {Number} value The integer to prefix, written as 4 bytes in big-endian order\n   * @param {Buffer} bytes The buffer to append after the integer prefix\n   * @returns {Buffer} The resulting buffer with the prefixed integer\n   * @private\n   */\n  private prefixedInt(value: number, bytes: Buffer): Buffer {\n    const buffer = Buffer.alloc(4 + bytes.length);\n    buffer.writeUInt32BE(value, 0);\n    Buffer.from(bytes).copy(buffer, 4);\n    return buffer;\n  }\n\n  /**\n   * Computes an SHA-256 Canton-style hash by prefixing the input with a purpose identifier,\n   * then hashing the resulting buffer and prepending a multi-prefix\n   *\n   * @param {Number} purpose A numeric identifier to prefix the hash input with\n   * @param {Buffer} bytes The buffer to be hashed\n   * @returns {String} A hexadecimal string representation of the resulting hash with multi-prefix\n   * @private\n   */\n  private computeSha256CantonHash(purpose: number, bytes: Buffer): string {\n    const hashInput = this.prefixedInt(purpose, bytes);\n    const hash = crypto.createHash('sha256').update(hashInput).digest();\n    const multiprefix = Buffer.from([0x12, 0x20]);\n    return Buffer.concat([multiprefix, hash]).toString('hex');\n  }\n\n  /**\n   * Decodes a Base64-encoded string into a Uint8Array\n   * @param {String} b64 The Base64-encoded string\n   * @returns {Uint8Array} The decoded byte array\n   * @private\n   */\n  private fromBase64(b64: string): Uint8Array {\n    return new Uint8Array(Buffer.from(b64, 'base64'));\n  }\n\n  /**\n   * Decodes a Base64-encoded prepared transaction into a structured object\n   * @param {String} base64 The Base64-encoded transaction data\n   * @returns {IPreparedTransaction} The decoded `IPreparedTransaction` object\n   * @private\n   */\n  private decodePreparedTransaction(base64: string): IPreparedTransaction {\n    const bytes = this.fromBase64(base64);\n    return PreparedTransaction.fromBinary(bytes);\n  }\n\n  /**\n   * Computes a deterministic combined hash from an array of individual Canton-style SHA-256 hashes\n   *\n   * Each hash is decoded from hex, sorted lexicographically (by hex), and prefixed with its length\n   * The final buffer includes the number of hashes followed by each (length-prefixed) hash\n   *\n   * @param {string[]} hashes - An array of Canton-prefixed SHA-256 hashes in hexadecimal string format\n   * @returns {Buffer} A binary buffer representing the combined hash input\n   */\n  private computeMultiHashForTopology(hashes: string[]): Buffer {\n    const sortedHashes = hashes\n      .map((hex) => Buffer.from(hex, 'hex'))\n      .sort((a, b) => a.toString('hex').localeCompare(b.toString('hex')));\n\n    const numHashesBytes = this.encodeInt32(sortedHashes.length);\n    const parts: Buffer[] = [numHashesBytes];\n\n    for (const h of sortedHashes) {\n      const lengthBytes = this.encodeInt32(h.length);\n      parts.push(lengthBytes, h);\n    }\n\n    return Buffer.concat(parts);\n  }\n\n  /**\n   * Encodes a 32-bit signed integer into a 4-byte big-endian Buffer\n   *\n   * @param {number} value - The integer to encode\n   * @returns {Buffer} A 4-byte buffer representing the integer in big-endian format\n   */\n  private encodeInt32(value: number): Buffer {\n    const buf = Buffer.alloc(4);\n    buf.writeInt32BE(value, 0);\n    return buf;\n  }\n}\n\nconst utils = new Utils();\n\nexport default utils;\n"]}
|
|
363
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import BigNumber from 'bignumber.js';\nimport crypto from 'crypto';\n\nimport { BaseUtils, isValidEd25519PublicKey, TransactionType } from '@bitgo-beta/sdk-core';\n\nimport { computePreparedTransaction } from '../../resources/hash/hash.js';\nimport { PreparedTransaction } from '../../resources/proto/preparedTransaction.js';\n\nimport { CryptoKeyFormat, SigningAlgorithmSpec, SigningKeySpec } from './constant';\nimport { PreparedTransaction as IPreparedTransaction, PreparedTxnParsedInfo } from './iface';\nimport { RecordField } from './resourcesInterface';\n\nexport class Utils implements BaseUtils {\n  /** @inheritdoc */\n  isValidAddress(address: string): boolean {\n    if (!address || address.trim() === '') return false;\n    const [partyHint, fingerprint] = address.trim().split('::');\n    if (!fingerprint) return false;\n    // all memoIds are considered valid as long as strings are passed\n    const [fingerprintPart] = fingerprint.trim().split('?memoId=');\n    if (!partyHint || !fingerprintPart) return false;\n    return this.isValidCantonHex(fingerprintPart);\n  }\n\n  /** @inheritdoc */\n  isValidBlockId(hash: string): boolean {\n    // In canton, there is no block hash, we store the height as the _id (hash)\n    // this will be of the form, <blockHeight>_<version>\n    const [height] = hash.split('_');\n    const blockHeight = Number(height);\n    return !isNaN(blockHeight) && blockHeight > 0;\n  }\n\n  /** @inheritdoc */\n  isValidPrivateKey(key: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritdoc */\n  isValidPublicKey(key: string): boolean {\n    return isValidEd25519PublicKey(key);\n  }\n\n  /** @inheritdoc */\n  isValidSignature(signature: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritdoc */\n  isValidTransactionId(txId: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Method to validate the input is a valid canton hex string\n   * @param {String} value the hex string value\n   * @returns {Boolean} true if valid\n   */\n  isValidCantonHex(value: string): boolean {\n    const regex = /^[a-fA-F0-9]{68}$/;\n    return regex.test(value);\n  }\n\n  /**\n   * Helper method to convert hex value to base64\n   * @param {String} hexString - hex encoded string\n   * @returns {String} base64 encoded string\n   */\n  getBase64FromHex(hexString: string): string {\n    return Buffer.from(hexString, 'hex').toString('base64');\n  }\n\n  /**\n   * Method to create fingerprint (part of the canton partyId) from public key\n   * @param {String} publicKey the public key\n   * @returns {String}\n   */\n  getAddressFromPublicKey(publicKey: string): string {\n    const key = this.signingPublicKeyFromEd25519(publicKey);\n    const hashPurpose = 12;\n    return this.computeSha256CantonHash(hashPurpose, key.publicKey);\n  }\n\n  /**\n   * Method to parse raw canton transaction & get required data\n   * @param {String} rawData base64 encoded string\n   * @param {TransactionType} txType the transaction type\n   * @returns {PreparedTxnParsedInfo}\n   */\n  parseRawCantonTransactionData(rawData: string, txType: TransactionType): PreparedTxnParsedInfo {\n    const decodedData = this.decodePreparedTransaction(rawData);\n    let sender = '';\n    let receiver = '';\n    let amount = '';\n    let memoId: string | undefined;\n    let preApprovalNode: RecordField[] = [];\n    let transferNode: RecordField[] = [];\n    let transferAcceptRejectNode: RecordField[] = [];\n    const nodes = decodedData.transaction?.nodes;\n\n    nodes?.forEach((node) => {\n      const versionedNode = node.versionedNode;\n      if (!versionedNode || versionedNode.oneofKind !== 'v1') return;\n      const v1Node = versionedNode.v1;\n      const nodeType = v1Node.nodeType;\n      if (nodeType.oneofKind !== 'create') return;\n      const createNode = nodeType.create;\n      const template = createNode.templateId;\n      const argSum = createNode.argument?.sum;\n      if (!argSum || argSum.oneofKind !== 'record') return;\n      const fields = argSum.record?.fields;\n      if (!fields) return;\n      if (\n        template?.entityName === 'TransferPreapprovalProposal' &&\n        !preApprovalNode.length &&\n        txType === TransactionType.OneStepPreApproval\n      ) {\n        preApprovalNode = fields;\n      }\n      if (\n        template?.entityName === 'Amulet' &&\n        !transferAcceptRejectNode.length &&\n        (txType === TransactionType.TransferAccept || txType === TransactionType.TransferReject)\n      ) {\n        transferAcceptRejectNode = fields;\n      }\n    });\n\n    nodes?.forEach((node) => {\n      const versionedNode = node.versionedNode;\n      if (!versionedNode || versionedNode.oneofKind !== 'v1') return;\n      const v1Node = versionedNode.v1;\n      const nodeType = v1Node.nodeType;\n      if (nodeType.oneofKind !== 'exercise') return;\n      const exerciseNode = nodeType.exercise;\n      const choiceId = exerciseNode.choiceId;\n      if (!choiceId || choiceId !== 'TransferFactory_Transfer') return;\n      const argSum = exerciseNode.chosenValue?.sum;\n      if (!argSum || argSum.oneofKind !== 'record') return;\n      const fields = argSum.record?.fields;\n      if (!fields) return;\n      transferNode = fields;\n    });\n\n    const getField = (fields: RecordField[], label: string) => fields.find((f) => f.label === label)?.value?.sum;\n\n    if (preApprovalNode.length) {\n      const receiverData = getField(preApprovalNode, 'receiver');\n      if (receiverData?.oneofKind === 'party') receiver = receiverData.party ?? '';\n      const providerData = getField(preApprovalNode, 'provider');\n      if (providerData?.oneofKind === 'party') sender = providerData.party ?? '';\n      amount = '0';\n    } else if (transferNode.length) {\n      const transferField = transferNode.find((f) => f.label === 'transfer');\n      const transferSum = transferField?.value?.sum;\n      if (transferSum && transferSum.oneofKind === 'record') {\n        const transferRecord = transferSum.record?.fields;\n        if (transferRecord?.length) {\n          const senderData = getField(transferRecord, 'sender');\n          if (senderData?.oneofKind === 'party') sender = senderData.party ?? '';\n\n          const receiverData = getField(transferRecord, 'receiver');\n          if (receiverData?.oneofKind === 'party') receiver = receiverData.party ?? '';\n\n          const amountData = getField(transferRecord, 'amount');\n          if (amountData?.oneofKind === 'numeric') amount = amountData.numeric ?? '';\n\n          const metaField = getField(transferRecord, 'meta');\n          if (metaField?.oneofKind === 'record') {\n            const metaFields = metaField.record?.fields;\n            if (metaFields && metaFields.length) {\n              const valuesField = getField(metaFields, 'values');\n              if (valuesField?.oneofKind === 'textMap') {\n                const entries = valuesField.textMap?.entries ?? [];\n                const memoEntry = entries.find((e) => e.key === 'splice.lfdecentralizedtrust.org/reason');\n                if (memoEntry) {\n                  const memoValue = memoEntry?.value?.sum;\n                  if (memoValue?.oneofKind === 'text') {\n                    memoId = memoValue.text;\n                  }\n                }\n              }\n            }\n          }\n        }\n      }\n    } else if (transferAcceptRejectNode.length) {\n      const dsoData = getField(transferAcceptRejectNode, 'dso');\n      if (dsoData?.oneofKind === 'party') sender = dsoData.party ?? '';\n      const ownerData = getField(transferAcceptRejectNode, 'owner');\n      if (ownerData?.oneofKind === 'party') receiver = ownerData.party ?? '';\n      const amountField = getField(transferAcceptRejectNode, 'amount');\n      if (amountField && amountField.oneofKind === 'record') {\n        const amountRecord = amountField.record?.fields;\n        if (amountRecord?.length) {\n          const initialAmountData = getField(amountRecord, 'initialAmount');\n          if (initialAmountData?.oneofKind === 'numeric') amount = initialAmountData.numeric ?? '';\n        }\n      }\n    }\n    if (!sender || !receiver || !amount) {\n      const missingFields: string[] = [];\n      if (!sender) missingFields.push('sender');\n      if (!receiver) missingFields.push('receiver');\n      if (!amount) missingFields.push('amount');\n      throw new Error(`invalid transaction data: missing ${missingFields.join(', ')}`);\n    }\n    const convertedAmount = this.convertAmountToLowestUnit(new BigNumber(amount));\n    const parsedData: PreparedTxnParsedInfo = {\n      sender,\n      receiver,\n      amount: convertedAmount,\n    };\n    if (memoId) {\n      parsedData.memoId = memoId;\n    }\n    return parsedData;\n  }\n\n  /**\n   * Computes the topology hash from the API response of the 'create party' endpoint.\n   *\n   * @param topologyTransactions - List of base64-encoded topology transactions from the Canton API.\n   * @returns The final base64-encoded topology transaction hash.\n   */\n  computeHashFromCreatePartyResponse(topologyTransactions: string[]): string {\n    const txBuffers = topologyTransactions.map((tx) => Buffer.from(tx, 'base64'));\n    return this.computeHashFromTopologyTransaction(txBuffers);\n  }\n\n  async computeHashFromPrepareSubmissionResponse(preparedTransactionBase64: string): Promise<string> {\n    const preparedTransaction = this.decodePreparedTransaction(preparedTransactionBase64);\n    const hash = await computePreparedTransaction(preparedTransaction);\n    return Buffer.from(hash).toString('base64');\n  }\n\n  /**\n   * Computes the final topology transaction hash for a list of prepared Canton transactions.\n   *\n   * Each transaction is first hashed with purpose `11`, then all hashes are combined and\n   * hashed again with purpose `55`, following the Canton topology hash rules.\n   *\n   * The resulting hash is encoded as a base64 string.\n   *\n   * @param {Buffer[]} preparedTransactions - An array of Canton transaction buffers.\n   * @returns {string} The final topology hash, base64-encoded.\n   */\n  private computeHashFromTopologyTransaction(preparedTransactions: Buffer[]): string {\n    const rawHashes = preparedTransactions.map((tx) => this.computeSha256CantonHash(11, tx));\n    const combinedHashes = this.computeMultiHashForTopology(rawHashes);\n    const computedHash = this.computeSha256CantonHash(55, combinedHashes);\n    return Buffer.from(computedHash, 'hex').toString('base64');\n  }\n\n  /**\n   * Converts a base64-encoded Ed25519 public key string into a structured signing public key object.\n   * @param {String} publicKey The base64-encoded Ed25519 public key\n   * @returns {Object} The structured signing key object formatted for use with cryptographic operations\n   * @private\n   */\n  private signingPublicKeyFromEd25519(publicKey: string): {\n    format: number;\n    publicKey: Buffer;\n    scheme: number;\n    keySpec: number;\n    usage: [];\n  } {\n    return {\n      format: CryptoKeyFormat.RAW,\n      publicKey: Buffer.from(publicKey, 'base64'),\n      scheme: SigningAlgorithmSpec.ED25519,\n      keySpec: SigningKeySpec.EC_CURVE25519,\n      usage: [],\n    };\n  }\n\n  /**\n   * Creates a buffer with a 4-byte big-endian integer prefix followed by the provided byte buffer\n   * @param {Number} value The integer to prefix, written as 4 bytes in big-endian order\n   * @param {Buffer} bytes The buffer to append after the integer prefix\n   * @returns {Buffer} The resulting buffer with the prefixed integer\n   * @private\n   */\n  private prefixedInt(value: number, bytes: Buffer): Buffer {\n    const buffer = Buffer.alloc(4 + bytes.length);\n    buffer.writeUInt32BE(value, 0);\n    Buffer.from(bytes).copy(buffer, 4);\n    return buffer;\n  }\n\n  /**\n   * Computes an SHA-256 Canton-style hash by prefixing the input with a purpose identifier,\n   * then hashing the resulting buffer and prepending a multi-prefix\n   *\n   * @param {Number} purpose A numeric identifier to prefix the hash input with\n   * @param {Buffer} bytes The buffer to be hashed\n   * @returns {String} A hexadecimal string representation of the resulting hash with multi-prefix\n   * @private\n   */\n  private computeSha256CantonHash(purpose: number, bytes: Buffer): string {\n    const hashInput = this.prefixedInt(purpose, bytes);\n    const hash = crypto.createHash('sha256').update(hashInput).digest();\n    const multiprefix = Buffer.from([0x12, 0x20]);\n    return Buffer.concat([multiprefix, hash]).toString('hex');\n  }\n\n  /**\n   * Decodes a Base64-encoded string into a Uint8Array\n   * @param {String} b64 The Base64-encoded string\n   * @returns {Uint8Array} The decoded byte array\n   * @private\n   */\n  private fromBase64(b64: string): Uint8Array {\n    return new Uint8Array(Buffer.from(b64, 'base64'));\n  }\n\n  /**\n   * Decodes a Base64-encoded prepared transaction into a structured object\n   * @param {String} base64 The Base64-encoded transaction data\n   * @returns {IPreparedTransaction} The decoded `IPreparedTransaction` object\n   * @private\n   */\n  private decodePreparedTransaction(base64: string): IPreparedTransaction {\n    const bytes = this.fromBase64(base64);\n    return PreparedTransaction.fromBinary(bytes);\n  }\n\n  /**\n   * Computes a deterministic combined hash from an array of individual Canton-style SHA-256 hashes\n   *\n   * Each hash is decoded from hex, sorted lexicographically (by hex), and prefixed with its length\n   * The final buffer includes the number of hashes followed by each (length-prefixed) hash\n   *\n   * @param {string[]} hashes - An array of Canton-prefixed SHA-256 hashes in hexadecimal string format\n   * @returns {Buffer} A binary buffer representing the combined hash input\n   */\n  private computeMultiHashForTopology(hashes: string[]): Buffer {\n    const sortedHashes = hashes\n      .map((hex) => Buffer.from(hex, 'hex'))\n      .sort((a, b) => a.toString('hex').localeCompare(b.toString('hex')));\n\n    const numHashesBytes = this.encodeInt32(sortedHashes.length);\n    const parts: Buffer[] = [numHashesBytes];\n\n    for (const h of sortedHashes) {\n      const lengthBytes = this.encodeInt32(h.length);\n      parts.push(lengthBytes, h);\n    }\n\n    return Buffer.concat(parts);\n  }\n\n  /**\n   * Encodes a 32-bit signed integer into a 4-byte big-endian Buffer\n   *\n   * @param {number} value - The integer to encode\n   * @returns {Buffer} A 4-byte buffer representing the integer in big-endian format\n   */\n  private encodeInt32(value: number): Buffer {\n    const buf = Buffer.alloc(4);\n    buf.writeInt32BE(value, 0);\n    return buf;\n  }\n\n  /**\n   * Convert to canton raw units\n   * @param {BigNumber} value\n   * @returns {String} the converted raw canton units\n   * @private\n   */\n  private convertAmountToLowestUnit(value: BigNumber): string {\n    return value.multipliedBy(new BigNumber(10).pow(10)).toFixed(0);\n  }\n}\n\nconst utils = new Utils();\n\nexport default utils;\n"]}
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import BigNumber from 'bignumber.js';
|
|
2
|
-
import { BaseAddress, BaseKey, BaseTransactionBuilder } from '@bitgo-beta/sdk-core';
|
|
2
|
+
import { BaseAddress, BaseKey, BaseTransactionBuilder, PublicKey, TransactionType } from '@bitgo-beta/sdk-core';
|
|
3
3
|
import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';
|
|
4
4
|
import { PreparedParty, WalletInitRequest } from './iface';
|
|
5
5
|
import { WalletInitTransaction } from './walletInitialization/walletInitTransaction';
|
|
6
6
|
export declare class WalletInitBuilder extends BaseTransactionBuilder {
|
|
7
7
|
private _transaction;
|
|
8
|
+
private _signatures;
|
|
8
9
|
private _publicKey;
|
|
9
|
-
private _synchronizer;
|
|
10
10
|
private _partyHint;
|
|
11
11
|
private _localParticipantObservationOnly;
|
|
12
12
|
private _otherConfirmingParticipantUids;
|
|
@@ -14,6 +14,7 @@ export declare class WalletInitBuilder extends BaseTransactionBuilder {
|
|
|
14
14
|
private _observingParticipantUids;
|
|
15
15
|
constructor(_coinConfig: Readonly<CoinConfig>);
|
|
16
16
|
initBuilder(tx: WalletInitTransaction): void;
|
|
17
|
+
protected get transactionType(): TransactionType;
|
|
17
18
|
protected buildImplementation(): Promise<WalletInitTransaction>;
|
|
18
19
|
protected fromImplementation(rawTransaction: any, isFirstSigner?: boolean): WalletInitTransaction;
|
|
19
20
|
protected signImplementation(key: BaseKey): WalletInitTransaction;
|
|
@@ -24,6 +25,8 @@ export declare class WalletInitBuilder extends BaseTransactionBuilder {
|
|
|
24
25
|
validateRawTransaction(rawTransaction: string[]): void;
|
|
25
26
|
validateTransaction(transaction?: WalletInitTransaction): void;
|
|
26
27
|
validateValue(value: BigNumber): void;
|
|
28
|
+
/** @inheritDoc */
|
|
29
|
+
addSignature(publicKey: PublicKey, signature: Buffer): void;
|
|
27
30
|
/**
|
|
28
31
|
* Sets the public key used for signing.
|
|
29
32
|
*
|
|
@@ -32,14 +35,6 @@ export declare class WalletInitBuilder extends BaseTransactionBuilder {
|
|
|
32
35
|
* @throws Error if key is not a valid public key
|
|
33
36
|
*/
|
|
34
37
|
publicKey(key: string): this;
|
|
35
|
-
/**
|
|
36
|
-
* Sets the synchronizer ID for the wallet initialization.
|
|
37
|
-
*
|
|
38
|
-
* @param id - The synchronizer identifier (must be a non-empty string).
|
|
39
|
-
* @returns The current builder instance for chaining.
|
|
40
|
-
* @throws Error if the synchronizer ID is empty.
|
|
41
|
-
*/
|
|
42
|
-
synchronizer(id: string): this;
|
|
43
38
|
/**
|
|
44
39
|
* Sets the party hint (alias or name) used during wallet initialization.
|
|
45
40
|
*
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"walletInitBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/walletInitBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,cAAc,CAAC;AAErC,OAAO,
|
|
1
|
+
{"version":3,"file":"walletInitBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/walletInitBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,cAAc,CAAC;AAErC,OAAO,EACL,WAAW,EACX,OAAO,EACP,sBAAsB,EAGtB,SAAS,EAET,eAAe,EAChB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAG7D,OAAO,EAAc,aAAa,EAAE,iBAAiB,EAAE,MAAM,SAAS,CAAC;AAEvE,OAAO,EAAE,qBAAqB,EAAE,MAAM,8CAA8C,CAAC;AAGrF,qBAAa,iBAAkB,SAAQ,sBAAsB;IAC3D,OAAO,CAAC,YAAY,CAAwB;IAC5C,OAAO,CAAC,WAAW,CAAmB;IAEtC,OAAO,CAAC,UAAU,CAAa;IAC/B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,gCAAgC,CAAS;IACjD,OAAO,CAAC,+BAA+B,CAAgB;IACvD,OAAO,CAAC,sBAAsB,CAAK;IACnC,OAAO,CAAC,yBAAyB,CAAgB;gBAErC,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAK7C,WAAW,CAAC,EAAE,EAAE,qBAAqB,GAAG,IAAI;IAI5C,SAAS,KAAK,eAAe,IAAI,eAAe,CAE/C;IAED,SAAS,CAAC,mBAAmB,IAAI,OAAO,CAAC,qBAAqB,CAAC;IAI/D,SAAS,CAAC,kBAAkB,CAAC,cAAc,EAAE,GAAG,EAAE,aAAa,CAAC,EAAE,OAAO,GAAG,qBAAqB;IAIjG,SAAS,CAAC,kBAAkB,CAAC,GAAG,EAAE,OAAO,GAAG,qBAAqB;IAIjE,IAAI,WAAW,IAAI,qBAAqB,CAEvC;IAED,IAAI,WAAW,CAAC,WAAW,EAAE,aAAa,EAEzC;IAED,eAAe,CAAC,OAAO,EAAE,WAAW,EAAE,aAAa,CAAC,EAAE,MAAM,GAAG,IAAI;IAMnE,WAAW,CAAC,GAAG,EAAE,OAAO,GAAG,IAAI;IAY/B,sBAAsB,CAAC,cAAc,EAAE,MAAM,EAAE,GAAG,IAAI;IAUtD,mBAAmB,CAAC,WAAW,CAAC,EAAE,qBAAqB,GAAG,IAAI;IAW9D,aAAa,CAAC,KAAK,EAAE,SAAS,GAAG,IAAI;IAIrC,kBAAkB;IAClB,YAAY,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI;IAQ3D;;;;;;OAMG;IACH,SAAS,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI;IAU5B;;;;;;OAMG;IACH,SAAS,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAY7B;;;;;OAKG;IACH,+BAA+B,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI;IAKpD;;;;;;OAMG;IACH,6BAA6B,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI;IAYhD;;;;;;OAMG;IACH,qBAAqB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAQ9C;;;;;;OAMG;IACH,uBAAuB,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI;IAY1C;;;;;;;;OAQG;IACH,eAAe,IAAI,iBAAiB;IAYpC;;;;;;;;;;;;OAYG;IACH,OAAO,CAAC,QAAQ;CAejB"}
|
|
@@ -12,6 +12,7 @@ const constant_1 = require("./constant");
|
|
|
12
12
|
class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
13
13
|
constructor(_coinConfig) {
|
|
14
14
|
super(_coinConfig);
|
|
15
|
+
this._signatures = [];
|
|
15
16
|
this._localParticipantObservationOnly = false;
|
|
16
17
|
this._otherConfirmingParticipantUids = [];
|
|
17
18
|
this._confirmationThreshold = 1;
|
|
@@ -21,6 +22,9 @@ class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
|
21
22
|
initBuilder(tx) {
|
|
22
23
|
this._transaction = tx;
|
|
23
24
|
}
|
|
25
|
+
get transactionType() {
|
|
26
|
+
return sdk_core_1.TransactionType.WalletInitialization;
|
|
27
|
+
}
|
|
24
28
|
buildImplementation() {
|
|
25
29
|
throw new Error('Not implemented');
|
|
26
30
|
}
|
|
@@ -75,6 +79,14 @@ class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
|
75
79
|
validateValue(value) {
|
|
76
80
|
throw new Error('Not implemented');
|
|
77
81
|
}
|
|
82
|
+
/** @inheritDoc */
|
|
83
|
+
addSignature(publicKey, signature) {
|
|
84
|
+
if (!this.transaction) {
|
|
85
|
+
throw new sdk_core_1.InvalidTransactionError('transaction is empty!');
|
|
86
|
+
}
|
|
87
|
+
this._signatures.push({ publicKey, signature });
|
|
88
|
+
this.transaction.signatures = signature.toString('base64');
|
|
89
|
+
}
|
|
78
90
|
/**
|
|
79
91
|
* Sets the public key used for signing.
|
|
80
92
|
*
|
|
@@ -91,20 +103,6 @@ class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
|
91
103
|
};
|
|
92
104
|
return this;
|
|
93
105
|
}
|
|
94
|
-
/**
|
|
95
|
-
* Sets the synchronizer ID for the wallet initialization.
|
|
96
|
-
*
|
|
97
|
-
* @param id - The synchronizer identifier (must be a non-empty string).
|
|
98
|
-
* @returns The current builder instance for chaining.
|
|
99
|
-
* @throws Error if the synchronizer ID is empty.
|
|
100
|
-
*/
|
|
101
|
-
synchronizer(id) {
|
|
102
|
-
if (!id.trim()) {
|
|
103
|
-
throw new Error('synchronizer must be a non-empty string');
|
|
104
|
-
}
|
|
105
|
-
this._synchronizer = id.trim();
|
|
106
|
-
return this;
|
|
107
|
-
}
|
|
108
106
|
/**
|
|
109
107
|
* Sets the party hint (alias or name) used during wallet initialization.
|
|
110
108
|
*
|
|
@@ -196,7 +194,6 @@ class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
|
196
194
|
this.validate();
|
|
197
195
|
return {
|
|
198
196
|
publicKey: this._publicKey,
|
|
199
|
-
synchronizer: this._synchronizer,
|
|
200
197
|
partyHint: this._partyHint,
|
|
201
198
|
localParticipantObservationOnly: this._localParticipantObservationOnly,
|
|
202
199
|
otherConfirmingParticipantUids: this._otherConfirmingParticipantUids,
|
|
@@ -218,8 +215,6 @@ class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
|
218
215
|
* @throws {Error} If any required field is missing or invalid.
|
|
219
216
|
*/
|
|
220
217
|
validate() {
|
|
221
|
-
if (!this._synchronizer)
|
|
222
|
-
throw new Error('Missing synchronizer');
|
|
223
218
|
if (!this._partyHint || this._partyHint.length > 5)
|
|
224
219
|
throw new Error('Invalid partyHint');
|
|
225
220
|
if (!this._publicKey || !this._publicKey.keyData || !this._publicKey.format || !this._publicKey.keySpec) {
|
|
@@ -237,4 +232,4 @@ class WalletInitBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
|
237
232
|
}
|
|
238
233
|
}
|
|
239
234
|
exports.WalletInitBuilder = WalletInitBuilder;
|
|
240
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"walletInitBuilder.js","sourceRoot":"","sources":["../../../src/lib/walletInitBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import BigNumber from 'bignumber.js';\n\nimport { BaseAddress, BaseKey, BaseTransactionBuilder, BuildTransactionError } from '@bitgo-beta/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\n\nimport { KeyPair } from './keyPair';\nimport { IPublicKey, PreparedParty, WalletInitRequest } from './iface';\nimport utils from './utils';\nimport { WalletInitTransaction } from './walletInitialization/walletInitTransaction';\nimport { PUBLIC_KEY_FORMAT, PUBLIC_KEY_SPEC } from './constant';\n\nexport class WalletInitBuilder extends BaseTransactionBuilder {\n  private _transaction: WalletInitTransaction;\n\n  private _publicKey: IPublicKey;\n  private _synchronizer: string;\n  private _partyHint: string;\n  private _localParticipantObservationOnly = false;\n  private _otherConfirmingParticipantUids: string[] = [];\n  private _confirmationThreshold = 1;\n  private _observingParticipantUids: string[] = [];\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new WalletInitTransaction(_coinConfig);\n  }\n\n  initBuilder(tx: WalletInitTransaction): void {\n    this._transaction = tx;\n  }\n\n  protected buildImplementation(): Promise<WalletInitTransaction> {\n    throw new Error('Not implemented');\n  }\n\n  protected fromImplementation(rawTransaction: any, isFirstSigner?: boolean): WalletInitTransaction {\n    throw new Error('Not implemented');\n  }\n\n  protected signImplementation(key: BaseKey): WalletInitTransaction {\n    throw new Error('Not implemented');\n  }\n\n  get transaction(): WalletInitTransaction {\n    return this._transaction;\n  }\n\n  set transaction(transaction: PreparedParty) {\n    this._transaction.preparedParty = transaction;\n  }\n\n  validateAddress(address: BaseAddress, addressFormat?: string): void {\n    if (!utils.isValidAddress(address.address)) {\n      throw new BuildTransactionError('Invalid address ' + address.address);\n    }\n  }\n\n  validateKey(key: BaseKey): void {\n    let keyPair: KeyPair;\n    try {\n      keyPair = new KeyPair({ prv: key.key });\n    } catch {\n      throw new BuildTransactionError('Invalid key');\n    }\n    if (!keyPair.getKeys().prv) {\n      throw new BuildTransactionError('Invalid key');\n    }\n  }\n\n  validateRawTransaction(rawTransaction: string[]): void {\n    if (!rawTransaction || !this._transaction.preparedParty) {\n      throw new BuildTransactionError('invalid raw transaction');\n    }\n    const localHash = utils.computeHashFromCreatePartyResponse(rawTransaction);\n    if (localHash !== this._transaction.preparedParty.multiHash) {\n      throw new BuildTransactionError('invalid raw transaction, hash not matching');\n    }\n  }\n\n  validateTransaction(transaction?: WalletInitTransaction): void {\n    if (!transaction?.preparedParty) {\n      return;\n    }\n    const localHash = utils.computeHashFromCreatePartyResponse(transaction.preparedParty.topologyTransactions);\n    if (localHash !== this._transaction.preparedParty.multiHash) {\n      throw new BuildTransactionError('invalid transaction');\n    }\n  }\n\n  // not implemented because wallet initialization doesn't require any value\n  validateValue(value: BigNumber): void {\n    throw new Error('Not implemented');\n  }\n\n  /**\n   * Sets the public key used for signing.\n   *\n   * @param key - The public key of the wallet.\n   * @returns The current builder instance for chaining.\n   * @throws Error if key is not a valid public key\n   */\n  publicKey(key: string): this {\n    utils.isValidPublicKey(key);\n    this._publicKey = {\n      format: PUBLIC_KEY_FORMAT,\n      keyData: key,\n      keySpec: PUBLIC_KEY_SPEC,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the synchronizer ID for the wallet initialization.\n   *\n   * @param id - The synchronizer identifier (must be a non-empty string).\n   * @returns The current builder instance for chaining.\n   * @throws Error if the synchronizer ID is empty.\n   */\n  synchronizer(id: string): this {\n    if (!id.trim()) {\n      throw new Error('synchronizer must be a non-empty string');\n    }\n    this._synchronizer = id.trim();\n    return this;\n  }\n\n  /**\n   * Sets the party hint (alias or name) used during wallet initialization.\n   *\n   * @param hint - A 5-character non-empty string representing the party hint.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the hint is empty or more than 5 characters long.\n   */\n  partyHint(hint: string): this {\n    const trimmedHint = hint.trim();\n    if (!trimmedHint) {\n      throw new Error('partyHint cannot be empty');\n    }\n    if (trimmedHint.length > 5) {\n      throw new Error('partyHint must be less than 6 characters long');\n    }\n    this._partyHint = trimmedHint;\n    return this;\n  }\n\n  /**\n   * Sets whether the local participant should be added in observation-only mode.\n   *\n   * @param flag - Boolean flag indicating observation-only status.\n   * @returns The current builder instance for chaining.\n   */\n  localParticipantObservationOnly(flag: boolean): this {\n    this._localParticipantObservationOnly = flag;\n    return this;\n  }\n\n  /**\n   * Adds a confirming participant UID to the list of other confirming participants.\n   *\n   * @param uid - A non-empty string UID of another confirming participant.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the UID is empty.\n   */\n  otherConfirmingParticipantUid(uid: string): this {\n    const trimmedUid = uid.trim();\n    if (!trimmedUid) {\n      throw new Error('otherConfirmingParticipantUid cannot be empty');\n    }\n    if (!this._otherConfirmingParticipantUids) {\n      this._otherConfirmingParticipantUids = [];\n    }\n    this._otherConfirmingParticipantUids.push(trimmedUid);\n    return this;\n  }\n\n  /**\n   * Sets the confirmation threshold for the wallet initialization.\n   *\n   * @param threshold - A positive integer indicating how many confirmations are required.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the threshold is not a positive integer.\n   */\n  confirmationThreshold(threshold: number): this {\n    if (!Number.isInteger(threshold) || threshold <= 0) {\n      throw new Error('confirmationThreshold must be a positive integer');\n    }\n    this._confirmationThreshold = threshold;\n    return this;\n  }\n\n  /**\n   * Adds an observing participant UID to the list of observers.\n   *\n   * @param uid - A non-empty string UID of the observing participant.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the UID is empty.\n   */\n  observingParticipantUid(uid: string): this {\n    const trimmedUid = uid.trim();\n    if (!trimmedUid) {\n      throw new Error('observingParticipantUid cannot be empty');\n    }\n    if (!this._observingParticipantUids) {\n      this._observingParticipantUids = [];\n    }\n    this._observingParticipantUids.push(trimmedUid);\n    return this;\n  }\n\n  /**\n   * Builds and returns the WalletInitRequest object from the builder's internal state.\n   *\n   * This method performs validation before constructing the object. If required fields are\n   * missing or invalid, it throws an error.\n   *\n   * @returns {WalletInitRequest} - A fully constructed and validated request object for wallet initialization.\n   * @throws {Error} If any required field is missing or fails validation.\n   */\n  toRequestObject(): WalletInitRequest {\n    this.validate();\n    return {\n      publicKey: this._publicKey,\n      synchronizer: this._synchronizer,\n      partyHint: this._partyHint,\n      localParticipantObservationOnly: this._localParticipantObservationOnly,\n      otherConfirmingParticipantUids: this._otherConfirmingParticipantUids,\n      confirmationThreshold: this._confirmationThreshold,\n      observingParticipantUids: this._observingParticipantUids,\n    };\n  }\n\n  /**\n   * Validates the internal state of the builder before building the request object.\n   *\n   * Checks for:\n   * - Presence of a synchronizer ID.\n   * - `partyHint` must not be empty and must be at most 5 characters long.\n   * - Public key must include `keyData`, `format`, and `keySpec`.\n   * - Confirmation threshold must be a positive integer.\n   * - Confirming and observing participant UID lists must be arrays.\n   *\n   * @private\n   * @throws {Error} If any required field is missing or invalid.\n   */\n  private validate(): void {\n    if (!this._synchronizer) throw new Error('Missing synchronizer');\n    if (!this._partyHint || this._partyHint.length > 5) throw new Error('Invalid partyHint');\n    if (!this._publicKey || !this._publicKey.keyData || !this._publicKey.format || !this._publicKey.keySpec) {\n      throw new Error('Invalid publicKey');\n    }\n    if (!Number.isInteger(this._confirmationThreshold) || this._confirmationThreshold <= 0) {\n      throw new Error('Invalid confirmationThreshold');\n    }\n    if (!Array.isArray(this._otherConfirmingParticipantUids)) {\n      throw new Error('otherConfirmingParticipantUids must be an array');\n    }\n    if (!Array.isArray(this._observingParticipantUids)) {\n      throw new Error('observingParticipantUids must be an array');\n    }\n  }\n}\n"]}
|
|
235
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"walletInitBuilder.js","sourceRoot":"","sources":["../../../src/lib/walletInitBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import BigNumber from 'bignumber.js';\n\nimport {\n  BaseAddress,\n  BaseKey,\n  BaseTransactionBuilder,\n  BuildTransactionError,\n  InvalidTransactionError,\n  PublicKey,\n  Signature,\n  TransactionType,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\n\nimport { KeyPair } from './keyPair';\nimport { IPublicKey, PreparedParty, WalletInitRequest } from './iface';\nimport utils from './utils';\nimport { WalletInitTransaction } from './walletInitialization/walletInitTransaction';\nimport { PUBLIC_KEY_FORMAT, PUBLIC_KEY_SPEC } from './constant';\n\nexport class WalletInitBuilder extends BaseTransactionBuilder {\n  private _transaction: WalletInitTransaction;\n  private _signatures: Signature[] = [];\n\n  private _publicKey: IPublicKey;\n  private _partyHint: string;\n  private _localParticipantObservationOnly = false;\n  private _otherConfirmingParticipantUids: string[] = [];\n  private _confirmationThreshold = 1;\n  private _observingParticipantUids: string[] = [];\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new WalletInitTransaction(_coinConfig);\n  }\n\n  initBuilder(tx: WalletInitTransaction): void {\n    this._transaction = tx;\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.WalletInitialization;\n  }\n\n  protected buildImplementation(): Promise<WalletInitTransaction> {\n    throw new Error('Not implemented');\n  }\n\n  protected fromImplementation(rawTransaction: any, isFirstSigner?: boolean): WalletInitTransaction {\n    throw new Error('Not implemented');\n  }\n\n  protected signImplementation(key: BaseKey): WalletInitTransaction {\n    throw new Error('Not implemented');\n  }\n\n  get transaction(): WalletInitTransaction {\n    return this._transaction;\n  }\n\n  set transaction(transaction: PreparedParty) {\n    this._transaction.preparedParty = transaction;\n  }\n\n  validateAddress(address: BaseAddress, addressFormat?: string): void {\n    if (!utils.isValidAddress(address.address)) {\n      throw new BuildTransactionError('Invalid address ' + address.address);\n    }\n  }\n\n  validateKey(key: BaseKey): void {\n    let keyPair: KeyPair;\n    try {\n      keyPair = new KeyPair({ prv: key.key });\n    } catch {\n      throw new BuildTransactionError('Invalid key');\n    }\n    if (!keyPair.getKeys().prv) {\n      throw new BuildTransactionError('Invalid key');\n    }\n  }\n\n  validateRawTransaction(rawTransaction: string[]): void {\n    if (!rawTransaction || !this._transaction.preparedParty) {\n      throw new BuildTransactionError('invalid raw transaction');\n    }\n    const localHash = utils.computeHashFromCreatePartyResponse(rawTransaction);\n    if (localHash !== this._transaction.preparedParty.multiHash) {\n      throw new BuildTransactionError('invalid raw transaction, hash not matching');\n    }\n  }\n\n  validateTransaction(transaction?: WalletInitTransaction): void {\n    if (!transaction?.preparedParty) {\n      return;\n    }\n    const localHash = utils.computeHashFromCreatePartyResponse(transaction.preparedParty.topologyTransactions);\n    if (localHash !== this._transaction.preparedParty.multiHash) {\n      throw new BuildTransactionError('invalid transaction');\n    }\n  }\n\n  // not implemented because wallet initialization doesn't require any value\n  validateValue(value: BigNumber): void {\n    throw new Error('Not implemented');\n  }\n\n  /** @inheritDoc */\n  addSignature(publicKey: PublicKey, signature: Buffer): void {\n    if (!this.transaction) {\n      throw new InvalidTransactionError('transaction is empty!');\n    }\n    this._signatures.push({ publicKey, signature });\n    this.transaction.signatures = signature.toString('base64');\n  }\n\n  /**\n   * Sets the public key used for signing.\n   *\n   * @param key - The public key of the wallet.\n   * @returns The current builder instance for chaining.\n   * @throws Error if key is not a valid public key\n   */\n  publicKey(key: string): this {\n    utils.isValidPublicKey(key);\n    this._publicKey = {\n      format: PUBLIC_KEY_FORMAT,\n      keyData: key,\n      keySpec: PUBLIC_KEY_SPEC,\n    };\n    return this;\n  }\n\n  /**\n   * Sets the party hint (alias or name) used during wallet initialization.\n   *\n   * @param hint - A 5-character non-empty string representing the party hint.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the hint is empty or more than 5 characters long.\n   */\n  partyHint(hint: string): this {\n    const trimmedHint = hint.trim();\n    if (!trimmedHint) {\n      throw new Error('partyHint cannot be empty');\n    }\n    if (trimmedHint.length > 5) {\n      throw new Error('partyHint must be less than 6 characters long');\n    }\n    this._partyHint = trimmedHint;\n    return this;\n  }\n\n  /**\n   * Sets whether the local participant should be added in observation-only mode.\n   *\n   * @param flag - Boolean flag indicating observation-only status.\n   * @returns The current builder instance for chaining.\n   */\n  localParticipantObservationOnly(flag: boolean): this {\n    this._localParticipantObservationOnly = flag;\n    return this;\n  }\n\n  /**\n   * Adds a confirming participant UID to the list of other confirming participants.\n   *\n   * @param uid - A non-empty string UID of another confirming participant.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the UID is empty.\n   */\n  otherConfirmingParticipantUid(uid: string): this {\n    const trimmedUid = uid.trim();\n    if (!trimmedUid) {\n      throw new Error('otherConfirmingParticipantUid cannot be empty');\n    }\n    if (!this._otherConfirmingParticipantUids) {\n      this._otherConfirmingParticipantUids = [];\n    }\n    this._otherConfirmingParticipantUids.push(trimmedUid);\n    return this;\n  }\n\n  /**\n   * Sets the confirmation threshold for the wallet initialization.\n   *\n   * @param threshold - A positive integer indicating how many confirmations are required.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the threshold is not a positive integer.\n   */\n  confirmationThreshold(threshold: number): this {\n    if (!Number.isInteger(threshold) || threshold <= 0) {\n      throw new Error('confirmationThreshold must be a positive integer');\n    }\n    this._confirmationThreshold = threshold;\n    return this;\n  }\n\n  /**\n   * Adds an observing participant UID to the list of observers.\n   *\n   * @param uid - A non-empty string UID of the observing participant.\n   * @returns The current builder instance for chaining.\n   * @throws Error if the UID is empty.\n   */\n  observingParticipantUid(uid: string): this {\n    const trimmedUid = uid.trim();\n    if (!trimmedUid) {\n      throw new Error('observingParticipantUid cannot be empty');\n    }\n    if (!this._observingParticipantUids) {\n      this._observingParticipantUids = [];\n    }\n    this._observingParticipantUids.push(trimmedUid);\n    return this;\n  }\n\n  /**\n   * Builds and returns the WalletInitRequest object from the builder's internal state.\n   *\n   * This method performs validation before constructing the object. If required fields are\n   * missing or invalid, it throws an error.\n   *\n   * @returns {WalletInitRequest} - A fully constructed and validated request object for wallet initialization.\n   * @throws {Error} If any required field is missing or fails validation.\n   */\n  toRequestObject(): WalletInitRequest {\n    this.validate();\n    return {\n      publicKey: this._publicKey,\n      partyHint: this._partyHint,\n      localParticipantObservationOnly: this._localParticipantObservationOnly,\n      otherConfirmingParticipantUids: this._otherConfirmingParticipantUids,\n      confirmationThreshold: this._confirmationThreshold,\n      observingParticipantUids: this._observingParticipantUids,\n    };\n  }\n\n  /**\n   * Validates the internal state of the builder before building the request object.\n   *\n   * Checks for:\n   * - Presence of a synchronizer ID.\n   * - `partyHint` must not be empty and must be at most 5 characters long.\n   * - Public key must include `keyData`, `format`, and `keySpec`.\n   * - Confirmation threshold must be a positive integer.\n   * - Confirming and observing participant UID lists must be arrays.\n   *\n   * @private\n   * @throws {Error} If any required field is missing or invalid.\n   */\n  private validate(): void {\n    if (!this._partyHint || this._partyHint.length > 5) throw new Error('Invalid partyHint');\n    if (!this._publicKey || !this._publicKey.keyData || !this._publicKey.format || !this._publicKey.keySpec) {\n      throw new Error('Invalid publicKey');\n    }\n    if (!Number.isInteger(this._confirmationThreshold) || this._confirmationThreshold <= 0) {\n      throw new Error('Invalid confirmationThreshold');\n    }\n    if (!Array.isArray(this._otherConfirmingParticipantUids)) {\n      throw new Error('otherConfirmingParticipantUids must be an array');\n    }\n    if (!Array.isArray(this._observingParticipantUids)) {\n      throw new Error('observingParticipantUids must be an array');\n    }\n  }\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { BaseKey, BaseTransaction } from '@bitgo-beta/sdk-core';
|
|
2
2
|
import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';
|
|
3
|
-
import { PreparedParty, WalletInitTxData } from '../iface';
|
|
3
|
+
import { PreparedParty, TransactionExplanation, WalletInitTxData } from '../iface';
|
|
4
4
|
export declare class WalletInitTransaction extends BaseTransaction {
|
|
5
5
|
private _preparedParty;
|
|
6
6
|
constructor(coinConfig: Readonly<CoinConfig>);
|
|
@@ -12,5 +12,6 @@ export declare class WalletInitTransaction extends BaseTransaction {
|
|
|
12
12
|
toJson(): WalletInitTxData;
|
|
13
13
|
get signablePayload(): Buffer;
|
|
14
14
|
fromRawTransaction(rawTx: string): void;
|
|
15
|
+
explainTransaction(): TransactionExplanation;
|
|
15
16
|
}
|
|
16
17
|
//# sourceMappingURL=walletInitTransaction.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"walletInitTransaction.d.ts","sourceRoot":"","sources":["../../../../src/lib/walletInitialization/walletInitTransaction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,eAAe,EAA4C,MAAM,sBAAsB,CAAC;AAC1G,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,
|
|
1
|
+
{"version":3,"file":"walletInitTransaction.d.ts","sourceRoot":"","sources":["../../../../src/lib/walletInitialization/walletInitTransaction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,eAAe,EAA4C,MAAM,sBAAsB,CAAC;AAC1G,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAGL,aAAa,EACb,sBAAsB,EAEtB,gBAAgB,EACjB,MAAM,UAAU,CAAC;AAGlB,qBAAa,qBAAsB,SAAQ,eAAe;IACxD,OAAO,CAAC,cAAc,CAAgB;gBAE1B,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC;IAI5C,IAAI,aAAa,IAAI,aAAa,CAEjC;IAED,IAAI,aAAa,CAAC,WAAW,EAAE,aAAa,EAI3C;IAED,OAAO,CAAC,GAAG,EAAE,OAAO,GAAG,OAAO;IAI9B,IAAI,UAAU,CAAC,SAAS,EAAE,MAAM,EAE/B;IAED,iBAAiB,IAAI,MAAM;IA6B3B,MAAM,IAAI,gBAAgB;IAY1B,IAAI,eAAe,IAAI,MAAM,CAK5B;IAED,kBAAkB,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAgBvC,kBAAkB,IAAI,sBAAsB;CAa7C"}
|