@bitgo-beta/abstract-utxo 1.6.1-alpha.420 → 1.6.1-alpha.422
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/cjs/src/abstractUtxoCoin.d.ts +6 -52
- package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/cjs/src/abstractUtxoCoin.js +14 -23
- package/dist/cjs/src/address/fixedScript.d.ts +2 -4
- package/dist/cjs/src/address/fixedScript.d.ts.map +1 -1
- package/dist/cjs/src/address/fixedScript.js +11 -6
- package/dist/cjs/src/impl/bch/bch.d.ts.map +1 -1
- package/dist/cjs/src/impl/bch/bch.js +5 -1
- package/dist/cjs/src/impl/bcha/bcha.d.ts +0 -1
- package/dist/cjs/src/impl/bcha/bcha.d.ts.map +1 -1
- package/dist/cjs/src/impl/bcha/bcha.js +1 -11
- package/dist/cjs/src/impl/doge/doge.d.ts +2 -1
- package/dist/cjs/src/impl/doge/doge.d.ts.map +1 -1
- package/dist/cjs/src/impl/doge/doge.js +1 -1
- package/dist/cjs/src/keychains.d.ts +6 -2
- package/dist/cjs/src/keychains.d.ts.map +1 -1
- package/dist/cjs/src/keychains.js +33 -1
- package/dist/cjs/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/backupKeyRecovery.js +20 -13
- package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts +2 -2
- package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts.map +1 -1
- package/dist/cjs/src/transaction/descriptor/explainPsbt.js +1 -1
- package/dist/cjs/src/transaction/descriptor/parse.d.ts +2 -1
- package/dist/cjs/src/transaction/descriptor/parse.d.ts.map +1 -1
- package/dist/cjs/src/transaction/descriptor/parse.js +1 -1
- package/dist/cjs/src/transaction/descriptor/parseToAmountType.d.ts +2 -1
- package/dist/cjs/src/transaction/descriptor/parseToAmountType.d.ts.map +1 -1
- package/dist/cjs/src/transaction/descriptor/parseToAmountType.js +1 -1
- package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts +2 -1
- package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/descriptor/verifyTransaction.js +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts +1 -0
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.js +42 -18
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts +15 -5
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +61 -38
- package/dist/cjs/src/transaction/fixedScript/parseOutput.d.ts +3 -1
- package/dist/cjs/src/transaction/fixedScript/parseOutput.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/parseOutput.js +14 -7
- package/dist/cjs/src/transaction/fixedScript/parseTransaction.d.ts +2 -1
- package/dist/cjs/src/transaction/fixedScript/parseTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/parseTransaction.js +56 -44
- package/dist/cjs/src/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/verifyTransaction.js +1 -1
- package/dist/cjs/src/transaction/index.d.ts +1 -0
- package/dist/cjs/src/transaction/index.d.ts.map +1 -1
- package/dist/cjs/src/transaction/index.js +2 -1
- package/dist/cjs/src/transaction/parseTransaction.d.ts +2 -1
- package/dist/cjs/src/transaction/parseTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/parseTransaction.js +1 -1
- package/dist/cjs/src/transaction/recipient.d.ts.map +1 -1
- package/dist/cjs/src/transaction/recipient.js +2 -2
- package/dist/cjs/src/transaction/types.d.ts +44 -0
- package/dist/cjs/src/transaction/types.d.ts.map +1 -0
- package/dist/cjs/src/transaction/types.js +3 -0
- package/dist/cjs/src/verifyKey.d.ts +2 -1
- package/dist/cjs/src/verifyKey.d.ts.map +1 -1
- package/dist/cjs/src/verifyKey.js +1 -1
- package/dist/cjs/test/unit/parseTransaction.js +1 -1
- package/dist/cjs/test/unit/recovery/backupKeyRecovery.js +2 -1
- package/dist/cjs/test/unit/transaction/descriptor/parse.js +1 -1
- package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +40 -16
- package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.d.ts +2 -0
- package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.d.ts.map +1 -0
- package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +288 -0
- package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts +3 -0
- package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -0
- package/dist/cjs/test/unit/transaction/fixedScript/util.js +46 -0
- package/dist/cjs/test/unit/txFormat.d.ts.map +1 -1
- package/dist/cjs/test/unit/txFormat.js +14 -2
- package/dist/cjs/tsconfig.tsbuildinfo +1 -1
- package/dist/esm/abstractUtxoCoin.d.ts +6 -52
- package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/esm/abstractUtxoCoin.js +12 -21
- package/dist/esm/address/fixedScript.d.ts +2 -4
- package/dist/esm/address/fixedScript.d.ts.map +1 -1
- package/dist/esm/address/fixedScript.js +12 -7
- package/dist/esm/impl/bch/bch.d.ts.map +1 -1
- package/dist/esm/impl/bch/bch.js +5 -1
- package/dist/esm/impl/bcha/bcha.d.ts +0 -1
- package/dist/esm/impl/bcha/bcha.d.ts.map +1 -1
- package/dist/esm/impl/bcha/bcha.js +1 -11
- package/dist/esm/impl/doge/doge.d.ts +2 -1
- package/dist/esm/impl/doge/doge.d.ts.map +1 -1
- package/dist/esm/impl/doge/doge.js +1 -1
- package/dist/esm/keychains.d.ts +6 -2
- package/dist/esm/keychains.d.ts.map +1 -1
- package/dist/esm/keychains.js +32 -1
- package/dist/esm/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/backupKeyRecovery.js +20 -13
- package/dist/esm/transaction/descriptor/explainPsbt.d.ts +2 -2
- package/dist/esm/transaction/descriptor/explainPsbt.d.ts.map +1 -1
- package/dist/esm/transaction/descriptor/explainPsbt.js +1 -1
- package/dist/esm/transaction/descriptor/parse.d.ts +2 -1
- package/dist/esm/transaction/descriptor/parse.d.ts.map +1 -1
- package/dist/esm/transaction/descriptor/parse.js +1 -1
- package/dist/esm/transaction/descriptor/parseToAmountType.d.ts +2 -1
- package/dist/esm/transaction/descriptor/parseToAmountType.d.ts.map +1 -1
- package/dist/esm/transaction/descriptor/parseToAmountType.js +1 -1
- package/dist/esm/transaction/descriptor/verifyTransaction.d.ts +2 -1
- package/dist/esm/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/descriptor/verifyTransaction.js +1 -1
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts +1 -0
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.js +42 -18
- package/dist/esm/transaction/fixedScript/explainTransaction.d.ts +15 -5
- package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/explainTransaction.js +61 -38
- package/dist/esm/transaction/fixedScript/parseOutput.d.ts +3 -1
- package/dist/esm/transaction/fixedScript/parseOutput.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/parseOutput.js +13 -7
- package/dist/esm/transaction/fixedScript/parseTransaction.d.ts +2 -1
- package/dist/esm/transaction/fixedScript/parseTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/parseTransaction.js +56 -44
- package/dist/esm/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/verifyTransaction.js +1 -1
- package/dist/esm/transaction/index.d.ts +1 -0
- package/dist/esm/transaction/index.d.ts.map +1 -1
- package/dist/esm/transaction/index.js +2 -1
- package/dist/esm/transaction/parseTransaction.d.ts +2 -1
- package/dist/esm/transaction/parseTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/parseTransaction.js +1 -1
- package/dist/esm/transaction/recipient.d.ts.map +1 -1
- package/dist/esm/transaction/recipient.js +2 -2
- package/dist/esm/transaction/types.d.ts +44 -0
- package/dist/esm/transaction/types.d.ts.map +1 -0
- package/dist/esm/transaction/types.js +2 -0
- package/dist/esm/verifyKey.d.ts +2 -1
- package/dist/esm/verifyKey.d.ts.map +1 -1
- package/dist/esm/verifyKey.js +1 -1
- package/package.json +11 -11
|
@@ -41,44 +41,52 @@ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
|
41
41
|
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
42
42
|
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
43
43
|
const fixedScript_1 = require("../../../../src/transaction/fixedScript");
|
|
44
|
-
|
|
45
|
-
return ![
|
|
46
|
-
utxolib.networks.bitcoincash,
|
|
47
|
-
utxolib.networks.bitcoingold,
|
|
48
|
-
utxolib.networks.ecash,
|
|
49
|
-
utxolib.networks.zcash,
|
|
50
|
-
].includes(utxolib.getMainnet(network));
|
|
51
|
-
}
|
|
44
|
+
const util_1 = require("./util");
|
|
52
45
|
function describeTransactionWith(acidTest) {
|
|
53
46
|
describe(`${acidTest.name}`, function () {
|
|
47
|
+
let psbt;
|
|
54
48
|
let psbtBytes;
|
|
49
|
+
let walletXpubs;
|
|
50
|
+
let customChangeWalletXpubs;
|
|
51
|
+
let wasmPsbt;
|
|
55
52
|
let refExplanation;
|
|
56
53
|
before('prepare', function () {
|
|
57
|
-
|
|
54
|
+
psbt = acidTest.createPsbt();
|
|
58
55
|
refExplanation = (0, fixedScript_1.explainPsbt)(psbt, { pubs: acidTest.rootWalletKeys }, acidTest.network, {
|
|
59
56
|
strict: true,
|
|
60
57
|
});
|
|
61
58
|
psbtBytes = psbt.toBuffer();
|
|
59
|
+
const networkName = utxolib.getNetworkName(acidTest.network);
|
|
60
|
+
(0, strict_1.default)(networkName);
|
|
61
|
+
walletXpubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
|
|
62
|
+
customChangeWalletXpubs = acidTest.otherWalletKeys.triple.map((k) => k.neutered().toBase58());
|
|
63
|
+
if ((0, util_1.hasWasmUtxoSupport)(acidTest.network)) {
|
|
64
|
+
wasmPsbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.fromBytes(psbtBytes, networkName);
|
|
65
|
+
}
|
|
62
66
|
});
|
|
63
67
|
it('should match the expected values for explainPsbt', function () {
|
|
64
68
|
// note: `outputs` means external outputs here
|
|
65
69
|
strict_1.default.strictEqual(refExplanation.outputs.length, 3);
|
|
66
70
|
strict_1.default.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);
|
|
67
|
-
strict_1.default.strictEqual(refExplanation.outputAmount, '
|
|
71
|
+
strict_1.default.strictEqual(refExplanation.outputAmount, '1800');
|
|
68
72
|
strict_1.default.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);
|
|
69
73
|
refExplanation.changeOutputs.forEach((change) => {
|
|
70
74
|
strict_1.default.strictEqual(change.amount, '900');
|
|
71
75
|
strict_1.default.strictEqual(typeof change.address, 'string');
|
|
72
76
|
});
|
|
73
77
|
});
|
|
78
|
+
it('reference implementation should support custom change outputs', function () {
|
|
79
|
+
const customChangeExplanation = (0, fixedScript_1.explainPsbt)(psbt, { pubs: acidTest.rootWalletKeys, customChangePubs: acidTest.otherWalletKeys }, acidTest.network, { strict: true });
|
|
80
|
+
strict_1.default.ok(customChangeExplanation.customChangeOutputs);
|
|
81
|
+
strict_1.default.strictEqual(customChangeExplanation.changeOutputs.length, refExplanation.changeOutputs.length);
|
|
82
|
+
strict_1.default.strictEqual(customChangeExplanation.outputs.length, refExplanation.outputs.length - 1);
|
|
83
|
+
strict_1.default.strictEqual(customChangeExplanation.customChangeOutputs.length, 1);
|
|
84
|
+
strict_1.default.strictEqual(customChangeExplanation.customChangeOutputs[0].amount, '900');
|
|
85
|
+
});
|
|
74
86
|
it('should match explainPsbtWasm', function () {
|
|
75
|
-
if (!hasWasmUtxoSupport(acidTest.network)) {
|
|
87
|
+
if (!(0, util_1.hasWasmUtxoSupport)(acidTest.network)) {
|
|
76
88
|
return this.skip();
|
|
77
89
|
}
|
|
78
|
-
const networkName = utxolib.getNetworkName(acidTest.network);
|
|
79
|
-
(0, strict_1.default)(networkName);
|
|
80
|
-
const wasmPsbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.fromBytes(psbtBytes, networkName);
|
|
81
|
-
const walletXpubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
|
|
82
90
|
const wasmExplanation = (0, fixedScript_1.explainPsbtWasm)(wasmPsbt, walletXpubs, {
|
|
83
91
|
replayProtection: {
|
|
84
92
|
outputScripts: [acidTest.getReplayProtectionOutputScript()],
|
|
@@ -100,9 +108,25 @@ function describeTransactionWith(acidTest) {
|
|
|
100
108
|
}
|
|
101
109
|
}
|
|
102
110
|
});
|
|
111
|
+
if (acidTest.network !== utxolib.networks.bitcoin) {
|
|
112
|
+
return;
|
|
113
|
+
}
|
|
114
|
+
// extended test suite for bitcoin
|
|
115
|
+
it('returns custom change outputs when parameter is set', function () {
|
|
116
|
+
const wasmExplanation = (0, fixedScript_1.explainPsbtWasm)(wasmPsbt, walletXpubs, {
|
|
117
|
+
replayProtection: {
|
|
118
|
+
outputScripts: [acidTest.getReplayProtectionOutputScript()],
|
|
119
|
+
},
|
|
120
|
+
customChangeWalletXpubs,
|
|
121
|
+
});
|
|
122
|
+
strict_1.default.ok(wasmExplanation.customChangeOutputs);
|
|
123
|
+
strict_1.default.deepStrictEqual(wasmExplanation.outputs.length, 2);
|
|
124
|
+
strict_1.default.deepStrictEqual(wasmExplanation.customChangeOutputs.length, 1);
|
|
125
|
+
strict_1.default.deepStrictEqual(wasmExplanation.customChangeOutputs[0].amount, '900');
|
|
126
|
+
});
|
|
103
127
|
});
|
|
104
128
|
}
|
|
105
129
|
describe('explainPsbt(Wasm)', function () {
|
|
106
130
|
utxo_lib_1.testutil.AcidTest.suite().forEach((test) => describeTransactionWith(test));
|
|
107
131
|
});
|
|
108
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXhwbGFpblBzYnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi90ZXN0L3VuaXQvdHJhbnNhY3Rpb24vZml4ZWRTY3JpcHQvZXhwbGFpblBzYnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxnRUFBd0M7QUFFeEMsOERBQWdEO0FBQ2hELG1EQUFnRDtBQUNoRCxnREFBNkQ7QUFHN0QseUVBQXVGO0FBRXZGLFNBQVMsa0JBQWtCLENBQUMsT0FBd0I7SUFDbEQsT0FBTyxDQUFDO1FBQ04sT0FBTyxDQUFDLFFBQVEsQ0FBQyxXQUFXO1FBQzVCLE9BQU8sQ0FBQyxRQUFRLENBQUMsV0FBVztRQUM1QixPQUFPLENBQUMsUUFBUSxDQUFDLEtBQUs7UUFDdEIsT0FBTyxDQUFDLFFBQVEsQ0FBQyxLQUFLO0tBQ3ZCLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztBQUMxQyxDQUFDO0FBRUQsU0FBUyx1QkFBdUIsQ0FBQyxRQUEyQjtJQUMxRCxRQUFRLENBQUMsR0FBRyxRQUFRLENBQUMsSUFBSSxFQUFFLEVBQUU7UUFDM0IsSUFBSSxTQUFpQixDQUFDO1FBQ3RCLElBQUksY0FBc0MsQ0FBQztRQUMzQyxNQUFNLENBQUMsU0FBUyxFQUFFO1lBQ2hCLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUNuQyxjQUFjLEdBQUcsSUFBQSx5QkFBVyxFQUFDLElBQUksRUFBRSxFQUFFLElBQUksRUFBRSxRQUFRLENBQUMsY0FBYyxFQUFFLEVBQUUsUUFBUSxDQUFDLE9BQU8sRUFBRTtnQkFDdEYsTUFBTSxFQUFFLElBQUk7YUFDYixDQUFDLENBQUM7WUFDSCxTQUFTLEdBQUcsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQzlCLENBQUMsQ0FBQyxDQUFDO1FBRUgsRUFBRSxD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QyxDQUFDLENBQUM7QUFDN0UsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgYXNzZXJ0IGZyb20gJ25vZGU6YXNzZXJ0L3N0cmljdCc7XG5cbmltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuaW1wb3J0IHsgdGVzdHV0aWwgfSBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5pbXBvcnQgeyBmaXhlZFNjcmlwdFdhbGxldCwgVHJpcGxlIH0gZnJvbSAnQGJpdGdvL3dhc20tdXR4byc7XG5cbmltcG9ydCB0eXBlIHsgVHJhbnNhY3Rpb25FeHBsYW5hdGlvbiB9IGZyb20gJy4uLy4uLy4uLy4uL3NyYy90cmFuc2FjdGlvbi9maXhlZFNjcmlwdC9leHBsYWluVHJhbnNhY3Rpb24nO1xuaW1wb3J0IHsgZXhwbGFpblBzYnQsIGV4cGxhaW5Qc2J0V2FzbSB9IGZyb20gJy4uLy4uLy4uLy4uL3NyYy90cmFuc2FjdGlvbi9maXhlZFNjcmlwdCc7XG5cbmZ1bmN0aW9uIGhhc1dhc21VdHhvU3VwcG9ydChuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmspOiBib29sZWFuIHtcbiAgcmV0dXJuICFbXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luY2FzaCxcbiAgICB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5nb2xkLFxuICAgIHV0eG9saWIubmV0d29ya3MuZWNhc2gsXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy56Y2FzaCxcbiAgXS5pbmNsdWRlcyh1dHhvbGliLmdldE1haW5uZXQobmV0d29yaykpO1xufVxuXG5mdW5jdGlvbiBkZXNjcmliZVRyYW5zYWN0aW9uV2l0aChhY2lkVGVzdDogdGVzdHV0aWwuQWNpZFRlc3QpIHtcbiAgZGVzY3JpYmUoYCR7YWNpZFRlc3QubmFtZX1gLCBmdW5jdGlvbiAoKSB7XG4gICAgbGV0IHBzYnRCeXRlczogQnVmZmVyO1xuICAgIGxldCByZWZFeHBsYW5hdGlvbjogVHJhbnNhY3Rpb25FeHBsYW5hdGlvbjtcbiAgICBiZWZvcmUoJ3ByZXBhcmUnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBwc2J0ID0gYWNpZFRlc3QuY3JlYXRlUHNidCgpO1xuICAgICAgcmVmRXhwbGFuYXRpb24gPSBleHBsYWluUHNidChwc2J0LCB7IHB1YnM6IGFjaWRUZXN0LnJvb3RXYWxsZXRLZXlzIH0sIGFjaWRUZXN0Lm5ldHdvcmssIHtcbiAgICAgICAgc3RyaWN0OiB0cnVlLFxuICAgICAgfSk7XG4gICAgICBwc2J0Qnl0ZXMgPSBwc2J0LnRvQnVmZmVyKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIG1hdGNoIHRoZSBleHBlY3RlZCB2YWx1ZXMgZm9yIGV4cGxhaW5Qc2J0JywgZnVuY3Rpb24gKCkge1xuICAgICAgLy8gbm90ZTogYG91dHB1dHNgIG1lYW5zIGV4dGVybmFsIG91dHB1dHMgaGVyZVxuICAgICAgYXNzZXJ0LnN0cmljdEVxdWFsKHJlZkV4cGxhbmF0aW9uLm91dHB1dHMubGVuZ3RoLCAzKTtcbiAgICAgIGFzc2VydC5zdHJpY3RFcXVhbChyZWZFeHBsYW5hdGlvbi5jaGFuZ2VPdXRwdXRzLmxlbmd0aCwgYWNpZFRlc3Qub3V0cHV0cy5sZW5ndGggLSAzKTtcbiAgICAgIGFzc2VydC5zdHJpY3RFcXVhbChyZWZFeHBsYW5hdGlvbi5vdXRwdXRBbW91bnQsICcyNzAwJyk7XG4gICAgICBhc3NlcnQuc3RyaWN0RXF1YWwocmVmRXhwbGFuYXRpb24uY2hhbmdlT3V0cHV0cy5sZW5ndGgsIGFjaWRUZXN0Lm91dHB1dHMubGVuZ3RoIC0gMyk7XG4gICAgICByZWZFeHBsYW5hdGlvbi5jaGFuZ2VPdXRwdXRzLmZvckVhY2goKGNoYW5nZSkgPT4ge1xuICAgICAgICBhc3NlcnQuc3RyaWN0RXF1YWwoY2hhbmdlLmFtb3VudCwgJzkwMCcpO1xuICAgICAgICBhc3NlcnQuc3RyaWN0RXF1YWwodHlwZW9mIGNoYW5nZS5hZGRyZXNzLCAnc3RyaW5nJyk7XG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgbWF0Y2ggZXhwbGFpblBzYnRXYXNtJywgZnVuY3Rpb24gKCkge1xuICAgICAgaWYgKCFoYXNXYXNtVXR4b1N1cHBvcnQoYWNpZFRlc3QubmV0d29yaykpIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuc2tpcCgpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBuZXR3b3JrTmFtZSA9IHV0eG9saWIuZ2V0TmV0d29ya05hbWUoYWNpZFRlc3QubmV0d29yayk7XG4gICAgICBhc3NlcnQobmV0d29ya05hbWUpO1xuICAgICAgY29uc3Qgd2FzbVBzYnQgPSBmaXhlZFNjcmlwdFdhbGxldC5CaXRHb1BzYnQuZnJvbUJ5dGVzKHBzYnRCeXRlcywgbmV0d29ya05hbWUpO1xuICAgICAgY29uc3Qgd2FsbGV0WHB1YnMgPSBhY2lkVGVzdC5yb290V2FsbGV0S2V5cy50cmlwbGUubWFwKChrKSA9PiBrLm5ldXRlcmVkKCkudG9CYXNlNTgoKSkgYXMgVHJpcGxlPHN0cmluZz47XG4gICAgICBjb25zdCB3YXNtRXhwbGFuYXRpb24gPSBleHBsYWluUHNidFdhc20od2FzbVBzYnQsIHdhbGxldFhwdWJzLCB7XG4gICAgICAgIHJlcGxheVByb3RlY3Rpb246IHtcbiAgICAgICAgICBvdXRwdXRTY3JpcHRzOiBbYWNpZFRlc3QuZ2V0UmVwbGF5UHJvdGVjdGlvbk91dHB1dFNjcmlwdCgpXSxcbiAgICAgICAgfSxcbiAgICAgIH0pO1xuXG4gICAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyhyZWZFeHBsYW5hdGlvbikpIHtcbiAgICAgICAgY29uc3QgcmVmVmFsdWUgPSByZWZFeHBsYW5hdGlvbltrZXldO1xuICAgICAgICBjb25zdCB3YXNtVmFsdWUgPSB3YXNtRXhwbGFuYXRpb25ba2V5XTtcbiAgICAgICAgc3dpdGNoIChrZXkpIHtcbiAgICAgICAgICBjYXNlICdkaXNwbGF5T3JkZXInOlxuICAgICAgICAgIGNhc2UgJ2lucHV0U2lnbmF0dXJlcyc6XG4gICAgICAgICAgY2FzZSAnc2lnbmF0dXJlcyc6XG4gICAgICAgICAgICAvLyB0aGVzZSBhcmUgZGVwcmVjYXRlZCBmaWVsZHMgdGhhdCB3ZSB3YW50IHRvIGdldCByaWQgb2ZcbiAgICAgICAgICAgIGFzc2VydC5kZWVwU3RyaWN0RXF1YWwod2FzbVZhbHVlLCB1bmRlZmluZWQpO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgIGFzc2VydC5kZWVwU3RyaWN0RXF1YWwod2FzbVZhbHVlLCByZWZWYWx1ZSwgYG1pc21hdGNoIGZvciBrZXkgJHtrZXl9YCk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0pO1xuICB9KTtcbn1cblxuZGVzY3JpYmUoJ2V4cGxhaW5Qc2J0KFdhc20pJywgZnVuY3Rpb24gKCkge1xuICB0ZXN0dXRpbC5BY2lkVGVzdC5zdWl0ZSgpLmZvckVhY2goKHRlc3QpID0+IGRlc2NyaWJlVHJhbnNhY3Rpb25XaXRoKHRlc3QpKTtcbn0pO1xuIl19
|
|
132
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"explainPsbt.js","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/explainPsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'node:assert/strict';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { testutil } from '@bitgo-beta/utxo-lib';\nimport { fixedScriptWallet, Triple } from '@bitgo/wasm-utxo';\n\nimport type { TransactionExplanation } from '../../../../src/transaction/fixedScript/explainTransaction';\nimport { explainPsbt, explainPsbtWasm } from '../../../../src/transaction/fixedScript';\n\nimport { hasWasmUtxoSupport } from './util';\n\nfunction describeTransactionWith(acidTest: testutil.AcidTest) {\n  describe(`${acidTest.name}`, function () {\n    let psbt: utxolib.bitgo.UtxoPsbt;\n    let psbtBytes: Buffer;\n    let walletXpubs: Triple<string>;\n    let customChangeWalletXpubs: Triple<string> | undefined;\n    let wasmPsbt: fixedScriptWallet.BitGoPsbt;\n    let refExplanation: TransactionExplanation;\n    before('prepare', function () {\n      psbt = acidTest.createPsbt();\n      refExplanation = explainPsbt(psbt, { pubs: acidTest.rootWalletKeys }, acidTest.network, {\n        strict: true,\n      });\n      psbtBytes = psbt.toBuffer();\n      const networkName = utxolib.getNetworkName(acidTest.network);\n      assert(networkName);\n      walletXpubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58()) as Triple<string>;\n      customChangeWalletXpubs = acidTest.otherWalletKeys.triple.map((k) => k.neutered().toBase58()) as Triple<string>;\n      if (hasWasmUtxoSupport(acidTest.network)) {\n        wasmPsbt = fixedScriptWallet.BitGoPsbt.fromBytes(psbtBytes, networkName);\n      }\n    });\n\n    it('should match the expected values for explainPsbt', function () {\n      // note: `outputs` means external outputs here\n      assert.strictEqual(refExplanation.outputs.length, 3);\n      assert.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);\n      assert.strictEqual(refExplanation.outputAmount, '1800');\n      assert.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);\n      refExplanation.changeOutputs.forEach((change) => {\n        assert.strictEqual(change.amount, '900');\n        assert.strictEqual(typeof change.address, 'string');\n      });\n    });\n\n    it('reference implementation should support custom change outputs', function () {\n      const customChangeExplanation = explainPsbt(\n        psbt,\n        { pubs: acidTest.rootWalletKeys, customChangePubs: acidTest.otherWalletKeys },\n        acidTest.network,\n        { strict: true }\n      );\n      assert.ok(customChangeExplanation.customChangeOutputs);\n      assert.strictEqual(customChangeExplanation.changeOutputs.length, refExplanation.changeOutputs.length);\n      assert.strictEqual(customChangeExplanation.outputs.length, refExplanation.outputs.length - 1);\n      assert.strictEqual(customChangeExplanation.customChangeOutputs.length, 1);\n      assert.strictEqual(customChangeExplanation.customChangeOutputs[0].amount, '900');\n    });\n\n    it('should match explainPsbtWasm', function () {\n      if (!hasWasmUtxoSupport(acidTest.network)) {\n        return this.skip();\n      }\n\n      const wasmExplanation = explainPsbtWasm(wasmPsbt, walletXpubs, {\n        replayProtection: {\n          outputScripts: [acidTest.getReplayProtectionOutputScript()],\n        },\n      });\n\n      for (const key of Object.keys(refExplanation)) {\n        const refValue = refExplanation[key];\n        const wasmValue = wasmExplanation[key];\n        switch (key) {\n          case 'displayOrder':\n          case 'inputSignatures':\n          case 'signatures':\n            // these are deprecated fields that we want to get rid of\n            assert.deepStrictEqual(wasmValue, undefined);\n            break;\n          default:\n            assert.deepStrictEqual(wasmValue, refValue, `mismatch for key ${key}`);\n            break;\n        }\n      }\n    });\n\n    if (acidTest.network !== utxolib.networks.bitcoin) {\n      return;\n    }\n\n    // extended test suite for bitcoin\n\n    it('returns custom change outputs when parameter is set', function () {\n      const wasmExplanation = explainPsbtWasm(wasmPsbt, walletXpubs, {\n        replayProtection: {\n          outputScripts: [acidTest.getReplayProtectionOutputScript()],\n        },\n        customChangeWalletXpubs,\n      });\n      assert.ok(wasmExplanation.customChangeOutputs);\n      assert.deepStrictEqual(wasmExplanation.outputs.length, 2);\n      assert.deepStrictEqual(wasmExplanation.customChangeOutputs.length, 1);\n      assert.deepStrictEqual(wasmExplanation.customChangeOutputs[0].amount, '900');\n    });\n  });\n}\n\ndescribe('explainPsbt(Wasm)', function () {\n  testutil.AcidTest.suite().forEach((test) => describeTransactionWith(test));\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parsePsbt.d.ts","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/parsePsbt.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const strict_1 = __importDefault(require("node:assert/strict"));
|
|
40
|
+
const sinon = __importStar(require("sinon"));
|
|
41
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
42
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
43
|
+
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
44
|
+
const parseTransaction_1 = require("../../../../src/transaction/fixedScript/parseTransaction");
|
|
45
|
+
const util_1 = require("../../util");
|
|
46
|
+
const fixedScript_1 = require("../../../../src/transaction/fixedScript");
|
|
47
|
+
const names_1 = require("../../../../src/names");
|
|
48
|
+
const util_2 = require("./util");
|
|
49
|
+
function getTxParamsFromExplanation(explanation, { externalCustomChangeAddress }) {
|
|
50
|
+
// The external outputs are the ones that are in outputs but not in changeOutputs
|
|
51
|
+
const changeAddresses = new Set(explanation.changeOutputs.map((o) => o.address));
|
|
52
|
+
let externalOutputs = explanation.outputs.filter((o) => o.address && !changeAddresses.has(o.address));
|
|
53
|
+
let changeAddress;
|
|
54
|
+
if (externalCustomChangeAddress) {
|
|
55
|
+
// convert an external output to a change output
|
|
56
|
+
//
|
|
57
|
+
// in combination with allowExternalChangeAddress, this allows an external
|
|
58
|
+
// output on the transaction without a size constraint
|
|
59
|
+
const externalOutput = externalOutputs[0];
|
|
60
|
+
if (!externalOutput) {
|
|
61
|
+
throw new Error('no external output found');
|
|
62
|
+
}
|
|
63
|
+
changeAddress = externalOutput.address;
|
|
64
|
+
externalOutputs = externalOutputs.slice(1);
|
|
65
|
+
}
|
|
66
|
+
return {
|
|
67
|
+
recipients: externalOutputs.map((output) => ({
|
|
68
|
+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
69
|
+
address: output.address,
|
|
70
|
+
amount: output.amount,
|
|
71
|
+
})),
|
|
72
|
+
changeAddress,
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
function getChangeInfoFromPsbt(psbt) {
|
|
76
|
+
try {
|
|
77
|
+
return utxolib.bitgo.findInternalOutputIndices(psbt).map((i) => {
|
|
78
|
+
const output = psbt.data.outputs[i];
|
|
79
|
+
const derivations = output.bip32Derivation ?? output.tapBip32Derivation ?? undefined;
|
|
80
|
+
if (!derivations || derivations.length !== 3) {
|
|
81
|
+
throw new Error('expected 3 derivation paths');
|
|
82
|
+
}
|
|
83
|
+
const path = derivations[0].path;
|
|
84
|
+
const { chain, index } = utxolib.bitgo.getChainAndIndexFromPath(path);
|
|
85
|
+
return {
|
|
86
|
+
address: utxolib.address.fromOutputScript(psbt.txOutputs[i].script, psbt.network),
|
|
87
|
+
chain,
|
|
88
|
+
index,
|
|
89
|
+
};
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
catch (e) {
|
|
93
|
+
if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {
|
|
94
|
+
return undefined;
|
|
95
|
+
}
|
|
96
|
+
throw e;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
function describeParseTransactionWith(acidTest, label, backend, { txParams, externalCustomChangeAddress = false, expectedExplicitExternalSpendAmount, expectedImplicitExternalSpendAmount, txFormat = 'psbt', }) {
|
|
100
|
+
describe(`${acidTest.name}/${label}`, function () {
|
|
101
|
+
let refParsedTransaction;
|
|
102
|
+
let coin;
|
|
103
|
+
let mockWallet;
|
|
104
|
+
let stubExplainTransaction;
|
|
105
|
+
before('prepare', async function () {
|
|
106
|
+
const coinName = (0, names_1.getChainFromNetwork)(acidTest.network);
|
|
107
|
+
coin = (0, util_1.getUtxoCoin)(coinName);
|
|
108
|
+
// Create PSBT and explanation
|
|
109
|
+
const psbt = acidTest.createPsbt();
|
|
110
|
+
const tx = psbt.getUnsignedTx();
|
|
111
|
+
const txHash = tx.getId();
|
|
112
|
+
let explanation;
|
|
113
|
+
if (txFormat === 'psbt') {
|
|
114
|
+
if (backend === 'utxolib') {
|
|
115
|
+
explanation = (0, fixedScript_1.explainPsbt)(psbt, { pubs: acidTest.rootWalletKeys }, acidTest.network, {
|
|
116
|
+
strict: true,
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
else if (backend === 'wasm') {
|
|
120
|
+
const wasmPsbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.fromBytes(psbt.toBuffer(), utxolib.getNetworkName(acidTest.network));
|
|
121
|
+
explanation = (0, fixedScript_1.explainPsbtWasm)(wasmPsbt, acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58()), {
|
|
122
|
+
replayProtection: {
|
|
123
|
+
outputScripts: [acidTest.getReplayProtectionOutputScript()],
|
|
124
|
+
},
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
else {
|
|
128
|
+
throw new Error(`Invalid backend: ${backend}`);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
else if (txFormat === 'legacy') {
|
|
132
|
+
const pubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
|
|
133
|
+
// Extract change info from PSBT to pass to explainLegacyTx
|
|
134
|
+
const changeInfo = getChangeInfoFromPsbt(psbt);
|
|
135
|
+
explanation = (0, fixedScript_1.explainLegacyTx)(tx, { pubs, changeInfo }, acidTest.network);
|
|
136
|
+
}
|
|
137
|
+
else {
|
|
138
|
+
throw new Error(`Invalid txFormat: ${txFormat}`);
|
|
139
|
+
}
|
|
140
|
+
// Determine txParams
|
|
141
|
+
let resolvedTxParams;
|
|
142
|
+
if (txParams === 'inferFromExplanation' || txParams === undefined) {
|
|
143
|
+
resolvedTxParams = getTxParamsFromExplanation(explanation, { externalCustomChangeAddress });
|
|
144
|
+
}
|
|
145
|
+
else if ('rbfTxIds' in txParams) {
|
|
146
|
+
// Replace placeholder txHash with actual computed txHash
|
|
147
|
+
resolvedTxParams = {
|
|
148
|
+
rbfTxIds: txParams.rbfTxIds.map((hash) => (hash === 'PLACEHOLDER' ? txHash : hash)),
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
else {
|
|
152
|
+
resolvedTxParams = txParams;
|
|
153
|
+
}
|
|
154
|
+
if (externalCustomChangeAddress) {
|
|
155
|
+
resolvedTxParams.allowExternalChangeAddress = true;
|
|
156
|
+
}
|
|
157
|
+
// Create mock wallet
|
|
158
|
+
mockWallet = sinon.createStubInstance(sdk_core_1.Wallet);
|
|
159
|
+
mockWallet.id.returns('test-wallet-id');
|
|
160
|
+
mockWallet.coin.returns(coin.getChain());
|
|
161
|
+
mockWallet.coinSpecific.returns(undefined);
|
|
162
|
+
// Mock getTransaction for RBF case
|
|
163
|
+
if ('rbfTxIds' in resolvedTxParams) {
|
|
164
|
+
const rbfTxParams = getTxParamsFromExplanation(explanation, { externalCustomChangeAddress: false });
|
|
165
|
+
mockWallet.getTransaction.resolves({
|
|
166
|
+
outputs: rbfTxParams.recipients.map((r) => ({
|
|
167
|
+
valueString: typeof r.amount === 'string' ? r.amount : r.amount.toString(),
|
|
168
|
+
address: r.address,
|
|
169
|
+
// wallet field is undefined for external outputs (not self-sends)
|
|
170
|
+
})),
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
// Mock verification options with keychains to disable networking
|
|
174
|
+
// Use the same keychains that were used to create the PSBT
|
|
175
|
+
const pubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
|
|
176
|
+
const verification = {
|
|
177
|
+
disableNetworking: true,
|
|
178
|
+
keychains: {
|
|
179
|
+
user: { id: '0', pub: pubs[0], type: 'independent' },
|
|
180
|
+
backup: { id: '1', pub: pubs[1], type: 'independent' },
|
|
181
|
+
bitgo: { id: '2', pub: pubs[2], type: 'independent' },
|
|
182
|
+
},
|
|
183
|
+
};
|
|
184
|
+
// Stub explainTransaction to return the explanation without making network calls
|
|
185
|
+
stubExplainTransaction = sinon.stub(coin, 'explainTransaction').resolves(explanation);
|
|
186
|
+
let txPrebuild;
|
|
187
|
+
if (txFormat === 'psbt') {
|
|
188
|
+
txPrebuild = {
|
|
189
|
+
txHex: psbt.toHex(),
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
else if (txFormat === 'legacy') {
|
|
193
|
+
txPrebuild = {
|
|
194
|
+
txHex: psbt.getUnsignedTx().toHex(),
|
|
195
|
+
};
|
|
196
|
+
}
|
|
197
|
+
else {
|
|
198
|
+
throw new Error(`Invalid txFormat: ${txFormat}`);
|
|
199
|
+
}
|
|
200
|
+
refParsedTransaction = await (0, parseTransaction_1.parseTransaction)(coin, {
|
|
201
|
+
wallet: mockWallet,
|
|
202
|
+
txParams: resolvedTxParams,
|
|
203
|
+
txPrebuild,
|
|
204
|
+
verification,
|
|
205
|
+
});
|
|
206
|
+
});
|
|
207
|
+
after('cleanup', function () {
|
|
208
|
+
if (stubExplainTransaction) {
|
|
209
|
+
stubExplainTransaction.restore();
|
|
210
|
+
}
|
|
211
|
+
});
|
|
212
|
+
it('should parse transaction without network calls', function () {
|
|
213
|
+
strict_1.default.ok(refParsedTransaction);
|
|
214
|
+
strict_1.default.ok(refParsedTransaction.keychains);
|
|
215
|
+
strict_1.default.ok(refParsedTransaction.outputs);
|
|
216
|
+
});
|
|
217
|
+
it('should have valid keychains', function () {
|
|
218
|
+
strict_1.default.ok(refParsedTransaction.keychains.user);
|
|
219
|
+
strict_1.default.ok(refParsedTransaction.keychains.backup);
|
|
220
|
+
strict_1.default.ok(refParsedTransaction.keychains.bitgo);
|
|
221
|
+
const pubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
|
|
222
|
+
strict_1.default.strictEqual(refParsedTransaction.keychains.user.pub, pubs[0]);
|
|
223
|
+
strict_1.default.strictEqual(refParsedTransaction.keychains.backup.pub, pubs[1]);
|
|
224
|
+
strict_1.default.strictEqual(refParsedTransaction.keychains.bitgo.pub, pubs[2]);
|
|
225
|
+
});
|
|
226
|
+
it('should have outputs classified as internal or external', function () {
|
|
227
|
+
// Since we didn't specify any recipients, outputs will be classified based on whether they can be
|
|
228
|
+
// verified as wallet addresses. Some may be external if address verification fails without a proper wallet setup.
|
|
229
|
+
const totalOutputs = refParsedTransaction.outputs.length;
|
|
230
|
+
const changeOutputs = refParsedTransaction.changeOutputs.length;
|
|
231
|
+
const externalOutputs = refParsedTransaction.outputs.filter((o) => o.external === true).length;
|
|
232
|
+
strict_1.default.strictEqual(externalOutputs, 3);
|
|
233
|
+
strict_1.default.ok(totalOutputs > 0, 'should have at least one output');
|
|
234
|
+
strict_1.default.strictEqual(changeOutputs + externalOutputs, totalOutputs, 'all outputs should be classified');
|
|
235
|
+
});
|
|
236
|
+
it('should have expected explicit and implicit external spend amounts', function () {
|
|
237
|
+
strict_1.default.strictEqual(BigInt(refParsedTransaction.explicitExternalSpendAmount), expectedExplicitExternalSpendAmount);
|
|
238
|
+
strict_1.default.strictEqual(BigInt(refParsedTransaction.implicitExternalSpendAmount), expectedImplicitExternalSpendAmount);
|
|
239
|
+
});
|
|
240
|
+
});
|
|
241
|
+
}
|
|
242
|
+
function describeTransaction(backend, filter = () => true) {
|
|
243
|
+
describe(`parseTransaction (${backend})`, function () {
|
|
244
|
+
utxolib.testutil.AcidTest.suite()
|
|
245
|
+
.filter(filter)
|
|
246
|
+
.forEach((test) => {
|
|
247
|
+
// Default case: psbt format, infer recipients from explanation
|
|
248
|
+
describeParseTransactionWith(test, 'default', backend, {
|
|
249
|
+
txParams: 'inferFromExplanation',
|
|
250
|
+
expectedExplicitExternalSpendAmount: 1800n,
|
|
251
|
+
expectedImplicitExternalSpendAmount: 0n,
|
|
252
|
+
});
|
|
253
|
+
if (test.network !== utxolib.networks.bitcoin) {
|
|
254
|
+
return;
|
|
255
|
+
}
|
|
256
|
+
// extended test suite for bitcoin
|
|
257
|
+
describeParseTransactionWith(test, 'legacy', backend, {
|
|
258
|
+
txFormat: 'legacy',
|
|
259
|
+
txParams: 'inferFromExplanation',
|
|
260
|
+
expectedExplicitExternalSpendAmount: 1800n,
|
|
261
|
+
expectedImplicitExternalSpendAmount: 0n,
|
|
262
|
+
});
|
|
263
|
+
describeParseTransactionWith(test, 'empty recipients', backend, {
|
|
264
|
+
txParams: {
|
|
265
|
+
recipients: [],
|
|
266
|
+
},
|
|
267
|
+
expectedExplicitExternalSpendAmount: 0n,
|
|
268
|
+
expectedImplicitExternalSpendAmount: 1800n,
|
|
269
|
+
});
|
|
270
|
+
describeParseTransactionWith(test, 'rbf', backend, {
|
|
271
|
+
txParams: {
|
|
272
|
+
rbfTxIds: ['PLACEHOLDER'],
|
|
273
|
+
},
|
|
274
|
+
expectedExplicitExternalSpendAmount: 1800n,
|
|
275
|
+
expectedImplicitExternalSpendAmount: 0n,
|
|
276
|
+
});
|
|
277
|
+
describeParseTransactionWith(test, 'allowExternalChangeAddress', backend, {
|
|
278
|
+
txParams: 'inferFromExplanation',
|
|
279
|
+
externalCustomChangeAddress: true,
|
|
280
|
+
expectedExplicitExternalSpendAmount: 1800n,
|
|
281
|
+
expectedImplicitExternalSpendAmount: 0n,
|
|
282
|
+
});
|
|
283
|
+
});
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
describeTransaction('utxolib');
|
|
287
|
+
describeTransaction('wasm', (test) => (0, util_2.hasWasmUtxoSupport)(test.network));
|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parsePsbt.js","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/parsePsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'node:assert/strict';\n\nimport * as sinon from 'sinon';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Wallet, VerificationOptions, ITransactionRecipient, Triple } from '@bitgo-beta/sdk-core';\nimport { fixedScriptWallet } from '@bitgo/wasm-utxo';\n\nimport { parseTransaction } from '../../../../src/transaction/fixedScript/parseTransaction';\nimport { ParsedTransaction } from '../../../../src/transaction/types';\nimport { UtxoWallet } from '../../../../src/wallet';\nimport { getUtxoCoin } from '../../util';\nimport { explainLegacyTx, explainPsbt, explainPsbtWasm } from '../../../../src/transaction/fixedScript';\nimport type {\n  TransactionExplanation,\n  ChangeAddressInfo,\n} from '../../../../src/transaction/fixedScript/explainTransaction';\nimport { getChainFromNetwork } from '../../../../src/names';\nimport { TransactionPrebuild } from '../../../../src/abstractUtxoCoin';\n\nimport { hasWasmUtxoSupport } from './util';\n\nfunction getTxParamsFromExplanation(\n  explanation: TransactionExplanation,\n  { externalCustomChangeAddress }: { externalCustomChangeAddress: boolean }\n): {\n  recipients: ITransactionRecipient[];\n  changeAddress?: string;\n} {\n  // The external outputs are the ones that are in outputs but not in changeOutputs\n  const changeAddresses = new Set(explanation.changeOutputs.map((o) => o.address));\n  let externalOutputs = explanation.outputs.filter((o) => o.address && !changeAddresses.has(o.address));\n\n  let changeAddress: string | undefined;\n  if (externalCustomChangeAddress) {\n    // convert an external output to a change output\n    //\n    // in combination with allowExternalChangeAddress, this allows an external\n    // output on the transaction without a size constraint\n    const externalOutput = externalOutputs[0];\n    if (!externalOutput) {\n      throw new Error('no external output found');\n    }\n    changeAddress = externalOutput.address;\n    externalOutputs = externalOutputs.slice(1);\n  }\n\n  return {\n    recipients: externalOutputs.map((output) => ({\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      address: output.address!,\n      amount: output.amount,\n    })),\n    changeAddress,\n  };\n}\n\nfunction getChangeInfoFromPsbt(psbt: utxolib.bitgo.UtxoPsbt): ChangeAddressInfo[] | undefined {\n  try {\n    return utxolib.bitgo.findInternalOutputIndices(psbt).map((i) => {\n      const output = psbt.data.outputs[i];\n      const derivations = output.bip32Derivation ?? output.tapBip32Derivation ?? undefined;\n      if (!derivations || derivations.length !== 3) {\n        throw new Error('expected 3 derivation paths');\n      }\n      const path = derivations[0].path;\n      const { chain, index } = utxolib.bitgo.getChainAndIndexFromPath(path);\n      return {\n        address: utxolib.address.fromOutputScript(psbt.txOutputs[i].script, psbt.network),\n        chain,\n        index,\n      };\n    });\n  } catch (e) {\n    if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {\n      return undefined;\n    }\n    throw e;\n  }\n}\n\nfunction describeParseTransactionWith(\n  acidTest: utxolib.testutil.AcidTest,\n  label: string,\n  backend: 'utxolib' | 'wasm',\n  {\n    txParams,\n    externalCustomChangeAddress = false,\n    expectedExplicitExternalSpendAmount,\n    expectedImplicitExternalSpendAmount,\n    txFormat = 'psbt',\n  }: {\n    txParams:\n      | {\n          recipients: ITransactionRecipient[];\n        }\n      | {\n          rbfTxIds: string[];\n        }\n      | 'inferFromExplanation';\n    externalCustomChangeAddress?: boolean;\n    expectedExplicitExternalSpendAmount: bigint;\n    expectedImplicitExternalSpendAmount: bigint;\n    txFormat?: 'psbt' | 'legacy';\n  }\n) {\n  describe(`${acidTest.name}/${label}`, function () {\n    let refParsedTransaction: ParsedTransaction<bigint>;\n    let coin: ReturnType<typeof getUtxoCoin>;\n    let mockWallet: sinon.SinonStubbedInstance<Wallet>;\n    let stubExplainTransaction: sinon.SinonStub;\n\n    before('prepare', async function () {\n      const coinName = getChainFromNetwork(acidTest.network);\n      coin = getUtxoCoin(coinName);\n\n      // Create PSBT and explanation\n      const psbt = acidTest.createPsbt();\n      const tx = psbt.getUnsignedTx();\n      const txHash = tx.getId();\n\n      let explanation: TransactionExplanation;\n      if (txFormat === 'psbt') {\n        if (backend === 'utxolib') {\n          explanation = explainPsbt(psbt, { pubs: acidTest.rootWalletKeys }, acidTest.network, {\n            strict: true,\n          });\n        } else if (backend === 'wasm') {\n          const wasmPsbt = fixedScriptWallet.BitGoPsbt.fromBytes(\n            psbt.toBuffer(),\n            utxolib.getNetworkName(acidTest.network)!\n          );\n          explanation = explainPsbtWasm(\n            wasmPsbt,\n            acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58()) as Triple<string>,\n            {\n              replayProtection: {\n                outputScripts: [acidTest.getReplayProtectionOutputScript()],\n              },\n            }\n          );\n        } else {\n          throw new Error(`Invalid backend: ${backend}`);\n        }\n      } else if (txFormat === 'legacy') {\n        const pubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());\n        // Extract change info from PSBT to pass to explainLegacyTx\n        const changeInfo = getChangeInfoFromPsbt(psbt);\n        explanation = explainLegacyTx(tx, { pubs, changeInfo }, acidTest.network);\n      } else {\n        throw new Error(`Invalid txFormat: ${txFormat}`);\n      }\n\n      // Determine txParams\n      let resolvedTxParams;\n      if (txParams === 'inferFromExplanation' || txParams === undefined) {\n        resolvedTxParams = getTxParamsFromExplanation(explanation, { externalCustomChangeAddress });\n      } else if ('rbfTxIds' in txParams) {\n        // Replace placeholder txHash with actual computed txHash\n        resolvedTxParams = {\n          rbfTxIds: txParams.rbfTxIds.map((hash) => (hash === 'PLACEHOLDER' ? txHash : hash)),\n        };\n      } else {\n        resolvedTxParams = txParams;\n      }\n\n      if (externalCustomChangeAddress) {\n        resolvedTxParams.allowExternalChangeAddress = true;\n      }\n\n      // Create mock wallet\n      mockWallet = sinon.createStubInstance(Wallet);\n      mockWallet.id.returns('test-wallet-id');\n      mockWallet.coin.returns(coin.getChain());\n      mockWallet.coinSpecific.returns(undefined);\n\n      // Mock getTransaction for RBF case\n      if ('rbfTxIds' in resolvedTxParams) {\n        const rbfTxParams = getTxParamsFromExplanation(explanation, { externalCustomChangeAddress: false });\n        mockWallet.getTransaction.resolves({\n          outputs: rbfTxParams.recipients.map((r) => ({\n            valueString: typeof r.amount === 'string' ? r.amount : r.amount.toString(),\n            address: r.address,\n            // wallet field is undefined for external outputs (not self-sends)\n          })),\n        });\n      }\n\n      // Mock verification options with keychains to disable networking\n      // Use the same keychains that were used to create the PSBT\n      const pubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());\n      const verification: VerificationOptions = {\n        disableNetworking: true,\n        keychains: {\n          user: { id: '0', pub: pubs[0], type: 'independent' },\n          backup: { id: '1', pub: pubs[1], type: 'independent' },\n          bitgo: { id: '2', pub: pubs[2], type: 'independent' },\n        },\n      };\n\n      // Stub explainTransaction to return the explanation without making network calls\n      stubExplainTransaction = sinon.stub(coin, 'explainTransaction').resolves(explanation);\n\n      let txPrebuild: TransactionPrebuild<bigint>;\n      if (txFormat === 'psbt') {\n        txPrebuild = {\n          txHex: psbt.toHex(),\n        };\n      } else if (txFormat === 'legacy') {\n        txPrebuild = {\n          txHex: psbt.getUnsignedTx().toHex(),\n        };\n      } else {\n        throw new Error(`Invalid txFormat: ${txFormat}`);\n      }\n\n      refParsedTransaction = await parseTransaction(coin, {\n        wallet: mockWallet as unknown as UtxoWallet,\n        txParams: resolvedTxParams,\n        txPrebuild,\n        verification,\n      });\n    });\n\n    after('cleanup', function () {\n      if (stubExplainTransaction) {\n        stubExplainTransaction.restore();\n      }\n    });\n\n    it('should parse transaction without network calls', function () {\n      assert.ok(refParsedTransaction);\n      assert.ok(refParsedTransaction.keychains);\n      assert.ok(refParsedTransaction.outputs);\n    });\n\n    it('should have valid keychains', function () {\n      assert.ok(refParsedTransaction.keychains.user);\n      assert.ok(refParsedTransaction.keychains.backup);\n      assert.ok(refParsedTransaction.keychains.bitgo);\n      const pubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());\n      assert.strictEqual(refParsedTransaction.keychains.user.pub, pubs[0]);\n      assert.strictEqual(refParsedTransaction.keychains.backup.pub, pubs[1]);\n      assert.strictEqual(refParsedTransaction.keychains.bitgo.pub, pubs[2]);\n    });\n\n    it('should have outputs classified as internal or external', function () {\n      // Since we didn't specify any recipients, outputs will be classified based on whether they can be\n      // verified as wallet addresses. Some may be external if address verification fails without a proper wallet setup.\n      const totalOutputs = refParsedTransaction.outputs.length;\n      const changeOutputs = refParsedTransaction.changeOutputs.length;\n      const externalOutputs = refParsedTransaction.outputs.filter((o) => o.external === true).length;\n      assert.strictEqual(externalOutputs, 3);\n\n      assert.ok(totalOutputs > 0, 'should have at least one output');\n      assert.strictEqual(changeOutputs + externalOutputs, totalOutputs, 'all outputs should be classified');\n    });\n\n    it('should have expected explicit and implicit external spend amounts', function () {\n      assert.strictEqual(BigInt(refParsedTransaction.explicitExternalSpendAmount), expectedExplicitExternalSpendAmount);\n      assert.strictEqual(BigInt(refParsedTransaction.implicitExternalSpendAmount), expectedImplicitExternalSpendAmount);\n    });\n  });\n}\n\nfunction describeTransaction(\n  backend: 'utxolib' | 'wasm',\n  filter: (test: utxolib.testutil.AcidTest) => boolean = () => true\n) {\n  describe(`parseTransaction (${backend})`, function () {\n    utxolib.testutil.AcidTest.suite()\n      .filter(filter)\n      .forEach((test) => {\n        // Default case: psbt format, infer recipients from explanation\n        describeParseTransactionWith(test, 'default', backend, {\n          txParams: 'inferFromExplanation',\n          expectedExplicitExternalSpendAmount: 1800n,\n          expectedImplicitExternalSpendAmount: 0n,\n        });\n\n        if (test.network !== utxolib.networks.bitcoin) {\n          return;\n        }\n        // extended test suite for bitcoin\n\n        describeParseTransactionWith(test, 'legacy', backend, {\n          txFormat: 'legacy',\n          txParams: 'inferFromExplanation',\n          expectedExplicitExternalSpendAmount: 1800n,\n          expectedImplicitExternalSpendAmount: 0n,\n        });\n\n        describeParseTransactionWith(test, 'empty recipients', backend, {\n          txParams: {\n            recipients: [],\n          },\n          expectedExplicitExternalSpendAmount: 0n,\n          expectedImplicitExternalSpendAmount: 1800n,\n        });\n\n        describeParseTransactionWith(test, 'rbf', backend, {\n          txParams: {\n            rbfTxIds: ['PLACEHOLDER'],\n          },\n          expectedExplicitExternalSpendAmount: 1800n,\n          expectedImplicitExternalSpendAmount: 0n,\n        });\n\n        describeParseTransactionWith(test, 'allowExternalChangeAddress', backend, {\n          txParams: 'inferFromExplanation',\n          externalCustomChangeAddress: true,\n          expectedExplicitExternalSpendAmount: 1800n,\n          expectedImplicitExternalSpendAmount: 0n,\n        });\n      });\n  });\n}\n\ndescribeTransaction('utxolib');\ndescribeTransaction('wasm', (test) => hasWasmUtxoSupport(test.network));\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"util.d.ts","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/util.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,OAAO,CAOpE"}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.hasWasmUtxoSupport = hasWasmUtxoSupport;
|
|
37
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
38
|
+
function hasWasmUtxoSupport(network) {
|
|
39
|
+
return ![
|
|
40
|
+
utxolib.networks.bitcoincash,
|
|
41
|
+
utxolib.networks.bitcoingold,
|
|
42
|
+
utxolib.networks.ecash,
|
|
43
|
+
utxolib.networks.zcash,
|
|
44
|
+
].includes(utxolib.getMainnet(network));
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Rlc3QvdW5pdC90cmFuc2FjdGlvbi9maXhlZFNjcmlwdC91dGlsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUEsZ0RBT0M7QUFURCw4REFBZ0Q7QUFFaEQsU0FBZ0Isa0JBQWtCLENBQUMsT0FBd0I7SUFDekQsT0FBTyxDQUFDO1FBQ04sT0FBTyxDQUFDLFFBQVEsQ0FBQyxXQUFXO1FBQzVCLE9BQU8sQ0FBQyxRQUFRLENBQUMsV0FBVztRQUM1QixPQUFPLENBQUMsUUFBUSxDQUFDLEtBQUs7UUFDdEIsT0FBTyxDQUFDLFFBQVEsQ0FBQyxLQUFLO0tBQ3ZCLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztBQUMxQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgdXR4b2xpYiBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5cbmV4cG9ydCBmdW5jdGlvbiBoYXNXYXNtVXR4b1N1cHBvcnQobmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogYm9vbGVhbiB7XG4gIHJldHVybiAhW1xuICAgIHV0eG9saWIubmV0d29ya3MuYml0Y29pbmNhc2gsXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luZ29sZCxcbiAgICB1dHhvbGliLm5ldHdvcmtzLmVjYXNoLFxuICAgIHV0eG9saWIubmV0d29ya3MuemNhc2gsXG4gIF0uaW5jbHVkZXModXR4b2xpYi5nZXRNYWlubmV0KG5ldHdvcmspKTtcbn1cbiJdfQ==
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"txFormat.d.ts","sourceRoot":"","sources":["../../../../test/unit/txFormat.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AAE9C,OAAO,EAAE,gBAAgB,
|
|
1
|
+
{"version":3,"file":"txFormat.d.ts","sourceRoot":"","sources":["../../../../test/unit/txFormat.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AAE9C,OAAO,EAAE,gBAAgB,EAA2B,QAAQ,EAAE,MAAM,WAAW,CAAC;AAIhF,KAAK,UAAU,GAAG,KAAK,GAAG,MAAM,GAAG,WAAW,GAAG,iBAAiB,GAAG,SAAS,CAAC;AAC/E,KAAK,aAAa,GAAG,oBAAoB,CAAC;AAC1C,KAAK,UAAU,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,CAAC;AAElD,KAAK,aAAa,GAAG;IACnB,IAAI,CAAC,EAAE,UAAU,CAAC;IAClB,OAAO,CAAC,EAAE,aAAa,CAAC;IACxB,WAAW,CAAC,EAAE,UAAU,EAAE,CAAC;CAC5B,CAAC;AAEF;;GAEG;AACH,wBAAgB,uBAAuB,IAAI,KAAK,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,aAAa,CAAA;CAAE,CAAC,CASzF;AAED;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,gBAAgB,EAAE,OAAO,GAAE,aAAkB,GAAG,MAAM,CAS5F;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,gBAAgB,EAAE,MAAM,EAAE,MAAM,EAAE,eAAe,CAAC,EAAE,QAAQ,GAAG,QAAQ,GAAG,SAAS,CAEpH"}
|