@bitgo-beta/sdk-coin-sol 2.4.3-beta.998 → 7.6.3
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/src/bigint-buffer-guard.d.ts +1 -0
- package/dist/src/bigint-buffer-guard.d.ts.map +1 -0
- package/dist/src/bigint-buffer-guard.js +29 -0
- package/dist/src/config/token2022StaticConfig.d.ts +3 -0
- package/dist/src/config/token2022StaticConfig.d.ts.map +1 -0
- package/dist/src/config/token2022StaticConfig.js +50 -0
- 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/constants.d.ts +35 -2
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +41 -2
- package/dist/src/lib/customInstructionBuilder.d.ts +72 -0
- package/dist/src/lib/customInstructionBuilder.d.ts.map +1 -0
- package/dist/src/lib/customInstructionBuilder.js +289 -0
- package/dist/src/lib/iface.d.ts +79 -5
- 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 +1 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +4 -2
- package/dist/src/lib/instructionParamsFactory.d.ts.map +1 -1
- package/dist/src/lib/instructionParamsFactory.js +364 -78
- package/dist/src/lib/jitoStakePoolOperations.d.ts +113 -0
- package/dist/src/lib/jitoStakePoolOperations.d.ts.map +1 -0
- package/dist/src/lib/jitoStakePoolOperations.js +200 -0
- package/dist/src/lib/solInstructionFactory.d.ts.map +1 -1
- package/dist/src/lib/solInstructionFactory.js +266 -69
- package/dist/src/lib/stakingActivateBuilder.d.ts +16 -5
- package/dist/src/lib/stakingActivateBuilder.d.ts.map +1 -1
- package/dist/src/lib/stakingActivateBuilder.js +23 -10
- package/dist/src/lib/stakingDeactivateBuilder.d.ts +16 -5
- package/dist/src/lib/stakingDeactivateBuilder.d.ts.map +1 -1
- package/dist/src/lib/stakingDeactivateBuilder.js +43 -20
- package/dist/src/lib/token2022Config.d.ts +44 -0
- package/dist/src/lib/token2022Config.d.ts.map +1 -0
- package/dist/src/lib/token2022Config.js +27 -0
- package/dist/src/lib/tokenTransferBuilder.js +7 -7
- package/dist/src/lib/transaction.d.ts +31 -4
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +134 -46
- package/dist/src/lib/transactionBuilder.d.ts +18 -2
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +78 -2
- package/dist/src/lib/transactionBuilderFactory.d.ts +5 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +10 -1
- package/dist/src/lib/utils.d.ts +34 -1
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +88 -24
- package/dist/src/sol.d.ts +36 -13
- package/dist/src/sol.d.ts.map +1 -1
- package/dist/src/sol.js +230 -38
- package/dist/test/fixtures/sol.d.ts +1152 -0
- package/dist/test/fixtures/sol.d.ts.map +1 -0
- package/dist/test/fixtures/sol.js +1433 -0
- package/dist/test/resources/sol.d.ts +238 -0
- package/dist/test/resources/sol.d.ts.map +1 -0
- package/dist/test/resources/sol.js +320 -0
- package/dist/test/unit/fixtures/solBackupKey.d.ts +5 -0
- package/dist/test/unit/fixtures/solBackupKey.d.ts.map +1 -0
- package/dist/test/unit/fixtures/solBackupKey.js +8 -0
- package/dist/test/unit/getBuilderFactory.d.ts +3 -0
- package/dist/test/unit/getBuilderFactory.d.ts.map +1 -0
- package/dist/test/unit/getBuilderFactory.js +10 -0
- package/dist/test/unit/instructionParamsFactory.d.ts +2 -0
- package/dist/test/unit/instructionParamsFactory.d.ts.map +1 -0
- package/dist/test/unit/instructionParamsFactory.js +412 -0
- package/dist/test/unit/instructionParamsFactory.staking.d.ts +2 -0
- package/dist/test/unit/instructionParamsFactory.staking.d.ts.map +1 -0
- package/dist/test/unit/instructionParamsFactory.staking.js +1059 -0
- package/dist/test/unit/keyPair.d.ts +2 -0
- package/dist/test/unit/keyPair.d.ts.map +1 -0
- package/dist/test/unit/keyPair.js +177 -0
- package/dist/test/unit/messages/messageBuilderFactory.d.ts +2 -0
- package/dist/test/unit/messages/messageBuilderFactory.d.ts.map +1 -0
- package/dist/test/unit/messages/messageBuilderFactory.js +118 -0
- package/dist/test/unit/messages/simpleMessageBuilder.d.ts +2 -0
- package/dist/test/unit/messages/simpleMessageBuilder.d.ts.map +1 -0
- package/dist/test/unit/messages/simpleMessageBuilder.js +194 -0
- package/dist/test/unit/sol.d.ts +2 -0
- package/dist/test/unit/sol.d.ts.map +1 -0
- package/dist/test/unit/sol.js +3108 -0
- package/dist/test/unit/solInstructionFactory.d.ts +2 -0
- package/dist/test/unit/solInstructionFactory.d.ts.map +1 -0
- package/dist/test/unit/solInstructionFactory.js +454 -0
- package/dist/test/unit/solToken.d.ts +2 -0
- package/dist/test/unit/solToken.d.ts.map +1 -0
- package/dist/test/unit/solToken.js +31 -0
- package/dist/test/unit/transaction.d.ts +2 -0
- package/dist/test/unit/transaction.d.ts.map +1 -0
- package/dist/test/unit/transaction.js +983 -0
- package/dist/test/unit/transactionBuilder/StakingWithdrawBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/StakingWithdrawBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/StakingWithdrawBuilder.js +202 -0
- package/dist/test/unit/transactionBuilder/ataInitBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/ataInitBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/ataInitBuilder.js +471 -0
- package/dist/test/unit/transactionBuilder/customInstructionBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/customInstructionBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/customInstructionBuilder.js +413 -0
- package/dist/test/unit/transactionBuilder/stakingActivateBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/stakingActivateBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/stakingActivateBuilder.js +430 -0
- package/dist/test/unit/transactionBuilder/stakingAuthorizeBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/stakingAuthorizeBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/stakingAuthorizeBuilder.js +157 -0
- package/dist/test/unit/transactionBuilder/stakingDeactivateBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/stakingDeactivateBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/stakingDeactivateBuilder.js +384 -0
- package/dist/test/unit/transactionBuilder/stakingDelegateBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/stakingDelegateBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/stakingDelegateBuilder.js +224 -0
- package/dist/test/unit/transactionBuilder/stakingRawMsgAuthorizeBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/stakingRawMsgAuthorizeBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/stakingRawMsgAuthorizeBuilder.js +259 -0
- package/dist/test/unit/transactionBuilder/tokenTransferBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/tokenTransferBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/tokenTransferBuilder.js +787 -0
- package/dist/test/unit/transactionBuilder/transactionBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/transactionBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/transactionBuilder.js +495 -0
- package/dist/test/unit/transactionBuilder/transferBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/transferBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/transferBuilder.js +286 -0
- package/dist/test/unit/transactionBuilder/transferBuilderV2.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/transferBuilderV2.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/transferBuilderV2.js +862 -0
- package/dist/test/unit/transactionBuilder/walletInitBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/walletInitBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/walletInitBuilder.js +259 -0
- package/dist/test/unit/utils.d.ts +2 -0
- package/dist/test/unit/utils.d.ts.map +1 -0
- package/dist/test/unit/utils.js +505 -0
- package/dist/test/unit/versionedTransaction.d.ts +2 -0
- package/dist/test/unit/versionedTransaction.d.ts.map +1 -0
- package/dist/test/unit/versionedTransaction.js +207 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +14 -9
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -1223
|
@@ -0,0 +1,3108 @@
|
|
|
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 assert_1 = __importDefault(require("assert"));
|
|
40
|
+
const _ = __importStar(require("lodash"));
|
|
41
|
+
const nock_1 = __importDefault(require("nock"));
|
|
42
|
+
const should = __importStar(require("should"));
|
|
43
|
+
const sinon = __importStar(require("sinon"));
|
|
44
|
+
const spl_token_1 = require("@solana/spl-token");
|
|
45
|
+
const sdk_api_1 = require("@bitgo-beta/sdk-api");
|
|
46
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
47
|
+
const sdk_test_1 = require("@bitgo-beta/sdk-test");
|
|
48
|
+
const statics_1 = require("@bitgo-beta/statics");
|
|
49
|
+
const src_1 = require("../../src");
|
|
50
|
+
const lib_1 = require("../../src/lib");
|
|
51
|
+
const utils_1 = require("../../src/lib/utils");
|
|
52
|
+
const testData = __importStar(require("../fixtures/sol"));
|
|
53
|
+
const resources = __importStar(require("../resources/sol"));
|
|
54
|
+
const solBackupKey_1 = require("./fixtures/solBackupKey");
|
|
55
|
+
const getBuilderFactory_1 = require("./getBuilderFactory");
|
|
56
|
+
describe('SOL:', function () {
|
|
57
|
+
let bitgo;
|
|
58
|
+
let basecoin;
|
|
59
|
+
let keyPair;
|
|
60
|
+
let newTxPrebuild;
|
|
61
|
+
let newTxPrebuildTokenTransfer;
|
|
62
|
+
let newTxParams;
|
|
63
|
+
let newTxParamsWithError;
|
|
64
|
+
let newTxParamsWithExtraData;
|
|
65
|
+
let newTxParamsTokenTransfer;
|
|
66
|
+
const badAddresses = resources.addresses.invalidAddresses;
|
|
67
|
+
const goodAddresses = resources.addresses.validAddresses;
|
|
68
|
+
const keypair = {
|
|
69
|
+
pub: resources.accountWithSeed.publicKey,
|
|
70
|
+
prv: resources.accountWithSeed.privateKey.base58,
|
|
71
|
+
};
|
|
72
|
+
const txPrebuild = {
|
|
73
|
+
recipients: [
|
|
74
|
+
{
|
|
75
|
+
address: 'lionteste212',
|
|
76
|
+
amount: '1000',
|
|
77
|
+
},
|
|
78
|
+
],
|
|
79
|
+
txBase64: resources.TRANSFER_UNSIGNED_TX_WITH_MEMO_AND_DURABLE_NONCE,
|
|
80
|
+
txInfo: {
|
|
81
|
+
feePayer: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
82
|
+
nonce: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
83
|
+
},
|
|
84
|
+
txid: '586c5b59b10b134d04c16ac1b273fe3c5529f34aef75db4456cd469c5cdac7e2',
|
|
85
|
+
isVotingTransaction: false,
|
|
86
|
+
coin: 'tsol',
|
|
87
|
+
};
|
|
88
|
+
const txParams = {
|
|
89
|
+
txPrebuild,
|
|
90
|
+
recipients: [
|
|
91
|
+
{
|
|
92
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
93
|
+
amount: '300000',
|
|
94
|
+
},
|
|
95
|
+
],
|
|
96
|
+
};
|
|
97
|
+
const memo = { value: 'test memo' };
|
|
98
|
+
const durableNonce = {
|
|
99
|
+
walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
100
|
+
authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
101
|
+
};
|
|
102
|
+
const errorDurableNonce = {
|
|
103
|
+
walletNonceAddress: '8YM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
104
|
+
authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
105
|
+
};
|
|
106
|
+
const txParamsWithError = {
|
|
107
|
+
txPrebuild,
|
|
108
|
+
recipients: [
|
|
109
|
+
{
|
|
110
|
+
address: 'CP5Dpaa42mMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
111
|
+
amount: '300000',
|
|
112
|
+
},
|
|
113
|
+
],
|
|
114
|
+
};
|
|
115
|
+
const txParamsWithExtraData = {
|
|
116
|
+
txPrebuild,
|
|
117
|
+
recipients: [
|
|
118
|
+
{
|
|
119
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
120
|
+
amount: '300000',
|
|
121
|
+
data: undefined,
|
|
122
|
+
},
|
|
123
|
+
],
|
|
124
|
+
};
|
|
125
|
+
const txPrebuildTokenTransfer = {
|
|
126
|
+
recipients: [
|
|
127
|
+
{
|
|
128
|
+
address: 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg',
|
|
129
|
+
amount: '1',
|
|
130
|
+
},
|
|
131
|
+
],
|
|
132
|
+
txHex: resources.TOKEN_TRANSFER_TO_NATIVE_UNSIGNED_TX_HEX,
|
|
133
|
+
txInfo: {
|
|
134
|
+
feePayer: '4DujymUFbQ8GBKtAwAZrQ6QqpvtBEivL48h4ta2oJGd2',
|
|
135
|
+
nonce: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
136
|
+
},
|
|
137
|
+
txid: '586c5b59b10b134d04c16ac1b273fe3c5529f34aef75db4456cd469c5cdac7e2',
|
|
138
|
+
isVotingTransaction: false,
|
|
139
|
+
coin: 'tsol',
|
|
140
|
+
};
|
|
141
|
+
const txParamsTokenTransfer = {
|
|
142
|
+
txPrebuild,
|
|
143
|
+
recipients: [
|
|
144
|
+
{
|
|
145
|
+
address: 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg',
|
|
146
|
+
amount: '1',
|
|
147
|
+
},
|
|
148
|
+
],
|
|
149
|
+
};
|
|
150
|
+
const errorMemo = { value: 'different memo' };
|
|
151
|
+
const errorFeePayer = '5hr5fisPi6DXCuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe';
|
|
152
|
+
const factory = (0, getBuilderFactory_1.getBuilderFactory)('tsol');
|
|
153
|
+
const wallet = new src_1.KeyPair(resources.authAccount).getKeys();
|
|
154
|
+
const stakeAccount = new src_1.KeyPair(resources.stakeAccount).getKeys();
|
|
155
|
+
const blockHash = resources.blockHashes.validBlockHashes[0];
|
|
156
|
+
const amount = '10000';
|
|
157
|
+
const validator = resources.validator;
|
|
158
|
+
before(function () {
|
|
159
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'mock' });
|
|
160
|
+
bitgo.safeRegister('sol', src_1.Tsol.createInstance);
|
|
161
|
+
bitgo.safeRegister('tsol', src_1.Tsol.createInstance);
|
|
162
|
+
bitgo.initializeTestVars();
|
|
163
|
+
basecoin = bitgo.coin('tsol');
|
|
164
|
+
keyPair = basecoin.generateKeyPair(resources.accountWithSeed.seed);
|
|
165
|
+
newTxPrebuild = () => {
|
|
166
|
+
return _.cloneDeep(txPrebuild);
|
|
167
|
+
};
|
|
168
|
+
newTxPrebuildTokenTransfer = () => {
|
|
169
|
+
return _.cloneDeep(txPrebuildTokenTransfer);
|
|
170
|
+
};
|
|
171
|
+
newTxParams = () => {
|
|
172
|
+
return _.cloneDeep(txParams);
|
|
173
|
+
};
|
|
174
|
+
newTxParamsWithError = () => {
|
|
175
|
+
return _.cloneDeep(txParamsWithError);
|
|
176
|
+
};
|
|
177
|
+
newTxParamsWithExtraData = () => {
|
|
178
|
+
return _.cloneDeep(txParamsWithExtraData);
|
|
179
|
+
};
|
|
180
|
+
newTxParamsTokenTransfer = () => {
|
|
181
|
+
return _.cloneDeep(txParamsTokenTransfer);
|
|
182
|
+
};
|
|
183
|
+
});
|
|
184
|
+
it('should instantiate the coin', async function () {
|
|
185
|
+
let localBasecoin = bitgo.coin('tsol');
|
|
186
|
+
localBasecoin.should.be.an.instanceof(src_1.Tsol);
|
|
187
|
+
localBasecoin = bitgo.coin('sol');
|
|
188
|
+
localBasecoin.should.be.an.instanceof(src_1.Sol);
|
|
189
|
+
});
|
|
190
|
+
it('should retun the right info', function () {
|
|
191
|
+
basecoin.getChain().should.equal('tsol');
|
|
192
|
+
basecoin.getFamily().should.equal('sol');
|
|
193
|
+
basecoin.getFullName().should.equal('Testnet Solana');
|
|
194
|
+
basecoin.getBaseFactor().should.equal(1000000000);
|
|
195
|
+
});
|
|
196
|
+
describe('verify transactions', () => {
|
|
197
|
+
const walletData = {
|
|
198
|
+
id: '5b34252f1bf349930e34020a00000000',
|
|
199
|
+
coin: 'tsol',
|
|
200
|
+
keys: [
|
|
201
|
+
'5b3424f91bf349930e34017500000000',
|
|
202
|
+
'5b3424f91bf349930e34017600000000',
|
|
203
|
+
'5b3424f91bf349930e34017700000000',
|
|
204
|
+
],
|
|
205
|
+
coinSpecific: {
|
|
206
|
+
rootAddress: wallet.pub,
|
|
207
|
+
},
|
|
208
|
+
multisigType: 'tss',
|
|
209
|
+
};
|
|
210
|
+
const walletObj = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
211
|
+
it('should verify transactions', async function () {
|
|
212
|
+
const txParams = newTxParams();
|
|
213
|
+
const txPrebuild = newTxPrebuild();
|
|
214
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
215
|
+
txParams,
|
|
216
|
+
txPrebuild,
|
|
217
|
+
memo,
|
|
218
|
+
durableNonce,
|
|
219
|
+
wallet: walletObj,
|
|
220
|
+
});
|
|
221
|
+
validTransaction.should.equal(true);
|
|
222
|
+
});
|
|
223
|
+
it('should verify consolidate transaction', async function () {
|
|
224
|
+
const txParams = newTxParams();
|
|
225
|
+
const txPrebuild = newTxPrebuild();
|
|
226
|
+
txPrebuild.consolidateId = 'consolidateId';
|
|
227
|
+
const walletData = {
|
|
228
|
+
id: '5b34252f1bf349930e34020a00000000',
|
|
229
|
+
coin: 'tsol',
|
|
230
|
+
keys: [
|
|
231
|
+
'5b3424f91bf349930e34017500000000',
|
|
232
|
+
'5b3424f91bf349930e34017600000000',
|
|
233
|
+
'5b3424f91bf349930e34017700000000',
|
|
234
|
+
],
|
|
235
|
+
coinSpecific: {
|
|
236
|
+
rootAddress: stakeAccount.pub,
|
|
237
|
+
},
|
|
238
|
+
multisigType: 'tss',
|
|
239
|
+
};
|
|
240
|
+
const walletWithDifferentAddress = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
241
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
242
|
+
txParams,
|
|
243
|
+
txPrebuild,
|
|
244
|
+
memo,
|
|
245
|
+
durableNonce,
|
|
246
|
+
wallet: walletWithDifferentAddress,
|
|
247
|
+
});
|
|
248
|
+
validTransaction.should.be.true();
|
|
249
|
+
});
|
|
250
|
+
it('should handle txBase64 and txHex interchangeably', async function () {
|
|
251
|
+
const txParams = newTxParams();
|
|
252
|
+
const txPrebuild = newTxPrebuild();
|
|
253
|
+
txPrebuild.txHex = txPrebuild.txBase64;
|
|
254
|
+
txPrebuild.txBase64 = undefined;
|
|
255
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
256
|
+
txParams,
|
|
257
|
+
txPrebuild,
|
|
258
|
+
memo,
|
|
259
|
+
durableNonce,
|
|
260
|
+
wallet: walletObj,
|
|
261
|
+
});
|
|
262
|
+
validTransaction.should.equal(true);
|
|
263
|
+
});
|
|
264
|
+
it('should convert serialized hex string to base64', async function () {
|
|
265
|
+
const txParams = newTxParams();
|
|
266
|
+
const txPrebuild = newTxPrebuild();
|
|
267
|
+
txPrebuild.txBase64 = Buffer.from(txPrebuild.txBase64, 'base64').toString('hex');
|
|
268
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
269
|
+
txParams,
|
|
270
|
+
txPrebuild,
|
|
271
|
+
memo,
|
|
272
|
+
durableNonce,
|
|
273
|
+
wallet: walletObj,
|
|
274
|
+
});
|
|
275
|
+
validTransaction.should.equal(true);
|
|
276
|
+
});
|
|
277
|
+
it('should verify when input `recipients` is absent', async function () {
|
|
278
|
+
const txParams = newTxParams();
|
|
279
|
+
txParams.recipients = undefined;
|
|
280
|
+
const txPrebuild = newTxPrebuild();
|
|
281
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
282
|
+
txParams,
|
|
283
|
+
txPrebuild,
|
|
284
|
+
memo,
|
|
285
|
+
durableNonce,
|
|
286
|
+
wallet: walletObj,
|
|
287
|
+
});
|
|
288
|
+
validTransaction.should.equal(true);
|
|
289
|
+
});
|
|
290
|
+
it('should fail verify transactions when have different memo', async function () {
|
|
291
|
+
const txParams = newTxParams();
|
|
292
|
+
const txPrebuild = newTxPrebuild();
|
|
293
|
+
await basecoin
|
|
294
|
+
.verifyTransaction({ txParams, txPrebuild, memo: errorMemo, wallet: walletObj })
|
|
295
|
+
.should.be.rejectedWith('Tx memo does not match with expected txParams recipient memo');
|
|
296
|
+
});
|
|
297
|
+
it('should pass if we pass PDA address', async function () {
|
|
298
|
+
const walletData = {
|
|
299
|
+
id: '67f8ddff4c9b8b57a2e16acffac9a3b5',
|
|
300
|
+
coin: 'tsol',
|
|
301
|
+
keys: [
|
|
302
|
+
'5b3424f91bf349930e34017500000000',
|
|
303
|
+
'5b3424f91bf349930e34017600000000',
|
|
304
|
+
'5b3424f91bf349930e34017700000000',
|
|
305
|
+
],
|
|
306
|
+
coinSpecific: {
|
|
307
|
+
rootAddress: '8zbsJA5c8HPR7BPjZkrSVrus2uMuXqCfzksGwB3Uscjb',
|
|
308
|
+
},
|
|
309
|
+
multisigType: 'tss',
|
|
310
|
+
};
|
|
311
|
+
const walletObj = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
312
|
+
const txPrebuild = {
|
|
313
|
+
recipients: [
|
|
314
|
+
{
|
|
315
|
+
address: '11111111111111111111111111111112',
|
|
316
|
+
amount: '1000000000',
|
|
317
|
+
tokenName: 'tsol:usdc',
|
|
318
|
+
},
|
|
319
|
+
],
|
|
320
|
+
txBase64: '02000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006ec1adcc89bb564f1f8225821140a9723efa80e8d506765770b7e201d66d8200d4f690e9a8163291b69f8c3827aad96cfd2105eee3aae76cbca38fcad2bf7f0a0201070c76c356cb069b66c2b35a8638b4d4afca75b303f29f0deeb4bff8528299a9c9d21c96172044f1217c3784e8f02f49e2c8fc3591e81294ab54394f9d22fd7b7a8f60129e6ecb20309c27dcba5fc6c441438d33a1568004a1860e22c16f071976a7d2e2008bd34b53a08aa9c8ec04eb2196745fc6029224447417e2fb0fced601240cabba4ce534c02fc154ba559ed2a02ac971e3385acb426ff63bb1040e2c2435000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000018c97258f4e2489f1bb3d1029148e0d830b5a1399daff1084048e7bd8dbe9f859d10389fbcee528f208611dccc734b31092540cb2b8d58d100f2eaa2cedb4da5e06a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000006a7d517192c5c51218cc94c3d4af17f58daee089ba1fd44e3dbd98a0000000006ddf6e1d765a193d9cbe146ceeb79ac1cb485ed5f5b37913a8cf5857eff00a9e680634533882f880a3e7dfa999dfb864b88968d242a0c9a90b5df149e42da050305030209010404000000070700030608050b0a000b04040803000a0c00ca9a3b0000000009',
|
|
321
|
+
txInfo: {
|
|
322
|
+
feePayer: '8zbsJA5c8HPR7BPjZkrSVrus2uMuXqCfzksGwB3Uscjb',
|
|
323
|
+
nonce: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
324
|
+
},
|
|
325
|
+
txid: '586c5b59b10b134d04c16ac1b273fe3c5529f34aef75db4456cd469c5cdac7e2',
|
|
326
|
+
isVotingTransaction: false,
|
|
327
|
+
coin: 'tsol',
|
|
328
|
+
};
|
|
329
|
+
const txParams = {
|
|
330
|
+
txPrebuild,
|
|
331
|
+
recipients: [
|
|
332
|
+
{
|
|
333
|
+
address: '11111111111111111111111111111112',
|
|
334
|
+
amount: '1000000000',
|
|
335
|
+
tokenName: 'tsol:usdc',
|
|
336
|
+
},
|
|
337
|
+
],
|
|
338
|
+
};
|
|
339
|
+
const memo = {
|
|
340
|
+
value: undefined,
|
|
341
|
+
};
|
|
342
|
+
const verifyTransaction = await basecoin.verifyTransaction({
|
|
343
|
+
txParams,
|
|
344
|
+
txPrebuild,
|
|
345
|
+
memo: memo,
|
|
346
|
+
wallet: walletObj,
|
|
347
|
+
});
|
|
348
|
+
verifyTransaction.should.equal(true);
|
|
349
|
+
});
|
|
350
|
+
it('should fail verify transactions when have different durableNonce', async function () {
|
|
351
|
+
const txParams = newTxParams();
|
|
352
|
+
const txPrebuild = newTxPrebuild();
|
|
353
|
+
await basecoin
|
|
354
|
+
.verifyTransaction({ txParams, txPrebuild, memo, durableNonce: errorDurableNonce, wallet: walletObj })
|
|
355
|
+
.should.be.rejectedWith('Tx durableNonce does not match with param durableNonce');
|
|
356
|
+
});
|
|
357
|
+
it('should fail verify transactions when have different feePayer', async function () {
|
|
358
|
+
const txParams = newTxParams();
|
|
359
|
+
const txPrebuild = newTxPrebuild();
|
|
360
|
+
const walletData = {
|
|
361
|
+
id: '5b34252f1bf349930e34020a00000000',
|
|
362
|
+
coin: 'tsol',
|
|
363
|
+
keys: [
|
|
364
|
+
'5b3424f91bf349930e34017500000000',
|
|
365
|
+
'5b3424f91bf349930e34017600000000',
|
|
366
|
+
'5b3424f91bf349930e34017700000000',
|
|
367
|
+
],
|
|
368
|
+
coinSpecific: {
|
|
369
|
+
rootAddress: stakeAccount.pub,
|
|
370
|
+
},
|
|
371
|
+
multisigType: 'tss',
|
|
372
|
+
};
|
|
373
|
+
const walletWithDifferentAddress = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
374
|
+
await basecoin
|
|
375
|
+
.verifyTransaction({ txParams, txPrebuild, memo, wallet: walletWithDifferentAddress })
|
|
376
|
+
.should.be.rejectedWith('Tx fee payer is not the wallet root address');
|
|
377
|
+
});
|
|
378
|
+
it('should fail verify transactions when have different recipients', async function () {
|
|
379
|
+
const txParams = newTxParamsWithError();
|
|
380
|
+
const txPrebuild = newTxPrebuild();
|
|
381
|
+
await basecoin
|
|
382
|
+
.verifyTransaction({ txParams, txPrebuild, memo, errorFeePayer, wallet: walletObj })
|
|
383
|
+
.should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');
|
|
384
|
+
});
|
|
385
|
+
it('should succeed to verify token transaction with native address recipient', async function () {
|
|
386
|
+
const txParams = newTxParamsTokenTransfer();
|
|
387
|
+
const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address
|
|
388
|
+
txParams.recipients = [{ address, amount: '1', tokenName: 'tsol:usdc' }];
|
|
389
|
+
const txPrebuild = newTxPrebuildTokenTransfer();
|
|
390
|
+
const feePayerWalletData = {
|
|
391
|
+
id: '5b34252f1bf349930e34020a00000000',
|
|
392
|
+
coin: 'tsol',
|
|
393
|
+
keys: [
|
|
394
|
+
'5b3424f91bf349930e34017500000000',
|
|
395
|
+
'5b3424f91bf349930e34017600000000',
|
|
396
|
+
'5b3424f91bf349930e34017700000000',
|
|
397
|
+
],
|
|
398
|
+
coinSpecific: {
|
|
399
|
+
rootAddress: '4DujymUFbQ8GBKtAwAZrQ6QqpvtBEivL48h4ta2oJGd2',
|
|
400
|
+
},
|
|
401
|
+
multisigType: 'tss',
|
|
402
|
+
};
|
|
403
|
+
const feePayerWallet = new sdk_core_1.Wallet(bitgo, basecoin, feePayerWalletData);
|
|
404
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
405
|
+
txParams,
|
|
406
|
+
txPrebuild,
|
|
407
|
+
wallet: feePayerWallet,
|
|
408
|
+
});
|
|
409
|
+
validTransaction.should.equal(true);
|
|
410
|
+
});
|
|
411
|
+
it('should succeed to verify token transaction with leading zero recipient amount', async function () {
|
|
412
|
+
const txParams = newTxParamsTokenTransfer();
|
|
413
|
+
const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address
|
|
414
|
+
txParams.recipients = [{ address, amount: '0001', tokenName: 'tsol:usdc' }];
|
|
415
|
+
const txPrebuild = newTxPrebuildTokenTransfer();
|
|
416
|
+
const feePayerWalletData = {
|
|
417
|
+
id: '5b34252f1bf349930e34020a00000000',
|
|
418
|
+
coin: 'tsol',
|
|
419
|
+
keys: [
|
|
420
|
+
'5b3424f91bf349930e34017500000000',
|
|
421
|
+
'5b3424f91bf349930e34017600000000',
|
|
422
|
+
'5b3424f91bf349930e34017700000000',
|
|
423
|
+
],
|
|
424
|
+
coinSpecific: {
|
|
425
|
+
rootAddress: '4DujymUFbQ8GBKtAwAZrQ6QqpvtBEivL48h4ta2oJGd2',
|
|
426
|
+
},
|
|
427
|
+
multisigType: 'tss',
|
|
428
|
+
};
|
|
429
|
+
const feePayerWallet = new sdk_core_1.Wallet(bitgo, basecoin, feePayerWalletData);
|
|
430
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
431
|
+
txParams,
|
|
432
|
+
txPrebuild,
|
|
433
|
+
wallet: feePayerWallet,
|
|
434
|
+
});
|
|
435
|
+
validTransaction.should.equal(true);
|
|
436
|
+
});
|
|
437
|
+
it('should fail to verify token transaction with different recipient tokenName', async function () {
|
|
438
|
+
const txParams = newTxParamsTokenTransfer();
|
|
439
|
+
const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address
|
|
440
|
+
txParams.recipients = [{ address, amount: '1', tokenName: 'tsol:usdt' }]; // Different tokenName, should fail to verify tx
|
|
441
|
+
const txPrebuild = newTxPrebuildTokenTransfer();
|
|
442
|
+
await basecoin
|
|
443
|
+
.verifyTransaction({
|
|
444
|
+
txParams,
|
|
445
|
+
txPrebuild,
|
|
446
|
+
wallet: walletObj,
|
|
447
|
+
})
|
|
448
|
+
.should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');
|
|
449
|
+
});
|
|
450
|
+
it('should fail to verify token transaction with different recipient amounts', async function () {
|
|
451
|
+
const txParams = newTxParamsTokenTransfer();
|
|
452
|
+
const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address
|
|
453
|
+
txParams.recipients = [{ address, amount: '2', tokenName: 'tsol:usdt' }];
|
|
454
|
+
const txPrebuild = newTxPrebuildTokenTransfer();
|
|
455
|
+
await basecoin
|
|
456
|
+
.verifyTransaction({
|
|
457
|
+
txParams,
|
|
458
|
+
txPrebuild,
|
|
459
|
+
wallet: walletObj,
|
|
460
|
+
})
|
|
461
|
+
.should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');
|
|
462
|
+
});
|
|
463
|
+
it('should fail to verify token transaction with different native address', async function () {
|
|
464
|
+
const txParams = newTxParamsTokenTransfer();
|
|
465
|
+
const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvX'; // Native SOL address, different than tx recipients
|
|
466
|
+
txParams.recipients = [{ address, amount: '1', tokenName: 'tsol:usdc' }];
|
|
467
|
+
const txPrebuild = newTxPrebuildTokenTransfer();
|
|
468
|
+
await basecoin
|
|
469
|
+
.verifyTransaction({
|
|
470
|
+
txParams,
|
|
471
|
+
txPrebuild,
|
|
472
|
+
wallet: walletObj,
|
|
473
|
+
})
|
|
474
|
+
.should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');
|
|
475
|
+
});
|
|
476
|
+
it('should succeed to verify transactions when recipients has extra data', async function () {
|
|
477
|
+
const txParams = newTxParamsWithExtraData();
|
|
478
|
+
const txPrebuild = newTxPrebuild();
|
|
479
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
480
|
+
txParams,
|
|
481
|
+
txPrebuild,
|
|
482
|
+
memo,
|
|
483
|
+
durableNonce,
|
|
484
|
+
wallet: walletObj,
|
|
485
|
+
});
|
|
486
|
+
validTransaction.should.equal(true);
|
|
487
|
+
});
|
|
488
|
+
it('should verify activate staking transaction', async function () {
|
|
489
|
+
const tx = await factory
|
|
490
|
+
.getStakingActivateBuilder()
|
|
491
|
+
.stakingAddress(stakeAccount.pub)
|
|
492
|
+
.sender(wallet.pub)
|
|
493
|
+
.nonce(blockHash)
|
|
494
|
+
.amount(amount)
|
|
495
|
+
.validator(validator.pub)
|
|
496
|
+
.memo('test memo')
|
|
497
|
+
.fee({ amount: 5000 })
|
|
498
|
+
.build();
|
|
499
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
500
|
+
const txParams = newTxParams();
|
|
501
|
+
const txPrebuild = newTxPrebuild();
|
|
502
|
+
txPrebuild.txBase64 = txToBroadcastFormat;
|
|
503
|
+
txPrebuild.txInfo.nonce = '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen';
|
|
504
|
+
txParams.recipients = [
|
|
505
|
+
{
|
|
506
|
+
address: '7dRuGFbU2y2kijP6o1LYNzVyz4yf13MooqoionCzv5Za',
|
|
507
|
+
amount: amount,
|
|
508
|
+
},
|
|
509
|
+
];
|
|
510
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
511
|
+
txParams,
|
|
512
|
+
txPrebuild,
|
|
513
|
+
memo,
|
|
514
|
+
wallet: walletObj,
|
|
515
|
+
});
|
|
516
|
+
validTransaction.should.equal(true);
|
|
517
|
+
});
|
|
518
|
+
it('should verify withdraw staking transaction', async function () {
|
|
519
|
+
const tx = await factory
|
|
520
|
+
.getStakingWithdrawBuilder()
|
|
521
|
+
.stakingAddress(stakeAccount.pub)
|
|
522
|
+
.sender(wallet.pub)
|
|
523
|
+
.nonce(blockHash)
|
|
524
|
+
.amount(amount)
|
|
525
|
+
.memo('test memo')
|
|
526
|
+
.fee({ amount: 5000 })
|
|
527
|
+
.build();
|
|
528
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
529
|
+
const txParams = newTxParams();
|
|
530
|
+
const txPrebuild = newTxPrebuild();
|
|
531
|
+
txPrebuild.txBase64 = txToBroadcastFormat;
|
|
532
|
+
txPrebuild.txInfo.nonce = '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen';
|
|
533
|
+
txParams.recipients = [
|
|
534
|
+
{
|
|
535
|
+
address: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
536
|
+
amount: amount,
|
|
537
|
+
},
|
|
538
|
+
];
|
|
539
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
540
|
+
txParams,
|
|
541
|
+
txPrebuild,
|
|
542
|
+
memo,
|
|
543
|
+
wallet: walletObj,
|
|
544
|
+
});
|
|
545
|
+
validTransaction.should.equal(true);
|
|
546
|
+
});
|
|
547
|
+
it('should verify deactivate staking transaction', async function () {
|
|
548
|
+
const tx = await factory
|
|
549
|
+
.getStakingDeactivateBuilder()
|
|
550
|
+
.stakingAddress(stakeAccount.pub)
|
|
551
|
+
.sender(wallet.pub)
|
|
552
|
+
.nonce(blockHash)
|
|
553
|
+
.memo('test memo')
|
|
554
|
+
.fee({ amount: 5000 })
|
|
555
|
+
.build();
|
|
556
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
557
|
+
const txParams = newTxParams();
|
|
558
|
+
const txPrebuild = newTxPrebuild();
|
|
559
|
+
txPrebuild.txBase64 = txToBroadcastFormat;
|
|
560
|
+
txPrebuild.txInfo.nonce = '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen';
|
|
561
|
+
txParams.recipients = [];
|
|
562
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
563
|
+
txParams,
|
|
564
|
+
txPrebuild,
|
|
565
|
+
memo,
|
|
566
|
+
wallet: walletObj,
|
|
567
|
+
});
|
|
568
|
+
validTransaction.should.equal(true);
|
|
569
|
+
});
|
|
570
|
+
it('should verify create associated token account transaction', async function () {
|
|
571
|
+
const tx = await factory
|
|
572
|
+
.getAtaInitializationBuilder()
|
|
573
|
+
.mint('tsol:usdc')
|
|
574
|
+
.sender(wallet.pub)
|
|
575
|
+
.nonce(blockHash)
|
|
576
|
+
.memo('test memo')
|
|
577
|
+
.fee({ amount: 5000 })
|
|
578
|
+
.build();
|
|
579
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
580
|
+
const txParams = newTxParams();
|
|
581
|
+
const txPrebuild = newTxPrebuild();
|
|
582
|
+
txPrebuild.txBase64 = txToBroadcastFormat;
|
|
583
|
+
txPrebuild.txInfo.nonce = blockHash;
|
|
584
|
+
txParams.recipients = [];
|
|
585
|
+
const validTransaction = await basecoin.verifyTransaction({
|
|
586
|
+
txParams,
|
|
587
|
+
txPrebuild,
|
|
588
|
+
memo,
|
|
589
|
+
wallet: walletObj,
|
|
590
|
+
});
|
|
591
|
+
validTransaction.should.equal(true);
|
|
592
|
+
});
|
|
593
|
+
});
|
|
594
|
+
it('should accept valid address', function () {
|
|
595
|
+
goodAddresses.forEach((addr) => {
|
|
596
|
+
basecoin.isValidAddress(addr).should.equal(true);
|
|
597
|
+
});
|
|
598
|
+
});
|
|
599
|
+
it('should reject invalid address', function () {
|
|
600
|
+
badAddresses.forEach((addr) => {
|
|
601
|
+
basecoin.isValidAddress(addr).should.equal(false);
|
|
602
|
+
});
|
|
603
|
+
});
|
|
604
|
+
it('should check valid pub keys', function () {
|
|
605
|
+
keyPair.should.have.property('pub');
|
|
606
|
+
basecoin.isValidPub(keyPair.pub).should.equal(true);
|
|
607
|
+
});
|
|
608
|
+
it('should check an invalid pub keys', function () {
|
|
609
|
+
const badPubKey = keyPair.pub.slice(0, keyPair.pub.length - 1) + '-';
|
|
610
|
+
basecoin.isValidPub(badPubKey).should.equal(false);
|
|
611
|
+
});
|
|
612
|
+
it('should check valid prv keys', function () {
|
|
613
|
+
keyPair.should.have.property('prv');
|
|
614
|
+
basecoin.isValidPrv(keyPair.prv).should.equal(true);
|
|
615
|
+
});
|
|
616
|
+
it('should check an invalid prv keys', function () {
|
|
617
|
+
const badPrvKey = keyPair.prv ? keyPair.prv.slice(0, keyPair.prv.length - 1) + '-' : undefined;
|
|
618
|
+
basecoin.isValidPrv(badPrvKey).should.equal(false);
|
|
619
|
+
});
|
|
620
|
+
describe('Parse Transactions:', () => {
|
|
621
|
+
it('should parse an unsigned transfer transaction', async function () {
|
|
622
|
+
const parsedTransaction = await basecoin.parseTransaction({
|
|
623
|
+
txBase64: testData.rawTransactions.transfer.unsigned,
|
|
624
|
+
feeInfo: {
|
|
625
|
+
fee: '5000',
|
|
626
|
+
},
|
|
627
|
+
});
|
|
628
|
+
parsedTransaction.should.deepEqual({
|
|
629
|
+
inputs: [
|
|
630
|
+
{
|
|
631
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
632
|
+
amount: 305000,
|
|
633
|
+
},
|
|
634
|
+
],
|
|
635
|
+
outputs: [
|
|
636
|
+
{
|
|
637
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
638
|
+
amount: '300000',
|
|
639
|
+
},
|
|
640
|
+
],
|
|
641
|
+
});
|
|
642
|
+
});
|
|
643
|
+
it('should parse a signed transfer transaction', async function () {
|
|
644
|
+
const parsedTransaction = await basecoin.parseTransaction({
|
|
645
|
+
txBase64: testData.rawTransactions.transfer.signed,
|
|
646
|
+
feeInfo: {
|
|
647
|
+
fee: '5000',
|
|
648
|
+
},
|
|
649
|
+
});
|
|
650
|
+
parsedTransaction.should.deepEqual({
|
|
651
|
+
inputs: [
|
|
652
|
+
{
|
|
653
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
654
|
+
amount: 305000,
|
|
655
|
+
},
|
|
656
|
+
],
|
|
657
|
+
outputs: [
|
|
658
|
+
{
|
|
659
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
660
|
+
amount: '300000',
|
|
661
|
+
},
|
|
662
|
+
],
|
|
663
|
+
});
|
|
664
|
+
});
|
|
665
|
+
it('should parse an unsigned wallet init transaction', async function () {
|
|
666
|
+
const parsedTransaction = await basecoin.parseTransaction({
|
|
667
|
+
txBase64: testData.rawTransactions.walletInit.unsigned,
|
|
668
|
+
feeInfo: {
|
|
669
|
+
fee: '5000',
|
|
670
|
+
},
|
|
671
|
+
});
|
|
672
|
+
parsedTransaction.should.deepEqual({
|
|
673
|
+
inputs: [
|
|
674
|
+
{
|
|
675
|
+
address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
676
|
+
amount: 310000,
|
|
677
|
+
},
|
|
678
|
+
],
|
|
679
|
+
outputs: [
|
|
680
|
+
{
|
|
681
|
+
address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
682
|
+
amount: '300000',
|
|
683
|
+
},
|
|
684
|
+
],
|
|
685
|
+
});
|
|
686
|
+
});
|
|
687
|
+
it('should parse a signed wallet init transaction', async function () {
|
|
688
|
+
const parsedTransaction = await basecoin.parseTransaction({
|
|
689
|
+
txBase64: testData.rawTransactions.walletInit.signed,
|
|
690
|
+
feeInfo: {
|
|
691
|
+
fee: '5000',
|
|
692
|
+
},
|
|
693
|
+
});
|
|
694
|
+
parsedTransaction.should.deepEqual({
|
|
695
|
+
inputs: [
|
|
696
|
+
{
|
|
697
|
+
address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
698
|
+
amount: 310000,
|
|
699
|
+
},
|
|
700
|
+
],
|
|
701
|
+
outputs: [
|
|
702
|
+
{
|
|
703
|
+
address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
704
|
+
amount: '300000',
|
|
705
|
+
},
|
|
706
|
+
],
|
|
707
|
+
});
|
|
708
|
+
});
|
|
709
|
+
it('should parse an unsigned transfer token transaction', async function () {
|
|
710
|
+
const parsedTransaction = await basecoin.parseTransaction({
|
|
711
|
+
txBase64: testData.rawTransactions.transferToken.unsigned,
|
|
712
|
+
feeInfo: {
|
|
713
|
+
fee: '5000',
|
|
714
|
+
},
|
|
715
|
+
});
|
|
716
|
+
parsedTransaction.should.deepEqual({
|
|
717
|
+
inputs: [
|
|
718
|
+
{
|
|
719
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
720
|
+
amount: 5000,
|
|
721
|
+
},
|
|
722
|
+
],
|
|
723
|
+
outputs: [
|
|
724
|
+
{
|
|
725
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
726
|
+
amount: '300000',
|
|
727
|
+
tokenName: 'tsol:usdc',
|
|
728
|
+
},
|
|
729
|
+
],
|
|
730
|
+
});
|
|
731
|
+
});
|
|
732
|
+
it('should parse a signed transfer token transaction', async function () {
|
|
733
|
+
const parsedTransaction = await basecoin.parseTransaction({
|
|
734
|
+
txBase64: testData.rawTransactions.transferToken.signed,
|
|
735
|
+
feeInfo: {
|
|
736
|
+
fee: '5000',
|
|
737
|
+
},
|
|
738
|
+
});
|
|
739
|
+
parsedTransaction.should.deepEqual({
|
|
740
|
+
inputs: [
|
|
741
|
+
{
|
|
742
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
743
|
+
amount: 5000,
|
|
744
|
+
},
|
|
745
|
+
],
|
|
746
|
+
outputs: [
|
|
747
|
+
{
|
|
748
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
749
|
+
amount: '300000',
|
|
750
|
+
tokenName: 'tsol:usdc',
|
|
751
|
+
},
|
|
752
|
+
],
|
|
753
|
+
});
|
|
754
|
+
});
|
|
755
|
+
});
|
|
756
|
+
describe('Explain Transactions:', () => {
|
|
757
|
+
it('should explain an unsigned transfer transaction', async function () {
|
|
758
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
759
|
+
txBase64: testData.rawTransactions.transfer.unsigned,
|
|
760
|
+
feeInfo: {
|
|
761
|
+
fee: '5000',
|
|
762
|
+
},
|
|
763
|
+
});
|
|
764
|
+
explainedTransaction.should.deepEqual({
|
|
765
|
+
displayOrder: [
|
|
766
|
+
'id',
|
|
767
|
+
'type',
|
|
768
|
+
'blockhash',
|
|
769
|
+
'durableNonce',
|
|
770
|
+
'outputAmount',
|
|
771
|
+
'changeAmount',
|
|
772
|
+
'outputs',
|
|
773
|
+
'changeOutputs',
|
|
774
|
+
'tokenEnablements',
|
|
775
|
+
'fee',
|
|
776
|
+
'memo',
|
|
777
|
+
],
|
|
778
|
+
id: 'UNAVAILABLE',
|
|
779
|
+
type: 'Send',
|
|
780
|
+
changeOutputs: [],
|
|
781
|
+
changeAmount: '0',
|
|
782
|
+
outputAmount: '300000',
|
|
783
|
+
outputs: [
|
|
784
|
+
{
|
|
785
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
786
|
+
amount: '300000',
|
|
787
|
+
},
|
|
788
|
+
],
|
|
789
|
+
fee: {
|
|
790
|
+
fee: '5000',
|
|
791
|
+
feeRate: 5000,
|
|
792
|
+
},
|
|
793
|
+
memo: 'test memo',
|
|
794
|
+
blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
795
|
+
durableNonce: {
|
|
796
|
+
authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
797
|
+
walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
798
|
+
},
|
|
799
|
+
tokenEnablements: [],
|
|
800
|
+
});
|
|
801
|
+
});
|
|
802
|
+
it('should explain a signed transfer transaction', async function () {
|
|
803
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
804
|
+
txBase64: testData.rawTransactions.transfer.signed,
|
|
805
|
+
feeInfo: {
|
|
806
|
+
fee: '5000',
|
|
807
|
+
},
|
|
808
|
+
});
|
|
809
|
+
explainedTransaction.should.deepEqual({
|
|
810
|
+
displayOrder: [
|
|
811
|
+
'id',
|
|
812
|
+
'type',
|
|
813
|
+
'blockhash',
|
|
814
|
+
'durableNonce',
|
|
815
|
+
'outputAmount',
|
|
816
|
+
'changeAmount',
|
|
817
|
+
'outputs',
|
|
818
|
+
'changeOutputs',
|
|
819
|
+
'tokenEnablements',
|
|
820
|
+
'fee',
|
|
821
|
+
'memo',
|
|
822
|
+
],
|
|
823
|
+
id: '2XFxGfXddKWnqGaMAsfNL8HgXqDvjBL2Ae28KWrRvg9bQBmCrpHYVDacuZFeAUyYwjXG6ey2jTARX5VQCnj7SF4L',
|
|
824
|
+
type: 'Send',
|
|
825
|
+
changeOutputs: [],
|
|
826
|
+
changeAmount: '0',
|
|
827
|
+
outputAmount: '300000',
|
|
828
|
+
outputs: [
|
|
829
|
+
{
|
|
830
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
831
|
+
amount: '300000',
|
|
832
|
+
},
|
|
833
|
+
],
|
|
834
|
+
fee: {
|
|
835
|
+
fee: '5000',
|
|
836
|
+
feeRate: 5000,
|
|
837
|
+
},
|
|
838
|
+
memo: 'test memo',
|
|
839
|
+
blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
840
|
+
durableNonce: {
|
|
841
|
+
authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
842
|
+
walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
843
|
+
},
|
|
844
|
+
tokenEnablements: [],
|
|
845
|
+
});
|
|
846
|
+
});
|
|
847
|
+
it('should explain an unsigned wallet init transaction', async function () {
|
|
848
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
849
|
+
txBase64: testData.rawTransactions.walletInit.unsigned,
|
|
850
|
+
feeInfo: {
|
|
851
|
+
fee: '5000',
|
|
852
|
+
},
|
|
853
|
+
});
|
|
854
|
+
explainedTransaction.should.deepEqual({
|
|
855
|
+
displayOrder: [
|
|
856
|
+
'id',
|
|
857
|
+
'type',
|
|
858
|
+
'blockhash',
|
|
859
|
+
'durableNonce',
|
|
860
|
+
'outputAmount',
|
|
861
|
+
'changeAmount',
|
|
862
|
+
'outputs',
|
|
863
|
+
'changeOutputs',
|
|
864
|
+
'tokenEnablements',
|
|
865
|
+
'fee',
|
|
866
|
+
'memo',
|
|
867
|
+
],
|
|
868
|
+
id: 'UNAVAILABLE',
|
|
869
|
+
type: 'WalletInitialization',
|
|
870
|
+
changeOutputs: [],
|
|
871
|
+
changeAmount: '0',
|
|
872
|
+
outputAmount: '300000',
|
|
873
|
+
outputs: [
|
|
874
|
+
{
|
|
875
|
+
address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
876
|
+
amount: '300000',
|
|
877
|
+
},
|
|
878
|
+
],
|
|
879
|
+
fee: {
|
|
880
|
+
fee: '10000',
|
|
881
|
+
feeRate: 5000,
|
|
882
|
+
},
|
|
883
|
+
blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
884
|
+
durableNonce: undefined,
|
|
885
|
+
memo: undefined,
|
|
886
|
+
tokenEnablements: [],
|
|
887
|
+
});
|
|
888
|
+
});
|
|
889
|
+
it('should explain a signed wallet init transaction', async function () {
|
|
890
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
891
|
+
txBase64: testData.rawTransactions.walletInit.signed,
|
|
892
|
+
feeInfo: {
|
|
893
|
+
fee: '5000',
|
|
894
|
+
},
|
|
895
|
+
});
|
|
896
|
+
explainedTransaction.should.deepEqual({
|
|
897
|
+
displayOrder: [
|
|
898
|
+
'id',
|
|
899
|
+
'type',
|
|
900
|
+
'blockhash',
|
|
901
|
+
'durableNonce',
|
|
902
|
+
'outputAmount',
|
|
903
|
+
'changeAmount',
|
|
904
|
+
'outputs',
|
|
905
|
+
'changeOutputs',
|
|
906
|
+
'tokenEnablements',
|
|
907
|
+
'fee',
|
|
908
|
+
'memo',
|
|
909
|
+
],
|
|
910
|
+
id: '7TkU8wLgXDeLFbVydtg6mqMsp9GatsetitSngysgjxFhofKSUcLPBoKPHciLeGEfJFMsqezpZmGRSFQTBy7ZDsg',
|
|
911
|
+
type: 'WalletInitialization',
|
|
912
|
+
changeOutputs: [],
|
|
913
|
+
changeAmount: '0',
|
|
914
|
+
outputAmount: '300000',
|
|
915
|
+
outputs: [
|
|
916
|
+
{
|
|
917
|
+
address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
918
|
+
amount: '300000',
|
|
919
|
+
},
|
|
920
|
+
],
|
|
921
|
+
fee: {
|
|
922
|
+
fee: '10000',
|
|
923
|
+
feeRate: 5000,
|
|
924
|
+
},
|
|
925
|
+
blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
926
|
+
durableNonce: undefined,
|
|
927
|
+
memo: undefined,
|
|
928
|
+
tokenEnablements: [],
|
|
929
|
+
});
|
|
930
|
+
});
|
|
931
|
+
it('should explain an unsigned token transfer transaction', async function () {
|
|
932
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
933
|
+
txBase64: testData.rawTransactions.transferToken.unsigned,
|
|
934
|
+
feeInfo: {
|
|
935
|
+
fee: '5000',
|
|
936
|
+
},
|
|
937
|
+
});
|
|
938
|
+
explainedTransaction.should.deepEqual({
|
|
939
|
+
displayOrder: [
|
|
940
|
+
'id',
|
|
941
|
+
'type',
|
|
942
|
+
'blockhash',
|
|
943
|
+
'durableNonce',
|
|
944
|
+
'outputAmount',
|
|
945
|
+
'changeAmount',
|
|
946
|
+
'outputs',
|
|
947
|
+
'changeOutputs',
|
|
948
|
+
'tokenEnablements',
|
|
949
|
+
'fee',
|
|
950
|
+
'memo',
|
|
951
|
+
],
|
|
952
|
+
id: 'UNAVAILABLE',
|
|
953
|
+
type: 'Send',
|
|
954
|
+
changeOutputs: [],
|
|
955
|
+
changeAmount: '0',
|
|
956
|
+
outputAmount: '0',
|
|
957
|
+
outputs: [
|
|
958
|
+
{
|
|
959
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
960
|
+
amount: '300000',
|
|
961
|
+
tokenName: 'tsol:usdc',
|
|
962
|
+
},
|
|
963
|
+
],
|
|
964
|
+
fee: {
|
|
965
|
+
fee: '5000',
|
|
966
|
+
feeRate: 5000,
|
|
967
|
+
},
|
|
968
|
+
memo: 'test memo',
|
|
969
|
+
blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
970
|
+
durableNonce: {
|
|
971
|
+
authWalletAddress: '12f6D3WubGVeQoH2m8kTvvcrasWdXWwtVzUCyRNDZxA2',
|
|
972
|
+
walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
973
|
+
},
|
|
974
|
+
tokenEnablements: [],
|
|
975
|
+
});
|
|
976
|
+
});
|
|
977
|
+
it('should explain a signed token transfer transaction', async function () {
|
|
978
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
979
|
+
txBase64: testData.rawTransactions.transferToken.signed,
|
|
980
|
+
feeInfo: {
|
|
981
|
+
fee: '5000',
|
|
982
|
+
},
|
|
983
|
+
});
|
|
984
|
+
explainedTransaction.should.deepEqual({
|
|
985
|
+
displayOrder: [
|
|
986
|
+
'id',
|
|
987
|
+
'type',
|
|
988
|
+
'blockhash',
|
|
989
|
+
'durableNonce',
|
|
990
|
+
'outputAmount',
|
|
991
|
+
'changeAmount',
|
|
992
|
+
'outputs',
|
|
993
|
+
'changeOutputs',
|
|
994
|
+
'tokenEnablements',
|
|
995
|
+
'fee',
|
|
996
|
+
'memo',
|
|
997
|
+
],
|
|
998
|
+
id: '2ticU4ZkEqdTHULr6LobTgWBhim6E7wSscDhM4gzyuGUmQyUwLYhoqaifuvwmNzzEf1T5aefVcgMQkSHdJ5nsrfZ',
|
|
999
|
+
type: 'Send',
|
|
1000
|
+
changeOutputs: [],
|
|
1001
|
+
changeAmount: '0',
|
|
1002
|
+
outputAmount: '0',
|
|
1003
|
+
outputs: [
|
|
1004
|
+
{
|
|
1005
|
+
address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',
|
|
1006
|
+
amount: '300000',
|
|
1007
|
+
tokenName: 'tsol:usdc',
|
|
1008
|
+
},
|
|
1009
|
+
],
|
|
1010
|
+
fee: {
|
|
1011
|
+
fee: '5000',
|
|
1012
|
+
feeRate: 5000,
|
|
1013
|
+
},
|
|
1014
|
+
memo: 'test memo',
|
|
1015
|
+
blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',
|
|
1016
|
+
durableNonce: {
|
|
1017
|
+
authWalletAddress: '12f6D3WubGVeQoH2m8kTvvcrasWdXWwtVzUCyRNDZxA2',
|
|
1018
|
+
walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',
|
|
1019
|
+
},
|
|
1020
|
+
tokenEnablements: [],
|
|
1021
|
+
});
|
|
1022
|
+
});
|
|
1023
|
+
it('should explain activate staking transaction', async function () {
|
|
1024
|
+
const tx = await factory
|
|
1025
|
+
.getStakingActivateBuilder()
|
|
1026
|
+
.stakingAddress(stakeAccount.pub)
|
|
1027
|
+
.sender(wallet.pub)
|
|
1028
|
+
.nonce(blockHash)
|
|
1029
|
+
.amount(amount)
|
|
1030
|
+
.validator(validator.pub)
|
|
1031
|
+
.memo('test memo')
|
|
1032
|
+
.fee({ amount: 5000 })
|
|
1033
|
+
.build();
|
|
1034
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
1035
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
1036
|
+
txBase64: txToBroadcastFormat,
|
|
1037
|
+
feeInfo: {
|
|
1038
|
+
fee: '5000',
|
|
1039
|
+
},
|
|
1040
|
+
});
|
|
1041
|
+
explainedTransaction.should.deepEqual({
|
|
1042
|
+
displayOrder: [
|
|
1043
|
+
'id',
|
|
1044
|
+
'type',
|
|
1045
|
+
'blockhash',
|
|
1046
|
+
'durableNonce',
|
|
1047
|
+
'outputAmount',
|
|
1048
|
+
'changeAmount',
|
|
1049
|
+
'outputs',
|
|
1050
|
+
'changeOutputs',
|
|
1051
|
+
'tokenEnablements',
|
|
1052
|
+
'fee',
|
|
1053
|
+
'memo',
|
|
1054
|
+
],
|
|
1055
|
+
id: 'UNAVAILABLE',
|
|
1056
|
+
type: 'StakingActivate',
|
|
1057
|
+
changeOutputs: [],
|
|
1058
|
+
changeAmount: '0',
|
|
1059
|
+
outputAmount: '10000',
|
|
1060
|
+
outputs: [
|
|
1061
|
+
{
|
|
1062
|
+
address: '7dRuGFbU2y2kijP6o1LYNzVyz4yf13MooqoionCzv5Za',
|
|
1063
|
+
amount: '10000',
|
|
1064
|
+
},
|
|
1065
|
+
],
|
|
1066
|
+
fee: {
|
|
1067
|
+
fee: '10000',
|
|
1068
|
+
feeRate: 5000,
|
|
1069
|
+
},
|
|
1070
|
+
memo: 'test memo',
|
|
1071
|
+
blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',
|
|
1072
|
+
durableNonce: undefined,
|
|
1073
|
+
tokenEnablements: [],
|
|
1074
|
+
});
|
|
1075
|
+
});
|
|
1076
|
+
it('should explain deactivate staking transaction', async function () {
|
|
1077
|
+
const tx = await factory
|
|
1078
|
+
.getStakingDeactivateBuilder()
|
|
1079
|
+
.stakingAddress(stakeAccount.pub)
|
|
1080
|
+
.sender(wallet.pub)
|
|
1081
|
+
.nonce(blockHash)
|
|
1082
|
+
.memo('test memo')
|
|
1083
|
+
.fee({ amount: 5000 })
|
|
1084
|
+
.build();
|
|
1085
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
1086
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
1087
|
+
txBase64: txToBroadcastFormat,
|
|
1088
|
+
feeInfo: {
|
|
1089
|
+
fee: '5000',
|
|
1090
|
+
},
|
|
1091
|
+
});
|
|
1092
|
+
explainedTransaction.should.deepEqual({
|
|
1093
|
+
displayOrder: [
|
|
1094
|
+
'id',
|
|
1095
|
+
'type',
|
|
1096
|
+
'blockhash',
|
|
1097
|
+
'durableNonce',
|
|
1098
|
+
'outputAmount',
|
|
1099
|
+
'changeAmount',
|
|
1100
|
+
'outputs',
|
|
1101
|
+
'changeOutputs',
|
|
1102
|
+
'tokenEnablements',
|
|
1103
|
+
'fee',
|
|
1104
|
+
'memo',
|
|
1105
|
+
],
|
|
1106
|
+
id: 'UNAVAILABLE',
|
|
1107
|
+
type: 'StakingDeactivate',
|
|
1108
|
+
changeOutputs: [],
|
|
1109
|
+
changeAmount: '0',
|
|
1110
|
+
outputAmount: '0',
|
|
1111
|
+
outputs: [],
|
|
1112
|
+
fee: {
|
|
1113
|
+
fee: '5000',
|
|
1114
|
+
feeRate: 5000,
|
|
1115
|
+
},
|
|
1116
|
+
memo: 'test memo',
|
|
1117
|
+
blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',
|
|
1118
|
+
durableNonce: undefined,
|
|
1119
|
+
tokenEnablements: [],
|
|
1120
|
+
});
|
|
1121
|
+
});
|
|
1122
|
+
it('should explain withdraw staking transaction', async function () {
|
|
1123
|
+
const tx = await factory
|
|
1124
|
+
.getStakingWithdrawBuilder()
|
|
1125
|
+
.stakingAddress(stakeAccount.pub)
|
|
1126
|
+
.sender(wallet.pub)
|
|
1127
|
+
.nonce(blockHash)
|
|
1128
|
+
.amount(amount)
|
|
1129
|
+
.memo('test memo')
|
|
1130
|
+
.fee({ amount: 5000 })
|
|
1131
|
+
.build();
|
|
1132
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
1133
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
1134
|
+
txBase64: txToBroadcastFormat,
|
|
1135
|
+
feeInfo: {
|
|
1136
|
+
fee: '5000',
|
|
1137
|
+
},
|
|
1138
|
+
});
|
|
1139
|
+
explainedTransaction.should.deepEqual({
|
|
1140
|
+
displayOrder: [
|
|
1141
|
+
'id',
|
|
1142
|
+
'type',
|
|
1143
|
+
'blockhash',
|
|
1144
|
+
'durableNonce',
|
|
1145
|
+
'outputAmount',
|
|
1146
|
+
'changeAmount',
|
|
1147
|
+
'outputs',
|
|
1148
|
+
'changeOutputs',
|
|
1149
|
+
'tokenEnablements',
|
|
1150
|
+
'fee',
|
|
1151
|
+
'memo',
|
|
1152
|
+
],
|
|
1153
|
+
id: 'UNAVAILABLE',
|
|
1154
|
+
type: 'StakingWithdraw',
|
|
1155
|
+
changeOutputs: [],
|
|
1156
|
+
changeAmount: '0',
|
|
1157
|
+
outputAmount: '10000',
|
|
1158
|
+
outputs: [
|
|
1159
|
+
{
|
|
1160
|
+
address: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',
|
|
1161
|
+
amount: '10000',
|
|
1162
|
+
},
|
|
1163
|
+
],
|
|
1164
|
+
fee: {
|
|
1165
|
+
fee: '5000',
|
|
1166
|
+
feeRate: 5000,
|
|
1167
|
+
},
|
|
1168
|
+
memo: 'test memo',
|
|
1169
|
+
blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',
|
|
1170
|
+
durableNonce: undefined,
|
|
1171
|
+
tokenEnablements: [],
|
|
1172
|
+
});
|
|
1173
|
+
});
|
|
1174
|
+
it('should explain create ATA transaction', async function () {
|
|
1175
|
+
const tokenName = 'tsol:usdc';
|
|
1176
|
+
const rentExemptAmount = '3000000';
|
|
1177
|
+
const tx = await factory
|
|
1178
|
+
.getAtaInitializationBuilder()
|
|
1179
|
+
.sender(wallet.pub)
|
|
1180
|
+
.nonce(blockHash)
|
|
1181
|
+
.mint(tokenName)
|
|
1182
|
+
.rentExemptAmount(rentExemptAmount)
|
|
1183
|
+
.memo('test memo')
|
|
1184
|
+
.fee({ amount: 5000 })
|
|
1185
|
+
.build();
|
|
1186
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
1187
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
1188
|
+
txBase64: txToBroadcastFormat,
|
|
1189
|
+
feeInfo: {
|
|
1190
|
+
fee: '5000',
|
|
1191
|
+
},
|
|
1192
|
+
tokenAccountRentExemptAmount: rentExemptAmount,
|
|
1193
|
+
});
|
|
1194
|
+
explainedTransaction.should.deepEqual({
|
|
1195
|
+
displayOrder: [
|
|
1196
|
+
'id',
|
|
1197
|
+
'type',
|
|
1198
|
+
'blockhash',
|
|
1199
|
+
'durableNonce',
|
|
1200
|
+
'outputAmount',
|
|
1201
|
+
'changeAmount',
|
|
1202
|
+
'outputs',
|
|
1203
|
+
'changeOutputs',
|
|
1204
|
+
'tokenEnablements',
|
|
1205
|
+
'fee',
|
|
1206
|
+
'memo',
|
|
1207
|
+
],
|
|
1208
|
+
id: 'UNAVAILABLE',
|
|
1209
|
+
type: 'AssociatedTokenAccountInitialization',
|
|
1210
|
+
changeOutputs: [],
|
|
1211
|
+
changeAmount: '0',
|
|
1212
|
+
outputAmount: '0',
|
|
1213
|
+
outputs: [],
|
|
1214
|
+
fee: {
|
|
1215
|
+
fee: '3005000',
|
|
1216
|
+
feeRate: 5000,
|
|
1217
|
+
},
|
|
1218
|
+
memo: 'test memo',
|
|
1219
|
+
blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',
|
|
1220
|
+
durableNonce: undefined,
|
|
1221
|
+
tokenEnablements: [
|
|
1222
|
+
{
|
|
1223
|
+
address: '141BFNem3pknc8CzPVLv1Ri3btgKdCsafYP5nXwmXfxU',
|
|
1224
|
+
tokenAddress: 'F4uLeXJoFz3hw13MposuwaQbMcZbCjqvEGPPeRRB1Byf',
|
|
1225
|
+
tokenName: 'tsol:usdc',
|
|
1226
|
+
},
|
|
1227
|
+
],
|
|
1228
|
+
});
|
|
1229
|
+
});
|
|
1230
|
+
it('should explain create multi ATA transaction', async function () {
|
|
1231
|
+
const recipients = [
|
|
1232
|
+
{
|
|
1233
|
+
ownerAddress: wallet.pub,
|
|
1234
|
+
tokenName: 'tsol:usdc',
|
|
1235
|
+
},
|
|
1236
|
+
{
|
|
1237
|
+
ownerAddress: durableNonce.walletNonceAddress,
|
|
1238
|
+
tokenName: 'tsol:ray',
|
|
1239
|
+
},
|
|
1240
|
+
];
|
|
1241
|
+
const rentExemptAmount = '3000000';
|
|
1242
|
+
const tx = await factory
|
|
1243
|
+
.getAtaInitializationBuilder()
|
|
1244
|
+
.sender(wallet.pub)
|
|
1245
|
+
.nonce(blockHash)
|
|
1246
|
+
.enableToken(recipients[0])
|
|
1247
|
+
.enableToken(recipients[1])
|
|
1248
|
+
.rentExemptAmount(rentExemptAmount)
|
|
1249
|
+
.memo('test memo')
|
|
1250
|
+
.fee({ amount: 5000 })
|
|
1251
|
+
.build();
|
|
1252
|
+
const txToBroadcastFormat = tx.toBroadcastFormat();
|
|
1253
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
1254
|
+
txBase64: txToBroadcastFormat,
|
|
1255
|
+
feeInfo: {
|
|
1256
|
+
fee: '5000',
|
|
1257
|
+
},
|
|
1258
|
+
tokenAccountRentExemptAmount: rentExemptAmount,
|
|
1259
|
+
});
|
|
1260
|
+
explainedTransaction.should.deepEqual({
|
|
1261
|
+
displayOrder: [
|
|
1262
|
+
'id',
|
|
1263
|
+
'type',
|
|
1264
|
+
'blockhash',
|
|
1265
|
+
'durableNonce',
|
|
1266
|
+
'outputAmount',
|
|
1267
|
+
'changeAmount',
|
|
1268
|
+
'outputs',
|
|
1269
|
+
'changeOutputs',
|
|
1270
|
+
'tokenEnablements',
|
|
1271
|
+
'fee',
|
|
1272
|
+
'memo',
|
|
1273
|
+
],
|
|
1274
|
+
id: 'UNAVAILABLE',
|
|
1275
|
+
type: 'AssociatedTokenAccountInitialization',
|
|
1276
|
+
changeOutputs: [],
|
|
1277
|
+
changeAmount: '0',
|
|
1278
|
+
outputAmount: '0',
|
|
1279
|
+
outputs: [],
|
|
1280
|
+
fee: {
|
|
1281
|
+
fee: '6005000',
|
|
1282
|
+
feeRate: 5000,
|
|
1283
|
+
},
|
|
1284
|
+
memo: 'test memo',
|
|
1285
|
+
blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',
|
|
1286
|
+
durableNonce: undefined,
|
|
1287
|
+
tokenEnablements: [
|
|
1288
|
+
{
|
|
1289
|
+
address: '141BFNem3pknc8CzPVLv1Ri3btgKdCsafYP5nXwmXfxU',
|
|
1290
|
+
tokenAddress: 'F4uLeXJoFz3hw13MposuwaQbMcZbCjqvEGPPeRRB1Byf',
|
|
1291
|
+
tokenName: 'tsol:usdc',
|
|
1292
|
+
},
|
|
1293
|
+
{
|
|
1294
|
+
address: '9KaLinZFNW5chL4J8UoKnTECppWVMz3ewgx4FAkxUDcf',
|
|
1295
|
+
tokenAddress: '9kLJoGbMgSteptkhKKuh7ken4JEvHrT83157ezEGrZ7R',
|
|
1296
|
+
tokenName: 'tsol:ray',
|
|
1297
|
+
},
|
|
1298
|
+
],
|
|
1299
|
+
});
|
|
1300
|
+
});
|
|
1301
|
+
it('should explain an unsigned token transfer with ATA creation transaction', async function () {
|
|
1302
|
+
const explainedTransaction = await basecoin.explainTransaction({
|
|
1303
|
+
txBase64: testData.rawTransactions.tokenTransferWithAtaCreation.unsigned,
|
|
1304
|
+
feeInfo: {
|
|
1305
|
+
fee: '5000',
|
|
1306
|
+
},
|
|
1307
|
+
tokenAccountRentExemptAmount: '3000000',
|
|
1308
|
+
});
|
|
1309
|
+
explainedTransaction.should.deepEqual({
|
|
1310
|
+
displayOrder: [
|
|
1311
|
+
'id',
|
|
1312
|
+
'type',
|
|
1313
|
+
'blockhash',
|
|
1314
|
+
'durableNonce',
|
|
1315
|
+
'outputAmount',
|
|
1316
|
+
'changeAmount',
|
|
1317
|
+
'outputs',
|
|
1318
|
+
'changeOutputs',
|
|
1319
|
+
'tokenEnablements',
|
|
1320
|
+
'fee',
|
|
1321
|
+
'memo',
|
|
1322
|
+
],
|
|
1323
|
+
id: 'UNAVAILABLE',
|
|
1324
|
+
type: 'Send',
|
|
1325
|
+
changeOutputs: [],
|
|
1326
|
+
changeAmount: '0',
|
|
1327
|
+
outputAmount: '0',
|
|
1328
|
+
outputs: [
|
|
1329
|
+
{
|
|
1330
|
+
address: '2eKjVtzV3oPTXFdtRSDj3Em9k1MV7k8WjKkBszQUwizS',
|
|
1331
|
+
amount: '10000',
|
|
1332
|
+
tokenName: 'tsol:usdc',
|
|
1333
|
+
},
|
|
1334
|
+
],
|
|
1335
|
+
fee: { fee: '3005000', feeRate: 5000 },
|
|
1336
|
+
memo: undefined,
|
|
1337
|
+
blockhash: '27E3MXFvXMUNYeMJeX1pAbERGsJfUbkaZTfgMgpmNN5g',
|
|
1338
|
+
durableNonce: undefined,
|
|
1339
|
+
tokenEnablements: [
|
|
1340
|
+
{
|
|
1341
|
+
address: '2eKjVtzV3oPTXFdtRSDj3Em9k1MV7k8WjKkBszQUwizS',
|
|
1342
|
+
tokenAddress: 'F4uLeXJoFz3hw13MposuwaQbMcZbCjqvEGPPeRRB1Byf',
|
|
1343
|
+
tokenName: 'tsol:usdc',
|
|
1344
|
+
},
|
|
1345
|
+
],
|
|
1346
|
+
});
|
|
1347
|
+
});
|
|
1348
|
+
});
|
|
1349
|
+
describe('Keypair:', () => {
|
|
1350
|
+
it('should generate a keypair from random seed', function () {
|
|
1351
|
+
should.throws(() => basecoin.generateKeyPair('placeholder'), 'generateKeyPair method not implemented');
|
|
1352
|
+
});
|
|
1353
|
+
it('should generate a keypair from a seed', function () {
|
|
1354
|
+
should.throws(() => basecoin.generateKeyPair('placeholder'), 'generateKeyPair method not implemented');
|
|
1355
|
+
});
|
|
1356
|
+
});
|
|
1357
|
+
describe('Sign transaction:', () => {
|
|
1358
|
+
it('should sign transaction', async function () {
|
|
1359
|
+
const signed = await basecoin.signTransaction({
|
|
1360
|
+
txPrebuild: {
|
|
1361
|
+
txBase64: resources.RAW_TX_UNSIGNED,
|
|
1362
|
+
keys: [resources.accountWithSeed.publicKey.toString()],
|
|
1363
|
+
},
|
|
1364
|
+
prv: resources.accountWithSeed.privateKey.base58,
|
|
1365
|
+
});
|
|
1366
|
+
signed.txHex.should.equal(resources.RAW_TX_SIGNED);
|
|
1367
|
+
});
|
|
1368
|
+
it('should handle txHex and txBase64 interchangeably', async function () {
|
|
1369
|
+
const signed = await basecoin.signTransaction({
|
|
1370
|
+
txPrebuild: {
|
|
1371
|
+
txHex: resources.RAW_TX_UNSIGNED,
|
|
1372
|
+
keys: [resources.accountWithSeed.publicKey.toString()],
|
|
1373
|
+
},
|
|
1374
|
+
prv: resources.accountWithSeed.privateKey.base58,
|
|
1375
|
+
});
|
|
1376
|
+
signed.txHex.should.equal(resources.RAW_TX_SIGNED);
|
|
1377
|
+
});
|
|
1378
|
+
it('should throw invalid transaction when sign with public key', async function () {
|
|
1379
|
+
await basecoin
|
|
1380
|
+
.signTransaction({
|
|
1381
|
+
txPrebuild: {
|
|
1382
|
+
txBase64: resources.RAW_TX_UNSIGNED,
|
|
1383
|
+
keys: [resources.accountWithSeed.publicKey.toString()],
|
|
1384
|
+
},
|
|
1385
|
+
prv: resources.accountWithSeed.publicKey,
|
|
1386
|
+
})
|
|
1387
|
+
.should.be.rejectedWith('Invalid key');
|
|
1388
|
+
});
|
|
1389
|
+
});
|
|
1390
|
+
describe('Sign message', () => {
|
|
1391
|
+
it('should sign message', async function () {
|
|
1392
|
+
const signed = await basecoin.signMessage(keypair, 'signed message');
|
|
1393
|
+
signed
|
|
1394
|
+
.toString('base64')
|
|
1395
|
+
.should.equal('s+7d/8aW/twfM/0wLSKOGxd9+LhDIiz/g0FfJ39ylJhQIkjK0RYPm/Y+gdeJ5DIy6K6h6gCXXESDomlv12DBBQ==');
|
|
1396
|
+
});
|
|
1397
|
+
it('shouldnt sign message when message is undefined', async function () {
|
|
1398
|
+
await basecoin
|
|
1399
|
+
.signMessage(keypair, undefined)
|
|
1400
|
+
.should.be.rejectedWith('The first argument must be of type string or an instance of Buffer, ArrayBuffer, or Array or an Array-like Object. Received undefined');
|
|
1401
|
+
});
|
|
1402
|
+
});
|
|
1403
|
+
describe('Get Signing Payload', () => {
|
|
1404
|
+
it('should return a valid signing payload', async function () {
|
|
1405
|
+
const factory = (0, getBuilderFactory_1.getBuilderFactory)(basecoin.getChain());
|
|
1406
|
+
const rebuiltSignablePayload = (await factory.from(resources.TRANSFER_UNSIGNED_TX_WITH_MEMO).build())
|
|
1407
|
+
.signablePayload;
|
|
1408
|
+
const signingPayload = await basecoin.getSignablePayload(resources.TRANSFER_UNSIGNED_TX_WITH_MEMO);
|
|
1409
|
+
signingPayload.should.be.deepEqual(rebuiltSignablePayload);
|
|
1410
|
+
});
|
|
1411
|
+
it('should build CloseAssociatedTokenAccount txn builder from raw txn', async function () {
|
|
1412
|
+
const factory = (0, getBuilderFactory_1.getBuilderFactory)(basecoin.getChain());
|
|
1413
|
+
const txnBuilder = factory.from(resources.TRANSFER_UNSIGNED_TX_CLOSE_ATA);
|
|
1414
|
+
assert_1.default.ok(txnBuilder);
|
|
1415
|
+
});
|
|
1416
|
+
});
|
|
1417
|
+
describe('Presign transaction', () => {
|
|
1418
|
+
const txRequestId = 'txRequestId';
|
|
1419
|
+
let sandbox;
|
|
1420
|
+
beforeEach(() => {
|
|
1421
|
+
sandbox = sinon.createSandbox();
|
|
1422
|
+
});
|
|
1423
|
+
afterEach(() => {
|
|
1424
|
+
sandbox.verifyAndRestore();
|
|
1425
|
+
});
|
|
1426
|
+
it('should rebuild tx request for hot wallets', async () => {
|
|
1427
|
+
const rebuiltTx = {
|
|
1428
|
+
txRequestId,
|
|
1429
|
+
unsignedTxs: [
|
|
1430
|
+
{
|
|
1431
|
+
serializedTxHex: 'deadbeef',
|
|
1432
|
+
signableHex: 'serializedTxHex',
|
|
1433
|
+
derivationPath: 'm/0',
|
|
1434
|
+
},
|
|
1435
|
+
],
|
|
1436
|
+
transactions: [],
|
|
1437
|
+
date: new Date().toISOString(),
|
|
1438
|
+
intent: {
|
|
1439
|
+
intentType: 'payment',
|
|
1440
|
+
},
|
|
1441
|
+
latest: true,
|
|
1442
|
+
state: 'pendingUserSignature',
|
|
1443
|
+
walletType: 'hot',
|
|
1444
|
+
walletId: 'walletId',
|
|
1445
|
+
policiesChecked: true,
|
|
1446
|
+
version: 1,
|
|
1447
|
+
userId: 'userId',
|
|
1448
|
+
};
|
|
1449
|
+
const stubTssUtils = sandbox.createStubInstance(sdk_core_1.TssUtils);
|
|
1450
|
+
stubTssUtils.deleteSignatureShares.resolves([]);
|
|
1451
|
+
stubTssUtils.getTxRequest.resolves(rebuiltTx);
|
|
1452
|
+
const hotWallet = {
|
|
1453
|
+
type: 'hot',
|
|
1454
|
+
};
|
|
1455
|
+
const presignedTransaction = await basecoin.presignTransaction({
|
|
1456
|
+
walletData: hotWallet,
|
|
1457
|
+
tssUtils: stubTssUtils,
|
|
1458
|
+
txPrebuild: {
|
|
1459
|
+
txRequestId,
|
|
1460
|
+
},
|
|
1461
|
+
});
|
|
1462
|
+
presignedTransaction.walletData.should.deepEqual(hotWallet);
|
|
1463
|
+
presignedTransaction.txPrebuild.should.deepEqual(rebuiltTx);
|
|
1464
|
+
presignedTransaction.txHex.should.equal(rebuiltTx.unsignedTxs[0].serializedTxHex);
|
|
1465
|
+
});
|
|
1466
|
+
it('should do nothing for non-hot wallets', async () => {
|
|
1467
|
+
const coldWallet = {
|
|
1468
|
+
type: 'cold',
|
|
1469
|
+
};
|
|
1470
|
+
const presignedTransaction = await basecoin.presignTransaction({
|
|
1471
|
+
walletData: coldWallet,
|
|
1472
|
+
});
|
|
1473
|
+
presignedTransaction.should.deepEqual({
|
|
1474
|
+
walletData: coldWallet,
|
|
1475
|
+
});
|
|
1476
|
+
});
|
|
1477
|
+
it('should error if txRequestId is missing', async () => {
|
|
1478
|
+
const hotWallet = {
|
|
1479
|
+
type: 'hot',
|
|
1480
|
+
};
|
|
1481
|
+
await basecoin
|
|
1482
|
+
.presignTransaction({
|
|
1483
|
+
walletData: hotWallet,
|
|
1484
|
+
txPrebuild: {},
|
|
1485
|
+
})
|
|
1486
|
+
.should.rejectedWith('Missing txRequestId');
|
|
1487
|
+
});
|
|
1488
|
+
});
|
|
1489
|
+
describe('API Key parameter:', () => {
|
|
1490
|
+
// Test the getPublicNodeUrl method directly
|
|
1491
|
+
it('should append apiKey to node URL when provided', function () {
|
|
1492
|
+
// Access the protected method using type casting
|
|
1493
|
+
const url = basecoin.getPublicNodeUrl('test-api-key-123');
|
|
1494
|
+
url.should.equal(`${sdk_core_1.Environments.test.solAlchemyNodeUrl}/test-api-key-123`);
|
|
1495
|
+
});
|
|
1496
|
+
it('should use regular node URL when apiKey is not provided', function () {
|
|
1497
|
+
// Access the protected method using type casting
|
|
1498
|
+
const url = basecoin.getPublicNodeUrl();
|
|
1499
|
+
url.should.equal(sdk_core_1.Environments.test.solNodeUrl);
|
|
1500
|
+
});
|
|
1501
|
+
});
|
|
1502
|
+
describe('Recover Transactions:', () => {
|
|
1503
|
+
const sandBox = sinon.createSandbox();
|
|
1504
|
+
const coin = statics_1.coins.get('tsol');
|
|
1505
|
+
const usdtMintAddress = '9cgpBeNZ2HnLda7NWaaU1i3NyTstk2c4zCMUcoAGsi9C';
|
|
1506
|
+
const t22mintAddress = '5NR1bQwLWqjbkhbQ1hx72HKJybbuvwkDnUZNoAZ2VhW6';
|
|
1507
|
+
let callBack;
|
|
1508
|
+
beforeEach(() => {
|
|
1509
|
+
callBack = sandBox.stub(src_1.Sol.prototype, 'getDataFromNode');
|
|
1510
|
+
callBack
|
|
1511
|
+
.withArgs({
|
|
1512
|
+
payload: {
|
|
1513
|
+
id: '1',
|
|
1514
|
+
jsonrpc: '2.0',
|
|
1515
|
+
method: 'getLatestBlockhash',
|
|
1516
|
+
params: [
|
|
1517
|
+
{
|
|
1518
|
+
commitment: 'finalized',
|
|
1519
|
+
},
|
|
1520
|
+
],
|
|
1521
|
+
},
|
|
1522
|
+
})
|
|
1523
|
+
.resolves(testData.SolResponses.getBlockhashResponse);
|
|
1524
|
+
callBack
|
|
1525
|
+
.withArgs({
|
|
1526
|
+
payload: {
|
|
1527
|
+
id: '1',
|
|
1528
|
+
jsonrpc: '2.0',
|
|
1529
|
+
method: 'getFeeForMessage',
|
|
1530
|
+
params: [
|
|
1531
|
+
sinon.match.string,
|
|
1532
|
+
{
|
|
1533
|
+
commitment: 'finalized',
|
|
1534
|
+
},
|
|
1535
|
+
],
|
|
1536
|
+
},
|
|
1537
|
+
})
|
|
1538
|
+
.resolves(testData.SolResponses.getFeesForMessageResponse);
|
|
1539
|
+
callBack
|
|
1540
|
+
.withArgs({
|
|
1541
|
+
payload: {
|
|
1542
|
+
id: '1',
|
|
1543
|
+
jsonrpc: '2.0',
|
|
1544
|
+
method: 'getMinimumBalanceForRentExemption',
|
|
1545
|
+
params: [165],
|
|
1546
|
+
},
|
|
1547
|
+
})
|
|
1548
|
+
.resolves(testData.SolResponses.getMinimumBalanceForRentExemptionResponse);
|
|
1549
|
+
callBack
|
|
1550
|
+
.withArgs({
|
|
1551
|
+
payload: {
|
|
1552
|
+
id: '1',
|
|
1553
|
+
jsonrpc: '2.0',
|
|
1554
|
+
method: 'getBalance',
|
|
1555
|
+
params: [testData.accountInfo.bs58EncodedPublicKey],
|
|
1556
|
+
},
|
|
1557
|
+
})
|
|
1558
|
+
.resolves(testData.SolResponses.getAccountBalanceResponse);
|
|
1559
|
+
callBack
|
|
1560
|
+
.withArgs({
|
|
1561
|
+
payload: {
|
|
1562
|
+
id: '1',
|
|
1563
|
+
jsonrpc: '2.0',
|
|
1564
|
+
method: 'getBalance',
|
|
1565
|
+
params: [testData.accountInfo.bs58EncodedPublicKeyNoFunds],
|
|
1566
|
+
},
|
|
1567
|
+
})
|
|
1568
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseNoFunds);
|
|
1569
|
+
callBack
|
|
1570
|
+
.withArgs({
|
|
1571
|
+
payload: {
|
|
1572
|
+
id: '1',
|
|
1573
|
+
jsonrpc: '2.0',
|
|
1574
|
+
method: 'getBalance',
|
|
1575
|
+
params: [testData.accountInfo.bs58EncodedPublicKeyM1Derivation],
|
|
1576
|
+
},
|
|
1577
|
+
})
|
|
1578
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseM1Derivation);
|
|
1579
|
+
callBack
|
|
1580
|
+
.withArgs({
|
|
1581
|
+
payload: {
|
|
1582
|
+
id: '1',
|
|
1583
|
+
jsonrpc: '2.0',
|
|
1584
|
+
method: 'getBalance',
|
|
1585
|
+
params: [testData.accountInfo.bs58EncodedPublicKeyM2Derivation],
|
|
1586
|
+
},
|
|
1587
|
+
})
|
|
1588
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);
|
|
1589
|
+
callBack
|
|
1590
|
+
.withArgs({
|
|
1591
|
+
payload: {
|
|
1592
|
+
id: '1',
|
|
1593
|
+
jsonrpc: '2.0',
|
|
1594
|
+
method: 'getBalance',
|
|
1595
|
+
params: [testData.accountInfo.bs58EncodedPublicKeyWithManyTokens],
|
|
1596
|
+
},
|
|
1597
|
+
})
|
|
1598
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);
|
|
1599
|
+
callBack
|
|
1600
|
+
.withArgs({
|
|
1601
|
+
payload: {
|
|
1602
|
+
id: '1',
|
|
1603
|
+
jsonrpc: '2.0',
|
|
1604
|
+
method: 'getBalance',
|
|
1605
|
+
params: [testData.closeATAkeys.closeAtaAddress],
|
|
1606
|
+
},
|
|
1607
|
+
})
|
|
1608
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);
|
|
1609
|
+
callBack
|
|
1610
|
+
.withArgs({
|
|
1611
|
+
payload: {
|
|
1612
|
+
id: '1',
|
|
1613
|
+
jsonrpc: '2.0',
|
|
1614
|
+
method: 'getBalance',
|
|
1615
|
+
params: [testData.closeATAkeys.bs58EncodedPublicKey],
|
|
1616
|
+
},
|
|
1617
|
+
})
|
|
1618
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);
|
|
1619
|
+
callBack
|
|
1620
|
+
.withArgs({
|
|
1621
|
+
payload: {
|
|
1622
|
+
id: '1',
|
|
1623
|
+
jsonrpc: '2.0',
|
|
1624
|
+
method: 'getAccountInfo',
|
|
1625
|
+
params: [
|
|
1626
|
+
testData.closeATAkeys.closeAtaAddress,
|
|
1627
|
+
{
|
|
1628
|
+
encoding: 'jsonParsed',
|
|
1629
|
+
},
|
|
1630
|
+
],
|
|
1631
|
+
},
|
|
1632
|
+
})
|
|
1633
|
+
.resolves(testData.SolResponses.getTokenInfoResponse);
|
|
1634
|
+
callBack
|
|
1635
|
+
.withArgs({
|
|
1636
|
+
payload: {
|
|
1637
|
+
id: '1',
|
|
1638
|
+
jsonrpc: '2.0',
|
|
1639
|
+
method: 'getAccountInfo',
|
|
1640
|
+
params: [
|
|
1641
|
+
testData.keys.durableNoncePubKey,
|
|
1642
|
+
{
|
|
1643
|
+
encoding: 'jsonParsed',
|
|
1644
|
+
},
|
|
1645
|
+
],
|
|
1646
|
+
},
|
|
1647
|
+
})
|
|
1648
|
+
.resolves(testData.SolResponses.getAccountInfoResponse);
|
|
1649
|
+
callBack
|
|
1650
|
+
.withArgs({
|
|
1651
|
+
payload: {
|
|
1652
|
+
id: '1',
|
|
1653
|
+
jsonrpc: '2.0',
|
|
1654
|
+
method: 'getTokenAccountsByOwner',
|
|
1655
|
+
params: [
|
|
1656
|
+
testData.keys.destinationPubKey,
|
|
1657
|
+
{
|
|
1658
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
1659
|
+
},
|
|
1660
|
+
{
|
|
1661
|
+
encoding: 'jsonParsed',
|
|
1662
|
+
},
|
|
1663
|
+
],
|
|
1664
|
+
},
|
|
1665
|
+
})
|
|
1666
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);
|
|
1667
|
+
callBack
|
|
1668
|
+
.withArgs({
|
|
1669
|
+
payload: {
|
|
1670
|
+
id: '1',
|
|
1671
|
+
jsonrpc: '2.0',
|
|
1672
|
+
method: 'getTokenAccountsByOwner',
|
|
1673
|
+
params: [
|
|
1674
|
+
testData.accountInfo.bs58EncodedPublicKey,
|
|
1675
|
+
{
|
|
1676
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
1677
|
+
},
|
|
1678
|
+
{
|
|
1679
|
+
encoding: 'jsonParsed',
|
|
1680
|
+
},
|
|
1681
|
+
],
|
|
1682
|
+
},
|
|
1683
|
+
})
|
|
1684
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);
|
|
1685
|
+
callBack
|
|
1686
|
+
.withArgs({
|
|
1687
|
+
payload: {
|
|
1688
|
+
id: '1',
|
|
1689
|
+
jsonrpc: '2.0',
|
|
1690
|
+
method: 'getTokenAccountsByOwner',
|
|
1691
|
+
params: [
|
|
1692
|
+
testData.keys.destinationPubKey2,
|
|
1693
|
+
{
|
|
1694
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
1695
|
+
},
|
|
1696
|
+
{
|
|
1697
|
+
encoding: 'jsonParsed',
|
|
1698
|
+
},
|
|
1699
|
+
],
|
|
1700
|
+
},
|
|
1701
|
+
})
|
|
1702
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponse);
|
|
1703
|
+
callBack
|
|
1704
|
+
.withArgs({
|
|
1705
|
+
payload: {
|
|
1706
|
+
id: '1',
|
|
1707
|
+
jsonrpc: '2.0',
|
|
1708
|
+
method: 'getTokenAccountsByOwner',
|
|
1709
|
+
params: [
|
|
1710
|
+
testData.keys.destinationPubKey2,
|
|
1711
|
+
{
|
|
1712
|
+
programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',
|
|
1713
|
+
},
|
|
1714
|
+
{
|
|
1715
|
+
encoding: 'jsonParsed',
|
|
1716
|
+
},
|
|
1717
|
+
],
|
|
1718
|
+
},
|
|
1719
|
+
})
|
|
1720
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerForSol2022Response2);
|
|
1721
|
+
callBack
|
|
1722
|
+
.withArgs({
|
|
1723
|
+
payload: {
|
|
1724
|
+
id: '1',
|
|
1725
|
+
jsonrpc: '2.0',
|
|
1726
|
+
method: 'getTokenAccountsByOwner',
|
|
1727
|
+
params: [
|
|
1728
|
+
testData.wrwUser.walletAddress0,
|
|
1729
|
+
{
|
|
1730
|
+
programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',
|
|
1731
|
+
},
|
|
1732
|
+
{
|
|
1733
|
+
encoding: 'jsonParsed',
|
|
1734
|
+
},
|
|
1735
|
+
],
|
|
1736
|
+
},
|
|
1737
|
+
})
|
|
1738
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerForSol2022Response);
|
|
1739
|
+
callBack
|
|
1740
|
+
.withArgs({
|
|
1741
|
+
payload: {
|
|
1742
|
+
id: '1',
|
|
1743
|
+
jsonrpc: '2.0',
|
|
1744
|
+
method: 'getTokenAccountsByOwner',
|
|
1745
|
+
params: [
|
|
1746
|
+
testData.wrwUser.walletAddress0,
|
|
1747
|
+
{
|
|
1748
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
1749
|
+
},
|
|
1750
|
+
{
|
|
1751
|
+
encoding: 'jsonParsed',
|
|
1752
|
+
},
|
|
1753
|
+
],
|
|
1754
|
+
},
|
|
1755
|
+
})
|
|
1756
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponse2);
|
|
1757
|
+
callBack
|
|
1758
|
+
.withArgs({
|
|
1759
|
+
payload: {
|
|
1760
|
+
id: '1',
|
|
1761
|
+
jsonrpc: '2.0',
|
|
1762
|
+
method: 'getTokenAccountsByOwner',
|
|
1763
|
+
params: [
|
|
1764
|
+
testData.wrwUser.walletAddress4,
|
|
1765
|
+
{
|
|
1766
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
1767
|
+
},
|
|
1768
|
+
{
|
|
1769
|
+
encoding: 'jsonParsed',
|
|
1770
|
+
},
|
|
1771
|
+
],
|
|
1772
|
+
},
|
|
1773
|
+
})
|
|
1774
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponse3);
|
|
1775
|
+
callBack
|
|
1776
|
+
.withArgs({
|
|
1777
|
+
payload: {
|
|
1778
|
+
id: '1',
|
|
1779
|
+
jsonrpc: '2.0',
|
|
1780
|
+
method: 'getBalance',
|
|
1781
|
+
params: [testData.wrwUser.walletAddress0],
|
|
1782
|
+
},
|
|
1783
|
+
})
|
|
1784
|
+
.resolves(testData.SolResponses.getAccountBalanceResponse);
|
|
1785
|
+
callBack
|
|
1786
|
+
.withArgs({
|
|
1787
|
+
payload: {
|
|
1788
|
+
id: '1',
|
|
1789
|
+
jsonrpc: '2.0',
|
|
1790
|
+
method: 'getTokenAccountsByOwner',
|
|
1791
|
+
params: [
|
|
1792
|
+
testData.keys.destinationPubKey,
|
|
1793
|
+
{
|
|
1794
|
+
programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',
|
|
1795
|
+
},
|
|
1796
|
+
{
|
|
1797
|
+
encoding: 'jsonParsed',
|
|
1798
|
+
},
|
|
1799
|
+
],
|
|
1800
|
+
},
|
|
1801
|
+
})
|
|
1802
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);
|
|
1803
|
+
callBack
|
|
1804
|
+
.withArgs({
|
|
1805
|
+
payload: {
|
|
1806
|
+
id: '1',
|
|
1807
|
+
jsonrpc: '2.0',
|
|
1808
|
+
method: 'sendTransaction',
|
|
1809
|
+
params: sinon.match.array,
|
|
1810
|
+
},
|
|
1811
|
+
})
|
|
1812
|
+
.onCall(0)
|
|
1813
|
+
.resolves(testData.SolResponses.broadcastTransactionResponse)
|
|
1814
|
+
.onCall(1)
|
|
1815
|
+
.resolves(testData.SolResponses.broadcastTransactionResponse1);
|
|
1816
|
+
});
|
|
1817
|
+
afterEach(() => {
|
|
1818
|
+
sandBox.restore();
|
|
1819
|
+
});
|
|
1820
|
+
it('should take OVC output and generate a signed sweep transaction', async function () {
|
|
1821
|
+
const params = testData.ovcResponse;
|
|
1822
|
+
const recoveryTxn = await basecoin.createBroadcastableSweepTransaction(params);
|
|
1823
|
+
recoveryTxn.transactions[0].serializedTx.should.equal('AvR+L909kzRq6NuaUe9F6Jt97MOiFs7jpW8MuOrwz4EbKF40d31dci/bgLTq4gpk/Hh3s5cA8FtbLkDQr15PqAE7yd8LOXvsLtO2REqMM/OCZ8wItfsqfTfia2xIfibRW3wHgw63jiaojbXeSqaYajJ/Ca7YwBUz5blydI3fYLgPAgECBsLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHxPX1mHv+JqpmAT79ltNjYPK0M2yR+ZMln7VgUTBWFNQvLqE/j/nXlY2/JpxuNr/fXLXEPeS04dPvt9qz1dAoYEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAADpiH20cxLj7KnOaoI5ANNoPxYjs472FdjDeMPft3kXdAgQDAgUBBAQAAAAEAgADDAIAAADwopo7AAAAAA==');
|
|
1824
|
+
(recoveryTxn.transactions[0].scanIndex ?? 0).should.equal(0);
|
|
1825
|
+
(recoveryTxn.lastScanIndex ?? 0).should.equal(0);
|
|
1826
|
+
});
|
|
1827
|
+
it('should take sol 2022 token OVC output and generate a signed sweep transaction', async function () {
|
|
1828
|
+
const params = testData.ovcResponse;
|
|
1829
|
+
const recoveryTxn = await basecoin.createBroadcastableSweepTransaction(params);
|
|
1830
|
+
recoveryTxn.transactions[0].serializedTx.should.equal('AvR+L909kzRq6NuaUe9F6Jt97MOiFs7jpW8MuOrwz4EbKF40d31dci/bgLTq4gpk/Hh3s5cA8FtbLkDQr15PqAE7yd8LOXvsLtO2REqMM/OCZ8wItfsqfTfia2xIfibRW3wHgw63jiaojbXeSqaYajJ/Ca7YwBUz5blydI3fYLgPAgECBsLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHxPX1mHv+JqpmAT79ltNjYPK0M2yR+ZMln7VgUTBWFNQvLqE/j/nXlY2/JpxuNr/fXLXEPeS04dPvt9qz1dAoYEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAADpiH20cxLj7KnOaoI5ANNoPxYjs472FdjDeMPft3kXdAgQDAgUBBAQAAAAEAgADDAIAAADwopo7AAAAAA==');
|
|
1831
|
+
(recoveryTxn.transactions[0].scanIndex ?? 0).should.equal(0);
|
|
1832
|
+
(recoveryTxn.lastScanIndex ?? 0).should.equal(0);
|
|
1833
|
+
});
|
|
1834
|
+
it('should take consolidation OVC output and generate multiple signed sweep transactions', async function () {
|
|
1835
|
+
const params = testData.ovcResponse2;
|
|
1836
|
+
const recoveryTxn = await basecoin.createBroadcastableSweepTransaction(params);
|
|
1837
|
+
recoveryTxn.transactions[0].serializedTx.should.equal('AtQPLzOmLuKwHY6N5XoJIZK/T7W10uYWm/MRte3GFUdl+w3gHLjSa9H66WSfFNubQxIPckxJDyltkP7ksLDf9QgBNJM2UWbBUH5wT0JJHILlhCs33HX8DeE/8Tdsw6tGfZoMhCnSKv6TPWtBxy7Sb6sW8ksCUPnAWuHGGKmgjEMBAgECBmLrqxJrY2kbN/tcrQw3P8P15OljFGabFJAKBrUO1grNBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHxPX1mHv+JqpmAT79ltNjYPK0M2yR+ZMln7VgUTBWFNQsLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAAIZQniiS73D6mwfpnfhVMC4lyYJtRSrmoZpF7yIlUdIDAgQDAgUBBAQAAAAEAgADDAIAAADwPc0dAAAAAA==');
|
|
1838
|
+
(recoveryTxn.transactions[0].scanIndex ?? 0).should.equal(1);
|
|
1839
|
+
recoveryTxn.transactions[1].serializedTx.should.equal('AuLhOA5zmOBZR85lo+nKdTopVwJAMrMp6NW+8UnGNsSBSpBkqfWZQqSg9s+7aTlXezm5vxol+Pl6t7PpVNTOHwLcp9xJp3TFHdivEbhwJKldR4Ny+pasoFx+Bgk8q6g1iNiq7XSi1Ov3bs7euMkTj7nDRFqP8lv7xLTcvrBm9OQJAgECBp14ImBCdmVROlw0UveYS1MvG/ljCRI3MJTFmsxuXEoWBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHw0hyxvpVwtIx9/zeX2O16eTrY+aKIh1mdKg4MMg0eyxMLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAAC7ws1XFslinwgtpISUViVWIVTHyD2Q0qj24YjKmrAmXAgQDAgUBBAQAAAAEAgADDAIAAADwPc0dAAAAAA==');
|
|
1840
|
+
(recoveryTxn.transactions[1].scanIndex ?? 0).should.equal(2);
|
|
1841
|
+
(recoveryTxn.lastScanIndex ?? 0).should.equal(20);
|
|
1842
|
+
});
|
|
1843
|
+
it('should recover a txn for non-bitgo recoveries (latest blockhash)', async function () {
|
|
1844
|
+
// Latest Blockhash Recovery (BitGo-less)
|
|
1845
|
+
const latestBlockHashTxn = await basecoin.recover({
|
|
1846
|
+
userKey: testData.keys.userKey,
|
|
1847
|
+
backupKey: testData.keys.backupKey,
|
|
1848
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1849
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1850
|
+
walletPassphrase: testData.keys.walletPassword,
|
|
1851
|
+
});
|
|
1852
|
+
latestBlockHashTxn.should.not.be.empty();
|
|
1853
|
+
latestBlockHashTxn.should.hasOwnProperty('serializedTx');
|
|
1854
|
+
latestBlockHashTxn.should.hasOwnProperty('scanIndex');
|
|
1855
|
+
should.equal(latestBlockHashTxn.scanIndex, 0);
|
|
1856
|
+
const latestBlockhashTxnDeserialize = new lib_1.Transaction(coin);
|
|
1857
|
+
latestBlockhashTxnDeserialize.fromRawTransaction(latestBlockHashTxn.serializedTx);
|
|
1858
|
+
const latestBlockhashTxnJson = latestBlockhashTxnDeserialize.toJson();
|
|
1859
|
+
should.equal(latestBlockhashTxnJson.nonce, testData.SolInputData.blockhash);
|
|
1860
|
+
should.equal(latestBlockhashTxnJson.feePayer, testData.accountInfo.bs58EncodedPublicKey);
|
|
1861
|
+
should.equal(latestBlockhashTxnJson.numSignatures, testData.SolInputData.latestBlockhashSignatures);
|
|
1862
|
+
const solCoin = basecoin;
|
|
1863
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 3);
|
|
1864
|
+
});
|
|
1865
|
+
it('should recover a txn for non-bitgo recoveries (durable nonce)', async function () {
|
|
1866
|
+
// Durable Nonce Recovery (BitGo-less)
|
|
1867
|
+
const durableNonceTxn = await basecoin.recover({
|
|
1868
|
+
userKey: testData.keys.userKey,
|
|
1869
|
+
backupKey: testData.keys.backupKey,
|
|
1870
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1871
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1872
|
+
walletPassphrase: testData.keys.walletPassword,
|
|
1873
|
+
durableNonce: {
|
|
1874
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
1875
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
1876
|
+
},
|
|
1877
|
+
});
|
|
1878
|
+
durableNonceTxn.should.not.be.empty();
|
|
1879
|
+
durableNonceTxn.should.hasOwnProperty('serializedTx');
|
|
1880
|
+
durableNonceTxn.should.hasOwnProperty('scanIndex');
|
|
1881
|
+
should.equal(durableNonceTxn.scanIndex, 0);
|
|
1882
|
+
const durableNonceTxnDeserialize = new lib_1.Transaction(coin);
|
|
1883
|
+
durableNonceTxnDeserialize.fromRawTransaction(durableNonceTxn.serializedTx);
|
|
1884
|
+
const durableNonceTxnJson = durableNonceTxnDeserialize.toJson();
|
|
1885
|
+
should.equal(durableNonceTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
1886
|
+
should.equal(durableNonceTxnJson.feePayer, testData.accountInfo.bs58EncodedPublicKey);
|
|
1887
|
+
should.equal(durableNonceTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
1888
|
+
const solCoin = basecoin;
|
|
1889
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 4);
|
|
1890
|
+
});
|
|
1891
|
+
it('should recover a txn for unsigned sweep recoveries', async function () {
|
|
1892
|
+
// Unsigned Sweep Recovery
|
|
1893
|
+
const unsignedSweepTxn = (await basecoin.recover({
|
|
1894
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1895
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1896
|
+
durableNonce: {
|
|
1897
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
1898
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
1899
|
+
},
|
|
1900
|
+
}));
|
|
1901
|
+
unsignedSweepTxn.should.not.be.empty();
|
|
1902
|
+
unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.should.hasOwnProperty('serializedTx');
|
|
1903
|
+
unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.should.hasOwnProperty('scanIndex');
|
|
1904
|
+
should.equal(unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.scanIndex, 0);
|
|
1905
|
+
const unsignedSweepTxnDeserialize = new lib_1.Transaction(coin);
|
|
1906
|
+
unsignedSweepTxnDeserialize.fromRawTransaction(unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.serializedTx);
|
|
1907
|
+
const unsignedSweepTxnJson = unsignedSweepTxnDeserialize.toJson();
|
|
1908
|
+
should.equal(unsignedSweepTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
1909
|
+
should.equal(unsignedSweepTxnJson.feePayer, testData.accountInfo.bs58EncodedPublicKey);
|
|
1910
|
+
should.equal(unsignedSweepTxnJson.numSignatures, testData.SolInputData.unsignedSweepSignatures);
|
|
1911
|
+
const solCoin = basecoin;
|
|
1912
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 4);
|
|
1913
|
+
});
|
|
1914
|
+
it('should handle error in recover function if a required field is missing/incorrect', async function () {
|
|
1915
|
+
// missing userkey
|
|
1916
|
+
await basecoin
|
|
1917
|
+
.recover({
|
|
1918
|
+
backupKey: testData.keys.backupKey,
|
|
1919
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1920
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1921
|
+
walletPassphrase: testData.keys.walletPassword,
|
|
1922
|
+
})
|
|
1923
|
+
.should.rejectedWith('missing userKey');
|
|
1924
|
+
// missing backupkey
|
|
1925
|
+
await basecoin
|
|
1926
|
+
.recover({
|
|
1927
|
+
userKey: testData.keys.userKey,
|
|
1928
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1929
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1930
|
+
walletPassphrase: testData.keys.walletPassword,
|
|
1931
|
+
})
|
|
1932
|
+
.should.rejectedWith('missing backupKey');
|
|
1933
|
+
// missing wallet passphrase
|
|
1934
|
+
await basecoin
|
|
1935
|
+
.recover({
|
|
1936
|
+
userKey: testData.keys.userKey,
|
|
1937
|
+
backupKey: testData.keys.backupKey,
|
|
1938
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1939
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1940
|
+
})
|
|
1941
|
+
.should.rejectedWith('missing wallet passphrase');
|
|
1942
|
+
// incorrect wallet passphrase, user key, backup key combination
|
|
1943
|
+
await basecoin
|
|
1944
|
+
.recover({
|
|
1945
|
+
userKey: testData.keys.userKey,
|
|
1946
|
+
backupKey: testData.keys.backupKey,
|
|
1947
|
+
bitgoKey: testData.keys.bitgoKey,
|
|
1948
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1949
|
+
walletPassphrase: testData.keys.walletPassword + 'incorrect',
|
|
1950
|
+
})
|
|
1951
|
+
.should.rejectedWith("Error decrypting user keychain: password error - ccm: tag doesn't match");
|
|
1952
|
+
// no wallet with sufficient funds
|
|
1953
|
+
await basecoin
|
|
1954
|
+
.recover({
|
|
1955
|
+
userKey: testData.keys.userKey,
|
|
1956
|
+
backupKey: testData.keys.backupKey,
|
|
1957
|
+
bitgoKey: testData.keys.bitgoKeyNoFunds,
|
|
1958
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1959
|
+
walletPassphrase: testData.keys.walletPassword,
|
|
1960
|
+
})
|
|
1961
|
+
.should.rejectedWith('Did not find address with funds to recover');
|
|
1962
|
+
});
|
|
1963
|
+
it('should recover sol tokens to recovery destination with no existing token accounts', async function () {
|
|
1964
|
+
const tokenTxn = await basecoin.recover({
|
|
1965
|
+
userKey: testData.wrwUser.userKey,
|
|
1966
|
+
backupKey: testData.wrwUser.backupKey,
|
|
1967
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
1968
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
1969
|
+
tokenContractAddress: usdtMintAddress,
|
|
1970
|
+
walletPassphrase: testData.wrwUser.walletPassphrase,
|
|
1971
|
+
durableNonce: {
|
|
1972
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
1973
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
1974
|
+
},
|
|
1975
|
+
});
|
|
1976
|
+
tokenTxn.should.not.be.empty();
|
|
1977
|
+
tokenTxn.should.hasOwnProperty('serializedTx');
|
|
1978
|
+
tokenTxn.should.hasOwnProperty('scanIndex');
|
|
1979
|
+
should.equal(tokenTxn.scanIndex, 0);
|
|
1980
|
+
const tokenTxnDeserialize = new lib_1.Transaction(coin);
|
|
1981
|
+
tokenTxnDeserialize.fromRawTransaction(tokenTxn.serializedTx);
|
|
1982
|
+
const tokenTxnJson = tokenTxnDeserialize.toJson();
|
|
1983
|
+
should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
1984
|
+
should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);
|
|
1985
|
+
should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
1986
|
+
const instructionsData = tokenTxnJson.instructionsData;
|
|
1987
|
+
should.equal(instructionsData.length, 3);
|
|
1988
|
+
should.equal(instructionsData[0].type, 'NonceAdvance');
|
|
1989
|
+
const destinationUSDTTokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(usdtMintAddress, testData.keys.destinationPubKey);
|
|
1990
|
+
should.equal(instructionsData[1].type, 'CreateAssociatedTokenAccount');
|
|
1991
|
+
should.equal(instructionsData[1].params.mintAddress, usdtMintAddress);
|
|
1992
|
+
should.equal(instructionsData[1].params.ataAddress, destinationUSDTTokenAccount);
|
|
1993
|
+
should.equal(instructionsData[1].params.ownerAddress, testData.keys.destinationPubKey);
|
|
1994
|
+
should.equal(instructionsData[1].params.tokenName, 'tsol:usdt');
|
|
1995
|
+
should.equal(instructionsData[1].params.payerAddress, testData.wrwUser.walletAddress0);
|
|
1996
|
+
const sourceUSDTTokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(usdtMintAddress, testData.wrwUser.walletAddress0);
|
|
1997
|
+
should.equal(instructionsData[2].type, 'TokenTransfer');
|
|
1998
|
+
should.equal(instructionsData[2].params.fromAddress, testData.wrwUser.walletAddress0);
|
|
1999
|
+
should.equal(instructionsData[2].params.toAddress, destinationUSDTTokenAccount);
|
|
2000
|
+
should.equal(instructionsData[2].params.amount, '2000000000');
|
|
2001
|
+
should.equal(instructionsData[2].params.tokenName, 'tsol:usdt');
|
|
2002
|
+
should.equal(instructionsData[2].params.sourceAddress, sourceUSDTTokenAccount);
|
|
2003
|
+
const solCoin = basecoin;
|
|
2004
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 7);
|
|
2005
|
+
});
|
|
2006
|
+
it('should recover sol 2022 tokens to recovery destination with no existing token accounts', async function () {
|
|
2007
|
+
const tokenTxn = await basecoin.recover({
|
|
2008
|
+
userKey: testData.wrwUser.userKey,
|
|
2009
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2010
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2011
|
+
recoveryDestination: testData.keys.destinationPubKey,
|
|
2012
|
+
tokenContractAddress: t22mintAddress,
|
|
2013
|
+
walletPassphrase: testData.wrwUser.walletPassphrase,
|
|
2014
|
+
durableNonce: {
|
|
2015
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
2016
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
2017
|
+
},
|
|
2018
|
+
programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',
|
|
2019
|
+
});
|
|
2020
|
+
tokenTxn.should.not.be.empty();
|
|
2021
|
+
tokenTxn.should.hasOwnProperty('serializedTx');
|
|
2022
|
+
tokenTxn.should.hasOwnProperty('scanIndex');
|
|
2023
|
+
should.equal(tokenTxn.scanIndex, 0);
|
|
2024
|
+
const tokenTxnDeserialize = new lib_1.Transaction(coin);
|
|
2025
|
+
tokenTxnDeserialize.fromRawTransaction(tokenTxn.serializedTx);
|
|
2026
|
+
const tokenTxnJson = tokenTxnDeserialize.toJson();
|
|
2027
|
+
should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
2028
|
+
should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);
|
|
2029
|
+
should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
2030
|
+
const instructionsData = tokenTxnJson.instructionsData;
|
|
2031
|
+
should.equal(instructionsData.length, 3);
|
|
2032
|
+
should.equal(instructionsData[0].type, 'NonceAdvance');
|
|
2033
|
+
const destinationTokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(t22mintAddress, testData.keys.destinationPubKey);
|
|
2034
|
+
should.equal(instructionsData[1].type, 'CreateAssociatedTokenAccount');
|
|
2035
|
+
should.equal(instructionsData[1].params.mintAddress, t22mintAddress);
|
|
2036
|
+
should.equal(instructionsData[1].params.ataAddress, destinationTokenAccount);
|
|
2037
|
+
should.equal(instructionsData[1].params.ownerAddress, testData.keys.destinationPubKey);
|
|
2038
|
+
should.equal(instructionsData[1].params.tokenName, 'tsol:t22mint');
|
|
2039
|
+
should.equal(instructionsData[1].params.payerAddress, testData.wrwUser.walletAddress0);
|
|
2040
|
+
const sourceTokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(t22mintAddress, testData.wrwUser.walletAddress0);
|
|
2041
|
+
should.equal(instructionsData[2].type, 'TokenTransfer');
|
|
2042
|
+
should.equal(instructionsData[2].params.fromAddress, testData.wrwUser.walletAddress0);
|
|
2043
|
+
should.equal(instructionsData[2].params.toAddress, destinationTokenAccount);
|
|
2044
|
+
should.equal(instructionsData[2].params.amount, '2000000000');
|
|
2045
|
+
should.equal(instructionsData[2].params.tokenName, 'tsol:t22mint');
|
|
2046
|
+
should.equal(instructionsData[2].params.sourceAddress, sourceTokenAccount);
|
|
2047
|
+
should.equal(instructionsData[2].params.programId, 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb');
|
|
2048
|
+
const solCoin = basecoin;
|
|
2049
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 7);
|
|
2050
|
+
});
|
|
2051
|
+
it('should recover sol tokens to recovery destination with existing token accounts', async function () {
|
|
2052
|
+
const tokenTxn = await basecoin.recover({
|
|
2053
|
+
userKey: testData.wrwUser.userKey,
|
|
2054
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2055
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2056
|
+
recoveryDestination: testData.keys.destinationPubKey2,
|
|
2057
|
+
tokenContractAddress: usdtMintAddress,
|
|
2058
|
+
walletPassphrase: testData.wrwUser.walletPassphrase,
|
|
2059
|
+
durableNonce: {
|
|
2060
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
2061
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
2062
|
+
},
|
|
2063
|
+
});
|
|
2064
|
+
tokenTxn.should.not.be.empty();
|
|
2065
|
+
tokenTxn.should.hasOwnProperty('serializedTx');
|
|
2066
|
+
tokenTxn.should.hasOwnProperty('scanIndex');
|
|
2067
|
+
should.equal(tokenTxn.scanIndex, 0);
|
|
2068
|
+
const tokenTxnDeserialize = new lib_1.Transaction(coin);
|
|
2069
|
+
tokenTxnDeserialize.fromRawTransaction(tokenTxn.serializedTx);
|
|
2070
|
+
const tokenTxnJson = tokenTxnDeserialize.toJson();
|
|
2071
|
+
should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
2072
|
+
should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);
|
|
2073
|
+
should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
2074
|
+
const instructionsData = tokenTxnJson.instructionsData;
|
|
2075
|
+
should.equal(instructionsData.length, 2);
|
|
2076
|
+
should.equal(instructionsData[0].type, 'NonceAdvance');
|
|
2077
|
+
const sourceUSDTTokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(usdtMintAddress, testData.wrwUser.walletAddress0);
|
|
2078
|
+
const destinationUSDTTokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(usdtMintAddress, testData.keys.destinationPubKey2);
|
|
2079
|
+
should.equal(instructionsData[1].type, 'TokenTransfer');
|
|
2080
|
+
should.equal(instructionsData[1].params.fromAddress, testData.wrwUser.walletAddress0);
|
|
2081
|
+
should.equal(instructionsData[1].params.toAddress, destinationUSDTTokenAccount);
|
|
2082
|
+
should.equal(instructionsData[1].params.amount, '2000000000');
|
|
2083
|
+
should.equal(instructionsData[1].params.tokenName, 'tsol:usdt');
|
|
2084
|
+
should.equal(instructionsData[1].params.sourceAddress, sourceUSDTTokenAccount);
|
|
2085
|
+
const solCoin = basecoin;
|
|
2086
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 7);
|
|
2087
|
+
});
|
|
2088
|
+
it('should recover sol 2022 tokens to recovery destination with existing token accounts', async function () {
|
|
2089
|
+
const tokenTxn = await basecoin.recover({
|
|
2090
|
+
userKey: testData.wrwUser.userKey,
|
|
2091
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2092
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2093
|
+
recoveryDestination: testData.keys.destinationPubKey2,
|
|
2094
|
+
tokenContractAddress: t22mintAddress,
|
|
2095
|
+
walletPassphrase: testData.wrwUser.walletPassphrase,
|
|
2096
|
+
durableNonce: {
|
|
2097
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
2098
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
2099
|
+
},
|
|
2100
|
+
programId: spl_token_1.TOKEN_2022_PROGRAM_ID.toString(),
|
|
2101
|
+
});
|
|
2102
|
+
tokenTxn.should.not.be.empty();
|
|
2103
|
+
tokenTxn.should.hasOwnProperty('serializedTx');
|
|
2104
|
+
tokenTxn.should.hasOwnProperty('scanIndex');
|
|
2105
|
+
should.equal(tokenTxn.scanIndex, 0);
|
|
2106
|
+
const tokenTxnDeserialize = new lib_1.Transaction(coin);
|
|
2107
|
+
tokenTxnDeserialize.fromRawTransaction(tokenTxn.serializedTx);
|
|
2108
|
+
const tokenTxnJson = tokenTxnDeserialize.toJson();
|
|
2109
|
+
console.log(tokenTxnJson);
|
|
2110
|
+
should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
2111
|
+
should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);
|
|
2112
|
+
should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
2113
|
+
const instructionsData = tokenTxnJson.instructionsData;
|
|
2114
|
+
should.equal(instructionsData.length, 2);
|
|
2115
|
+
should.equal(instructionsData[0].type, 'NonceAdvance');
|
|
2116
|
+
const source2022TokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(t22mintAddress, testData.wrwUser.walletAddress0, false, spl_token_1.TOKEN_2022_PROGRAM_ID.toString());
|
|
2117
|
+
const destination2022TokenAccount = await (0, utils_1.getAssociatedTokenAccountAddress)(t22mintAddress, testData.keys.destinationPubKey2, false, spl_token_1.TOKEN_2022_PROGRAM_ID.toString());
|
|
2118
|
+
should.equal(instructionsData[1].type, 'TokenTransfer');
|
|
2119
|
+
should.equal(instructionsData[1].params.fromAddress, testData.wrwUser.walletAddress0);
|
|
2120
|
+
should.equal(instructionsData[1].params.toAddress, destination2022TokenAccount);
|
|
2121
|
+
should.equal(instructionsData[1].params.amount, '2000000000');
|
|
2122
|
+
should.equal(instructionsData[1].params.tokenName, 'tsol:t22mint');
|
|
2123
|
+
should.equal(instructionsData[1].params.sourceAddress, source2022TokenAccount);
|
|
2124
|
+
const solCoin = basecoin;
|
|
2125
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 7);
|
|
2126
|
+
});
|
|
2127
|
+
it('should recover sol tokens to recovery destination with existing token accounts for unsigned sweep recoveries', async function () {
|
|
2128
|
+
const feeResponse = testData.SolResponses.getFeesForMessageResponse;
|
|
2129
|
+
feeResponse.body.result.value = 10000;
|
|
2130
|
+
callBack
|
|
2131
|
+
.withArgs({
|
|
2132
|
+
payload: {
|
|
2133
|
+
id: '1',
|
|
2134
|
+
jsonrpc: '2.0',
|
|
2135
|
+
method: 'getFeeForMessage',
|
|
2136
|
+
params: [
|
|
2137
|
+
sinon.match.string,
|
|
2138
|
+
{
|
|
2139
|
+
commitment: 'finalized',
|
|
2140
|
+
},
|
|
2141
|
+
],
|
|
2142
|
+
},
|
|
2143
|
+
})
|
|
2144
|
+
.resolves(feeResponse);
|
|
2145
|
+
const tokenTxn = (await basecoin.recover({
|
|
2146
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2147
|
+
recoveryDestination: testData.keys.destinationPubKey2,
|
|
2148
|
+
durableNonce: {
|
|
2149
|
+
publicKey: testData.keys.durableNoncePubKey,
|
|
2150
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
2151
|
+
},
|
|
2152
|
+
tokenContractAddress: testData.tokenAddress.TestUSDC,
|
|
2153
|
+
}));
|
|
2154
|
+
// 2 signatures and no rent exemption fee since the destination already has token accounts
|
|
2155
|
+
const expectedFee = 5000 + 5000;
|
|
2156
|
+
const { serializedTx, scanIndex, feeInfo, parsedTx } = tokenTxn.txRequests[0].transactions[0].unsignedTx;
|
|
2157
|
+
assert_1.default.ok(serializedTx);
|
|
2158
|
+
assert_1.default.strictEqual(scanIndex, 0);
|
|
2159
|
+
assert_1.default.ok(feeInfo);
|
|
2160
|
+
assert_1.default.strictEqual(feeInfo.feeString, expectedFee.toString());
|
|
2161
|
+
assert_1.default.strictEqual(feeInfo.fee, expectedFee);
|
|
2162
|
+
assert_1.default.ok(parsedTx);
|
|
2163
|
+
assert_1.default.ok(parsedTx.inputs instanceof Array && parsedTx.inputs.length === 1);
|
|
2164
|
+
assert_1.default.ok(parsedTx.outputs instanceof Array && parsedTx.outputs.length === 1);
|
|
2165
|
+
const tokenTxnDeserialize = new lib_1.Transaction(coin);
|
|
2166
|
+
tokenTxnDeserialize.fromRawTransaction(tokenTxn.txRequests[0].transactions[0].unsignedTx.serializedTx);
|
|
2167
|
+
const tokenTxnJson = tokenTxnDeserialize.toJson();
|
|
2168
|
+
assert_1.default.strictEqual(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);
|
|
2169
|
+
assert_1.default.strictEqual(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);
|
|
2170
|
+
assert_1.default.strictEqual(tokenTxnJson.numSignatures, testData.SolInputData.unsignedSweepSignatures);
|
|
2171
|
+
const solCoin = basecoin;
|
|
2172
|
+
sandBox.assert.callCount(solCoin.getDataFromNode, 7);
|
|
2173
|
+
});
|
|
2174
|
+
it('should recover sol funds from ATA address for non-bitgo recoveries', async function () {
|
|
2175
|
+
// close ATA address instruction type txn
|
|
2176
|
+
const closeATATxns = await basecoin.recoverCloseATA({
|
|
2177
|
+
userKey: testData.closeATAkeys.userKey,
|
|
2178
|
+
backupKey: testData.closeATAkeys.backupKey,
|
|
2179
|
+
bitgoKey: testData.closeATAkeys.bitgoKey,
|
|
2180
|
+
recoveryDestination: testData.closeATAkeys.destinationPubKey,
|
|
2181
|
+
walletPassphrase: testData.closeATAkeys.walletPassword,
|
|
2182
|
+
closeAtaAddress: testData.closeATAkeys.closeAtaAddress,
|
|
2183
|
+
recoveryDestinationAtaAddress: testData.closeATAkeys.recoveryDestinationAtaAddress,
|
|
2184
|
+
});
|
|
2185
|
+
closeATATxns.should.not.be.empty();
|
|
2186
|
+
should.equal(closeATATxns[0].txId, '2id3YC2jK9G5Wo2phDx4gJVAew8DcY5NAojnVuao8rkxwPYPe8cSwE5GzhEgJA2y8fVjDEo6iR6ykBvDxrTQrtpb');
|
|
2187
|
+
should.equal(closeATATxns[1].txId, '5oUBgXX4enGmFEspG64goy3PRysjfrekZGg3rZNkBHUCQFd482vrVWbfDcRYMBEJt65JXymfEPm8M6d89X4xV79n');
|
|
2188
|
+
});
|
|
2189
|
+
});
|
|
2190
|
+
describe('Build Consolidation Recoveries:', () => {
|
|
2191
|
+
const sandBox = sinon.createSandbox();
|
|
2192
|
+
const coin = statics_1.coins.get('tsol');
|
|
2193
|
+
const usdtMintAddress = '9cgpBeNZ2HnLda7NWaaU1i3NyTstk2c4zCMUcoAGsi9C';
|
|
2194
|
+
const durableNonces = {
|
|
2195
|
+
publicKeys: [
|
|
2196
|
+
testData.keys.durableNoncePubKey,
|
|
2197
|
+
testData.keys.durableNoncePubKey2,
|
|
2198
|
+
testData.keys.durableNoncePubKey3,
|
|
2199
|
+
],
|
|
2200
|
+
secretKey: testData.keys.durableNoncePrivKey,
|
|
2201
|
+
};
|
|
2202
|
+
beforeEach(() => {
|
|
2203
|
+
const callBack = sandBox.stub(src_1.Sol.prototype, 'getDataFromNode');
|
|
2204
|
+
callBack
|
|
2205
|
+
.withArgs({
|
|
2206
|
+
payload: {
|
|
2207
|
+
id: '1',
|
|
2208
|
+
jsonrpc: '2.0',
|
|
2209
|
+
method: 'getLatestBlockhash',
|
|
2210
|
+
params: [
|
|
2211
|
+
{
|
|
2212
|
+
commitment: 'finalized',
|
|
2213
|
+
},
|
|
2214
|
+
],
|
|
2215
|
+
},
|
|
2216
|
+
})
|
|
2217
|
+
.resolves(testData.SolResponses.getBlockhashResponse);
|
|
2218
|
+
callBack
|
|
2219
|
+
.withArgs({
|
|
2220
|
+
payload: {
|
|
2221
|
+
id: '1',
|
|
2222
|
+
jsonrpc: '2.0',
|
|
2223
|
+
method: 'getFeeForMessage',
|
|
2224
|
+
params: [
|
|
2225
|
+
sinon.match.string,
|
|
2226
|
+
{
|
|
2227
|
+
commitment: 'finalized',
|
|
2228
|
+
},
|
|
2229
|
+
],
|
|
2230
|
+
},
|
|
2231
|
+
})
|
|
2232
|
+
.resolves(testData.SolResponses.getFeesForMessageResponse);
|
|
2233
|
+
callBack
|
|
2234
|
+
.withArgs({
|
|
2235
|
+
payload: {
|
|
2236
|
+
id: '1',
|
|
2237
|
+
jsonrpc: '2.0',
|
|
2238
|
+
method: 'getBalance',
|
|
2239
|
+
params: [testData.wrwUser.walletAddress1],
|
|
2240
|
+
},
|
|
2241
|
+
})
|
|
2242
|
+
.resolves(testData.SolResponses.getAccountBalanceResponseNoFunds);
|
|
2243
|
+
callBack
|
|
2244
|
+
.withArgs({
|
|
2245
|
+
payload: {
|
|
2246
|
+
id: '1',
|
|
2247
|
+
jsonrpc: '2.0',
|
|
2248
|
+
method: 'getBalance',
|
|
2249
|
+
params: [testData.wrwUser.walletAddress2],
|
|
2250
|
+
},
|
|
2251
|
+
})
|
|
2252
|
+
.resolves(testData.SolResponses.getAccountBalanceResponse);
|
|
2253
|
+
callBack
|
|
2254
|
+
.withArgs({
|
|
2255
|
+
payload: {
|
|
2256
|
+
id: '1',
|
|
2257
|
+
jsonrpc: '2.0',
|
|
2258
|
+
method: 'getBalance',
|
|
2259
|
+
params: [testData.wrwUser.walletAddress3],
|
|
2260
|
+
},
|
|
2261
|
+
})
|
|
2262
|
+
.resolves(testData.SolResponses.getAccountBalanceResponse);
|
|
2263
|
+
callBack
|
|
2264
|
+
.withArgs({
|
|
2265
|
+
payload: {
|
|
2266
|
+
id: '1',
|
|
2267
|
+
jsonrpc: '2.0',
|
|
2268
|
+
method: 'getBalance',
|
|
2269
|
+
params: [testData.wrwUser.walletAddress5],
|
|
2270
|
+
},
|
|
2271
|
+
})
|
|
2272
|
+
.resolves(testData.SolResponses.getAccountBalanceResponse);
|
|
2273
|
+
callBack
|
|
2274
|
+
.withArgs({
|
|
2275
|
+
payload: {
|
|
2276
|
+
id: '1',
|
|
2277
|
+
jsonrpc: '2.0',
|
|
2278
|
+
method: 'getMinimumBalanceForRentExemption',
|
|
2279
|
+
params: [165],
|
|
2280
|
+
},
|
|
2281
|
+
})
|
|
2282
|
+
.resolves(testData.SolResponses.getMinimumBalanceForRentExemptionResponse);
|
|
2283
|
+
callBack
|
|
2284
|
+
.withArgs({
|
|
2285
|
+
payload: {
|
|
2286
|
+
id: '1',
|
|
2287
|
+
jsonrpc: '2.0',
|
|
2288
|
+
method: 'getAccountInfo',
|
|
2289
|
+
params: [
|
|
2290
|
+
testData.keys.durableNoncePubKey,
|
|
2291
|
+
{
|
|
2292
|
+
encoding: 'jsonParsed',
|
|
2293
|
+
},
|
|
2294
|
+
],
|
|
2295
|
+
},
|
|
2296
|
+
})
|
|
2297
|
+
.resolves(testData.SolResponses.getAccountInfoResponse);
|
|
2298
|
+
callBack
|
|
2299
|
+
.withArgs({
|
|
2300
|
+
payload: {
|
|
2301
|
+
id: '1',
|
|
2302
|
+
jsonrpc: '2.0',
|
|
2303
|
+
method: 'getAccountInfo',
|
|
2304
|
+
params: [
|
|
2305
|
+
testData.keys.durableNoncePubKey2,
|
|
2306
|
+
{
|
|
2307
|
+
encoding: 'jsonParsed',
|
|
2308
|
+
},
|
|
2309
|
+
],
|
|
2310
|
+
},
|
|
2311
|
+
})
|
|
2312
|
+
.resolves(testData.SolResponses.getAccountInfoResponse2);
|
|
2313
|
+
callBack
|
|
2314
|
+
.withArgs({
|
|
2315
|
+
payload: {
|
|
2316
|
+
id: '1',
|
|
2317
|
+
jsonrpc: '2.0',
|
|
2318
|
+
method: 'getTokenAccountsByOwner',
|
|
2319
|
+
params: [
|
|
2320
|
+
testData.wrwUser.walletAddress1,
|
|
2321
|
+
{
|
|
2322
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
2323
|
+
},
|
|
2324
|
+
{
|
|
2325
|
+
encoding: 'jsonParsed',
|
|
2326
|
+
},
|
|
2327
|
+
],
|
|
2328
|
+
},
|
|
2329
|
+
})
|
|
2330
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);
|
|
2331
|
+
callBack
|
|
2332
|
+
.withArgs({
|
|
2333
|
+
payload: {
|
|
2334
|
+
id: '1',
|
|
2335
|
+
jsonrpc: '2.0',
|
|
2336
|
+
method: 'getTokenAccountsByOwner',
|
|
2337
|
+
params: [
|
|
2338
|
+
testData.wrwUser.walletAddress2,
|
|
2339
|
+
{
|
|
2340
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
2341
|
+
},
|
|
2342
|
+
{
|
|
2343
|
+
encoding: 'jsonParsed',
|
|
2344
|
+
},
|
|
2345
|
+
],
|
|
2346
|
+
},
|
|
2347
|
+
})
|
|
2348
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);
|
|
2349
|
+
callBack
|
|
2350
|
+
.withArgs({
|
|
2351
|
+
payload: {
|
|
2352
|
+
id: '1',
|
|
2353
|
+
jsonrpc: '2.0',
|
|
2354
|
+
method: 'getTokenAccountsByOwner',
|
|
2355
|
+
params: [
|
|
2356
|
+
testData.wrwUser.walletAddress3,
|
|
2357
|
+
{
|
|
2358
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
2359
|
+
},
|
|
2360
|
+
{
|
|
2361
|
+
encoding: 'jsonParsed',
|
|
2362
|
+
},
|
|
2363
|
+
],
|
|
2364
|
+
},
|
|
2365
|
+
})
|
|
2366
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);
|
|
2367
|
+
callBack
|
|
2368
|
+
.withArgs({
|
|
2369
|
+
payload: {
|
|
2370
|
+
id: '1',
|
|
2371
|
+
jsonrpc: '2.0',
|
|
2372
|
+
method: 'getTokenAccountsByOwner',
|
|
2373
|
+
params: [
|
|
2374
|
+
testData.wrwUser.walletAddress5,
|
|
2375
|
+
{
|
|
2376
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
2377
|
+
},
|
|
2378
|
+
{
|
|
2379
|
+
encoding: 'jsonParsed',
|
|
2380
|
+
},
|
|
2381
|
+
],
|
|
2382
|
+
},
|
|
2383
|
+
})
|
|
2384
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponse);
|
|
2385
|
+
callBack
|
|
2386
|
+
.withArgs({
|
|
2387
|
+
payload: {
|
|
2388
|
+
id: '1',
|
|
2389
|
+
jsonrpc: '2.0',
|
|
2390
|
+
method: 'getTokenAccountsByOwner',
|
|
2391
|
+
params: [
|
|
2392
|
+
testData.wrwUser.walletAddress0,
|
|
2393
|
+
{
|
|
2394
|
+
programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',
|
|
2395
|
+
},
|
|
2396
|
+
{
|
|
2397
|
+
encoding: 'jsonParsed',
|
|
2398
|
+
},
|
|
2399
|
+
],
|
|
2400
|
+
},
|
|
2401
|
+
})
|
|
2402
|
+
.resolves(testData.SolResponses.getTokenAccountsByOwnerResponse);
|
|
2403
|
+
});
|
|
2404
|
+
afterEach(() => {
|
|
2405
|
+
sandBox.restore();
|
|
2406
|
+
});
|
|
2407
|
+
it('should build signed consolidation recoveries', async function () {
|
|
2408
|
+
const res = (await basecoin.recoverConsolidations({
|
|
2409
|
+
userKey: testData.wrwUser.userKey,
|
|
2410
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2411
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2412
|
+
walletPassphrase: testData.wrwUser.walletPassphrase,
|
|
2413
|
+
startingScanIndex: 1,
|
|
2414
|
+
endingScanIndex: 4,
|
|
2415
|
+
durableNonces: durableNonces,
|
|
2416
|
+
}));
|
|
2417
|
+
res.should.not.be.empty();
|
|
2418
|
+
res.transactions.length.should.equal(2);
|
|
2419
|
+
(res.lastScanIndex ?? 0).should.equal(3);
|
|
2420
|
+
const txn1 = res.transactions[0];
|
|
2421
|
+
const latestBlockhashTxnDeserialize1 = new lib_1.Transaction(coin);
|
|
2422
|
+
latestBlockhashTxnDeserialize1.fromRawTransaction(txn1.serializedTx);
|
|
2423
|
+
const latestBlockhashTxnJson1 = latestBlockhashTxnDeserialize1.toJson();
|
|
2424
|
+
const nonce1 = testData.SolResponses.getAccountInfoResponse.body.result.value.data.parsed.info.blockhash;
|
|
2425
|
+
should.equal(latestBlockhashTxnJson1.nonce, nonce1);
|
|
2426
|
+
should.equal(latestBlockhashTxnJson1.feePayer, testData.wrwUser.walletAddress2);
|
|
2427
|
+
should.equal(latestBlockhashTxnJson1.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
2428
|
+
const txn2 = res.transactions[1];
|
|
2429
|
+
const latestBlockhashTxnDeserialize2 = new lib_1.Transaction(coin);
|
|
2430
|
+
latestBlockhashTxnDeserialize2.fromRawTransaction(txn2.serializedTx);
|
|
2431
|
+
const latestBlockhashTxnJson2 = latestBlockhashTxnDeserialize2.toJson();
|
|
2432
|
+
const nonce2 = testData.SolResponses.getAccountInfoResponse2.body.result.value.data.parsed.info.blockhash;
|
|
2433
|
+
should.equal(latestBlockhashTxnJson2.nonce, nonce2);
|
|
2434
|
+
should.equal(latestBlockhashTxnJson2.feePayer, testData.wrwUser.walletAddress3);
|
|
2435
|
+
should.equal(latestBlockhashTxnJson2.numSignatures, testData.SolInputData.durableNonceSignatures);
|
|
2436
|
+
});
|
|
2437
|
+
it('should build unsigned consolidation recoveries', async function () {
|
|
2438
|
+
const res = (await basecoin.recoverConsolidations({
|
|
2439
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2440
|
+
startingScanIndex: 1,
|
|
2441
|
+
endingScanIndex: 4,
|
|
2442
|
+
durableNonces: durableNonces,
|
|
2443
|
+
}));
|
|
2444
|
+
res.should.not.be.empty();
|
|
2445
|
+
res.txRequests.length.should.equal(2);
|
|
2446
|
+
const txn1 = res.txRequests[0].transactions[0].unsignedTx;
|
|
2447
|
+
txn1.should.hasOwnProperty('serializedTx');
|
|
2448
|
+
txn1.should.hasOwnProperty('signableHex');
|
|
2449
|
+
txn1.should.hasOwnProperty('scanIndex');
|
|
2450
|
+
(txn1.scanIndex ?? 0).should.equal(2);
|
|
2451
|
+
txn1.should.hasOwnProperty('coin');
|
|
2452
|
+
txn1.coin?.should.equal('tsol');
|
|
2453
|
+
txn1.should.hasOwnProperty('derivationPath');
|
|
2454
|
+
txn1.derivationPath?.should.equal('m/2');
|
|
2455
|
+
txn1.should.hasOwnProperty('coinSpecific');
|
|
2456
|
+
const coinSpecific1 = txn1.coinSpecific;
|
|
2457
|
+
coinSpecific1?.should.hasOwnProperty('commonKeychain');
|
|
2458
|
+
const latestBlockhashTxnDeserialize1 = new lib_1.Transaction(coin);
|
|
2459
|
+
latestBlockhashTxnDeserialize1.fromRawTransaction(txn1.serializedTx);
|
|
2460
|
+
const latestBlockhashTxnJson1 = latestBlockhashTxnDeserialize1.toJson();
|
|
2461
|
+
const nonce1 = testData.SolResponses.getAccountInfoResponse.body.result.value.data.parsed.info.blockhash;
|
|
2462
|
+
should.equal(latestBlockhashTxnJson1.nonce, nonce1);
|
|
2463
|
+
should.equal(latestBlockhashTxnJson1.feePayer, testData.wrwUser.walletAddress2);
|
|
2464
|
+
should.equal(latestBlockhashTxnJson1.numSignatures, testData.SolInputData.unsignedSweepSignatures);
|
|
2465
|
+
const txn2 = res.txRequests[1].transactions[0].unsignedTx;
|
|
2466
|
+
txn2.should.hasOwnProperty('serializedTx');
|
|
2467
|
+
txn2.should.hasOwnProperty('signableHex');
|
|
2468
|
+
txn2.should.hasOwnProperty('scanIndex');
|
|
2469
|
+
(txn2.scanIndex ?? 0).should.equal(3);
|
|
2470
|
+
txn2.should.hasOwnProperty('coin');
|
|
2471
|
+
txn2.coin?.should.equal('tsol');
|
|
2472
|
+
txn2.should.hasOwnProperty('derivationPath');
|
|
2473
|
+
txn2.derivationPath?.should.equal('m/3');
|
|
2474
|
+
txn2.should.hasOwnProperty('coinSpecific');
|
|
2475
|
+
const coinSpecific2 = txn2.coinSpecific;
|
|
2476
|
+
coinSpecific2?.should.hasOwnProperty('commonKeychain');
|
|
2477
|
+
coinSpecific2?.should.hasOwnProperty('lastScanIndex');
|
|
2478
|
+
coinSpecific2?.lastScanIndex?.should.equal(3);
|
|
2479
|
+
const latestBlockhashTxnDeserialize2 = new lib_1.Transaction(coin);
|
|
2480
|
+
latestBlockhashTxnDeserialize2.fromRawTransaction(txn2.serializedTx);
|
|
2481
|
+
const latestBlockhashTxnJson2 = latestBlockhashTxnDeserialize2.toJson();
|
|
2482
|
+
const nonce2 = testData.SolResponses.getAccountInfoResponse2.body.result.value.data.parsed.info.blockhash;
|
|
2483
|
+
should.equal(latestBlockhashTxnJson2.nonce, nonce2);
|
|
2484
|
+
should.equal(latestBlockhashTxnJson2.feePayer, testData.wrwUser.walletAddress3);
|
|
2485
|
+
should.equal(latestBlockhashTxnJson2.numSignatures, testData.SolInputData.unsignedSweepSignatures);
|
|
2486
|
+
});
|
|
2487
|
+
it('should build unsigned token consolidation recoveries', async function () {
|
|
2488
|
+
const res = (await basecoin.recoverConsolidations({
|
|
2489
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2490
|
+
startingScanIndex: 3,
|
|
2491
|
+
endingScanIndex: 5,
|
|
2492
|
+
tokenContractAddress: usdtMintAddress,
|
|
2493
|
+
durableNonces: durableNonces,
|
|
2494
|
+
}));
|
|
2495
|
+
res.should.not.be.empty();
|
|
2496
|
+
res.txRequests.length.should.equal(1);
|
|
2497
|
+
const txn1 = res.txRequests[0].transactions[0].unsignedTx;
|
|
2498
|
+
txn1.should.hasOwnProperty('serializedTx');
|
|
2499
|
+
txn1.should.hasOwnProperty('signableHex');
|
|
2500
|
+
txn1.should.hasOwnProperty('scanIndex');
|
|
2501
|
+
(txn1.scanIndex ?? 0).should.equal(4);
|
|
2502
|
+
txn1.should.hasOwnProperty('coin');
|
|
2503
|
+
txn1.coin?.should.equal('tsol');
|
|
2504
|
+
txn1.should.hasOwnProperty('derivationPath');
|
|
2505
|
+
txn1.derivationPath?.should.equal('m/4');
|
|
2506
|
+
txn1.should.hasOwnProperty('coinSpecific');
|
|
2507
|
+
const coinSpecific1 = txn1.coinSpecific;
|
|
2508
|
+
coinSpecific1?.should.hasOwnProperty('commonKeychain');
|
|
2509
|
+
const latestBlockhashTxnDeserialize1 = new lib_1.Transaction(coin);
|
|
2510
|
+
latestBlockhashTxnDeserialize1.fromRawTransaction(txn1.serializedTx);
|
|
2511
|
+
const latestBlockhashTxnJson1 = latestBlockhashTxnDeserialize1.toJson();
|
|
2512
|
+
const nonce1 = testData.SolResponses.getAccountInfoResponse.body.result.value.data.parsed.info.blockhash;
|
|
2513
|
+
should.equal(latestBlockhashTxnJson1.nonce, nonce1);
|
|
2514
|
+
should.equal(latestBlockhashTxnJson1.feePayer, testData.wrwUser.walletAddress5);
|
|
2515
|
+
should.equal(latestBlockhashTxnJson1.numSignatures, testData.SolInputData.unsignedSweepSignatures);
|
|
2516
|
+
});
|
|
2517
|
+
it('should skip building consolidate transaction if balance is equal to zero', async function () {
|
|
2518
|
+
await basecoin
|
|
2519
|
+
.recoverConsolidations({
|
|
2520
|
+
userKey: testData.wrwUser.userKey,
|
|
2521
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2522
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2523
|
+
walletPassphrase: testData.wrwUser.walletPassphrase,
|
|
2524
|
+
startingScanIndex: 1,
|
|
2525
|
+
endingScanIndex: 2,
|
|
2526
|
+
durableNonces: durableNonces,
|
|
2527
|
+
})
|
|
2528
|
+
.should.rejectedWith('Did not find an address with funds to recover');
|
|
2529
|
+
});
|
|
2530
|
+
it('should throw if startingScanIndex is not ge to 1', async () => {
|
|
2531
|
+
await basecoin
|
|
2532
|
+
.recoverConsolidations({
|
|
2533
|
+
userKey: testData.wrwUser.userKey,
|
|
2534
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2535
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2536
|
+
startingScanIndex: -1,
|
|
2537
|
+
durableNonces: durableNonces,
|
|
2538
|
+
})
|
|
2539
|
+
.should.be.rejectedWith('Invalid starting or ending index to scan for addresses. startingScanIndex: -1, endingScanIndex: 19.');
|
|
2540
|
+
});
|
|
2541
|
+
it('should throw if scan factor is too high', async () => {
|
|
2542
|
+
await basecoin
|
|
2543
|
+
.recoverConsolidations({
|
|
2544
|
+
userKey: testData.wrwUser.userKey,
|
|
2545
|
+
backupKey: testData.wrwUser.backupKey,
|
|
2546
|
+
bitgoKey: testData.wrwUser.bitgoKey,
|
|
2547
|
+
startingScanIndex: 1,
|
|
2548
|
+
endingScanIndex: 300,
|
|
2549
|
+
durableNonces: durableNonces,
|
|
2550
|
+
})
|
|
2551
|
+
.should.be.rejectedWith('Invalid starting or ending index to scan for addresses. startingScanIndex: 1, endingScanIndex: 300.');
|
|
2552
|
+
});
|
|
2553
|
+
});
|
|
2554
|
+
describe('broadcastTransaction', function () {
|
|
2555
|
+
const sandBox = sinon.createSandbox();
|
|
2556
|
+
afterEach(() => {
|
|
2557
|
+
sandBox.restore();
|
|
2558
|
+
});
|
|
2559
|
+
it('should broadcast a transaction succesfully', async function () {
|
|
2560
|
+
const serializedSignedTransaction = testData.rawTransactions.transfer.signed;
|
|
2561
|
+
const broadcastStub = sandBox
|
|
2562
|
+
.stub(src_1.Sol.prototype, 'getDataFromNode')
|
|
2563
|
+
.withArgs({
|
|
2564
|
+
payload: {
|
|
2565
|
+
id: '1',
|
|
2566
|
+
jsonrpc: '2.0',
|
|
2567
|
+
method: 'sendTransaction',
|
|
2568
|
+
params: [
|
|
2569
|
+
serializedSignedTransaction,
|
|
2570
|
+
{
|
|
2571
|
+
encoding: 'base64',
|
|
2572
|
+
},
|
|
2573
|
+
],
|
|
2574
|
+
},
|
|
2575
|
+
})
|
|
2576
|
+
.resolves(testData.SolResponses.broadcastTransactionResponse);
|
|
2577
|
+
const broadcastTxn = await basecoin.broadcastTransaction({ serializedSignedTransaction });
|
|
2578
|
+
assert_1.default.ok(broadcastTxn);
|
|
2579
|
+
assert_1.default.ok(broadcastTxn.txId);
|
|
2580
|
+
assert_1.default.strictEqual(broadcastTxn.txId, '2id3YC2jK9G5Wo2phDx4gJVAew8DcY5NAojnVuao8rkxwPYPe8cSwE5GzhEgJA2y8fVjDEo6iR6ykBvDxrTQrtpb');
|
|
2581
|
+
assert_1.default.strictEqual(broadcastStub.callCount, 1);
|
|
2582
|
+
});
|
|
2583
|
+
it('should throw if got an error from the node', async function () {
|
|
2584
|
+
const serializedSignedTransaction = testData.rawTransactions.transfer.signed;
|
|
2585
|
+
const broadcastStub = sandBox
|
|
2586
|
+
.stub(src_1.Sol.prototype, 'getDataFromNode')
|
|
2587
|
+
.withArgs({
|
|
2588
|
+
payload: {
|
|
2589
|
+
id: '1',
|
|
2590
|
+
jsonrpc: '2.0',
|
|
2591
|
+
method: 'sendTransaction',
|
|
2592
|
+
params: [
|
|
2593
|
+
serializedSignedTransaction,
|
|
2594
|
+
{
|
|
2595
|
+
encoding: 'base64',
|
|
2596
|
+
},
|
|
2597
|
+
],
|
|
2598
|
+
},
|
|
2599
|
+
})
|
|
2600
|
+
.resolves(testData.SolResponses.broadcastTransactionResponseError);
|
|
2601
|
+
await assert_1.default.rejects(async () => {
|
|
2602
|
+
await basecoin.broadcastTransaction({ serializedSignedTransaction });
|
|
2603
|
+
}, { message: 'Error broadcasting transaction: Transaction simulation failed: Blockhash not found' });
|
|
2604
|
+
assert_1.default.strictEqual(broadcastStub.callCount, 1);
|
|
2605
|
+
});
|
|
2606
|
+
it('should throw if is not a valid transaction', async function () {
|
|
2607
|
+
const serializedSignedTransaction = 'randomstring';
|
|
2608
|
+
await assert_1.default.rejects(async () => {
|
|
2609
|
+
await basecoin.broadcastTransaction({ serializedSignedTransaction });
|
|
2610
|
+
}, { message: 'Invalid raw transaction' });
|
|
2611
|
+
});
|
|
2612
|
+
it('should throw if is not a signed transaction', async function () {
|
|
2613
|
+
const serializedSignedTransaction = testData.rawTransactions.transfer.unsigned;
|
|
2614
|
+
await assert_1.default.rejects(async () => {
|
|
2615
|
+
await basecoin.broadcastTransaction({ serializedSignedTransaction });
|
|
2616
|
+
}, { message: 'Invalid raw transaction' });
|
|
2617
|
+
});
|
|
2618
|
+
});
|
|
2619
|
+
describe('AuditKey', () => {
|
|
2620
|
+
const { key: keyString, commonKeychain } = solBackupKey_1.solBackupKey;
|
|
2621
|
+
const key = keyString.replace(/\s/g, '');
|
|
2622
|
+
const walletPassphrase = 'kAm[EFQ6o=SxlcLFDw%,';
|
|
2623
|
+
const multiSigType = 'tss';
|
|
2624
|
+
it('should return for valid inputs', () => {
|
|
2625
|
+
basecoin.assertIsValidKey({
|
|
2626
|
+
encryptedPrv: key,
|
|
2627
|
+
publicKey: commonKeychain,
|
|
2628
|
+
walletPassphrase,
|
|
2629
|
+
multiSigType,
|
|
2630
|
+
});
|
|
2631
|
+
});
|
|
2632
|
+
it('should throw error if the commonKeychain is invalid', () => {
|
|
2633
|
+
const alteredCommonKeychain = (0, sdk_core_1.generateRandomPassword)(10);
|
|
2634
|
+
assert_1.default.throws(() => basecoin.assertIsValidKey({
|
|
2635
|
+
encryptedPrv: key,
|
|
2636
|
+
publicKey: alteredCommonKeychain,
|
|
2637
|
+
walletPassphrase,
|
|
2638
|
+
multiSigType,
|
|
2639
|
+
}), {
|
|
2640
|
+
message: 'Invalid common keychain',
|
|
2641
|
+
});
|
|
2642
|
+
});
|
|
2643
|
+
it('should throw error if the walletPassphrase is incorrect', () => {
|
|
2644
|
+
const incorrectPassphrase = 'foo';
|
|
2645
|
+
assert_1.default.throws(() => basecoin.assertIsValidKey({
|
|
2646
|
+
encryptedPrv: key,
|
|
2647
|
+
publicKey: commonKeychain,
|
|
2648
|
+
walletPassphrase: incorrectPassphrase,
|
|
2649
|
+
multiSigType,
|
|
2650
|
+
}), {
|
|
2651
|
+
message: "failed to decrypt prv: ccm: tag doesn't match",
|
|
2652
|
+
});
|
|
2653
|
+
});
|
|
2654
|
+
it('should throw error if the key is altered', () => {
|
|
2655
|
+
const alteredKey = key.replace(/[0-9]/g, '0');
|
|
2656
|
+
assert_1.default.throws(() => basecoin.assertIsValidKey({
|
|
2657
|
+
encryptedPrv: alteredKey,
|
|
2658
|
+
publicKey: commonKeychain,
|
|
2659
|
+
walletPassphrase,
|
|
2660
|
+
multiSigType,
|
|
2661
|
+
}), {
|
|
2662
|
+
message: 'failed to decrypt prv: json decrypt: invalid parameters',
|
|
2663
|
+
});
|
|
2664
|
+
});
|
|
2665
|
+
it('should verify consolidation transaction', async function () {
|
|
2666
|
+
// Set up wallet data
|
|
2667
|
+
const walletData = {
|
|
2668
|
+
id: '5b34252f1bf349930e34020a00000000',
|
|
2669
|
+
coin: 'tsol',
|
|
2670
|
+
keys: [
|
|
2671
|
+
'5b3424f91bf349930e34017500000000',
|
|
2672
|
+
'5b3424f91bf349930e34017600000000',
|
|
2673
|
+
'5b3424f91bf349930e34017700000000',
|
|
2674
|
+
],
|
|
2675
|
+
coinSpecific: {
|
|
2676
|
+
rootAddress: wallet.pub,
|
|
2677
|
+
},
|
|
2678
|
+
multisigType: 'tss',
|
|
2679
|
+
};
|
|
2680
|
+
const fakePrv = (0, sdk_api_1.encrypt)('password', 'prv');
|
|
2681
|
+
const walletObj = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
2682
|
+
const bgUrl = sdk_core_1.common.Environments['mock'].uri;
|
|
2683
|
+
(0, nock_1.default)(bgUrl)
|
|
2684
|
+
.get('/api/v2/tsol/key/5b3424f91bf349930e34017500000000')
|
|
2685
|
+
.reply(200, [
|
|
2686
|
+
{
|
|
2687
|
+
encryptedPrv: fakePrv,
|
|
2688
|
+
},
|
|
2689
|
+
]);
|
|
2690
|
+
// Mock the API response for buildAccountConsolidations
|
|
2691
|
+
(0, nock_1.default)(bgUrl)
|
|
2692
|
+
.post('/api/v2/tsol/wallet/5b34252f1bf349930e34020a00000000/consolidateAccount/build')
|
|
2693
|
+
.reply(200, [
|
|
2694
|
+
{
|
|
2695
|
+
txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',
|
|
2696
|
+
walletId: '63068ed4efa63a000877f02fd4b0fa6d',
|
|
2697
|
+
txHex: '02000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006f50130abf8d0d8943c5b9a51a574886a7d7b3d8db18f0ddb4ab8b6d3ec27e3c2f36f3339bb92d4296af6ae4d3abfbb07877f77d0033c883de08fa4a2eea670d0201020674a9df2b94aa4b4ada1202dc2891be366501d0acb4a01ca3e02e7fd6c1f505a71c96172044f1217c3784e8f02f49e2c8fc3591e81294ab54394f9d22fd7b7a8f8401c3f67cfa52518b34a09b08f4ea77e1c4fb9d89bfaccdc33cf8b8a9cf8d7bf0e04c89c50428e4eda5cbb759427c370f0a29a50bb0d1407e57924b0cc5b36f000000000000000000000000000000000000000000000000000000000000000006a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000040c9568195d67eb6b396fdbe97ba2e276622ef0023af09f23e6e0292abb678d90204030305010404000000040200020c020000000100000000000000',
|
|
2698
|
+
feeInfo: {
|
|
2699
|
+
fee: 10000,
|
|
2700
|
+
feeString: '10000',
|
|
2701
|
+
},
|
|
2702
|
+
txInfo: {
|
|
2703
|
+
minerFee: '10000',
|
|
2704
|
+
spendAmount: '1547864',
|
|
2705
|
+
spendAmounts: [
|
|
2706
|
+
{
|
|
2707
|
+
coinName: 'tsol',
|
|
2708
|
+
amountString: '1547864',
|
|
2709
|
+
},
|
|
2710
|
+
],
|
|
2711
|
+
payGoFee: '0',
|
|
2712
|
+
outputs: [
|
|
2713
|
+
{
|
|
2714
|
+
address: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',
|
|
2715
|
+
value: 1547864,
|
|
2716
|
+
wallet: '63068ed4efa63a000877f02f',
|
|
2717
|
+
wallets: ['63068ed4efa63a000877f02f'],
|
|
2718
|
+
enterprise: '62d71a6b86068f0008f029fd',
|
|
2719
|
+
enterprises: ['62d71a6b86068f0008f029fd'],
|
|
2720
|
+
valueString: '1547864',
|
|
2721
|
+
coinName: 'tsol',
|
|
2722
|
+
walletType: 'hot',
|
|
2723
|
+
walletTypes: ['hot'],
|
|
2724
|
+
},
|
|
2725
|
+
],
|
|
2726
|
+
inputs: [
|
|
2727
|
+
{
|
|
2728
|
+
value: 1547864,
|
|
2729
|
+
address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',
|
|
2730
|
+
valueString: '1547864',
|
|
2731
|
+
},
|
|
2732
|
+
{
|
|
2733
|
+
value: 10000,
|
|
2734
|
+
address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',
|
|
2735
|
+
valueString: '10000',
|
|
2736
|
+
},
|
|
2737
|
+
],
|
|
2738
|
+
type: 'Send',
|
|
2739
|
+
},
|
|
2740
|
+
consolidateId: '68a7d5d0c66e74e216b97173bd558c6d',
|
|
2741
|
+
coin: 'tsol',
|
|
2742
|
+
},
|
|
2743
|
+
]);
|
|
2744
|
+
// Call the function to test
|
|
2745
|
+
await assert_1.default.rejects(async () => {
|
|
2746
|
+
await walletObj.sendAccountConsolidations({
|
|
2747
|
+
walletPassphrase: 'password',
|
|
2748
|
+
});
|
|
2749
|
+
}, {
|
|
2750
|
+
message: 'tx outputs does not match with expected address',
|
|
2751
|
+
});
|
|
2752
|
+
});
|
|
2753
|
+
it('should verify valid a consolidation transaction', async () => {
|
|
2754
|
+
const consolidationTx = {
|
|
2755
|
+
txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',
|
|
2756
|
+
walletId: '63068ed4efa63a000877f02fd4b0fa6d',
|
|
2757
|
+
txHex: '02000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002010206aeda253a331de489838246df93879440af8c62ac4967658edc2bb5d52b9759d91c96172044f1217c3784e8f02f49e2c8fc3591e81294ab54394f9d22fd7b7a8f74a9df2b94aa4b4ada1202dc2891be366501d0acb4a01ca3e02e7fd6c1f505a7d2734f3952f3eb4aefcf6c7a6092e979dd3fe5563ccfaca1cc92652a15ddd393000000000000000000000000000000000000000000000000000000000000000006a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea9400000428dad41bbedeb38018379e4ceeb7e80757d74dd00ae8c47c38d597477339ad80204030305010404000000040200020c02000000589e170000000000',
|
|
2758
|
+
feeInfo: {
|
|
2759
|
+
fee: 10000,
|
|
2760
|
+
feeString: '10000',
|
|
2761
|
+
},
|
|
2762
|
+
txInfo: {
|
|
2763
|
+
minerFee: '10000',
|
|
2764
|
+
spendAmount: '1547864',
|
|
2765
|
+
spendAmounts: [
|
|
2766
|
+
{
|
|
2767
|
+
coinName: 'tsol',
|
|
2768
|
+
amountString: '1547864',
|
|
2769
|
+
},
|
|
2770
|
+
],
|
|
2771
|
+
payGoFee: '0',
|
|
2772
|
+
outputs: [
|
|
2773
|
+
{
|
|
2774
|
+
address: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',
|
|
2775
|
+
value: 1547864,
|
|
2776
|
+
wallet: '63068ed4efa63a000877f02f',
|
|
2777
|
+
wallets: ['63068ed4efa63a000877f02f'],
|
|
2778
|
+
enterprise: '62d71a6b86068f0008f029fd',
|
|
2779
|
+
enterprises: ['62d71a6b86068f0008f029fd'],
|
|
2780
|
+
valueString: '1547864',
|
|
2781
|
+
coinName: 'tsol',
|
|
2782
|
+
walletType: 'hot',
|
|
2783
|
+
walletTypes: ['hot'],
|
|
2784
|
+
},
|
|
2785
|
+
],
|
|
2786
|
+
inputs: [
|
|
2787
|
+
{
|
|
2788
|
+
value: 1547864,
|
|
2789
|
+
address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',
|
|
2790
|
+
valueString: '1547864',
|
|
2791
|
+
},
|
|
2792
|
+
{
|
|
2793
|
+
value: 10000,
|
|
2794
|
+
address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',
|
|
2795
|
+
valueString: '10000',
|
|
2796
|
+
},
|
|
2797
|
+
],
|
|
2798
|
+
type: 'Send',
|
|
2799
|
+
},
|
|
2800
|
+
consolidateId: '68a7d5d0c66e74e216b97173bd558c6d',
|
|
2801
|
+
coin: 'tsol',
|
|
2802
|
+
};
|
|
2803
|
+
const mockedWallet = {
|
|
2804
|
+
coinSpecific: () => {
|
|
2805
|
+
const cs = {
|
|
2806
|
+
rootAddress: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',
|
|
2807
|
+
};
|
|
2808
|
+
return cs;
|
|
2809
|
+
},
|
|
2810
|
+
};
|
|
2811
|
+
try {
|
|
2812
|
+
if (!(await basecoin.verifyTransaction({
|
|
2813
|
+
blockhash: '',
|
|
2814
|
+
feePayer: '',
|
|
2815
|
+
txParams: {},
|
|
2816
|
+
txPrebuild: consolidationTx,
|
|
2817
|
+
walletType: 'tss',
|
|
2818
|
+
wallet: mockedWallet,
|
|
2819
|
+
verification: {
|
|
2820
|
+
consolidationToBaseAddress: true,
|
|
2821
|
+
},
|
|
2822
|
+
}))) {
|
|
2823
|
+
assert_1.default.fail('Transaction should pass verification');
|
|
2824
|
+
}
|
|
2825
|
+
}
|
|
2826
|
+
catch (e) {
|
|
2827
|
+
assert_1.default.fail('Transaction should pass verification');
|
|
2828
|
+
}
|
|
2829
|
+
});
|
|
2830
|
+
it('should verify a token consolidation transaction', async () => {
|
|
2831
|
+
const consolidationTx = {
|
|
2832
|
+
txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',
|
|
2833
|
+
walletId: '63068ed4efa63a000877f02fd4b0fa6d',
|
|
2834
|
+
txHex: '02b7c2c7829eded4e8f947c90ed3b9afce71f616eb47dfcfbf4b765778149060013acb33b9f67fdd9c2512f48fac4e4c049eab93829b69404f3bd166fe3242c90700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000020104096da690bd558fd8634ac14f1645d8095afd0caea5953578c596e3c0dea38305ee7298bfac55101f177735659d42ed6be890ef3a1d204d9e33f32e24c5635327ca66d1fe00826e5a4f759f87b279c1aee19cce5301af4ed66ae17db48b201ed6c2a0e8a28bf565627f1ab8a34b1a95ee1b0a2a39084f1f0e2acb1c394b20185d8e0b22657c8d9c4ce5f6495efb6410c199011530f90e3ab9d8d1e4206f9ae0ffeb0000000000000000000000000000000000000000000000000000000000000000c5f9fb32f49111ab20c33f2598fc836c113e291881ac21ee29169394011244e406a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000006ddf6e1d765a193d9cbe146ceeb79ac1cb485ed5f5b37913a8cf5857eff00a9de13c74d2b4d948e1608ea6eebdafe75bc2f995aad11b21d1e2c94f2a2d12f6802050303070004040000000804020604010a0ce0076bb20400000006',
|
|
2835
|
+
feeInfo: {
|
|
2836
|
+
fee: 10000,
|
|
2837
|
+
feeString: '10000',
|
|
2838
|
+
},
|
|
2839
|
+
txInfo: {
|
|
2840
|
+
inputs: [
|
|
2841
|
+
{
|
|
2842
|
+
address: '8iLa26KSbdpBUzNK7uYq8FvyuyA5h4k4erDHsDcPbHus',
|
|
2843
|
+
value: 2.0173228e10,
|
|
2844
|
+
valueString: '20173228000',
|
|
2845
|
+
},
|
|
2846
|
+
{
|
|
2847
|
+
address: '8P2kX7Tyh9eS3RKdaBhqbEtQGAX58DXzL7mhDQABGX2d',
|
|
2848
|
+
value: 10000,
|
|
2849
|
+
valueString: '10000',
|
|
2850
|
+
},
|
|
2851
|
+
],
|
|
2852
|
+
minerFee: '10000',
|
|
2853
|
+
outputs: [
|
|
2854
|
+
{
|
|
2855
|
+
address: 'HBxZShcE86UMmF93KUM8eWJKqeEXi5cqWCLYLMMhqMYm',
|
|
2856
|
+
coinName: 'sol:wif',
|
|
2857
|
+
enterprise: {
|
|
2858
|
+
$oid: '5553ba8ae7a5c77006719661',
|
|
2859
|
+
},
|
|
2860
|
+
enterprises: [
|
|
2861
|
+
{
|
|
2862
|
+
$oid: '5553ba8ae7a5c77006719661',
|
|
2863
|
+
},
|
|
2864
|
+
],
|
|
2865
|
+
value: 2.0173228e10,
|
|
2866
|
+
valueString: '20173228000',
|
|
2867
|
+
wallet: {
|
|
2868
|
+
$oid: '62f4c3720d92c50008257eb5',
|
|
2869
|
+
},
|
|
2870
|
+
walletType: 'hot',
|
|
2871
|
+
wallets: [
|
|
2872
|
+
{
|
|
2873
|
+
$oid: '62f4c3720d92c50008257eb5',
|
|
2874
|
+
},
|
|
2875
|
+
],
|
|
2876
|
+
},
|
|
2877
|
+
],
|
|
2878
|
+
payGoFee: '0',
|
|
2879
|
+
spendAmount: '20173228000',
|
|
2880
|
+
spendAmounts: [
|
|
2881
|
+
{
|
|
2882
|
+
amountString: '20173228000',
|
|
2883
|
+
coinName: 'sol:wif',
|
|
2884
|
+
},
|
|
2885
|
+
],
|
|
2886
|
+
type: 'Send',
|
|
2887
|
+
},
|
|
2888
|
+
consolidateId: '6712d7fda6de4906d658c04aebbf8f9b',
|
|
2889
|
+
coin: 'tsol',
|
|
2890
|
+
};
|
|
2891
|
+
const mockedWallet = {
|
|
2892
|
+
coinSpecific: () => {
|
|
2893
|
+
const cs = {
|
|
2894
|
+
rootAddress: 'HBxZShcE86UMmF93KUM8eWJKqeEXi5cqWCLYLMMhqMYm',
|
|
2895
|
+
};
|
|
2896
|
+
return cs;
|
|
2897
|
+
},
|
|
2898
|
+
};
|
|
2899
|
+
try {
|
|
2900
|
+
if (!(await basecoin.verifyTransaction({
|
|
2901
|
+
blockhash: '',
|
|
2902
|
+
feePayer: '',
|
|
2903
|
+
txParams: {},
|
|
2904
|
+
txPrebuild: consolidationTx,
|
|
2905
|
+
walletType: 'tss',
|
|
2906
|
+
wallet: mockedWallet,
|
|
2907
|
+
verification: {
|
|
2908
|
+
consolidationToBaseAddress: true,
|
|
2909
|
+
},
|
|
2910
|
+
}))) {
|
|
2911
|
+
assert_1.default.fail('Transaction should pass verification');
|
|
2912
|
+
}
|
|
2913
|
+
}
|
|
2914
|
+
catch (e) {
|
|
2915
|
+
assert_1.default.fail('Transaction should pass verification');
|
|
2916
|
+
}
|
|
2917
|
+
});
|
|
2918
|
+
it('should verify a spoofed token consolidation transaction', async () => {
|
|
2919
|
+
const consolidationTx = {
|
|
2920
|
+
txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',
|
|
2921
|
+
walletId: '63068ed4efa63a000877f02fd4b0fa6d',
|
|
2922
|
+
txHex: '02b7c2c7829eded4e8f947c90ed3b9afce71f616eb47dfcfbf4b765778149060013acb33b9f67fdd9c2512f48fac4e4c049eab93829b69404f3bd166fe3242c90700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000020104096da690bd558fd8634ac14f1645d8095afd0caea5953578c596e3c0dea38305ee7298bfac55101f177735659d42ed6be890ef3a1d204d9e33f32e24c5635327ca66d1fe00826e5a4f759f87b279c1aee19cce5301af4ed66ae17db48b201ed6c2a0e8a28bf565627f1ab8a34b1a95ee1b0a2a39084f1f0e2acb1c394b20185d8e0b22657c8d9c4ce5f6495efb6410c199011530f90e3ab9d8d1e4206f9ae0ffeb0000000000000000000000000000000000000000000000000000000000000000c5f9fb32f49111ab20c33f2598fc836c113e291881ac21ee29169394011244e406a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000006ddf6e1d765a193d9cbe146ceeb79ac1cb485ed5f5b37913a8cf5857eff00a9de13c74d2b4d948e1608ea6eebdafe75bc2f995aad11b21d1e2c94f2a2d12f6802050303070004040000000804020604010a0ce0076bb20400000006',
|
|
2923
|
+
feeInfo: {
|
|
2924
|
+
fee: 10000,
|
|
2925
|
+
feeString: '10000',
|
|
2926
|
+
},
|
|
2927
|
+
txInfo: {
|
|
2928
|
+
inputs: [
|
|
2929
|
+
{
|
|
2930
|
+
address: '8iLa26KSbdpBUzNK7uYq8FvyuyA5h4k4erDHsDcPbHus',
|
|
2931
|
+
value: 2.0173228e10,
|
|
2932
|
+
valueString: '20173228000',
|
|
2933
|
+
},
|
|
2934
|
+
{
|
|
2935
|
+
address: '8P2kX7Tyh9eS3RKdaBhqbEtQGAX58DXzL7mhDQABGX2d',
|
|
2936
|
+
value: 10000,
|
|
2937
|
+
valueString: '10000',
|
|
2938
|
+
},
|
|
2939
|
+
],
|
|
2940
|
+
minerFee: '10000',
|
|
2941
|
+
outputs: [
|
|
2942
|
+
{
|
|
2943
|
+
address: 'HBxZShcE86UMmF93KUM8eWJKqeEXi5cqWCLYLMMhqMYm',
|
|
2944
|
+
coinName: 'sol:wif',
|
|
2945
|
+
enterprise: {
|
|
2946
|
+
$oid: '5553ba8ae7a5c77006719661',
|
|
2947
|
+
},
|
|
2948
|
+
enterprises: [
|
|
2949
|
+
{
|
|
2950
|
+
$oid: '5553ba8ae7a5c77006719661',
|
|
2951
|
+
},
|
|
2952
|
+
],
|
|
2953
|
+
value: 2.0173228e10,
|
|
2954
|
+
valueString: '20173228000',
|
|
2955
|
+
wallet: {
|
|
2956
|
+
$oid: '62f4c3720d92c50008257eb5',
|
|
2957
|
+
},
|
|
2958
|
+
walletType: 'hot',
|
|
2959
|
+
wallets: [
|
|
2960
|
+
{
|
|
2961
|
+
$oid: '62f4c3720d92c50008257eb5',
|
|
2962
|
+
},
|
|
2963
|
+
],
|
|
2964
|
+
},
|
|
2965
|
+
],
|
|
2966
|
+
payGoFee: '0',
|
|
2967
|
+
spendAmount: '20173228000',
|
|
2968
|
+
spendAmounts: [
|
|
2969
|
+
{
|
|
2970
|
+
amountString: '20173228000',
|
|
2971
|
+
coinName: 'sol:wif',
|
|
2972
|
+
},
|
|
2973
|
+
],
|
|
2974
|
+
type: 'Send',
|
|
2975
|
+
},
|
|
2976
|
+
consolidateId: '6712d7fda6de4906d658c04aebbf8f9b',
|
|
2977
|
+
coin: 'tsol',
|
|
2978
|
+
};
|
|
2979
|
+
const mockedWallet = {
|
|
2980
|
+
coinSpecific: () => {
|
|
2981
|
+
const cs = {
|
|
2982
|
+
rootAddress: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',
|
|
2983
|
+
};
|
|
2984
|
+
return cs;
|
|
2985
|
+
},
|
|
2986
|
+
};
|
|
2987
|
+
await assert_1.default.rejects(async () => basecoin.verifyTransaction({
|
|
2988
|
+
blockhash: '',
|
|
2989
|
+
feePayer: '',
|
|
2990
|
+
txParams: {},
|
|
2991
|
+
txPrebuild: consolidationTx,
|
|
2992
|
+
walletType: 'tss',
|
|
2993
|
+
wallet: mockedWallet,
|
|
2994
|
+
verification: {
|
|
2995
|
+
consolidationToBaseAddress: true,
|
|
2996
|
+
},
|
|
2997
|
+
}), {
|
|
2998
|
+
message: 'tx outputs does not match with expected address',
|
|
2999
|
+
});
|
|
3000
|
+
});
|
|
3001
|
+
});
|
|
3002
|
+
describe('blind signing token enablement protection', () => {
|
|
3003
|
+
it('should verify as valid the enabletoken intent when prebuild tx matchs user intent ', async function () {
|
|
3004
|
+
const { txParams, txPrebuildRaw, walletData } = testData.enableTokenFixtures;
|
|
3005
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
3006
|
+
const sameIntentTx = await basecoin.verifyTransaction({
|
|
3007
|
+
txParams,
|
|
3008
|
+
txPrebuild: txPrebuildRaw,
|
|
3009
|
+
wallet,
|
|
3010
|
+
verification: { verifyTokenEnablement: true },
|
|
3011
|
+
});
|
|
3012
|
+
sameIntentTx.should.equal(true);
|
|
3013
|
+
});
|
|
3014
|
+
it('should thrown an error when tampered prebuild tx type ', async function () {
|
|
3015
|
+
const { txParams, txPrebuildRaw, sendTxHex, walletData } = testData.enableTokenFixtures;
|
|
3016
|
+
const tamperedTxPrebuild = { ...txPrebuildRaw, txHex: sendTxHex };
|
|
3017
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
3018
|
+
await assert_1.default.rejects(async () => await basecoin.verifyTransaction({
|
|
3019
|
+
txParams,
|
|
3020
|
+
txPrebuild: tamperedTxPrebuild,
|
|
3021
|
+
wallet,
|
|
3022
|
+
verification: { verifyTokenEnablement: true },
|
|
3023
|
+
}), {
|
|
3024
|
+
message: 'Invalid transaction type on token enablement: expected "AssociatedTokenAccountInitialization", got "Send".',
|
|
3025
|
+
});
|
|
3026
|
+
});
|
|
3027
|
+
it('should verify that tokenName matches between user intent and hex', async function () {
|
|
3028
|
+
const { txParams, txPrebuildRaw, wrongTokenNameTxHex, walletData } = testData.enableTokenFixtures;
|
|
3029
|
+
const tamperedTxPrebuild = { ...txPrebuildRaw, txHex: wrongTokenNameTxHex };
|
|
3030
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
3031
|
+
await assert_1.default.rejects(async () => basecoin.verifyTransaction({
|
|
3032
|
+
txParams,
|
|
3033
|
+
txPrebuild: tamperedTxPrebuild,
|
|
3034
|
+
wallet,
|
|
3035
|
+
verification: { verifyTokenEnablement: true },
|
|
3036
|
+
}), { message: 'Invalid token name: expected tsol:ray, got tsol:t22mint on token enablement tx' });
|
|
3037
|
+
});
|
|
3038
|
+
it('should verify that tokenAddr matches between user intent and hex', async function () {
|
|
3039
|
+
const { txParams, txPrebuildRaw, wrongAddrTxHex, walletData } = testData.enableTokenFixtures;
|
|
3040
|
+
const tamperedTxPrebuild = { ...txPrebuildRaw, txHex: wrongAddrTxHex };
|
|
3041
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
3042
|
+
await assert_1.default.rejects(async () => basecoin.verifyTransaction({
|
|
3043
|
+
txParams,
|
|
3044
|
+
txPrebuild: tamperedTxPrebuild,
|
|
3045
|
+
wallet,
|
|
3046
|
+
verification: { verifyTokenEnablement: true },
|
|
3047
|
+
}), {
|
|
3048
|
+
message: 'Invalid token address: expected 4bTYvvv2Hk4v2kQW8HZFFS4SzYPztQshw9Gm1suXmaBj, got G1LEgANAwKo7b8NfxTsMzrbBYDkXqi5REVJY8thrMRQm on token enablement tx',
|
|
3049
|
+
});
|
|
3050
|
+
});
|
|
3051
|
+
it('should fail sendTokenEnablement call on spoofed data', async function () {
|
|
3052
|
+
const { sendTokenEnablementPayload, walletData, wrongTokenNameTxHex } = testData.enableTokenFixtures;
|
|
3053
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
|
|
3054
|
+
(0, nock_1.default)('https://bitgo.fakeurl').get('/api/v2/tsol/key/68bafed588671cf94ed8a5dbba882ad3').reply(200, {});
|
|
3055
|
+
await assert_1.default.rejects(async () => wallet.sendTokenEnablement({
|
|
3056
|
+
verification: { verifyTokenEnablement: true },
|
|
3057
|
+
...sendTokenEnablementPayload,
|
|
3058
|
+
prebuildTx: { ...sendTokenEnablementPayload.prebuildTx, txHex: wrongTokenNameTxHex },
|
|
3059
|
+
}), {
|
|
3060
|
+
message: 'Invalid token name: expected tsol:ray, got tsol:t22mint on token enablement tx',
|
|
3061
|
+
});
|
|
3062
|
+
});
|
|
3063
|
+
});
|
|
3064
|
+
describe('isWalletAddress', () => {
|
|
3065
|
+
it('should verify valid wallet address with correct keychain and index', async function () {
|
|
3066
|
+
const address = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';
|
|
3067
|
+
const commonKeychain = '8ea32ecacfc83effbd2e2790ee44fa7c59b4d86c29a12f09fb613d8195f93f4e21875cad3b98adada40c040c54c3569467df41a020881a6184096378701862bd';
|
|
3068
|
+
const index = '1';
|
|
3069
|
+
const keychains = [{ id: '1', type: 'tss', commonKeychain }];
|
|
3070
|
+
const result = await basecoin.isWalletAddress({ keychains, address, index });
|
|
3071
|
+
result.should.equal(true);
|
|
3072
|
+
});
|
|
3073
|
+
it('should return false for address with incorrect keychain', async function () {
|
|
3074
|
+
const address = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';
|
|
3075
|
+
const wrongKeychain = '0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000';
|
|
3076
|
+
const index = '1';
|
|
3077
|
+
const keychains = [{ id: '1', type: 'tss', commonKeychain: wrongKeychain }];
|
|
3078
|
+
const result = await basecoin.isWalletAddress({ keychains, address, index });
|
|
3079
|
+
result.should.equal(false);
|
|
3080
|
+
});
|
|
3081
|
+
it('should return false for address with incorrect index', async function () {
|
|
3082
|
+
const address = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';
|
|
3083
|
+
const commonKeychain = '8ea32ecacfc83effbd2e2790ee44fa7c59b4d86c29a12f09fb613d8195f93f4e21875cad3b98adada40c040c54c3569467df41a020881a6184096378701862bd';
|
|
3084
|
+
const wrongIndex = '999';
|
|
3085
|
+
const keychains = [{ id: '1', type: 'tss', commonKeychain }];
|
|
3086
|
+
const result = await basecoin.isWalletAddress({ keychains, address, index: wrongIndex });
|
|
3087
|
+
result.should.equal(false);
|
|
3088
|
+
});
|
|
3089
|
+
it('should throw error for invalid address', async function () {
|
|
3090
|
+
const invalidAddress = 'invalidaddress';
|
|
3091
|
+
const commonKeychain = '8ea32ecacfc83effbd2e2790ee44fa7c59b4d86c29a12f09fb613d8195f93f4e21875cad3b98adada40c040c54c3569467df41a020881a6184096378701862bd';
|
|
3092
|
+
const index = '1';
|
|
3093
|
+
const keychains = [{ id: '1', type: 'tss', commonKeychain }];
|
|
3094
|
+
await assert_1.default.rejects(async () => await basecoin.isWalletAddress({ keychains, address: invalidAddress, index }), {
|
|
3095
|
+
message: `invalid address: ${invalidAddress}`,
|
|
3096
|
+
});
|
|
3097
|
+
});
|
|
3098
|
+
});
|
|
3099
|
+
describe('getAddressFromPublicKey', () => {
|
|
3100
|
+
it('should convert public key to base58 address', function () {
|
|
3101
|
+
const publicKey = '61220a9394802b1d1df37b35f7a3197970f48081092cee011fc98f7b71b2bd43';
|
|
3102
|
+
const expectedAddress = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';
|
|
3103
|
+
const address = basecoin.getAddressFromPublicKey(publicKey);
|
|
3104
|
+
address.should.equal(expectedAddress);
|
|
3105
|
+
});
|
|
3106
|
+
});
|
|
3107
|
+
});
|
|
3108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sol.js","sourceRoot":"","sources":["../../../test/unit/sol.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,oDAA4B;AAC5B,0CAA4B;AAC5B,gDAAwB;AACxB,+CAAiC;AACjC,6CAA+B;AAE/B,iDAA0D;AAE1D,iDAAwD;AACxD,mDAc8B;AAC9B,mDAA+D;AAC/D,iDAA4C;AAC5C,mCAA4E;AAC5E,uCAA4C;AAE5C,+CAAuE;AACvE,0DAA4C;AAC5C,4DAA8C;AAC9C,0DAAuD;AACvD,2DAAwD;AAExD,QAAQ,CAAC,MAAM,EAAE;IACf,IAAI,KAAmB,CAAC;IACxB,IAAI,QAAa,CAAC;IAClB,IAAI,OAAO,CAAC;IACZ,IAAI,aAAa,CAAC;IAClB,IAAI,0BAA0B,CAAC;IAC/B,IAAI,WAAW,CAAC;IAChB,IAAI,oBAAoB,CAAC;IACzB,IAAI,wBAAwB,CAAC;IAC7B,IAAI,wBAAwB,CAAC;IAC7B,MAAM,YAAY,GAAG,SAAS,CAAC,SAAS,CAAC,gBAAgB,CAAC;IAC1D,MAAM,aAAa,GAAG,SAAS,CAAC,SAAS,CAAC,cAAc,CAAC;IAEzD,MAAM,OAAO,GAAG;QACd,GAAG,EAAE,SAAS,CAAC,eAAe,CAAC,SAAS;QACxC,GAAG,EAAE,SAAS,CAAC,eAAe,CAAC,UAAU,CAAC,MAAM;KACjD,CAAC;IACF,MAAM,UAAU,GAAG;QACjB,UAAU,EAAE;YACV;gBACE,OAAO,EAAE,cAAc;gBACvB,MAAM,EAAE,MAAM;aACf;SACF;QACD,QAAQ,EAAE,SAAS,CAAC,gDAAgD;QACpE,MAAM,EAAE;YACN,QAAQ,EAAE,8CAA8C;YACxD,KAAK,EAAE,8CAA8C;SACtD;QACD,IAAI,EAAE,kEAAkE;QACxE,mBAAmB,EAAE,KAAK;QAC1B,IAAI,EAAE,MAAM;KACb,CAAC;IACF,MAAM,QAAQ,GAAG;QACf,UAAU;QACV,UAAU,EAAE;YACV;gBACE,OAAO,EAAE,8CAA8C;gBACvD,MAAM,EAAE,QAAQ;aACjB;SACF;KACF,CAAC;IACF,MAAM,IAAI,GAAG,EAAE,KAAK,EAAE,WAAW,EAAE,CAAC;IACpC,MAAM,YAAY,GAAG;QACnB,kBAAkB,EAAE,8CAA8C;QAClE,iBAAiB,EAAE,8CAA8C;KAClE,CAAC;IACF,MAAM,iBAAiB,GAAG;QACxB,kBAAkB,EAAE,4CAA4C;QAChE,iBAAiB,EAAE,8CAA8C;KAClE,CAAC;IACF,MAAM,iBAAiB,GAAG;QACxB,UAAU;QACV,UAAU,EAAE;YACV;gBACE,OAAO,EAAE,2CAA2C;gBACpD,MAAM,EAAE,QAAQ;aACjB;SACF;KACF,CAAC;IACF,MAAM,qBAAqB,GAAG;QAC5B,UAAU;QACV,UAAU,EAAE;YACV;gBACE,OAAO,EAAE,8CAA8C;gBACvD,MAAM,EAAE,QAAQ;gBAChB,IAAI,EAAE,SAAS;aAChB;SACF;KACF,CAAC;IACF,MAAM,uBAAuB,GAAG;QAC9B,UAAU,EAAE;YACV;gBACE,OAAO,EAAE,8CAA8C;gBACvD,MAAM,EAAE,GAAG;aACZ;SACF;QACD,KAAK,EAAE,SAAS,CAAC,wCAAwC;QACzD,MAAM,EAAE;YACN,QAAQ,EAAE,8CAA8C;YACxD,KAAK,EAAE,8CAA8C;SACtD;QACD,IAAI,EAAE,kEAAkE;QACxE,mBAAmB,EAAE,KAAK;QAC1B,IAAI,EAAE,MAAM;KACb,CAAC;IACF,MAAM,qBAAqB,GAAG;QAC5B,UAAU;QACV,UAAU,EAAE;YACV;gBACE,OAAO,EAAE,8CAA8C;gBACvD,MAAM,EAAE,GAAG;aACZ;SACF;KACF,CAAC;IACF,MAAM,SAAS,GAAG,EAAE,KAAK,EAAE,gBAAgB,EAAE,CAAC;IAC9C,MAAM,aAAa,GAAG,8CAA8C,CAAC;IACrE,MAAM,OAAO,GAAG,IAAA,qCAAiB,EAAC,MAAM,CAAC,CAAC;IAC1C,MAAM,MAAM,GAAG,IAAI,aAAO,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,OAAO,EAAE,CAAC;IAC5D,MAAM,YAAY,GAAG,IAAI,aAAO,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC,OAAO,EAAE,CAAC;IACnE,MAAM,SAAS,GAAG,SAAS,CAAC,WAAW,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC;IAC5D,MAAM,MAAM,GAAG,OAAO,CAAC;IACvB,MAAM,SAAS,GAAG,SAAS,CAAC,SAAS,CAAC;IAEtC,MAAM,CAAC;QACL,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,kBAAQ,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;QACtD,KAAK,CAAC,YAAY,CAAC,KAAK,EAAE,UAAI,CAAC,cAAc,CAAC,CAAC;QAC/C,KAAK,CAAC,YAAY,CAAC,MAAM,EAAE,UAAI,CAAC,cAAc,CAAC,CAAC;QAChD,KAAK,CAAC,kBAAkB,EAAE,CAAC;QAC3B,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAS,CAAC;QACtC,OAAO,GAAG,QAAQ,CAAC,eAAe,CAAC,SAAS,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;QACnE,aAAa,GAAG,GAAG,EAAE;YACnB,OAAO,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QACjC,CAAC,CAAC;QACF,0BAA0B,GAAG,GAAG,EAAE;YAChC,OAAO,CAAC,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;QAC9C,CAAC,CAAC;QACF,WAAW,GAAG,GAAG,EAAE;YACjB,OAAO,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QAC/B,CAAC,CAAC;QACF,oBAAoB,GAAG,GAAG,EAAE;YAC1B,OAAO,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QACxC,CAAC,CAAC;QACF,wBAAwB,GAAG,GAAG,EAAE;YAC9B,OAAO,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;QAC5C,CAAC,CAAC;QACF,wBAAwB,GAAG,GAAG,EAAE;YAC9B,OAAO,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC;QAC5C,CAAC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE,KAAK;QACrC,IAAI,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACvC,aAAa,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,UAAU,CAAC,UAAI,CAAC,CAAC;QAE5C,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAClC,aAAa,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,UAAU,CAAC,SAAG,CAAC,CAAC;IAC7C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE;QAChC,QAAQ,CAAC,QAAQ,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACzC,QAAQ,CAAC,SAAS,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACzC,QAAQ,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;QACtD,QAAQ,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IACH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,MAAM,UAAU,GAAG;YACjB,EAAE,EAAE,kCAAkC;YACtC,IAAI,EAAE,MAAM;YACZ,IAAI,EAAE;gBACJ,kCAAkC;gBAClC,kCAAkC;gBAClC,kCAAkC;aACnC;YACD,YAAY,EAAE;gBACZ,WAAW,EAAE,MAAM,CAAC,GAAG;aACxB;YACD,YAAY,EAAE,KAAK;SACpB,CAAC;QACF,MAAM,SAAS,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;QAE1D,EAAE,CAAC,4BAA4B,EAAE,KAAK;YACpC,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,YAAY;gBACZ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE,KAAK;YAC/C,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,aAAa,GAAG,eAAe,CAAC;YAE3C,MAAM,UAAU,GAAG;gBACjB,EAAE,EAAE,kCAAkC;gBACtC,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE;oBACJ,kCAAkC;oBAClC,kCAAkC;oBAClC,kCAAkC;iBACnC;gBACD,YAAY,EAAE;oBACZ,WAAW,EAAE,YAAY,CAAC,GAAG;iBAC9B;gBACD,YAAY,EAAE,KAAK;aACpB,CAAC;YACF,MAAM,0BAA0B,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAE3E,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,YAAY;gBACZ,MAAM,EAAE,0BAA0B;aAC5B,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QACpC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kDAAkD,EAAE,KAAK;YAC1D,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,KAAK,GAAG,UAAU,CAAC,QAAQ,CAAC;YACvC,UAAU,CAAC,QAAQ,GAAG,SAAS,CAAC;YAChC,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,YAAY;gBACZ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK;YACxD,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACjF,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,YAAY;gBACZ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK;YACzD,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,QAAQ,CAAC,UAAU,GAAG,SAAS,CAAC;YAChC,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,YAAY;gBACZ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0DAA0D,EAAE,KAAK;YAClE,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,QAAQ;iBACX,iBAAiB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAS,CAAC;iBACtF,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,8DAA8D,CAAC,CAAC;QAC5F,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oCAAoC,EAAE,KAAK;YAC5C,MAAM,UAAU,GAAG;gBACjB,EAAE,EAAE,kCAAkC;gBACtC,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE;oBACJ,kCAAkC;oBAClC,kCAAkC;oBAClC,kCAAkC;iBACnC;gBACD,YAAY,EAAE;oBACZ,WAAW,EAAE,8CAA8C;iBAC5D;gBACD,YAAY,EAAE,KAAK;aACpB,CAAC;YACF,MAAM,SAAS,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAC1D,MAAM,UAAU,GAAG;gBACjB,UAAU,EAAE;oBACV;wBACE,OAAO,EAAE,kCAAkC;wBAC3C,MAAM,EAAE,YAAY;wBACpB,SAAS,EAAE,WAAW;qBACvB;iBACF;gBACD,QAAQ,EACN,wpCAAwpC;gBAC1pC,MAAM,EAAE;oBACN,QAAQ,EAAE,8CAA8C;oBACxD,KAAK,EAAE,8CAA8C;iBACtD;gBACD,IAAI,EAAE,kEAAkE;gBACxE,mBAAmB,EAAE,KAAK;gBAC1B,IAAI,EAAE,MAAM;aACb,CAAC;YACF,MAAM,QAAQ,GAAG;gBACf,UAAU;gBACV,UAAU,EAAE;oBACV;wBACE,OAAO,EAAE,kCAAkC;wBAC3C,MAAM,EAAE,YAAY;wBACpB,SAAS,EAAE,WAAW;qBACvB;iBACF;aACF,CAAC;YACF,MAAM,IAAI,GAAG;gBACX,KAAK,EAAE,SAAS;aACjB,CAAC;YACF,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACzD,QAAQ;gBACR,UAAU;gBACV,IAAI,EAAE,IAAI;gBACV,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,iBAAiB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACvC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kEAAkE,EAAE,KAAK;YAC1E,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,QAAQ;iBACX,iBAAiB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,EAAE,SAAS,EAAS,CAAC;iBAC5G,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,wDAAwD,CAAC,CAAC;QACtF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8DAA8D,EAAE,KAAK;YACtE,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,UAAU,GAAG;gBACjB,EAAE,EAAE,kCAAkC;gBACtC,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE;oBACJ,kCAAkC;oBAClC,kCAAkC;oBAClC,kCAAkC;iBACnC;gBACD,YAAY,EAAE;oBACZ,WAAW,EAAE,YAAY,CAAC,GAAG;iBAC9B;gBACD,YAAY,EAAE,KAAK;aACpB,CAAC;YACF,MAAM,0BAA0B,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAE3E,MAAM,QAAQ;iBACX,iBAAiB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,EAAE,0BAA0B,EAAS,CAAC;iBAC5F,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6CAA6C,CAAC,CAAC;QAC3E,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gEAAgE,EAAE,KAAK;YACxE,MAAM,QAAQ,GAAG,oBAAoB,EAAE,CAAC;YACxC,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,QAAQ;iBACX,iBAAiB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,aAAa,EAAE,MAAM,EAAE,SAAS,EAAS,CAAC;iBAC1F,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6DAA6D,CAAC,CAAC;QAC3F,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0EAA0E,EAAE,KAAK;YAClF,MAAM,QAAQ,GAAG,wBAAwB,EAAE,CAAC;YAC5C,MAAM,OAAO,GAAG,8CAA8C,CAAC,CAAC,qBAAqB;YACrF,QAAQ,CAAC,UAAU,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,WAAW,EAAE,CAAC,CAAC;YACzE,MAAM,UAAU,GAAG,0BAA0B,EAAE,CAAC;YAChD,MAAM,kBAAkB,GAAG;gBACzB,EAAE,EAAE,kCAAkC;gBACtC,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE;oBACJ,kCAAkC;oBAClC,kCAAkC;oBAClC,kCAAkC;iBACnC;gBACD,YAAY,EAAE;oBACZ,WAAW,EAAE,8CAA8C;iBAC5D;gBACD,YAAY,EAAE,KAAK;aACpB,CAAC;YACF,MAAM,cAAc,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,kBAAkB,CAAC,CAAC;YACvE,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,MAAM,EAAE,cAAc;aAChB,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+EAA+E,EAAE,KAAK;YACvF,MAAM,QAAQ,GAAG,wBAAwB,EAAE,CAAC;YAC5C,MAAM,OAAO,GAAG,8CAA8C,CAAC,CAAC,qBAAqB;YACrF,QAAQ,CAAC,UAAU,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,WAAW,EAAE,CAAC,CAAC;YAC5E,MAAM,UAAU,GAAG,0BAA0B,EAAE,CAAC;YAChD,MAAM,kBAAkB,GAAG;gBACzB,EAAE,EAAE,kCAAkC;gBACtC,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE;oBACJ,kCAAkC;oBAClC,kCAAkC;oBAClC,kCAAkC;iBACnC;gBACD,YAAY,EAAE;oBACZ,WAAW,EAAE,8CAA8C;iBAC5D;gBACD,YAAY,EAAE,KAAK;aACpB,CAAC;YACF,MAAM,cAAc,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,kBAAkB,CAAC,CAAC;YACvE,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,MAAM,EAAE,cAAc;aAChB,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4EAA4E,EAAE,KAAK;YACpF,MAAM,QAAQ,GAAG,wBAAwB,EAAE,CAAC;YAC5C,MAAM,OAAO,GAAG,8CAA8C,CAAC,CAAC,qBAAqB;YACrF,QAAQ,CAAC,UAAU,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,WAAW,EAAE,CAAC,CAAC,CAAC,gDAAgD;YAC1H,MAAM,UAAU,GAAG,0BAA0B,EAAE,CAAC;YAChD,MAAM,QAAQ;iBACX,iBAAiB,CAAC;gBACjB,QAAQ;gBACR,UAAU;gBACV,MAAM,EAAE,SAAS;aACX,CAAC;iBACR,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6DAA6D,CAAC,CAAC;QAC3F,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0EAA0E,EAAE,KAAK;YAClF,MAAM,QAAQ,GAAG,wBAAwB,EAAE,CAAC;YAC5C,MAAM,OAAO,GAAG,8CAA8C,CAAC,CAAC,qBAAqB;YACrF,QAAQ,CAAC,UAAU,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,WAAW,EAAE,CAAC,CAAC;YACzE,MAAM,UAAU,GAAG,0BAA0B,EAAE,CAAC;YAChD,MAAM,QAAQ;iBACX,iBAAiB,CAAC;gBACjB,QAAQ;gBACR,UAAU;gBACV,MAAM,EAAE,SAAS;aACX,CAAC;iBACR,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6DAA6D,CAAC,CAAC;QAC3F,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uEAAuE,EAAE,KAAK;YAC/E,MAAM,QAAQ,GAAG,wBAAwB,EAAE,CAAC;YAC5C,MAAM,OAAO,GAAG,8CAA8C,CAAC,CAAC,mDAAmD;YACnH,QAAQ,CAAC,UAAU,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,WAAW,EAAE,CAAC,CAAC;YACzE,MAAM,UAAU,GAAG,0BAA0B,EAAE,CAAC;YAChD,MAAM,QAAQ;iBACX,iBAAiB,CAAC;gBACjB,QAAQ;gBACR,UAAU;gBACV,MAAM,EAAE,SAAS;aACX,CAAC;iBACR,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6DAA6D,CAAC,CAAC;QAC3F,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sEAAsE,EAAE,KAAK;YAC9E,MAAM,QAAQ,GAAG,wBAAwB,EAAE,CAAC;YAC5C,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,YAAY;gBACZ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;YACpD,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,yBAAyB,EAAE;iBAC3B,cAAc,CAAC,YAAY,CAAC,GAAG,CAAC;iBAChC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,MAAM,CAAC,MAAM,CAAC;iBACd,SAAS,CAAC,SAAS,CAAC,GAAG,CAAC;iBACxB,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,QAAQ,GAAG,mBAAmB,CAAC;YAC1C,UAAU,CAAC,MAAM,CAAC,KAAK,GAAG,8CAA8C,CAAC;YACzE,QAAQ,CAAC,UAAU,GAAG;gBACpB;oBACE,OAAO,EAAE,8CAA8C;oBACvD,MAAM,EAAE,MAAM;iBACf;aACF,CAAC;YACF,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;YACpD,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,yBAAyB,EAAE;iBAC3B,cAAc,CAAC,YAAY,CAAC,GAAG,CAAC;iBAChC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,MAAM,CAAC,MAAM,CAAC;iBACd,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,QAAQ,GAAG,mBAAmB,CAAC;YAC1C,UAAU,CAAC,MAAM,CAAC,KAAK,GAAG,8CAA8C,CAAC;YACzE,QAAQ,CAAC,UAAU,GAAG;gBACpB;oBACE,OAAO,EAAE,8CAA8C;oBACvD,MAAM,EAAE,MAAM;iBACf;aACF,CAAC;YACF,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK;YACtD,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,2BAA2B,EAAE;iBAC7B,cAAc,CAAC,YAAY,CAAC,GAAG,CAAC;iBAChC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,QAAQ,GAAG,mBAAmB,CAAC;YAC1C,UAAU,CAAC,MAAM,CAAC,KAAK,GAAG,8CAA8C,CAAC;YACzE,QAAQ,CAAC,UAAU,GAAG,EAAE,CAAC;YACzB,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2DAA2D,EAAE,KAAK;YACnE,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,2BAA2B,EAAE;iBAC7B,IAAI,CAAC,WAAW,CAAC;iBACjB,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,aAAa,EAAE,CAAC;YACnC,UAAU,CAAC,QAAQ,GAAG,mBAAmB,CAAC;YAC1C,UAAU,CAAC,MAAM,CAAC,KAAK,GAAG,SAAS,CAAC;YACpC,QAAQ,CAAC,UAAU,GAAG,EAAE,CAAC;YACzB,MAAM,gBAAgB,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,QAAQ;gBACR,UAAU;gBACV,IAAI;gBACJ,MAAM,EAAE,SAAS;aACX,CAAC,CAAC;YACV,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE;QAChC,aAAa,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;YAC7B,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+BAA+B,EAAE;QAClC,YAAY,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;YAC5B,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE;QAChC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACpC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACtD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kCAAkC,EAAE;QACrC,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC;QACrE,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACrD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE;QAChC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACpC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACtD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kCAAkC,EAAE;QACrC,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC;QAC/F,QAAQ,CAAC,UAAU,CAAC,SAAmB,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,EAAE,CAAC,+CAA+C,EAAE,KAAK;YACvD,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,gBAAgB,CAAC;gBACxD,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,QAAQ;gBACpD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,iBAAiB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACjC,MAAM,EAAE;oBACN;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,MAAM;qBACf;iBACF;gBACD,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;YACpD,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,gBAAgB,CAAC;gBACxD,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,MAAM;gBAClD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,iBAAiB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACjC,MAAM,EAAE;oBACN;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,MAAM;qBACf;iBACF;gBACD,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kDAAkD,EAAE,KAAK;YAC1D,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,gBAAgB,CAAC;gBACxD,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,UAAU,CAAC,QAAQ;gBACtD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,iBAAiB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACjC,MAAM,EAAE;oBACN;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,MAAM;qBACf;iBACF;gBACD,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+CAA+C,EAAE,KAAK;YACvD,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,gBAAgB,CAAC;gBACxD,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,UAAU,CAAC,MAAM;gBACpD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,iBAAiB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACjC,MAAM,EAAE;oBACN;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,MAAM;qBACf;iBACF;gBACD,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qDAAqD,EAAE,KAAK;YAC7D,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,gBAAgB,CAAC;gBACxD,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,aAAa,CAAC,QAAQ;gBACzD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,iBAAiB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACjC,MAAM,EAAE;oBACN;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,IAAI;qBACb;iBACF;gBACD,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;wBAChB,SAAS,EAAE,WAAW;qBACvB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kDAAkD,EAAE,KAAK;YAC1D,MAAM,iBAAiB,GAAG,MAAM,QAAQ,CAAC,gBAAgB,CAAC;gBACxD,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,aAAa,CAAC,MAAM;gBACvD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,iBAAiB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACjC,MAAM,EAAE;oBACN;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,IAAI;qBACb;iBACF;gBACD,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;wBAChB,SAAS,EAAE,WAAW;qBACvB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;QACrC,EAAE,CAAC,iDAAiD,EAAE,KAAK;YACzD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,QAAQ;gBACpD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,MAAM;gBACZ,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,QAAQ;gBACtB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,MAAM;oBACX,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE;oBACZ,iBAAiB,EAAE,8CAA8C;oBACjE,kBAAkB,EAAE,8CAA8C;iBACnE;gBACD,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK;YACtD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,MAAM;gBAClD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,0FAA0F;gBAC9F,IAAI,EAAE,MAAM;gBACZ,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,QAAQ;gBACtB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,MAAM;oBACX,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE;oBACZ,iBAAiB,EAAE,8CAA8C;oBACjE,kBAAkB,EAAE,8CAA8C;iBACnE;gBACD,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oDAAoD,EAAE,KAAK;YAC5D,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,UAAU,CAAC,QAAQ;gBACtD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,sBAAsB;gBAC5B,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,QAAQ;gBACtB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,OAAO;oBACZ,OAAO,EAAE,IAAI;iBACd;gBACD,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,IAAI,EAAE,SAAS;gBACf,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK;YACzD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,UAAU,CAAC,MAAM;gBACpD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YAEH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,yFAAyF;gBAC7F,IAAI,EAAE,sBAAsB;gBAC5B,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,QAAQ;gBACtB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;qBACjB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,OAAO;oBACZ,OAAO,EAAE,IAAI;iBACd;gBACD,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,IAAI,EAAE,SAAS;gBACf,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uDAAuD,EAAE,KAAK;YAC/D,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,aAAa,CAAC,QAAQ;gBACzD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,MAAM;gBACZ,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,GAAG;gBACjB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;wBAChB,SAAS,EAAE,WAAW;qBACvB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,MAAM;oBACX,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE;oBACZ,iBAAiB,EAAE,8CAA8C;oBACjE,kBAAkB,EAAE,8CAA8C;iBACnE;gBACD,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oDAAoD,EAAE,KAAK;YAC5D,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,aAAa,CAAC,MAAM;gBACvD,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,0FAA0F;gBAC9F,IAAI,EAAE,MAAM;gBACZ,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,GAAG;gBACjB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,QAAQ;wBAChB,SAAS,EAAE,WAAW;qBACvB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,MAAM;oBACX,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE;oBACZ,iBAAiB,EAAE,8CAA8C;oBACjE,kBAAkB,EAAE,8CAA8C;iBACnE;gBACD,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6CAA6C,EAAE,KAAK;YACrD,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,yBAAyB,EAAE;iBAC3B,cAAc,CAAC,YAAY,CAAC,GAAG,CAAC;iBAChC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,MAAM,CAAC,MAAM,CAAC;iBACd,SAAS,CAAC,SAAS,CAAC,GAAG,CAAC;iBACxB,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,mBAAmB;gBAC7B,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,iBAAiB;gBACvB,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,OAAO;gBACrB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,OAAO;qBAChB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,OAAO;oBACZ,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+CAA+C,EAAE,KAAK;YACvD,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,2BAA2B,EAAE;iBAC7B,cAAc,CAAC,YAAY,CAAC,GAAG,CAAC;iBAChC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,mBAAmB;gBAC7B,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,mBAAmB;gBACzB,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,GAAG;gBACjB,OAAO,EAAE,EAAE;gBACX,GAAG,EAAE;oBACH,GAAG,EAAE,MAAM;oBACX,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6CAA6C,EAAE,KAAK;YACrD,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,yBAAyB,EAAE;iBAC3B,cAAc,CAAC,YAAY,CAAC,GAAG,CAAC;iBAChC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,MAAM,CAAC,MAAM,CAAC;iBACd,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,mBAAmB;gBAC7B,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;aACF,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,iBAAiB;gBACvB,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,OAAO;gBACrB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,OAAO;qBAChB;iBACF;gBACD,GAAG,EAAE;oBACH,GAAG,EAAE,MAAM;oBACX,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,gBAAgB,EAAE,EAAE;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE,KAAK;YAC/C,MAAM,SAAS,GAAG,WAAW,CAAC;YAC9B,MAAM,gBAAgB,GAAG,SAAS,CAAC;YACnC,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,2BAA2B,EAAE;iBAC7B,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,IAAI,CAAC,SAAS,CAAC;iBACf,gBAAgB,CAAC,gBAAgB,CAAC;iBAClC,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,mBAAmB;gBAC7B,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;gBACD,4BAA4B,EAAE,gBAAgB;aAC/C,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,sCAAsC;gBAC5C,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,GAAG;gBACjB,OAAO,EAAE,EAAE;gBACX,GAAG,EAAE;oBACH,GAAG,EAAE,SAAS;oBACd,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,gBAAgB,EAAE;oBAChB;wBACE,OAAO,EAAE,8CAA8C;wBACvD,YAAY,EAAE,8CAA8C;wBAC5D,SAAS,EAAE,WAAW;qBACvB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6CAA6C,EAAE,KAAK;YACrD,MAAM,UAAU,GAAG;gBACjB;oBACE,YAAY,EAAE,MAAM,CAAC,GAAG;oBACxB,SAAS,EAAE,WAAW;iBACvB;gBACD;oBACE,YAAY,EAAE,YAAY,CAAC,kBAAkB;oBAC7C,SAAS,EAAE,UAAU;iBACtB;aACF,CAAC;YACF,MAAM,gBAAgB,GAAG,SAAS,CAAC;YACnC,MAAM,EAAE,GAAG,MAAM,OAAO;iBACrB,2BAA2B,EAAE;iBAC7B,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;iBAClB,KAAK,CAAC,SAAS,CAAC;iBAChB,WAAW,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;iBAC1B,WAAW,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;iBAC1B,gBAAgB,CAAC,gBAAgB,CAAC;iBAClC,IAAI,CAAC,WAAW,CAAC;iBACjB,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;iBACrB,KAAK,EAAE,CAAC;YACX,MAAM,mBAAmB,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC;YACnD,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,mBAAmB;gBAC7B,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;gBACD,4BAA4B,EAAE,gBAAgB;aAC/C,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,sCAAsC;gBAC5C,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,GAAG;gBACjB,OAAO,EAAE,EAAE;gBACX,GAAG,EAAE;oBACH,GAAG,EAAE,SAAS;oBACd,OAAO,EAAE,IAAI;iBACd;gBACD,IAAI,EAAE,WAAW;gBACjB,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,gBAAgB,EAAE;oBAChB;wBACE,OAAO,EAAE,8CAA8C;wBACvD,YAAY,EAAE,8CAA8C;wBAC5D,SAAS,EAAE,WAAW;qBACvB;oBACD;wBACE,OAAO,EAAE,8CAA8C;wBACvD,YAAY,EAAE,8CAA8C;wBAC5D,SAAS,EAAE,UAAU;qBACtB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yEAAyE,EAAE,KAAK;YACjF,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,4BAA4B,CAAC,QAAQ;gBACxE,OAAO,EAAE;oBACP,GAAG,EAAE,MAAM;iBACZ;gBACD,4BAA4B,EAAE,SAAS;aACxC,CAAC,CAAC;YACH,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,YAAY,EAAE;oBACZ,IAAI;oBACJ,MAAM;oBACN,WAAW;oBACX,cAAc;oBACd,cAAc;oBACd,cAAc;oBACd,SAAS;oBACT,eAAe;oBACf,kBAAkB;oBAClB,KAAK;oBACL,MAAM;iBACP;gBACD,EAAE,EAAE,aAAa;gBACjB,IAAI,EAAE,MAAM;gBACZ,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,GAAG;gBACjB,YAAY,EAAE,GAAG;gBACjB,OAAO,EAAE;oBACP;wBACE,OAAO,EAAE,8CAA8C;wBACvD,MAAM,EAAE,OAAO;wBACf,SAAS,EAAE,WAAW;qBACvB;iBACF;gBACD,GAAG,EAAE,EAAE,GAAG,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,EAAE;gBACtC,IAAI,EAAE,SAAS;gBACf,SAAS,EAAE,8CAA8C;gBACzD,YAAY,EAAE,SAAS;gBACvB,gBAAgB,EAAE;oBAChB;wBACE,OAAO,EAAE,8CAA8C;wBACvD,YAAY,EAAE,8CAA8C;wBAC5D,SAAS,EAAE,WAAW;qBACvB;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,UAAU,EAAE,GAAG,EAAE;QACxB,EAAE,CAAC,4CAA4C,EAAE;YAC/C,MAAM,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,QAAQ,CAAC,eAAe,CAAC,aAAoB,CAAC,EAAE,wCAAwC,CAAC,CAAC;QAChH,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE;YAC1C,MAAM,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,QAAQ,CAAC,eAAe,CAAC,aAAoB,CAAC,EAAE,wCAAwC,CAAC,CAAC;QAChH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;QACjC,EAAE,CAAC,yBAAyB,EAAE,KAAK;YACjC,MAAM,MAAM,GAAQ,MAAM,QAAQ,CAAC,eAAe,CAAC;gBACjD,UAAU,EAAE;oBACV,QAAQ,EAAE,SAAS,CAAC,eAAe;oBACnC,IAAI,EAAE,CAAC,SAAS,CAAC,eAAe,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;iBACvD;gBACD,GAAG,EAAE,SAAS,CAAC,eAAe,CAAC,UAAU,CAAC,MAAM;aAC1C,CAAC,CAAC;YACV,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kDAAkD,EAAE,KAAK;YAC1D,MAAM,MAAM,GAAQ,MAAM,QAAQ,CAAC,eAAe,CAAC;gBACjD,UAAU,EAAE;oBACV,KAAK,EAAE,SAAS,CAAC,eAAe;oBAChC,IAAI,EAAE,CAAC,SAAS,CAAC,eAAe,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;iBACvD;gBACD,GAAG,EAAE,SAAS,CAAC,eAAe,CAAC,UAAU,CAAC,MAAM;aAC1C,CAAC,CAAC;YACV,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4DAA4D,EAAE,KAAK;YACpE,MAAM,QAAQ;iBACX,eAAe,CAAC;gBACf,UAAU,EAAE;oBACV,QAAQ,EAAE,SAAS,CAAC,eAAe;oBACnC,IAAI,EAAE,CAAC,SAAS,CAAC,eAAe,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;iBACvD;gBACD,GAAG,EAAE,SAAS,CAAC,eAAe,CAAC,SAAS;aAClC,CAAC;iBACR,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,aAAa,CAAC,CAAC;QAC3C,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,cAAc,EAAE,GAAG,EAAE;QAC5B,EAAE,CAAC,qBAAqB,EAAE,KAAK;YAC7B,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,WAAW,CAAC,OAAO,EAAE,gBAAgB,CAAC,CAAC;YACrE,MAAM;iBACH,QAAQ,CAAC,QAAQ,CAAC;iBAClB,MAAM,CAAC,KAAK,CAAC,0FAA0F,CAAC,CAAC;QAC9G,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,iDAAiD,EAAE,KAAK;YACzD,MAAM,QAAQ;iBACX,WAAW,CAAC,OAAO,EAAE,SAAgB,CAAC;iBACtC,MAAM,CAAC,EAAE,CAAC,YAAY,CACrB,uIAAuI,CACxI,CAAC;QACN,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,EAAE,CAAC,uCAAuC,EAAE,KAAK;YAC/C,MAAM,OAAO,GAAG,IAAA,qCAAiB,EAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAC;YACvD,MAAM,sBAAsB,GAAG,CAAC,MAAM,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,8BAA8B,CAAC,CAAC,KAAK,EAAE,CAAC;iBAClG,eAAe,CAAC;YACnB,MAAM,cAAc,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC,SAAS,CAAC,8BAA8B,CAAC,CAAC;YACnG,cAAc,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,CAAC,sBAAsB,CAAC,CAAC;QAC7D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mEAAmE,EAAE,KAAK;YAC3E,MAAM,OAAO,GAAG,IAAA,qCAAiB,EAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAC;YACvD,MAAM,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,8BAA8B,CAAC,CAAC;YAC1E,gBAAM,CAAC,EAAE,CAAC,UAAU,CAAC,CAAC;QACxB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,MAAM,WAAW,GAAG,aAAa,CAAC;QAClC,IAAI,OAA2B,CAAC;QAEhC,UAAU,CAAC,GAAG,EAAE;YACd,OAAO,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,SAAS,CAAC,GAAG,EAAE;YACb,OAAO,CAAC,gBAAgB,EAAE,CAAC;QAC7B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2CAA2C,EAAE,KAAK,IAAI,EAAE;YACzD,MAAM,SAAS,GAAc;gBAC3B,WAAW;gBACX,WAAW,EAAE;oBACX;wBACE,eAAe,EAAE,UAAU;wBAC3B,WAAW,EAAE,iBAAiB;wBAC9B,cAAc,EAAE,KAAK;qBACtB;iBACF;gBACD,YAAY,EAAE,EAAE;gBAChB,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;gBAC9B,MAAM,EAAE;oBACN,UAAU,EAAE,SAAS;iBACtB;gBACD,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,sBAAsB;gBAC7B,UAAU,EAAE,KAAK;gBACjB,QAAQ,EAAE,UAAU;gBACpB,eAAe,EAAE,IAAI;gBACrB,OAAO,EAAE,CAAC;gBACV,MAAM,EAAE,QAAQ;aACjB,CAAC;YAEF,MAAM,YAAY,GAAG,OAAO,CAAC,kBAAkB,CAAC,mBAAQ,CAAC,CAAC;YAC1D,YAAY,CAAC,qBAAqB,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;YAChD,YAAY,CAAC,YAAY,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YAE9C,MAAM,SAAS,GAAG;gBAChB,IAAI,EAAE,KAAK;aACZ,CAAC;YACF,MAAM,oBAAoB,GAAQ,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAClE,UAAU,EAAE,SAAS;gBACrB,QAAQ,EAAE,YAAY;gBACtB,UAAU,EAAE;oBACV,WAAW;iBACZ;aACK,CAAC,CAAC;YAEV,oBAAoB,CAAC,UAAU,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YAC5D,oBAAoB,CAAC,UAAU,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YAC5D,oBAAoB,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC;QACpF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE,KAAK,IAAI,EAAE;YACrD,MAAM,UAAU,GAAG;gBACjB,IAAI,EAAE,MAAM;aACb,CAAC;YAEF,MAAM,oBAAoB,GAAG,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAC7D,UAAU,EAAE,UAAU;aAChB,CAAC,CAAC;YACV,oBAAoB,CAAC,MAAM,CAAC,SAAS,CAAC;gBACpC,UAAU,EAAE,UAAU;aACvB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wCAAwC,EAAE,KAAK,IAAI,EAAE;YACtD,MAAM,SAAS,GAAG;gBAChB,IAAI,EAAE,KAAK;aACZ,CAAC;YACF,MAAM,QAAQ;iBACX,kBAAkB,CAAC;gBAClB,UAAU,EAAE,SAAS;gBACrB,UAAU,EAAE,EAAE;aACR,CAAC;iBACR,MAAM,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;QAClC,4CAA4C;QAC5C,EAAE,CAAC,gDAAgD,EAAE;YACnD,iDAAiD;YACjD,MAAM,GAAG,GAAI,QAAgB,CAAC,gBAAgB,CAAC,kBAAkB,CAAC,CAAC;YACnE,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,uBAAY,CAAC,IAAI,CAAC,iBAAiB,mBAAmB,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yDAAyD,EAAE;YAC5D,iDAAiD;YACjD,MAAM,GAAG,GAAI,QAAgB,CAAC,gBAAgB,EAAE,CAAC;YACjD,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,uBAAY,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;QACrC,MAAM,OAAO,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC;QACtC,MAAM,IAAI,GAAG,eAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAC/B,MAAM,eAAe,GAAG,8CAA8C,CAAC;QACvE,MAAM,cAAc,GAAG,8CAA8C,CAAC;QACtE,IAAI,QAAyB,CAAC;QAE9B,UAAU,CAAC,GAAG,EAAE;YACd,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC,SAAG,CAAC,SAAS,EAAE,iBAA8B,CAAC,CAAC;YAEvE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,oBAAoB;oBAC5B,MAAM,EAAE;wBACN;4BACE,UAAU,EAAE,WAAW;yBACxB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,oBAAoB,CAAC,CAAC;YACxD,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,kBAAkB;oBAC1B,MAAM,EAAE;wBACN,KAAK,CAAC,KAAK,CAAC,MAAM;wBAClB;4BACE,UAAU,EAAE,WAAW;yBACxB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,mCAAmC;oBAC3C,MAAM,EAAE,CAAC,GAAG,CAAC;iBACd;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,oBAAoB,CAAC;iBACpD;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,2BAA2B,CAAC;iBAC3D;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,gCAAgC,CAAC,CAAC;YACpE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,gCAAgC,CAAC;iBAChE;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,qCAAqC,CAAC,CAAC;YACzE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,gCAAgC,CAAC;iBAChE;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,qCAAqC,CAAC,CAAC;YACzE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,kCAAkC,CAAC;iBAClE;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,qCAAqC,CAAC,CAAC;YACzE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,YAAY,CAAC,eAAe,CAAC;iBAChD;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,qCAAqC,CAAC,CAAC;YACzE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,YAAY,CAAC,oBAAoB,CAAC;iBACrD;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,qCAAqC,CAAC,CAAC;YACzE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,gBAAgB;oBACxB,MAAM,EAAE;wBACN,QAAQ,CAAC,YAAY,CAAC,eAAe;wBACrC;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,oBAAoB,CAAC,CAAC;YACxD,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,gBAAgB;oBACxB,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,kBAAkB;wBAChC;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAC1D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,iBAAiB;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,WAAW,CAAC,oBAAoB;wBACzC;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,kBAAkB;wBAChC;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,+BAA+B,CAAC,CAAC;YACnE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,kBAAkB;wBAChC;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,0CAA0C,CAAC,CAAC;YAC9E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,gCAAgC,CAAC,CAAC;YACpE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,gCAAgC,CAAC,CAAC;YACpE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC;iBAC1C;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,iBAAiB;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,iBAAiB;oBACzB,MAAM,EAAE,KAAK,CAAC,KAAK,CAAC,KAAK;iBAC1B;aACF,CAAC;iBACD,MAAM,CAAC,CAAC,CAAC;iBACT,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,4BAA4B,CAAC;iBAC5D,MAAM,CAAC,CAAC,CAAC;iBACT,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,6BAA6B,CAAC,CAAC;QACnE,CAAC,CAAC,CAAC;QAEH,SAAS,CAAC,GAAG,EAAE;YACb,OAAO,CAAC,OAAO,EAAE,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gEAAgE,EAAE,KAAK;YACxE,MAAM,MAAM,GAAG,QAAQ,CAAC,WAAW,CAAC;YACpC,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,mCAAmC,CAAC,MAAM,CAAC,CAAC;YAC/E,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CACnD,sgBAAsgB,CACvgB,CAAC;YACF,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC7D,CAAC,WAAW,CAAC,aAAa,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+EAA+E,EAAE,KAAK;YACvF,MAAM,MAAM,GAAG,QAAQ,CAAC,WAAW,CAAC;YACpC,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,mCAAmC,CAAC,MAAM,CAAC,CAAC;YAC/E,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CACnD,sgBAAsgB,CACvgB,CAAC;YACF,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC7D,CAAC,WAAW,CAAC,aAAa,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sFAAsF,EAAE,KAAK;YAC9F,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC;YACrC,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,mCAAmC,CAAC,MAAM,CAAC,CAAC;YAC/E,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CACnD,sgBAAsgB,CACvgB,CAAC;YACF,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC7D,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CACnD,sgBAAsgB,CACvgB,CAAC;YACF,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC7D,CAAC,WAAW,CAAC,aAAa,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kEAAkE,EAAE,KAAK;YAC1E,yCAAyC;YACzC,MAAM,kBAAkB,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC;gBAChD,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO;gBAC9B,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS;gBAClC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,cAAc;aAC/C,CAAC,CAAC;YACH,kBAAkB,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YACzC,kBAAkB,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YACzD,kBAAkB,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YACtD,MAAM,CAAC,KAAK,CAAE,kBAA4B,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAEzD,MAAM,6BAA6B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC5D,6BAA6B,CAAC,kBAAkB,CAAE,kBAA4B,CAAC,YAAY,CAAC,CAAC;YAC7F,MAAM,sBAAsB,GAAG,6BAA6B,CAAC,MAAM,EAAE,CAAC;YAEtE,MAAM,CAAC,KAAK,CAAC,sBAAsB,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;YAC5E,MAAM,CAAC,KAAK,CAAC,sBAAsB,CAAC,QAAQ,EAAE,QAAQ,CAAC,WAAW,CAAC,oBAAoB,CAAC,CAAC;YACzF,MAAM,CAAC,KAAK,CAAC,sBAAsB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YACpG,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+DAA+D,EAAE,KAAK;YACvE,sCAAsC;YACtC,MAAM,eAAe,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC;gBAC7C,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO;gBAC9B,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS;gBAClC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,cAAc;gBAC9C,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;aACF,CAAC,CAAC;YAEH,eAAe,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YACtC,eAAe,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YACtD,eAAe,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YACnD,MAAM,CAAC,KAAK,CAAE,eAAyB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAEtD,MAAM,0BAA0B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YACzD,0BAA0B,CAAC,kBAAkB,CAAE,eAAyB,CAAC,YAAY,CAAC,CAAC;YACvF,MAAM,mBAAmB,GAAG,0BAA0B,CAAC,MAAM,EAAE,CAAC;YAEhE,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YACrF,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,QAAQ,EAAE,QAAQ,CAAC,WAAW,CAAC,oBAAoB,CAAC,CAAC;YACtF,MAAM,CAAC,KAAK,CAAC,mBAAmB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAC9F,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oDAAoD,EAAE,KAAK;YAC5D,0BAA0B;YAC1B,MAAM,gBAAgB,GAAG,CAAC,MAAM,QAAQ,CAAC,OAAO,CAAC;gBAC/C,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;aACF,CAAC,CAAgB,CAAC;YAEnB,gBAAgB,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YACvC,gBAAgB,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAChG,gBAAgB,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAC7F,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAErF,MAAM,2BAA2B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC1D,2BAA2B,CAAC,kBAAkB,CAC5C,gBAAgB,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,YAAY,CACvE,CAAC;YACF,MAAM,oBAAoB,GAAG,2BAA2B,CAAC,MAAM,EAAE,CAAC;YAElE,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YACtF,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,QAAQ,EAAE,QAAQ,CAAC,WAAW,CAAC,oBAAoB,CAAC,CAAC;YACvF,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,CAAC;YAChG,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kFAAkF,EAAE,KAAK;YAC1F,kBAAkB;YAClB,MAAM,QAAQ;iBACX,OAAO,CAAC;gBACP,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS;gBAClC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,cAAc;aAC/C,CAAC;iBACD,MAAM,CAAC,YAAY,CAAC,iBAAiB,CAAC,CAAC;YAE1C,oBAAoB;YACpB,MAAM,QAAQ;iBACX,OAAO,CAAC;gBACP,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO;gBAC9B,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,cAAc;aAC/C,CAAC;iBACD,MAAM,CAAC,YAAY,CAAC,mBAAmB,CAAC,CAAC;YAE5C,4BAA4B;YAC5B,MAAM,QAAQ;iBACX,OAAO,CAAC;gBACP,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO;gBAC9B,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS;gBAClC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;aACrD,CAAC;iBACD,MAAM,CAAC,YAAY,CAAC,2BAA2B,CAAC,CAAC;YAEpD,gEAAgE;YAChE,MAAM,QAAQ;iBACX,OAAO,CAAC;gBACP,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO;gBAC9B,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS;gBAClC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,QAAQ;gBAChC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,cAAc,GAAG,WAAW;aAC7D,CAAC;iBACD,MAAM,CAAC,YAAY,CAAC,yEAAyE,CAAC,CAAC;YAElG,kCAAkC;YAClC,MAAM,QAAQ;iBACX,OAAO,CAAC;gBACP,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO;gBAC9B,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS;gBAClC,QAAQ,EAAE,QAAQ,CAAC,IAAI,CAAC,eAAe;gBACvC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,cAAc;aAC/C,CAAC;iBACD,MAAM,CAAC,YAAY,CAAC,4CAA4C,CAAC,CAAC;QACvE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mFAAmF,EAAE,KAAK;YAC3F,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC;gBACtC,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,oBAAoB,EAAE,eAAe;gBACrC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,gBAAgB;gBACnD,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;aACF,CAAC,CAAC;YAEH,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC/B,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC/C,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAC5C,MAAM,CAAC,KAAK,CAAE,QAAkB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAE/C,MAAM,mBAAmB,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAClD,mBAAmB,CAAC,kBAAkB,CAAE,QAAkB,CAAC,YAAY,CAAC,CAAC;YACzE,MAAM,YAAY,GAAG,mBAAmB,CAAC,MAAM,EAAE,CAAC;YAElD,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YAC9E,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACrE,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAEvF,MAAM,gBAAgB,GAAG,YAAY,CAAC,gBAAuC,CAAC;YAC9E,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACzC,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;YAEvD,MAAM,2BAA2B,GAAG,MAAM,IAAA,wCAAgC,EACxE,eAAe,EACf,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAChC,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,8BAA8B,CAAC,CAAC;YACvE,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,WAAW,EAAE,eAAe,CAAC,CAAC;YACnF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,UAAU,EAAE,2BAA2B,CAAC,CAAC;YAC9F,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,YAAY,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YACpG,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;YAC7E,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,YAAY,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAEpG,MAAM,sBAAsB,GAAG,MAAM,IAAA,wCAAgC,EACnE,eAAe,EACf,QAAQ,CAAC,OAAO,CAAC,cAAc,CAChC,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC;YACxD,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACzG,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,SAAS,EAAE,2BAA2B,CAAC,CAAC;YACnG,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;YACjF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;YACnF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,aAAa,EAAE,sBAAsB,CAAC,CAAC;YAElG,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wFAAwF,EAAE,KAAK;YAChG,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC;gBACtC,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB;gBACpD,oBAAoB,EAAE,cAAc;gBACpC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,gBAAgB;gBACnD,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;gBACD,SAAS,EAAE,6CAA6C;aACzD,CAAC,CAAC;YAEH,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC/B,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC/C,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAC5C,MAAM,CAAC,KAAK,CAAE,QAAkB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAE/C,MAAM,mBAAmB,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAClD,mBAAmB,CAAC,kBAAkB,CAAE,QAAkB,CAAC,YAAY,CAAC,CAAC;YACzE,MAAM,YAAY,GAAG,mBAAmB,CAAC,MAAM,EAAE,CAAC;YAElD,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YAC9E,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACrE,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAEvF,MAAM,gBAAgB,GAAG,YAAY,CAAC,gBAAuC,CAAC;YAC9E,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACzC,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;YAEvD,MAAM,uBAAuB,GAAG,MAAM,IAAA,wCAAgC,EACpE,cAAc,EACd,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAChC,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,8BAA8B,CAAC,CAAC;YACvE,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,WAAW,EAAE,cAAc,CAAC,CAAC;YAClF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,UAAU,EAAE,uBAAuB,CAAC,CAAC;YAC1F,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,YAAY,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YACpG,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAa,CAAC,MAAM,CAAC,YAAY,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAEpG,MAAM,kBAAkB,GAAG,MAAM,IAAA,wCAAgC,EAC/D,cAAc,EACd,QAAQ,CAAC,OAAO,CAAC,cAAc,CAChC,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC;YACxD,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACzG,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,SAAS,EAAE,uBAAuB,CAAC,CAAC;YAC/F,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;YACjF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;YACtF,MAAM,CAAC,KAAK,CAAE,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,aAAa,EAAE,kBAAkB,CAAC,CAAC;YAC9F,MAAM,CAAC,KAAK,CACT,gBAAgB,CAAC,CAAC,CAAmB,CAAC,MAAM,CAAC,SAAS,EACvD,6CAA6C,CAC9C,CAAC;YAEF,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gFAAgF,EAAE,KAAK;YACxF,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC;gBACtC,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;gBACrD,oBAAoB,EAAE,eAAe;gBACrC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,gBAAgB;gBACnD,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;aACF,CAAC,CAAC;YAEH,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC/B,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC/C,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAC5C,MAAM,CAAC,KAAK,CAAE,QAAkB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAE/C,MAAM,mBAAmB,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAClD,mBAAmB,CAAC,kBAAkB,CAAE,QAAkB,CAAC,YAAY,CAAC,CAAC;YACzE,MAAM,YAAY,GAAG,mBAAmB,CAAC,MAAM,EAAE,CAAC;YAElD,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YAC9E,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACrE,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAEvF,MAAM,gBAAgB,GAAG,YAAY,CAAC,gBAAmC,CAAC;YAC1E,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACzC,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;YAEvD,MAAM,sBAAsB,GAAG,MAAM,IAAA,wCAAgC,EACnE,eAAe,EACf,QAAQ,CAAC,OAAO,CAAC,cAAc,CAChC,CAAC;YACF,MAAM,2BAA2B,GAAG,MAAM,IAAA,wCAAgC,EACxE,eAAe,EACf,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CACjC,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC;YACxD,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACtF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,EAAE,2BAA2B,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;YAC9D,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;YAChE,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,aAAa,EAAE,sBAAsB,CAAC,CAAC;YAE/E,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qFAAqF,EAAE,KAAK;YAC7F,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC;gBACtC,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;gBACrD,oBAAoB,EAAE,cAAc;gBACpC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,gBAAgB;gBACnD,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;gBACD,SAAS,EAAE,iCAAqB,CAAC,QAAQ,EAAE;aAC5C,CAAC,CAAC;YAEH,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC/B,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC/C,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAC5C,MAAM,CAAC,KAAK,CAAE,QAAkB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAE/C,MAAM,mBAAmB,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAClD,mBAAmB,CAAC,kBAAkB,CAAE,QAAkB,CAAC,YAAY,CAAC,CAAC;YACzE,MAAM,YAAY,GAAG,mBAAmB,CAAC,MAAM,EAAE,CAAC;YAClD,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YAC1B,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YAC9E,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACrE,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAEvF,MAAM,gBAAgB,GAAG,YAAY,CAAC,gBAAmC,CAAC;YAC1E,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACzC,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;YAEvD,MAAM,sBAAsB,GAAG,MAAM,IAAA,wCAAgC,EACnE,cAAc,EACd,QAAQ,CAAC,OAAO,CAAC,cAAc,EAC/B,KAAK,EACL,iCAAqB,CAAC,QAAQ,EAAE,CACjC,CAAC;YACF,MAAM,2BAA2B,GAAG,MAAM,IAAA,wCAAgC,EACxE,cAAc,EACd,QAAQ,CAAC,IAAI,CAAC,kBAAkB,EAChC,KAAK,EACL,iCAAqB,CAAC,QAAQ,EAAE,CACjC,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC;YACxD,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACtF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,EAAE,2BAA2B,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;YAC9D,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;YACnE,MAAM,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,aAAa,EAAE,sBAAsB,CAAC,CAAC;YAC/E,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8GAA8G,EAAE,KAAK;YACtH,MAAM,WAAW,GAAG,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC;YACpE,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC;YACtC,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,kBAAkB;oBAC1B,MAAM,EAAE;wBACN,KAAK,CAAC,KAAK,CAAC,MAAM;wBAClB;4BACE,UAAU,EAAE,WAAW;yBACxB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,WAAW,CAAC,CAAC;YAEzB,MAAM,QAAQ,GAAG,CAAC,MAAM,QAAQ,CAAC,OAAO,CAAC;gBACvC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,mBAAmB,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;gBACrD,YAAY,EAAE;oBACZ,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,kBAAkB;oBAC3C,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;iBAC7C;gBACD,oBAAoB,EAAE,QAAQ,CAAC,YAAY,CAAC,QAAQ;aACrD,CAAC,CAAgB,CAAC;YAEnB,0FAA0F;YAC1F,MAAM,WAAW,GAAG,IAAI,GAAG,IAAI,CAAC;YAEhC,MAAM,EAAE,YAAY,EAAE,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE,GAAG,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC;YACzG,gBAAM,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;YACxB,gBAAM,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YACjC,gBAAM,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC;YACnB,gBAAM,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC;YAC9D,gBAAM,CAAC,WAAW,CAAC,OAAO,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;YAC7C,gBAAM,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC;YACpB,gBAAM,CAAC,EAAE,CAAC,QAAQ,CAAC,MAAM,YAAY,KAAK,IAAI,QAAQ,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC5E,gBAAM,CAAC,EAAE,CAAC,QAAQ,CAAC,OAAO,YAAY,KAAK,IAAI,QAAQ,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAE9E,MAAM,mBAAmB,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAClD,mBAAmB,CAAC,kBAAkB,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;YACvG,MAAM,YAAY,GAAG,mBAAmB,CAAC,MAAM,EAAE,CAAC;YAElD,gBAAM,CAAC,WAAW,CAAC,YAAY,CAAC,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,qBAAqB,CAAC,CAAC;YACpF,gBAAM,CAAC,WAAW,CAAC,YAAY,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAC3E,gBAAM,CAAC,WAAW,CAAC,YAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,CAAC;YAC9F,MAAM,OAAO,GAAG,QAAe,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oEAAoE,EAAE,KAAK;YAC5E,yCAAyC;YACzC,MAAM,YAAY,GAAG,MAAM,QAAQ,CAAC,eAAe,CAAC;gBAClD,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,OAAO;gBACtC,SAAS,EAAE,QAAQ,CAAC,YAAY,CAAC,SAAS;gBAC1C,QAAQ,EAAE,QAAQ,CAAC,YAAY,CAAC,QAAQ;gBACxC,mBAAmB,EAAE,QAAQ,CAAC,YAAY,CAAC,iBAAiB;gBAC5D,gBAAgB,EAAE,QAAQ,CAAC,YAAY,CAAC,cAAc;gBACtD,eAAe,EAAE,QAAQ,CAAC,YAAY,CAAC,eAAe;gBACtD,6BAA6B,EAAE,QAAQ,CAAC,YAAY,CAAC,6BAA6B;aACnF,CAAC,CAAC;YACH,YAAY,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YACnC,MAAM,CAAC,KAAK,CACV,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,EACpB,0FAA0F,CAC3F,CAAC;YACF,MAAM,CAAC,KAAK,CACV,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,EACpB,0FAA0F,CAC3F,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iCAAiC,EAAE,GAAG,EAAE;QAC/C,MAAM,OAAO,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC;QACtC,MAAM,IAAI,GAAG,eAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAC/B,MAAM,eAAe,GAAG,8CAA8C,CAAC;QACvE,MAAM,aAAa,GAAG;YACpB,UAAU,EAAE;gBACV,QAAQ,CAAC,IAAI,CAAC,kBAAkB;gBAChC,QAAQ,CAAC,IAAI,CAAC,mBAAmB;gBACjC,QAAQ,CAAC,IAAI,CAAC,mBAAmB;aAClC;YACD,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,mBAAmB;SAC7C,CAAC;QAEF,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC,SAAG,CAAC,SAAS,EAAE,iBAA8B,CAAC,CAAC;YAE7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,oBAAoB;oBAC5B,MAAM,EAAE;wBACN;4BACE,UAAU,EAAE,WAAW;yBACxB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,oBAAoB,CAAC,CAAC;YACxD,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,kBAAkB;oBAC1B,MAAM,EAAE;wBACN,KAAK,CAAC,KAAK,CAAC,MAAM;wBAClB;4BACE,UAAU,EAAE,WAAW;yBACxB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC;iBAC1C;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,gCAAgC,CAAC,CAAC;YACpE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC;iBAC1C;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC;iBAC1C;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,YAAY;oBACpB,MAAM,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC;iBAC1C;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yBAAyB,CAAC,CAAC;YAC7D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,mCAAmC;oBAC3C,MAAM,EAAE,CAAC,GAAG,CAAC;iBACd;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,gBAAgB;oBACxB,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,kBAAkB;wBAChC;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAC1D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,gBAAgB;oBACxB,MAAM,EAAE;wBACN,QAAQ,CAAC,IAAI,CAAC,mBAAmB;wBACjC;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,CAAC;YAC3D,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,yCAAyC,CAAC,CAAC;YAC7E,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,+BAA+B,CAAC,CAAC;YACnE,QAAQ;iBACL,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,yBAAyB;oBACjC,MAAM,EAAE;wBACN,QAAQ,CAAC,OAAO,CAAC,cAAc;wBAC/B;4BACE,SAAS,EAAE,6CAA6C;yBACzD;wBACD;4BACE,QAAQ,EAAE,YAAY;yBACvB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,+BAA+B,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;QAEH,SAAS,CAAC,GAAG,EAAE;YACb,OAAO,CAAC,OAAO,EAAE,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK;YACtD,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,qBAAqB,CAAC;gBAChD,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,gBAAgB;gBACnD,iBAAiB,EAAE,CAAC;gBACpB,eAAe,EAAE,CAAC;gBAClB,aAAa,EAAE,aAAa;aAC7B,CAAC,CAAW,CAAC;YACd,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC1B,GAAG,CAAC,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACxC,CAAC,GAAG,CAAC,aAAa,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAEzC,MAAM,IAAI,GAAG,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;YACjC,MAAM,8BAA8B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC7D,8BAA8B,CAAC,kBAAkB,CAAE,IAAc,CAAC,YAAY,CAAC,CAAC;YAChF,MAAM,uBAAuB,GAAG,8BAA8B,CAAC,MAAM,EAAE,CAAC;YAExE,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC;YACzG,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACpD,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YAElG,MAAM,IAAI,GAAG,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;YACjC,MAAM,8BAA8B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC7D,8BAA8B,CAAC,kBAAkB,CAAE,IAAc,CAAC,YAAY,CAAC,CAAC;YAChF,MAAM,uBAAuB,GAAG,8BAA8B,CAAC,MAAM,EAAE,CAAC;YAExE,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC;YAC1G,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACpD,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;QACpG,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK;YACxD,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,qBAAqB,CAAC;gBAChD,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,iBAAiB,EAAE,CAAC;gBACpB,eAAe,EAAE,CAAC;gBAClB,aAAa,EAAE,aAAa;aAC7B,CAAC,CAAgB,CAAC;YACnB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC1B,GAAG,CAAC,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAEtC,MAAM,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC;YAC1D,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC3C,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,aAAa,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YACxC,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACtC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;YACnC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAChC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;YAC7C,IAAI,CAAC,cAAc,EAAE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YAEzC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC3C,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC;YACxC,aAAa,EAAE,MAAM,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;YAEvD,MAAM,8BAA8B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC7D,8BAA8B,CAAC,kBAAkB,CAAE,IAAc,CAAC,YAAY,CAAC,CAAC;YAChF,MAAM,uBAAuB,GAAG,8BAA8B,CAAC,MAAM,EAAE,CAAC;YAExE,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC;YACzG,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACpD,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,CAAC;YAEnG,MAAM,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC;YAC1D,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC3C,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,aAAa,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YACxC,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACtC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;YACnC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAChC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;YAC7C,IAAI,CAAC,cAAc,EAAE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YAEzC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC3C,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC;YACxC,aAAa,EAAE,MAAM,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;YACvD,aAAa,EAAE,MAAM,CAAC,cAAc,CAAC,eAAe,CAAC,CAAC;YACtD,aAAa,EAAE,aAAa,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAE9C,MAAM,8BAA8B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC7D,8BAA8B,CAAC,kBAAkB,CAAE,IAAc,CAAC,YAAY,CAAC,CAAC;YAChF,MAAM,uBAAuB,GAAG,8BAA8B,CAAC,MAAM,EAAE,CAAC;YAExE,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC;YAC1G,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACpD,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,CAAC;QACrG,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sDAAsD,EAAE,KAAK;YAC9D,MAAM,GAAG,GAAG,CAAC,MAAM,QAAQ,CAAC,qBAAqB,CAAC;gBAChD,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,iBAAiB,EAAE,CAAC;gBACpB,eAAe,EAAE,CAAC;gBAClB,oBAAoB,EAAE,eAAe;gBACrC,aAAa,EAAE,aAAa;aAC7B,CAAC,CAAgB,CAAC;YACnB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAC1B,GAAG,CAAC,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAEtC,MAAM,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC;YAC1D,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC3C,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,aAAa,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YACxC,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACtC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;YACnC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAChC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;YAC7C,IAAI,CAAC,cAAc,EAAE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YAEzC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAC3C,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC;YACxC,aAAa,EAAE,MAAM,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAC;YAEvD,MAAM,8BAA8B,GAAG,IAAI,iBAAW,CAAC,IAAI,CAAC,CAAC;YAC7D,8BAA8B,CAAC,kBAAkB,CAAE,IAAc,CAAC,YAAY,CAAC,CAAC;YAChF,MAAM,uBAAuB,GAAG,8BAA8B,CAAC,MAAM,EAAE,CAAC;YAExE,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC,sBAAsB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC;YACzG,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACpD,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAChF,MAAM,CAAC,KAAK,CAAC,uBAAuB,CAAC,aAAa,EAAE,QAAQ,CAAC,YAAY,CAAC,uBAAuB,CAAC,CAAC;QACrG,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0EAA0E,EAAE,KAAK;YAClF,MAAM,QAAQ;iBACX,qBAAqB,CAAC;gBACrB,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,gBAAgB;gBACnD,iBAAiB,EAAE,CAAC;gBACpB,eAAe,EAAE,CAAC;gBAClB,aAAa,EAAE,aAAa;aAC7B,CAAC;iBACD,MAAM,CAAC,YAAY,CAAC,+CAA+C,CAAC,CAAC;QAC1E,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kDAAkD,EAAE,KAAK,IAAI,EAAE;YAChE,MAAM,QAAQ;iBACX,qBAAqB,CAAC;gBACrB,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,iBAAiB,EAAE,CAAC,CAAC;gBACrB,aAAa,EAAE,aAAa;aAC7B,CAAC;iBACD,MAAM,CAAC,EAAE,CAAC,YAAY,CACrB,qGAAqG,CACtG,CAAC;QACN,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yCAAyC,EAAE,KAAK,IAAI,EAAE;YACvD,MAAM,QAAQ;iBACX,qBAAqB,CAAC;gBACrB,OAAO,EAAE,QAAQ,CAAC,OAAO,CAAC,OAAO;gBACjC,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,SAAS;gBACrC,QAAQ,EAAE,QAAQ,CAAC,OAAO,CAAC,QAAQ;gBACnC,iBAAiB,EAAE,CAAC;gBACpB,eAAe,EAAE,GAAG;gBACpB,aAAa,EAAE,aAAa;aAC7B,CAAC;iBACD,MAAM,CAAC,EAAE,CAAC,YAAY,CACrB,qGAAqG,CACtG,CAAC;QACN,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,sBAAsB,EAAE;QAC/B,MAAM,OAAO,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC;QAEtC,SAAS,CAAC,GAAG,EAAE;YACb,OAAO,CAAC,OAAO,EAAE,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;YACpD,MAAM,2BAA2B,GAAG,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,MAAM,CAAC;YAC7E,MAAM,aAAa,GAAG,OAAO;iBAC1B,IAAI,CAAC,SAAG,CAAC,SAAS,EAAE,iBAA8B,CAAC;iBACnD,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,iBAAiB;oBACzB,MAAM,EAAE;wBACN,2BAA2B;wBAC3B;4BACE,QAAQ,EAAE,QAAQ;yBACnB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,4BAA4B,CAAC,CAAC;YAEhE,MAAM,YAAY,GAAG,MAAM,QAAQ,CAAC,oBAAoB,CAAC,EAAE,2BAA2B,EAAE,CAAC,CAAC;YAC1F,gBAAM,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;YACxB,gBAAM,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;YAC7B,gBAAM,CAAC,WAAW,CAChB,YAAY,CAAC,IAAI,EACjB,0FAA0F,CAC3F,CAAC;YACF,gBAAM,CAAC,WAAW,CAAC,aAAa,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;YACpD,MAAM,2BAA2B,GAAG,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,MAAM,CAAC;YAC7E,MAAM,aAAa,GAAG,OAAO;iBAC1B,IAAI,CAAC,SAAG,CAAC,SAAS,EAAE,iBAA8B,CAAC;iBACnD,QAAQ,CAAC;gBACR,OAAO,EAAE;oBACP,EAAE,EAAE,GAAG;oBACP,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,iBAAiB;oBACzB,MAAM,EAAE;wBACN,2BAA2B;wBAC3B;4BACE,QAAQ,EAAE,QAAQ;yBACnB;qBACF;iBACF;aACF,CAAC;iBACD,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,iCAAiC,CAAC,CAAC;YAErE,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE;gBACT,MAAM,QAAQ,CAAC,oBAAoB,CAAC,EAAE,2BAA2B,EAAE,CAAC,CAAC;YACvE,CAAC,EACD,EAAE,OAAO,EAAE,oFAAoF,EAAE,CAClG,CAAC;YACF,gBAAM,CAAC,WAAW,CAAC,aAAa,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK;YACpD,MAAM,2BAA2B,GAAG,cAAc,CAAC;YAEnD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE;gBACT,MAAM,QAAQ,CAAC,oBAAoB,CAAC,EAAE,2BAA2B,EAAE,CAAC,CAAC;YACvE,CAAC,EACD,EAAE,OAAO,EAAE,yBAAyB,EAAE,CACvC,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6CAA6C,EAAE,KAAK;YACrD,MAAM,2BAA2B,GAAG,QAAQ,CAAC,eAAe,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAE/E,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE;gBACT,MAAM,QAAQ,CAAC,oBAAoB,CAAC,EAAE,2BAA2B,EAAE,CAAC,CAAC;YACvE,CAAC,EACD,EAAE,OAAO,EAAE,yBAAyB,EAAE,CACvC,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,UAAU,EAAE,GAAG,EAAE;QACxB,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,cAAc,EAAE,GAAG,2BAAY,CAAC;QACxD,MAAM,GAAG,GAAG,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACzC,MAAM,gBAAgB,GAAG,sBAAsB,CAAC;QAChD,MAAM,YAAY,GAAG,KAAK,CAAC;QAE3B,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;YACxC,QAAQ,CAAC,gBAAgB,CAAC;gBACxB,YAAY,EAAE,GAAG;gBACjB,SAAS,EAAE,cAAc;gBACzB,gBAAgB;gBAChB,YAAY;aACb,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qDAAqD,EAAE,GAAG,EAAE;YAC7D,MAAM,qBAAqB,GAAG,IAAA,iCAAsB,EAAC,EAAE,CAAC,CAAC;YACzD,gBAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,QAAQ,CAAC,gBAAgB,CAAC;gBACxB,YAAY,EAAE,GAAG;gBACjB,SAAS,EAAE,qBAAqB;gBAChC,gBAAgB;gBAChB,YAAY;aACb,CAAC,EACJ;gBACE,OAAO,EAAE,yBAAyB;aACnC,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yDAAyD,EAAE,GAAG,EAAE;YACjE,MAAM,mBAAmB,GAAG,KAAK,CAAC;YAClC,gBAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,QAAQ,CAAC,gBAAgB,CAAC;gBACxB,YAAY,EAAE,GAAG;gBACjB,SAAS,EAAE,cAAc;gBACzB,gBAAgB,EAAE,mBAAmB;gBACrC,YAAY;aACb,CAAC,EACJ;gBACE,OAAO,EAAE,+CAA+C;aACzD,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,MAAM,UAAU,GAAG,GAAG,CAAC,OAAO,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;YAC9C,gBAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,QAAQ,CAAC,gBAAgB,CAAC;gBACxB,YAAY,EAAE,UAAU;gBACxB,SAAS,EAAE,cAAc;gBACzB,gBAAgB;gBAChB,YAAY;aACb,CAAC,EACJ;gBACE,OAAO,EAAE,yDAAyD;aACnE,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yCAAyC,EAAE,KAAK;YACjD,qBAAqB;YACrB,MAAM,UAAU,GAAG;gBACjB,EAAE,EAAE,kCAAkC;gBACtC,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE;oBACJ,kCAAkC;oBAClC,kCAAkC;oBAClC,kCAAkC;iBACnC;gBACD,YAAY,EAAE;oBACZ,WAAW,EAAE,MAAM,CAAC,GAAG;iBACxB;gBACD,YAAY,EAAE,KAAK;aACpB,CAAC;YACF,MAAM,OAAO,GAAG,IAAA,iBAAO,EAAC,UAAU,EAAE,KAAK,CAAC,CAAC;YAE3C,MAAM,SAAS,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAC1D,MAAM,KAAK,GAAG,iBAAM,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC;YAE9C,IAAA,cAAI,EAAC,KAAK,CAAC;iBACR,GAAG,CAAC,mDAAmD,CAAC;iBACxD,KAAK,CAAC,GAAG,EAAE;gBACV;oBACE,YAAY,EAAE,OAAO;iBACtB;aACF,CAAC,CAAC;YAEL,uDAAuD;YACvD,IAAA,cAAI,EAAC,KAAK,CAAC;iBACR,IAAI,CAAC,+EAA+E,CAAC;iBACrF,KAAK,CAAC,GAAG,EAAE;gBACV;oBACE,WAAW,EAAE,sCAAsC;oBACnD,QAAQ,EAAE,kCAAkC;oBAC5C,KAAK,EACH,owBAAowB;oBACtwB,OAAO,EAAE;wBACP,GAAG,EAAE,KAAK;wBACV,SAAS,EAAE,OAAO;qBACnB;oBACD,MAAM,EAAE;wBACN,QAAQ,EAAE,OAAO;wBACjB,WAAW,EAAE,SAAS;wBACtB,YAAY,EAAE;4BACZ;gCACE,QAAQ,EAAE,MAAM;gCAChB,YAAY,EAAE,SAAS;6BACxB;yBACF;wBACD,QAAQ,EAAE,GAAG;wBACb,OAAO,EAAE;4BACP;gCACE,OAAO,EAAE,8CAA8C;gCACvD,KAAK,EAAE,OAAO;gCACd,MAAM,EAAE,0BAA0B;gCAClC,OAAO,EAAE,CAAC,0BAA0B,CAAC;gCACrC,UAAU,EAAE,0BAA0B;gCACtC,WAAW,EAAE,CAAC,0BAA0B,CAAC;gCACzC,WAAW,EAAE,SAAS;gCACtB,QAAQ,EAAE,MAAM;gCAChB,UAAU,EAAE,KAAK;gCACjB,WAAW,EAAE,CAAC,KAAK,CAAC;6BACrB;yBACF;wBACD,MAAM,EAAE;4BACN;gCACE,KAAK,EAAE,OAAO;gCACd,OAAO,EAAE,8CAA8C;gCACvD,WAAW,EAAE,SAAS;6BACvB;4BACD;gCACE,KAAK,EAAE,KAAK;gCACZ,OAAO,EAAE,8CAA8C;gCACvD,WAAW,EAAE,OAAO;6BACrB;yBACF;wBACD,IAAI,EAAE,MAAM;qBACb;oBACD,aAAa,EAAE,kCAAkC;oBACjD,IAAI,EAAE,MAAM;iBACb;aACF,CAAC,CAAC;YAEL,4BAA4B;YAC5B,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE;gBACT,MAAM,SAAS,CAAC,yBAAyB,CAAC;oBACxC,gBAAgB,EAAE,UAAU;iBAC7B,CAAC,CAAC;YACL,CAAC,EACD;gBACE,OAAO,EAAE,iDAAiD;aAC3D,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK,IAAI,EAAE;YAC/D,MAAM,eAAe,GAAG;gBACtB,WAAW,EAAE,sCAAsC;gBACnD,QAAQ,EAAE,kCAAkC;gBAC5C,KAAK,EACH,owBAAowB;gBACtwB,OAAO,EAAE;oBACP,GAAG,EAAE,KAAK;oBACV,SAAS,EAAE,OAAO;iBACnB;gBACD,MAAM,EAAE;oBACN,QAAQ,EAAE,OAAO;oBACjB,WAAW,EAAE,SAAS;oBACtB,YAAY,EAAE;wBACZ;4BACE,QAAQ,EAAE,MAAM;4BAChB,YAAY,EAAE,SAAS;yBACxB;qBACF;oBACD,QAAQ,EAAE,GAAG;oBACb,OAAO,EAAE;wBACP;4BACE,OAAO,EAAE,8CAA8C;4BACvD,KAAK,EAAE,OAAO;4BACd,MAAM,EAAE,0BAA0B;4BAClC,OAAO,EAAE,CAAC,0BAA0B,CAAC;4BACrC,UAAU,EAAE,0BAA0B;4BACtC,WAAW,EAAE,CAAC,0BAA0B,CAAC;4BACzC,WAAW,EAAE,SAAS;4BACtB,QAAQ,EAAE,MAAM;4BAChB,UAAU,EAAE,KAAK;4BACjB,WAAW,EAAE,CAAC,KAAK,CAAC;yBACrB;qBACF;oBACD,MAAM,EAAE;wBACN;4BACE,KAAK,EAAE,OAAO;4BACd,OAAO,EAAE,8CAA8C;4BACvD,WAAW,EAAE,SAAS;yBACvB;wBACD;4BACE,KAAK,EAAE,KAAK;4BACZ,OAAO,EAAE,8CAA8C;4BACvD,WAAW,EAAE,OAAO;yBACrB;qBACF;oBACD,IAAI,EAAE,MAAM;iBACb;gBACD,aAAa,EAAE,kCAAkC;gBACjD,IAAI,EAAE,MAAM;aACb,CAAC;YAEF,MAAM,YAAY,GAAqB;gBACrC,YAAY,EAAE,GAAG,EAAE;oBACjB,MAAM,EAAE,GAAG;wBACT,WAAW,EAAE,8CAA8C;qBACtC,CAAC;oBACxB,OAAO,EAAE,CAAC;gBACZ,CAAC;aACF,CAAC;YAEF,IAAI,CAAC;gBACH,IACE,CAAC,CAAC,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBACjC,SAAS,EAAE,EAAE;oBACb,QAAQ,EAAE,EAAE;oBACZ,QAAQ,EAAE,EAAE;oBACZ,UAAU,EAAE,eAAiD;oBAC7D,UAAU,EAAE,KAAK;oBACjB,MAAM,EAAE,YAAuB;oBAC/B,YAAY,EAAE;wBACZ,0BAA0B,EAAE,IAAI;qBACjC;iBACF,CAAC,CAAC,EACH,CAAC;oBACD,gBAAM,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;gBACtD,CAAC;YACH,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,gBAAM,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK,IAAI,EAAE;YAC/D,MAAM,eAAe,GAAG;gBACtB,WAAW,EAAE,sCAAsC;gBACnD,QAAQ,EAAE,kCAAkC;gBAC5C,KAAK,EACH,o8BAAo8B;gBACt8B,OAAO,EAAE;oBACP,GAAG,EAAE,KAAK;oBACV,SAAS,EAAE,OAAO;iBACnB;gBACD,MAAM,EAAE;oBACN,MAAM,EAAE;wBACN;4BACE,OAAO,EAAE,8CAA8C;4BACvD,KAAK,EAAE,YAAY;4BACnB,WAAW,EAAE,aAAa;yBAC3B;wBACD;4BACE,OAAO,EAAE,8CAA8C;4BACvD,KAAK,EAAE,KAAK;4BACZ,WAAW,EAAE,OAAO;yBACrB;qBACF;oBACD,QAAQ,EAAE,OAAO;oBACjB,OAAO,EAAE;wBACP;4BACE,OAAO,EAAE,8CAA8C;4BACvD,QAAQ,EAAE,SAAS;4BACnB,UAAU,EAAE;gCACV,IAAI,EAAE,0BAA0B;6BACjC;4BACD,WAAW,EAAE;gCACX;oCACE,IAAI,EAAE,0BAA0B;iCACjC;6BACF;4BACD,KAAK,EAAE,YAAY;4BACnB,WAAW,EAAE,aAAa;4BAC1B,MAAM,EAAE;gCACN,IAAI,EAAE,0BAA0B;6BACjC;4BACD,UAAU,EAAE,KAAK;4BACjB,OAAO,EAAE;gCACP;oCACE,IAAI,EAAE,0BAA0B;iCACjC;6BACF;yBACF;qBACF;oBACD,QAAQ,EAAE,GAAG;oBACb,WAAW,EAAE,aAAa;oBAC1B,YAAY,EAAE;wBACZ;4BACE,YAAY,EAAE,aAAa;4BAC3B,QAAQ,EAAE,SAAS;yBACpB;qBACF;oBACD,IAAI,EAAE,MAAM;iBACb;gBACD,aAAa,EAAE,kCAAkC;gBACjD,IAAI,EAAE,MAAM;aACb,CAAC;YAEF,MAAM,YAAY,GAAqB;gBACrC,YAAY,EAAE,GAAG,EAAE;oBACjB,MAAM,EAAE,GAAG;wBACT,WAAW,EAAE,8CAA8C;qBACtC,CAAC;oBACxB,OAAO,EAAE,CAAC;gBACZ,CAAC;aACF,CAAC;YAEF,IAAI,CAAC;gBACH,IACE,CAAC,CAAC,MAAM,QAAQ,CAAC,iBAAiB,CAAC;oBACjC,SAAS,EAAE,EAAE;oBACb,QAAQ,EAAE,EAAE;oBACZ,QAAQ,EAAE,EAAE;oBACZ,UAAU,EAAE,eAAiD;oBAC7D,UAAU,EAAE,KAAK;oBACjB,MAAM,EAAE,YAAuB;oBAC/B,YAAY,EAAE;wBACZ,0BAA0B,EAAE,IAAI;qBACjC;iBACF,CAAC,CAAC,EACH,CAAC;oBACD,gBAAM,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;gBACtD,CAAC;YACH,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,gBAAM,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yDAAyD,EAAE,KAAK,IAAI,EAAE;YACvE,MAAM,eAAe,GAAG;gBACtB,WAAW,EAAE,sCAAsC;gBACnD,QAAQ,EAAE,kCAAkC;gBAC5C,KAAK,EACH,o8BAAo8B;gBACt8B,OAAO,EAAE;oBACP,GAAG,EAAE,KAAK;oBACV,SAAS,EAAE,OAAO;iBACnB;gBACD,MAAM,EAAE;oBACN,MAAM,EAAE;wBACN;4BACE,OAAO,EAAE,8CAA8C;4BACvD,KAAK,EAAE,YAAY;4BACnB,WAAW,EAAE,aAAa;yBAC3B;wBACD;4BACE,OAAO,EAAE,8CAA8C;4BACvD,KAAK,EAAE,KAAK;4BACZ,WAAW,EAAE,OAAO;yBACrB;qBACF;oBACD,QAAQ,EAAE,OAAO;oBACjB,OAAO,EAAE;wBACP;4BACE,OAAO,EAAE,8CAA8C;4BACvD,QAAQ,EAAE,SAAS;4BACnB,UAAU,EAAE;gCACV,IAAI,EAAE,0BAA0B;6BACjC;4BACD,WAAW,EAAE;gCACX;oCACE,IAAI,EAAE,0BAA0B;iCACjC;6BACF;4BACD,KAAK,EAAE,YAAY;4BACnB,WAAW,EAAE,aAAa;4BAC1B,MAAM,EAAE;gCACN,IAAI,EAAE,0BAA0B;6BACjC;4BACD,UAAU,EAAE,KAAK;4BACjB,OAAO,EAAE;gCACP;oCACE,IAAI,EAAE,0BAA0B;iCACjC;6BACF;yBACF;qBACF;oBACD,QAAQ,EAAE,GAAG;oBACb,WAAW,EAAE,aAAa;oBAC1B,YAAY,EAAE;wBACZ;4BACE,YAAY,EAAE,aAAa;4BAC3B,QAAQ,EAAE,SAAS;yBACpB;qBACF;oBACD,IAAI,EAAE,MAAM;iBACb;gBACD,aAAa,EAAE,kCAAkC;gBACjD,IAAI,EAAE,MAAM;aACb,CAAC;YAEF,MAAM,YAAY,GAAqB;gBACrC,YAAY,EAAE,GAAG,EAAE;oBACjB,MAAM,EAAE,GAAG;wBACT,WAAW,EAAE,8CAA8C;qBACtC,CAAC;oBACxB,OAAO,EAAE,CAAC;gBACZ,CAAC;aACF,CAAC;YAEF,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,QAAQ,CAAC,iBAAiB,CAAC;gBACzB,SAAS,EAAE,EAAE;gBACb,QAAQ,EAAE,EAAE;gBACZ,QAAQ,EAAE,EAAE;gBACZ,UAAU,EAAE,eAAiD;gBAC7D,UAAU,EAAE,KAAK;gBACjB,MAAM,EAAE,YAAuB;gBAC/B,YAAY,EAAE;oBACZ,0BAA0B,EAAE,IAAI;iBACjC;aACF,CAAC,EACJ;gBACE,OAAO,EAAE,iDAAiD;aAC3D,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACzD,EAAE,CAAC,oFAAoF,EAAE,KAAK;YAC5F,MAAM,EAAE,QAAQ,EAAE,aAAa,EAAE,UAAU,EAAE,GAAG,QAAQ,CAAC,mBAAmB,CAAC;YAC7E,MAAM,MAAM,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YACvD,MAAM,YAAY,GAAG,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBACpD,QAAQ;gBACR,UAAU,EAAE,aAAa;gBACzB,MAAM;gBACN,YAAY,EAAE,EAAE,qBAAqB,EAAE,IAAI,EAAE;aACJ,CAAC,CAAC;YAE7C,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wDAAwD,EAAE,KAAK;YAChE,MAAM,EAAE,QAAQ,EAAE,aAAa,EAAE,SAAS,EAAE,UAAU,EAAE,GAAG,QAAQ,CAAC,mBAAmB,CAAC;YACxF,MAAM,kBAAkB,GAAG,EAAE,GAAG,aAAa,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC;YAElE,MAAM,MAAM,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAEvD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,MAAM,QAAQ,CAAC,iBAAiB,CAAC;gBAC/B,QAAQ;gBACR,UAAU,EAAE,kBAAkB;gBAC9B,MAAM;gBACN,YAAY,EAAE,EAAE,qBAAqB,EAAE,IAAI,EAAE;aACJ,CAAC,EAC9C;gBACE,OAAO,EACL,4GAA4G;aAC/G,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kEAAkE,EAAE,KAAK;YAC1E,MAAM,EAAE,QAAQ,EAAE,aAAa,EAAE,mBAAmB,EAAE,UAAU,EAAE,GAAG,QAAQ,CAAC,mBAAmB,CAAC;YAClG,MAAM,kBAAkB,GAAG,EAAE,GAAG,aAAa,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC;YAC5E,MAAM,MAAM,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAEvD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,QAAQ,CAAC,iBAAiB,CAAC;gBACzB,QAAQ;gBACR,UAAU,EAAE,kBAAkB;gBAC9B,MAAM;gBACN,YAAY,EAAE,EAAE,qBAAqB,EAAE,IAAI,EAAE;aACJ,CAAC,EAC9C,EAAE,OAAO,EAAE,gFAAgF,EAAE,CAC9F,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kEAAkE,EAAE,KAAK;YAC1E,MAAM,EAAE,QAAQ,EAAE,aAAa,EAAE,cAAc,EAAE,UAAU,EAAE,GAAG,QAAQ,CAAC,mBAAmB,CAAC;YAC7F,MAAM,kBAAkB,GAAG,EAAE,GAAG,aAAa,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC;YAEvE,MAAM,MAAM,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YACvD,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,QAAQ,CAAC,iBAAiB,CAAC;gBACzB,QAAQ;gBACR,UAAU,EAAE,kBAAkB;gBAC9B,MAAM;gBACN,YAAY,EAAE,EAAE,qBAAqB,EAAE,IAAI,EAAE;aACJ,CAAC,EAC9C;gBACE,OAAO,EACL,uJAAuJ;aAC1J,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sDAAsD,EAAE,KAAK;YAC9D,MAAM,EAAE,0BAA0B,EAAE,UAAU,EAAE,mBAAmB,EAAE,GAAG,QAAQ,CAAC,mBAAmB,CAAC;YACrG,MAAM,MAAM,GAAG,IAAI,iBAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YACvD,IAAA,cAAI,EAAC,uBAAuB,CAAC,CAAC,GAAG,CAAC,mDAAmD,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;YACtG,MAAM,gBAAM,CAAC,OAAO,CAClB,KAAK,IAAI,EAAE,CACT,MAAM,CAAC,mBAAmB,CAAC;gBACzB,YAAY,EAAE,EAAE,qBAAqB,EAAE,IAAI,EAAE;gBAC7C,GAAG,0BAA0B;gBAC7B,UAAU,EAAE,EAAE,GAAG,0BAA0B,CAAC,UAAU,EAAE,KAAK,EAAE,mBAAmB,EAAE;aACrC,CAAC,EACpD;gBACE,OAAO,EAAE,gFAAgF;aAC1F,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;QAC/B,EAAE,CAAC,oEAAoE,EAAE,KAAK;YAC5E,MAAM,OAAO,GAAG,8CAA8C,CAAC;YAC/D,MAAM,cAAc,GAClB,kIAAkI,CAAC;YACrI,MAAM,KAAK,GAAG,GAAG,CAAC;YAClB,MAAM,SAAS,GAAG,CAAC,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,KAAc,EAAE,cAAc,EAAE,CAAC,CAAC;YAEtE,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,eAAe,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YAC7E,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC5B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yDAAyD,EAAE,KAAK;YACjE,MAAM,OAAO,GAAG,8CAA8C,CAAC;YAC/D,MAAM,aAAa,GACjB,iIAAiI,CAAC;YACpI,MAAM,KAAK,GAAG,GAAG,CAAC;YAClB,MAAM,SAAS,GAAG,CAAC,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,KAAc,EAAE,cAAc,EAAE,aAAa,EAAE,CAAC,CAAC;YAErF,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,eAAe,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;YAC7E,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sDAAsD,EAAE,KAAK;YAC9D,MAAM,OAAO,GAAG,8CAA8C,CAAC;YAC/D,MAAM,cAAc,GAClB,kIAAkI,CAAC;YACrI,MAAM,UAAU,GAAG,KAAK,CAAC;YACzB,MAAM,SAAS,GAAG,CAAC,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,KAAc,EAAE,cAAc,EAAE,CAAC,CAAC;YAEtE,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,eAAe,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC,CAAC;YACzF,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wCAAwC,EAAE,KAAK;YAChD,MAAM,cAAc,GAAG,gBAAgB,CAAC;YACxC,MAAM,cAAc,GAClB,kIAAkI,CAAC;YACrI,MAAM,KAAK,GAAG,GAAG,CAAC;YAClB,MAAM,SAAS,GAAG,CAAC,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,KAAc,EAAE,cAAc,EAAE,CAAC,CAAC;YAEtE,MAAM,gBAAM,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC,MAAM,QAAQ,CAAC,eAAe,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,cAAc,EAAE,KAAK,EAAE,CAAC,EAAE;gBAC9G,OAAO,EAAE,oBAAoB,cAAc,EAAE;aAC9C,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,yBAAyB,EAAE,GAAG,EAAE;QACvC,EAAE,CAAC,6CAA6C,EAAE;YAChD,MAAM,SAAS,GAAG,kEAAkE,CAAC;YACrF,MAAM,eAAe,GAAG,8CAA8C,CAAC;YAEvE,MAAM,OAAO,GAAG,QAAQ,CAAC,uBAAuB,CAAC,SAAS,CAAC,CAAC;YAC5D,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import assert from 'assert';\nimport * as _ from 'lodash';\nimport nock from 'nock';\nimport * as should from 'should';\nimport * as sinon from 'sinon';\n\nimport { TOKEN_2022_PROGRAM_ID } from '@solana/spl-token';\n\nimport { BitGoAPI, encrypt } from '@bitgo-beta/sdk-api';\nimport {\n  common,\n  Environments,\n  generateRandomPassword,\n  IWallet,\n  MPCSweepTxs,\n  MPCTx,\n  MPCTxs,\n  PrebuildAndSignTransactionOptions,\n  TransactionPrebuild,\n  TssUtils,\n  TxRequest,\n  Wallet,\n  WalletCoinSpecific,\n} from '@bitgo-beta/sdk-core';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { coins } from '@bitgo-beta/statics';\nimport { KeyPair, Sol, SolVerifyTransactionOptions, Tsol } from '../../src';\nimport { Transaction } from '../../src/lib';\nimport { AtaInit, InstructionParams, TokenTransfer } from '../../src/lib/iface';\nimport { getAssociatedTokenAccountAddress } from '../../src/lib/utils';\nimport * as testData from '../fixtures/sol';\nimport * as resources from '../resources/sol';\nimport { solBackupKey } from './fixtures/solBackupKey';\nimport { getBuilderFactory } from './getBuilderFactory';\n\ndescribe('SOL:', function () {\n  let bitgo: TestBitGoAPI;\n  let basecoin: Sol;\n  let keyPair;\n  let newTxPrebuild;\n  let newTxPrebuildTokenTransfer;\n  let newTxParams;\n  let newTxParamsWithError;\n  let newTxParamsWithExtraData;\n  let newTxParamsTokenTransfer;\n  const badAddresses = resources.addresses.invalidAddresses;\n  const goodAddresses = resources.addresses.validAddresses;\n\n  const keypair = {\n    pub: resources.accountWithSeed.publicKey,\n    prv: resources.accountWithSeed.privateKey.base58,\n  };\n  const txPrebuild = {\n    recipients: [\n      {\n        address: 'lionteste212',\n        amount: '1000',\n      },\n    ],\n    txBase64: resources.TRANSFER_UNSIGNED_TX_WITH_MEMO_AND_DURABLE_NONCE,\n    txInfo: {\n      feePayer: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n      nonce: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n    },\n    txid: '586c5b59b10b134d04c16ac1b273fe3c5529f34aef75db4456cd469c5cdac7e2',\n    isVotingTransaction: false,\n    coin: 'tsol',\n  };\n  const txParams = {\n    txPrebuild,\n    recipients: [\n      {\n        address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n        amount: '300000',\n      },\n    ],\n  };\n  const memo = { value: 'test memo' };\n  const durableNonce = {\n    walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n    authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n  };\n  const errorDurableNonce = {\n    walletNonceAddress: '8YM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n    authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n  };\n  const txParamsWithError = {\n    txPrebuild,\n    recipients: [\n      {\n        address: 'CP5Dpaa42mMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n        amount: '300000',\n      },\n    ],\n  };\n  const txParamsWithExtraData = {\n    txPrebuild,\n    recipients: [\n      {\n        address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n        amount: '300000',\n        data: undefined,\n      },\n    ],\n  };\n  const txPrebuildTokenTransfer = {\n    recipients: [\n      {\n        address: 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg',\n        amount: '1',\n      },\n    ],\n    txHex: resources.TOKEN_TRANSFER_TO_NATIVE_UNSIGNED_TX_HEX,\n    txInfo: {\n      feePayer: '4DujymUFbQ8GBKtAwAZrQ6QqpvtBEivL48h4ta2oJGd2',\n      nonce: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n    },\n    txid: '586c5b59b10b134d04c16ac1b273fe3c5529f34aef75db4456cd469c5cdac7e2',\n    isVotingTransaction: false,\n    coin: 'tsol',\n  };\n  const txParamsTokenTransfer = {\n    txPrebuild,\n    recipients: [\n      {\n        address: 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg',\n        amount: '1',\n      },\n    ],\n  };\n  const errorMemo = { value: 'different memo' };\n  const errorFeePayer = '5hr5fisPi6DXCuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe';\n  const factory = getBuilderFactory('tsol');\n  const wallet = new KeyPair(resources.authAccount).getKeys();\n  const stakeAccount = new KeyPair(resources.stakeAccount).getKeys();\n  const blockHash = resources.blockHashes.validBlockHashes[0];\n  const amount = '10000';\n  const validator = resources.validator;\n\n  before(function () {\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'mock' });\n    bitgo.safeRegister('sol', Tsol.createInstance);\n    bitgo.safeRegister('tsol', Tsol.createInstance);\n    bitgo.initializeTestVars();\n    basecoin = bitgo.coin('tsol') as Tsol;\n    keyPair = basecoin.generateKeyPair(resources.accountWithSeed.seed);\n    newTxPrebuild = () => {\n      return _.cloneDeep(txPrebuild);\n    };\n    newTxPrebuildTokenTransfer = () => {\n      return _.cloneDeep(txPrebuildTokenTransfer);\n    };\n    newTxParams = () => {\n      return _.cloneDeep(txParams);\n    };\n    newTxParamsWithError = () => {\n      return _.cloneDeep(txParamsWithError);\n    };\n    newTxParamsWithExtraData = () => {\n      return _.cloneDeep(txParamsWithExtraData);\n    };\n    newTxParamsTokenTransfer = () => {\n      return _.cloneDeep(txParamsTokenTransfer);\n    };\n  });\n\n  it('should instantiate the coin', async function () {\n    let localBasecoin = bitgo.coin('tsol');\n    localBasecoin.should.be.an.instanceof(Tsol);\n\n    localBasecoin = bitgo.coin('sol');\n    localBasecoin.should.be.an.instanceof(Sol);\n  });\n\n  it('should retun the right info', function () {\n    basecoin.getChain().should.equal('tsol');\n    basecoin.getFamily().should.equal('sol');\n    basecoin.getFullName().should.equal('Testnet Solana');\n    basecoin.getBaseFactor().should.equal(1000000000);\n  });\n  describe('verify transactions', () => {\n    const walletData = {\n      id: '5b34252f1bf349930e34020a00000000',\n      coin: 'tsol',\n      keys: [\n        '5b3424f91bf349930e34017500000000',\n        '5b3424f91bf349930e34017600000000',\n        '5b3424f91bf349930e34017700000000',\n      ],\n      coinSpecific: {\n        rootAddress: wallet.pub,\n      },\n      multisigType: 'tss',\n    };\n    const walletObj = new Wallet(bitgo, basecoin, walletData);\n\n    it('should verify transactions', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        durableNonce,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should verify consolidate transaction', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.consolidateId = 'consolidateId';\n\n      const walletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {\n          rootAddress: stakeAccount.pub,\n        },\n        multisigType: 'tss',\n      };\n      const walletWithDifferentAddress = new Wallet(bitgo, basecoin, walletData);\n\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        durableNonce,\n        wallet: walletWithDifferentAddress,\n      } as any);\n      validTransaction.should.be.true();\n    });\n\n    it('should handle txBase64 and txHex interchangeably', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.txHex = txPrebuild.txBase64;\n      txPrebuild.txBase64 = undefined;\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        durableNonce,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should convert serialized hex string to base64', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.txBase64 = Buffer.from(txPrebuild.txBase64, 'base64').toString('hex');\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        durableNonce,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should verify when input `recipients` is absent', async function () {\n      const txParams = newTxParams();\n      txParams.recipients = undefined;\n      const txPrebuild = newTxPrebuild();\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        durableNonce,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should fail verify transactions when have different memo', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, memo: errorMemo, wallet: walletObj } as any)\n        .should.be.rejectedWith('Tx memo does not match with expected txParams recipient memo');\n    });\n\n    it('should pass if we pass PDA address', async function () {\n      const walletData = {\n        id: '67f8ddff4c9b8b57a2e16acffac9a3b5',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {\n          rootAddress: '8zbsJA5c8HPR7BPjZkrSVrus2uMuXqCfzksGwB3Uscjb',\n        },\n        multisigType: 'tss',\n      };\n      const walletObj = new Wallet(bitgo, basecoin, walletData);\n      const txPrebuild = {\n        recipients: [\n          {\n            address: '11111111111111111111111111111112',\n            amount: '1000000000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n        txBase64:\n          '02000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006ec1adcc89bb564f1f8225821140a9723efa80e8d506765770b7e201d66d8200d4f690e9a8163291b69f8c3827aad96cfd2105eee3aae76cbca38fcad2bf7f0a0201070c76c356cb069b66c2b35a8638b4d4afca75b303f29f0deeb4bff8528299a9c9d21c96172044f1217c3784e8f02f49e2c8fc3591e81294ab54394f9d22fd7b7a8f60129e6ecb20309c27dcba5fc6c441438d33a1568004a1860e22c16f071976a7d2e2008bd34b53a08aa9c8ec04eb2196745fc6029224447417e2fb0fced601240cabba4ce534c02fc154ba559ed2a02ac971e3385acb426ff63bb1040e2c2435000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000018c97258f4e2489f1bb3d1029148e0d830b5a1399daff1084048e7bd8dbe9f859d10389fbcee528f208611dccc734b31092540cb2b8d58d100f2eaa2cedb4da5e06a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000006a7d517192c5c51218cc94c3d4af17f58daee089ba1fd44e3dbd98a0000000006ddf6e1d765a193d9cbe146ceeb79ac1cb485ed5f5b37913a8cf5857eff00a9e680634533882f880a3e7dfa999dfb864b88968d242a0c9a90b5df149e42da050305030209010404000000070700030608050b0a000b04040803000a0c00ca9a3b0000000009',\n        txInfo: {\n          feePayer: '8zbsJA5c8HPR7BPjZkrSVrus2uMuXqCfzksGwB3Uscjb',\n          nonce: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        },\n        txid: '586c5b59b10b134d04c16ac1b273fe3c5529f34aef75db4456cd469c5cdac7e2',\n        isVotingTransaction: false,\n        coin: 'tsol',\n      };\n      const txParams = {\n        txPrebuild,\n        recipients: [\n          {\n            address: '11111111111111111111111111111112',\n            amount: '1000000000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n      };\n      const memo = {\n        value: undefined,\n      };\n      const verifyTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo: memo,\n        wallet: walletObj,\n      } as any);\n      verifyTransaction.should.equal(true);\n    });\n\n    it('should fail verify transactions when have different durableNonce', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, memo, durableNonce: errorDurableNonce, wallet: walletObj } as any)\n        .should.be.rejectedWith('Tx durableNonce does not match with param durableNonce');\n    });\n\n    it('should fail verify transactions when have different feePayer', async function () {\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      const walletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {\n          rootAddress: stakeAccount.pub,\n        },\n        multisigType: 'tss',\n      };\n      const walletWithDifferentAddress = new Wallet(bitgo, basecoin, walletData);\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, memo, wallet: walletWithDifferentAddress } as any)\n        .should.be.rejectedWith('Tx fee payer is not the wallet root address');\n    });\n\n    it('should fail verify transactions when have different recipients', async function () {\n      const txParams = newTxParamsWithError();\n      const txPrebuild = newTxPrebuild();\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, memo, errorFeePayer, wallet: walletObj } as any)\n        .should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');\n    });\n\n    it('should succeed to verify token transaction with native address recipient', async function () {\n      const txParams = newTxParamsTokenTransfer();\n      const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address\n      txParams.recipients = [{ address, amount: '1', tokenName: 'tsol:usdc' }];\n      const txPrebuild = newTxPrebuildTokenTransfer();\n      const feePayerWalletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {\n          rootAddress: '4DujymUFbQ8GBKtAwAZrQ6QqpvtBEivL48h4ta2oJGd2',\n        },\n        multisigType: 'tss',\n      };\n      const feePayerWallet = new Wallet(bitgo, basecoin, feePayerWalletData);\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        wallet: feePayerWallet,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should succeed to verify token transaction with leading zero recipient amount', async function () {\n      const txParams = newTxParamsTokenTransfer();\n      const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address\n      txParams.recipients = [{ address, amount: '0001', tokenName: 'tsol:usdc' }];\n      const txPrebuild = newTxPrebuildTokenTransfer();\n      const feePayerWalletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {\n          rootAddress: '4DujymUFbQ8GBKtAwAZrQ6QqpvtBEivL48h4ta2oJGd2',\n        },\n        multisigType: 'tss',\n      };\n      const feePayerWallet = new Wallet(bitgo, basecoin, feePayerWalletData);\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        wallet: feePayerWallet,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should fail to verify token transaction with different recipient tokenName', async function () {\n      const txParams = newTxParamsTokenTransfer();\n      const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address\n      txParams.recipients = [{ address, amount: '1', tokenName: 'tsol:usdt' }]; // Different tokenName, should fail to verify tx\n      const txPrebuild = newTxPrebuildTokenTransfer();\n      await basecoin\n        .verifyTransaction({\n          txParams,\n          txPrebuild,\n          wallet: walletObj,\n        } as any)\n        .should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');\n    });\n\n    it('should fail to verify token transaction with different recipient amounts', async function () {\n      const txParams = newTxParamsTokenTransfer();\n      const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvg'; // Native SOL address\n      txParams.recipients = [{ address, amount: '2', tokenName: 'tsol:usdt' }];\n      const txPrebuild = newTxPrebuildTokenTransfer();\n      await basecoin\n        .verifyTransaction({\n          txParams,\n          txPrebuild,\n          wallet: walletObj,\n        } as any)\n        .should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');\n    });\n\n    it('should fail to verify token transaction with different native address', async function () {\n      const txParams = newTxParamsTokenTransfer();\n      const address = 'AF5H6vBkFnJuVqChRPgPQ4JRcQ5Gk25HBFhQQkyojmvX'; // Native SOL address, different than tx recipients\n      txParams.recipients = [{ address, amount: '1', tokenName: 'tsol:usdc' }];\n      const txPrebuild = newTxPrebuildTokenTransfer();\n      await basecoin\n        .verifyTransaction({\n          txParams,\n          txPrebuild,\n          wallet: walletObj,\n        } as any)\n        .should.be.rejectedWith('Tx outputs does not match with expected txParams recipients');\n    });\n\n    it('should succeed to verify transactions when recipients has extra data', async function () {\n      const txParams = newTxParamsWithExtraData();\n      const txPrebuild = newTxPrebuild();\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        durableNonce,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should verify activate staking transaction', async function () {\n      const tx = await factory\n        .getStakingActivateBuilder()\n        .stakingAddress(stakeAccount.pub)\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .amount(amount)\n        .validator(validator.pub)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.txBase64 = txToBroadcastFormat;\n      txPrebuild.txInfo.nonce = '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen';\n      txParams.recipients = [\n        {\n          address: '7dRuGFbU2y2kijP6o1LYNzVyz4yf13MooqoionCzv5Za',\n          amount: amount,\n        },\n      ];\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should verify withdraw staking transaction', async function () {\n      const tx = await factory\n        .getStakingWithdrawBuilder()\n        .stakingAddress(stakeAccount.pub)\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .amount(amount)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.txBase64 = txToBroadcastFormat;\n      txPrebuild.txInfo.nonce = '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen';\n      txParams.recipients = [\n        {\n          address: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n          amount: amount,\n        },\n      ];\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should verify deactivate staking transaction', async function () {\n      const tx = await factory\n        .getStakingDeactivateBuilder()\n        .stakingAddress(stakeAccount.pub)\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.txBase64 = txToBroadcastFormat;\n      txPrebuild.txInfo.nonce = '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen';\n      txParams.recipients = [];\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n\n    it('should verify create associated token account transaction', async function () {\n      const tx = await factory\n        .getAtaInitializationBuilder()\n        .mint('tsol:usdc')\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const txParams = newTxParams();\n      const txPrebuild = newTxPrebuild();\n      txPrebuild.txBase64 = txToBroadcastFormat;\n      txPrebuild.txInfo.nonce = blockHash;\n      txParams.recipients = [];\n      const validTransaction = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild,\n        memo,\n        wallet: walletObj,\n      } as any);\n      validTransaction.should.equal(true);\n    });\n  });\n\n  it('should accept valid address', function () {\n    goodAddresses.forEach((addr) => {\n      basecoin.isValidAddress(addr).should.equal(true);\n    });\n  });\n\n  it('should reject invalid address', function () {\n    badAddresses.forEach((addr) => {\n      basecoin.isValidAddress(addr).should.equal(false);\n    });\n  });\n\n  it('should check valid pub keys', function () {\n    keyPair.should.have.property('pub');\n    basecoin.isValidPub(keyPair.pub).should.equal(true);\n  });\n\n  it('should check an invalid pub keys', function () {\n    const badPubKey = keyPair.pub.slice(0, keyPair.pub.length - 1) + '-';\n    basecoin.isValidPub(badPubKey).should.equal(false);\n  });\n\n  it('should check valid prv keys', function () {\n    keyPair.should.have.property('prv');\n    basecoin.isValidPrv(keyPair.prv).should.equal(true);\n  });\n\n  it('should check an invalid prv keys', function () {\n    const badPrvKey = keyPair.prv ? keyPair.prv.slice(0, keyPair.prv.length - 1) + '-' : undefined;\n    basecoin.isValidPrv(badPrvKey as string).should.equal(false);\n  });\n\n  describe('Parse Transactions:', () => {\n    it('should parse an unsigned transfer transaction', async function () {\n      const parsedTransaction = await basecoin.parseTransaction({\n        txBase64: testData.rawTransactions.transfer.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      parsedTransaction.should.deepEqual({\n        inputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: 305000,\n          },\n        ],\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n          },\n        ],\n      });\n    });\n\n    it('should parse a signed transfer transaction', async function () {\n      const parsedTransaction = await basecoin.parseTransaction({\n        txBase64: testData.rawTransactions.transfer.signed,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      parsedTransaction.should.deepEqual({\n        inputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: 305000,\n          },\n        ],\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n          },\n        ],\n      });\n    });\n\n    it('should parse an unsigned wallet init transaction', async function () {\n      const parsedTransaction = await basecoin.parseTransaction({\n        txBase64: testData.rawTransactions.walletInit.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      parsedTransaction.should.deepEqual({\n        inputs: [\n          {\n            address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n            amount: 310000,\n          },\n        ],\n        outputs: [\n          {\n            address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n            amount: '300000',\n          },\n        ],\n      });\n    });\n\n    it('should parse a signed wallet init transaction', async function () {\n      const parsedTransaction = await basecoin.parseTransaction({\n        txBase64: testData.rawTransactions.walletInit.signed,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      parsedTransaction.should.deepEqual({\n        inputs: [\n          {\n            address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n            amount: 310000,\n          },\n        ],\n        outputs: [\n          {\n            address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n            amount: '300000',\n          },\n        ],\n      });\n    });\n\n    it('should parse an unsigned transfer token transaction', async function () {\n      const parsedTransaction = await basecoin.parseTransaction({\n        txBase64: testData.rawTransactions.transferToken.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      parsedTransaction.should.deepEqual({\n        inputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: 5000,\n          },\n        ],\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n      });\n    });\n\n    it('should parse a signed transfer token transaction', async function () {\n      const parsedTransaction = await basecoin.parseTransaction({\n        txBase64: testData.rawTransactions.transferToken.signed,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      parsedTransaction.should.deepEqual({\n        inputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: 5000,\n          },\n        ],\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n      });\n    });\n  });\n\n  describe('Explain Transactions:', () => {\n    it('should explain an unsigned transfer transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.transfer.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'Send',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '300000',\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n          },\n        ],\n        fee: {\n          fee: '5000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        durableNonce: {\n          authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n          walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n        },\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain a signed transfer transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.transfer.signed,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: '2XFxGfXddKWnqGaMAsfNL8HgXqDvjBL2Ae28KWrRvg9bQBmCrpHYVDacuZFeAUyYwjXG6ey2jTARX5VQCnj7SF4L',\n        type: 'Send',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '300000',\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n          },\n        ],\n        fee: {\n          fee: '5000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        durableNonce: {\n          authWalletAddress: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n          walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n        },\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain an unsigned wallet init transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.walletInit.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'WalletInitialization',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '300000',\n        outputs: [\n          {\n            address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n            amount: '300000',\n          },\n        ],\n        fee: {\n          fee: '10000',\n          feeRate: 5000,\n        },\n        blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        durableNonce: undefined,\n        memo: undefined,\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain a signed wallet init transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.walletInit.signed,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: '7TkU8wLgXDeLFbVydtg6mqMsp9GatsetitSngysgjxFhofKSUcLPBoKPHciLeGEfJFMsqezpZmGRSFQTBy7ZDsg',\n        type: 'WalletInitialization',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '300000',\n        outputs: [\n          {\n            address: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n            amount: '300000',\n          },\n        ],\n        fee: {\n          fee: '10000',\n          feeRate: 5000,\n        },\n        blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        durableNonce: undefined,\n        memo: undefined,\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain an unsigned token transfer transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.transferToken.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'Send',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '0',\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n        fee: {\n          fee: '5000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        durableNonce: {\n          authWalletAddress: '12f6D3WubGVeQoH2m8kTvvcrasWdXWwtVzUCyRNDZxA2',\n          walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n        },\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain a signed token transfer transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.transferToken.signed,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: '2ticU4ZkEqdTHULr6LobTgWBhim6E7wSscDhM4gzyuGUmQyUwLYhoqaifuvwmNzzEf1T5aefVcgMQkSHdJ5nsrfZ',\n        type: 'Send',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '0',\n        outputs: [\n          {\n            address: 'CP5Dpaa42RtJmMuKqCQsLwma5Yh3knuvKsYDFX85F41S',\n            amount: '300000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n        fee: {\n          fee: '5000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: 'GHtXQBsoZHVnNFa9YevAzFr17DJjgHXk3ycTKD5xD3Zi',\n        durableNonce: {\n          authWalletAddress: '12f6D3WubGVeQoH2m8kTvvcrasWdXWwtVzUCyRNDZxA2',\n          walletNonceAddress: '8Y7RM6JfcX4ASSNBkrkrmSbRu431YVi9Y3oLFnzC2dCh',\n        },\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain activate staking transaction', async function () {\n      const tx = await factory\n        .getStakingActivateBuilder()\n        .stakingAddress(stakeAccount.pub)\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .amount(amount)\n        .validator(validator.pub)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: txToBroadcastFormat,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'StakingActivate',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '10000',\n        outputs: [\n          {\n            address: '7dRuGFbU2y2kijP6o1LYNzVyz4yf13MooqoionCzv5Za',\n            amount: '10000',\n          },\n        ],\n        fee: {\n          fee: '10000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',\n        durableNonce: undefined,\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain deactivate staking transaction', async function () {\n      const tx = await factory\n        .getStakingDeactivateBuilder()\n        .stakingAddress(stakeAccount.pub)\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: txToBroadcastFormat,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'StakingDeactivate',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '0',\n        outputs: [],\n        fee: {\n          fee: '5000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',\n        durableNonce: undefined,\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain withdraw staking transaction', async function () {\n      const tx = await factory\n        .getStakingWithdrawBuilder()\n        .stakingAddress(stakeAccount.pub)\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .amount(amount)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: txToBroadcastFormat,\n        feeInfo: {\n          fee: '5000',\n        },\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'StakingWithdraw',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '10000',\n        outputs: [\n          {\n            address: '5hr5fisPi6DXNuuRpm5XUbzpiEnmdyxXuBDTwzwZj5Pe',\n            amount: '10000',\n          },\n        ],\n        fee: {\n          fee: '5000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',\n        durableNonce: undefined,\n        tokenEnablements: [],\n      });\n    });\n\n    it('should explain create ATA transaction', async function () {\n      const tokenName = 'tsol:usdc';\n      const rentExemptAmount = '3000000';\n      const tx = await factory\n        .getAtaInitializationBuilder()\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .mint(tokenName)\n        .rentExemptAmount(rentExemptAmount)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: txToBroadcastFormat,\n        feeInfo: {\n          fee: '5000',\n        },\n        tokenAccountRentExemptAmount: rentExemptAmount,\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'AssociatedTokenAccountInitialization',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '0',\n        outputs: [],\n        fee: {\n          fee: '3005000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',\n        durableNonce: undefined,\n        tokenEnablements: [\n          {\n            address: '141BFNem3pknc8CzPVLv1Ri3btgKdCsafYP5nXwmXfxU',\n            tokenAddress: 'F4uLeXJoFz3hw13MposuwaQbMcZbCjqvEGPPeRRB1Byf',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n      });\n    });\n\n    it('should explain create multi ATA transaction', async function () {\n      const recipients = [\n        {\n          ownerAddress: wallet.pub,\n          tokenName: 'tsol:usdc',\n        },\n        {\n          ownerAddress: durableNonce.walletNonceAddress,\n          tokenName: 'tsol:ray',\n        },\n      ];\n      const rentExemptAmount = '3000000';\n      const tx = await factory\n        .getAtaInitializationBuilder()\n        .sender(wallet.pub)\n        .nonce(blockHash)\n        .enableToken(recipients[0])\n        .enableToken(recipients[1])\n        .rentExemptAmount(rentExemptAmount)\n        .memo('test memo')\n        .fee({ amount: 5000 })\n        .build();\n      const txToBroadcastFormat = tx.toBroadcastFormat();\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: txToBroadcastFormat,\n        feeInfo: {\n          fee: '5000',\n        },\n        tokenAccountRentExemptAmount: rentExemptAmount,\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'AssociatedTokenAccountInitialization',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '0',\n        outputs: [],\n        fee: {\n          fee: '6005000',\n          feeRate: 5000,\n        },\n        memo: 'test memo',\n        blockhash: '5ne7phA48Jrvpn39AtupB8ZkCCAy8gLTfpGihZPuDqen',\n        durableNonce: undefined,\n        tokenEnablements: [\n          {\n            address: '141BFNem3pknc8CzPVLv1Ri3btgKdCsafYP5nXwmXfxU',\n            tokenAddress: 'F4uLeXJoFz3hw13MposuwaQbMcZbCjqvEGPPeRRB1Byf',\n            tokenName: 'tsol:usdc',\n          },\n          {\n            address: '9KaLinZFNW5chL4J8UoKnTECppWVMz3ewgx4FAkxUDcf',\n            tokenAddress: '9kLJoGbMgSteptkhKKuh7ken4JEvHrT83157ezEGrZ7R',\n            tokenName: 'tsol:ray',\n          },\n        ],\n      });\n    });\n\n    it('should explain an unsigned token transfer with ATA creation transaction', async function () {\n      const explainedTransaction = await basecoin.explainTransaction({\n        txBase64: testData.rawTransactions.tokenTransferWithAtaCreation.unsigned,\n        feeInfo: {\n          fee: '5000',\n        },\n        tokenAccountRentExemptAmount: '3000000',\n      });\n      explainedTransaction.should.deepEqual({\n        displayOrder: [\n          'id',\n          'type',\n          'blockhash',\n          'durableNonce',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'tokenEnablements',\n          'fee',\n          'memo',\n        ],\n        id: 'UNAVAILABLE',\n        type: 'Send',\n        changeOutputs: [],\n        changeAmount: '0',\n        outputAmount: '0',\n        outputs: [\n          {\n            address: '2eKjVtzV3oPTXFdtRSDj3Em9k1MV7k8WjKkBszQUwizS',\n            amount: '10000',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n        fee: { fee: '3005000', feeRate: 5000 },\n        memo: undefined,\n        blockhash: '27E3MXFvXMUNYeMJeX1pAbERGsJfUbkaZTfgMgpmNN5g',\n        durableNonce: undefined,\n        tokenEnablements: [\n          {\n            address: '2eKjVtzV3oPTXFdtRSDj3Em9k1MV7k8WjKkBszQUwizS',\n            tokenAddress: 'F4uLeXJoFz3hw13MposuwaQbMcZbCjqvEGPPeRRB1Byf',\n            tokenName: 'tsol:usdc',\n          },\n        ],\n      });\n    });\n  });\n\n  describe('Keypair:', () => {\n    it('should generate a keypair from random seed', function () {\n      should.throws(() => basecoin.generateKeyPair('placeholder' as any), 'generateKeyPair method not implemented');\n    });\n\n    it('should generate a keypair from a seed', function () {\n      should.throws(() => basecoin.generateKeyPair('placeholder' as any), 'generateKeyPair method not implemented');\n    });\n  });\n\n  describe('Sign transaction:', () => {\n    it('should sign transaction', async function () {\n      const signed: any = await basecoin.signTransaction({\n        txPrebuild: {\n          txBase64: resources.RAW_TX_UNSIGNED,\n          keys: [resources.accountWithSeed.publicKey.toString()],\n        },\n        prv: resources.accountWithSeed.privateKey.base58,\n      } as any);\n      signed.txHex.should.equal(resources.RAW_TX_SIGNED);\n    });\n\n    it('should handle txHex and txBase64 interchangeably', async function () {\n      const signed: any = await basecoin.signTransaction({\n        txPrebuild: {\n          txHex: resources.RAW_TX_UNSIGNED,\n          keys: [resources.accountWithSeed.publicKey.toString()],\n        },\n        prv: resources.accountWithSeed.privateKey.base58,\n      } as any);\n      signed.txHex.should.equal(resources.RAW_TX_SIGNED);\n    });\n\n    it('should throw invalid transaction when sign with public key', async function () {\n      await basecoin\n        .signTransaction({\n          txPrebuild: {\n            txBase64: resources.RAW_TX_UNSIGNED,\n            keys: [resources.accountWithSeed.publicKey.toString()],\n          },\n          prv: resources.accountWithSeed.publicKey,\n        } as any)\n        .should.be.rejectedWith('Invalid key');\n    });\n  });\n\n  describe('Sign message', () => {\n    it('should sign message', async function () {\n      const signed = await basecoin.signMessage(keypair, 'signed message');\n      signed\n        .toString('base64')\n        .should.equal('s+7d/8aW/twfM/0wLSKOGxd9+LhDIiz/g0FfJ39ylJhQIkjK0RYPm/Y+gdeJ5DIy6K6h6gCXXESDomlv12DBBQ==');\n    });\n    it('shouldnt sign message when message is undefined', async function () {\n      await basecoin\n        .signMessage(keypair, undefined as any)\n        .should.be.rejectedWith(\n          'The first argument must be of type string or an instance of Buffer, ArrayBuffer, or Array or an Array-like Object. Received undefined'\n        );\n    });\n  });\n\n  describe('Get Signing Payload', () => {\n    it('should return a valid signing payload', async function () {\n      const factory = getBuilderFactory(basecoin.getChain());\n      const rebuiltSignablePayload = (await factory.from(resources.TRANSFER_UNSIGNED_TX_WITH_MEMO).build())\n        .signablePayload;\n      const signingPayload = await basecoin.getSignablePayload(resources.TRANSFER_UNSIGNED_TX_WITH_MEMO);\n      signingPayload.should.be.deepEqual(rebuiltSignablePayload);\n    });\n\n    it('should build CloseAssociatedTokenAccount txn builder from raw txn', async function () {\n      const factory = getBuilderFactory(basecoin.getChain());\n      const txnBuilder = factory.from(resources.TRANSFER_UNSIGNED_TX_CLOSE_ATA);\n      assert.ok(txnBuilder);\n    });\n  });\n\n  describe('Presign transaction', () => {\n    const txRequestId = 'txRequestId';\n    let sandbox: sinon.SinonSandbox;\n\n    beforeEach(() => {\n      sandbox = sinon.createSandbox();\n    });\n\n    afterEach(() => {\n      sandbox.verifyAndRestore();\n    });\n\n    it('should rebuild tx request for hot wallets', async () => {\n      const rebuiltTx: TxRequest = {\n        txRequestId,\n        unsignedTxs: [\n          {\n            serializedTxHex: 'deadbeef',\n            signableHex: 'serializedTxHex',\n            derivationPath: 'm/0',\n          },\n        ],\n        transactions: [],\n        date: new Date().toISOString(),\n        intent: {\n          intentType: 'payment',\n        },\n        latest: true,\n        state: 'pendingUserSignature',\n        walletType: 'hot',\n        walletId: 'walletId',\n        policiesChecked: true,\n        version: 1,\n        userId: 'userId',\n      };\n\n      const stubTssUtils = sandbox.createStubInstance(TssUtils);\n      stubTssUtils.deleteSignatureShares.resolves([]);\n      stubTssUtils.getTxRequest.resolves(rebuiltTx);\n\n      const hotWallet = {\n        type: 'hot',\n      };\n      const presignedTransaction: any = await basecoin.presignTransaction({\n        walletData: hotWallet,\n        tssUtils: stubTssUtils,\n        txPrebuild: {\n          txRequestId,\n        },\n      } as any);\n\n      presignedTransaction.walletData.should.deepEqual(hotWallet);\n      presignedTransaction.txPrebuild.should.deepEqual(rebuiltTx);\n      presignedTransaction.txHex.should.equal(rebuiltTx.unsignedTxs[0].serializedTxHex);\n    });\n\n    it('should do nothing for non-hot wallets', async () => {\n      const coldWallet = {\n        type: 'cold',\n      };\n\n      const presignedTransaction = await basecoin.presignTransaction({\n        walletData: coldWallet,\n      } as any);\n      presignedTransaction.should.deepEqual({\n        walletData: coldWallet,\n      });\n    });\n\n    it('should error if txRequestId is missing', async () => {\n      const hotWallet = {\n        type: 'hot',\n      };\n      await basecoin\n        .presignTransaction({\n          walletData: hotWallet,\n          txPrebuild: {},\n        } as any)\n        .should.rejectedWith('Missing txRequestId');\n    });\n  });\n\n  describe('API Key parameter:', () => {\n    // Test the getPublicNodeUrl method directly\n    it('should append apiKey to node URL when provided', function () {\n      // Access the protected method using type casting\n      const url = (basecoin as any).getPublicNodeUrl('test-api-key-123');\n      url.should.equal(`${Environments.test.solAlchemyNodeUrl}/test-api-key-123`);\n    });\n\n    it('should use regular node URL when apiKey is not provided', function () {\n      // Access the protected method using type casting\n      const url = (basecoin as any).getPublicNodeUrl();\n      url.should.equal(Environments.test.solNodeUrl);\n    });\n  });\n\n  describe('Recover Transactions:', () => {\n    const sandBox = sinon.createSandbox();\n    const coin = coins.get('tsol');\n    const usdtMintAddress = '9cgpBeNZ2HnLda7NWaaU1i3NyTstk2c4zCMUcoAGsi9C';\n    const t22mintAddress = '5NR1bQwLWqjbkhbQ1hx72HKJybbuvwkDnUZNoAZ2VhW6';\n    let callBack: sinon.SinonStub;\n\n    beforeEach(() => {\n      callBack = sandBox.stub(Sol.prototype, 'getDataFromNode' as keyof Sol);\n\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getLatestBlockhash',\n            params: [\n              {\n                commitment: 'finalized',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getBlockhashResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getFeeForMessage',\n            params: [\n              sinon.match.string,\n              {\n                commitment: 'finalized',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getFeesForMessageResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getMinimumBalanceForRentExemption',\n            params: [165],\n          },\n        })\n        .resolves(testData.SolResponses.getMinimumBalanceForRentExemptionResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.accountInfo.bs58EncodedPublicKey],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.accountInfo.bs58EncodedPublicKeyNoFunds],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseNoFunds);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.accountInfo.bs58EncodedPublicKeyM1Derivation],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseM1Derivation);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.accountInfo.bs58EncodedPublicKeyM2Derivation],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.accountInfo.bs58EncodedPublicKeyWithManyTokens],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.closeATAkeys.closeAtaAddress],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.closeATAkeys.bs58EncodedPublicKey],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseM2Derivation);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getAccountInfo',\n            params: [\n              testData.closeATAkeys.closeAtaAddress,\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenInfoResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getAccountInfo',\n            params: [\n              testData.keys.durableNoncePubKey,\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountInfoResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.keys.destinationPubKey,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.accountInfo.bs58EncodedPublicKey,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.keys.destinationPubKey2,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.keys.destinationPubKey2,\n              {\n                programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerForSol2022Response2);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress0,\n              {\n                programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerForSol2022Response);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress0,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponse2);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress4,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponse3);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.wrwUser.walletAddress0],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.keys.destinationPubKey,\n              {\n                programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'sendTransaction',\n            params: sinon.match.array,\n          },\n        })\n        .onCall(0)\n        .resolves(testData.SolResponses.broadcastTransactionResponse)\n        .onCall(1)\n        .resolves(testData.SolResponses.broadcastTransactionResponse1);\n    });\n\n    afterEach(() => {\n      sandBox.restore();\n    });\n\n    it('should take OVC output and generate a signed sweep transaction', async function () {\n      const params = testData.ovcResponse;\n      const recoveryTxn = await basecoin.createBroadcastableSweepTransaction(params);\n      recoveryTxn.transactions[0].serializedTx.should.equal(\n        'AvR+L909kzRq6NuaUe9F6Jt97MOiFs7jpW8MuOrwz4EbKF40d31dci/bgLTq4gpk/Hh3s5cA8FtbLkDQr15PqAE7yd8LOXvsLtO2REqMM/OCZ8wItfsqfTfia2xIfibRW3wHgw63jiaojbXeSqaYajJ/Ca7YwBUz5blydI3fYLgPAgECBsLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHxPX1mHv+JqpmAT79ltNjYPK0M2yR+ZMln7VgUTBWFNQvLqE/j/nXlY2/JpxuNr/fXLXEPeS04dPvt9qz1dAoYEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAADpiH20cxLj7KnOaoI5ANNoPxYjs472FdjDeMPft3kXdAgQDAgUBBAQAAAAEAgADDAIAAADwopo7AAAAAA=='\n      );\n      (recoveryTxn.transactions[0].scanIndex ?? 0).should.equal(0);\n      (recoveryTxn.lastScanIndex ?? 0).should.equal(0);\n    });\n\n    it('should take sol 2022 token OVC output and generate a signed sweep transaction', async function () {\n      const params = testData.ovcResponse;\n      const recoveryTxn = await basecoin.createBroadcastableSweepTransaction(params);\n      recoveryTxn.transactions[0].serializedTx.should.equal(\n        'AvR+L909kzRq6NuaUe9F6Jt97MOiFs7jpW8MuOrwz4EbKF40d31dci/bgLTq4gpk/Hh3s5cA8FtbLkDQr15PqAE7yd8LOXvsLtO2REqMM/OCZ8wItfsqfTfia2xIfibRW3wHgw63jiaojbXeSqaYajJ/Ca7YwBUz5blydI3fYLgPAgECBsLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHxPX1mHv+JqpmAT79ltNjYPK0M2yR+ZMln7VgUTBWFNQvLqE/j/nXlY2/JpxuNr/fXLXEPeS04dPvt9qz1dAoYEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAADpiH20cxLj7KnOaoI5ANNoPxYjs472FdjDeMPft3kXdAgQDAgUBBAQAAAAEAgADDAIAAADwopo7AAAAAA=='\n      );\n      (recoveryTxn.transactions[0].scanIndex ?? 0).should.equal(0);\n      (recoveryTxn.lastScanIndex ?? 0).should.equal(0);\n    });\n\n    it('should take consolidation OVC output and generate multiple signed sweep transactions', async function () {\n      const params = testData.ovcResponse2;\n      const recoveryTxn = await basecoin.createBroadcastableSweepTransaction(params);\n      recoveryTxn.transactions[0].serializedTx.should.equal(\n        'AtQPLzOmLuKwHY6N5XoJIZK/T7W10uYWm/MRte3GFUdl+w3gHLjSa9H66WSfFNubQxIPckxJDyltkP7ksLDf9QgBNJM2UWbBUH5wT0JJHILlhCs33HX8DeE/8Tdsw6tGfZoMhCnSKv6TPWtBxy7Sb6sW8ksCUPnAWuHGGKmgjEMBAgECBmLrqxJrY2kbN/tcrQw3P8P15OljFGabFJAKBrUO1grNBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHxPX1mHv+JqpmAT79ltNjYPK0M2yR+ZMln7VgUTBWFNQsLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAAIZQniiS73D6mwfpnfhVMC4lyYJtRSrmoZpF7yIlUdIDAgQDAgUBBAQAAAAEAgADDAIAAADwPc0dAAAAAA=='\n      );\n      (recoveryTxn.transactions[0].scanIndex ?? 0).should.equal(1);\n      recoveryTxn.transactions[1].serializedTx.should.equal(\n        'AuLhOA5zmOBZR85lo+nKdTopVwJAMrMp6NW+8UnGNsSBSpBkqfWZQqSg9s+7aTlXezm5vxol+Pl6t7PpVNTOHwLcp9xJp3TFHdivEbhwJKldR4Ny+pasoFx+Bgk8q6g1iNiq7XSi1Ov3bs7euMkTj7nDRFqP8lv7xLTcvrBm9OQJAgECBp14ImBCdmVROlw0UveYS1MvG/ljCRI3MJTFmsxuXEoWBRo0vIrNQ4djl2+Wh2EVBQ9zgoVTVm0RHXrIv/6/WHw0hyxvpVwtIx9/zeX2O16eTrY+aKIh1mdKg4MMg0eyxMLVtfT7mpvNii8wPk0G942N7TAHE/RW2iq/8LPqAYWqAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGp9UXGSxWjuCKhF9z0peIzwNcMUWyGrNE2AYuqUAAAC7ws1XFslinwgtpISUViVWIVTHyD2Q0qj24YjKmrAmXAgQDAgUBBAQAAAAEAgADDAIAAADwPc0dAAAAAA=='\n      );\n      (recoveryTxn.transactions[1].scanIndex ?? 0).should.equal(2);\n      (recoveryTxn.lastScanIndex ?? 0).should.equal(20);\n    });\n\n    it('should recover a txn for non-bitgo recoveries (latest blockhash)', async function () {\n      // Latest Blockhash Recovery (BitGo-less)\n      const latestBlockHashTxn = await basecoin.recover({\n        userKey: testData.keys.userKey,\n        backupKey: testData.keys.backupKey,\n        bitgoKey: testData.keys.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey,\n        walletPassphrase: testData.keys.walletPassword,\n      });\n      latestBlockHashTxn.should.not.be.empty();\n      latestBlockHashTxn.should.hasOwnProperty('serializedTx');\n      latestBlockHashTxn.should.hasOwnProperty('scanIndex');\n      should.equal((latestBlockHashTxn as MPCTx).scanIndex, 0);\n\n      const latestBlockhashTxnDeserialize = new Transaction(coin);\n      latestBlockhashTxnDeserialize.fromRawTransaction((latestBlockHashTxn as MPCTx).serializedTx);\n      const latestBlockhashTxnJson = latestBlockhashTxnDeserialize.toJson();\n\n      should.equal(latestBlockhashTxnJson.nonce, testData.SolInputData.blockhash);\n      should.equal(latestBlockhashTxnJson.feePayer, testData.accountInfo.bs58EncodedPublicKey);\n      should.equal(latestBlockhashTxnJson.numSignatures, testData.SolInputData.latestBlockhashSignatures);\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 3);\n    });\n\n    it('should recover a txn for non-bitgo recoveries (durable nonce)', async function () {\n      // Durable Nonce Recovery (BitGo-less)\n      const durableNonceTxn = await basecoin.recover({\n        userKey: testData.keys.userKey,\n        backupKey: testData.keys.backupKey,\n        bitgoKey: testData.keys.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey,\n        walletPassphrase: testData.keys.walletPassword,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n      });\n\n      durableNonceTxn.should.not.be.empty();\n      durableNonceTxn.should.hasOwnProperty('serializedTx');\n      durableNonceTxn.should.hasOwnProperty('scanIndex');\n      should.equal((durableNonceTxn as MPCTx).scanIndex, 0);\n\n      const durableNonceTxnDeserialize = new Transaction(coin);\n      durableNonceTxnDeserialize.fromRawTransaction((durableNonceTxn as MPCTx).serializedTx);\n      const durableNonceTxnJson = durableNonceTxnDeserialize.toJson();\n\n      should.equal(durableNonceTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      should.equal(durableNonceTxnJson.feePayer, testData.accountInfo.bs58EncodedPublicKey);\n      should.equal(durableNonceTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 4);\n    });\n\n    it('should recover a txn for unsigned sweep recoveries', async function () {\n      // Unsigned Sweep Recovery\n      const unsignedSweepTxn = (await basecoin.recover({\n        bitgoKey: testData.keys.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n      })) as MPCSweepTxs;\n\n      unsignedSweepTxn.should.not.be.empty();\n      unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.should.hasOwnProperty('serializedTx');\n      unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.should.hasOwnProperty('scanIndex');\n      should.equal(unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.scanIndex, 0);\n\n      const unsignedSweepTxnDeserialize = new Transaction(coin);\n      unsignedSweepTxnDeserialize.fromRawTransaction(\n        unsignedSweepTxn.txRequests[0].transactions[0].unsignedTx.serializedTx\n      );\n      const unsignedSweepTxnJson = unsignedSweepTxnDeserialize.toJson();\n\n      should.equal(unsignedSweepTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      should.equal(unsignedSweepTxnJson.feePayer, testData.accountInfo.bs58EncodedPublicKey);\n      should.equal(unsignedSweepTxnJson.numSignatures, testData.SolInputData.unsignedSweepSignatures);\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 4);\n    });\n\n    it('should handle error in recover function if a required field is missing/incorrect', async function () {\n      // missing userkey\n      await basecoin\n        .recover({\n          backupKey: testData.keys.backupKey,\n          bitgoKey: testData.keys.bitgoKey,\n          recoveryDestination: testData.keys.destinationPubKey,\n          walletPassphrase: testData.keys.walletPassword,\n        })\n        .should.rejectedWith('missing userKey');\n\n      // missing backupkey\n      await basecoin\n        .recover({\n          userKey: testData.keys.userKey,\n          bitgoKey: testData.keys.bitgoKey,\n          recoveryDestination: testData.keys.destinationPubKey,\n          walletPassphrase: testData.keys.walletPassword,\n        })\n        .should.rejectedWith('missing backupKey');\n\n      // missing wallet passphrase\n      await basecoin\n        .recover({\n          userKey: testData.keys.userKey,\n          backupKey: testData.keys.backupKey,\n          bitgoKey: testData.keys.bitgoKey,\n          recoveryDestination: testData.keys.destinationPubKey,\n        })\n        .should.rejectedWith('missing wallet passphrase');\n\n      // incorrect wallet passphrase, user key, backup key combination\n      await basecoin\n        .recover({\n          userKey: testData.keys.userKey,\n          backupKey: testData.keys.backupKey,\n          bitgoKey: testData.keys.bitgoKey,\n          recoveryDestination: testData.keys.destinationPubKey,\n          walletPassphrase: testData.keys.walletPassword + 'incorrect',\n        })\n        .should.rejectedWith(\"Error decrypting user keychain: password error - ccm: tag doesn't match\");\n\n      // no wallet with sufficient funds\n      await basecoin\n        .recover({\n          userKey: testData.keys.userKey,\n          backupKey: testData.keys.backupKey,\n          bitgoKey: testData.keys.bitgoKeyNoFunds,\n          recoveryDestination: testData.keys.destinationPubKey,\n          walletPassphrase: testData.keys.walletPassword,\n        })\n        .should.rejectedWith('Did not find address with funds to recover');\n    });\n\n    it('should recover sol tokens to recovery destination with no existing token accounts', async function () {\n      const tokenTxn = await basecoin.recover({\n        userKey: testData.wrwUser.userKey,\n        backupKey: testData.wrwUser.backupKey,\n        bitgoKey: testData.wrwUser.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey,\n        tokenContractAddress: usdtMintAddress,\n        walletPassphrase: testData.wrwUser.walletPassphrase,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n      });\n\n      tokenTxn.should.not.be.empty();\n      tokenTxn.should.hasOwnProperty('serializedTx');\n      tokenTxn.should.hasOwnProperty('scanIndex');\n      should.equal((tokenTxn as MPCTx).scanIndex, 0);\n\n      const tokenTxnDeserialize = new Transaction(coin);\n      tokenTxnDeserialize.fromRawTransaction((tokenTxn as MPCTx).serializedTx);\n      const tokenTxnJson = tokenTxnDeserialize.toJson();\n\n      should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);\n      should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);\n\n      const instructionsData = tokenTxnJson.instructionsData as InstructionParams[];\n      should.equal(instructionsData.length, 3);\n      should.equal(instructionsData[0].type, 'NonceAdvance');\n\n      const destinationUSDTTokenAccount = await getAssociatedTokenAccountAddress(\n        usdtMintAddress,\n        testData.keys.destinationPubKey\n      );\n      should.equal(instructionsData[1].type, 'CreateAssociatedTokenAccount');\n      should.equal((instructionsData[1] as AtaInit).params.mintAddress, usdtMintAddress);\n      should.equal((instructionsData[1] as AtaInit).params.ataAddress, destinationUSDTTokenAccount);\n      should.equal((instructionsData[1] as AtaInit).params.ownerAddress, testData.keys.destinationPubKey);\n      should.equal((instructionsData[1] as AtaInit).params.tokenName, 'tsol:usdt');\n      should.equal((instructionsData[1] as AtaInit).params.payerAddress, testData.wrwUser.walletAddress0);\n\n      const sourceUSDTTokenAccount = await getAssociatedTokenAccountAddress(\n        usdtMintAddress,\n        testData.wrwUser.walletAddress0\n      );\n      should.equal(instructionsData[2].type, 'TokenTransfer');\n      should.equal((instructionsData[2] as TokenTransfer).params.fromAddress, testData.wrwUser.walletAddress0);\n      should.equal((instructionsData[2] as TokenTransfer).params.toAddress, destinationUSDTTokenAccount);\n      should.equal((instructionsData[2] as TokenTransfer).params.amount, '2000000000');\n      should.equal((instructionsData[2] as TokenTransfer).params.tokenName, 'tsol:usdt');\n      should.equal((instructionsData[2] as TokenTransfer).params.sourceAddress, sourceUSDTTokenAccount);\n\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 7);\n    });\n\n    it('should recover sol 2022 tokens to recovery destination with no existing token accounts', async function () {\n      const tokenTxn = await basecoin.recover({\n        userKey: testData.wrwUser.userKey,\n        backupKey: testData.wrwUser.backupKey,\n        bitgoKey: testData.wrwUser.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey,\n        tokenContractAddress: t22mintAddress,\n        walletPassphrase: testData.wrwUser.walletPassphrase,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n        programId: 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb',\n      });\n\n      tokenTxn.should.not.be.empty();\n      tokenTxn.should.hasOwnProperty('serializedTx');\n      tokenTxn.should.hasOwnProperty('scanIndex');\n      should.equal((tokenTxn as MPCTx).scanIndex, 0);\n\n      const tokenTxnDeserialize = new Transaction(coin);\n      tokenTxnDeserialize.fromRawTransaction((tokenTxn as MPCTx).serializedTx);\n      const tokenTxnJson = tokenTxnDeserialize.toJson();\n\n      should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);\n      should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);\n\n      const instructionsData = tokenTxnJson.instructionsData as InstructionParams[];\n      should.equal(instructionsData.length, 3);\n      should.equal(instructionsData[0].type, 'NonceAdvance');\n\n      const destinationTokenAccount = await getAssociatedTokenAccountAddress(\n        t22mintAddress,\n        testData.keys.destinationPubKey\n      );\n      should.equal(instructionsData[1].type, 'CreateAssociatedTokenAccount');\n      should.equal((instructionsData[1] as AtaInit).params.mintAddress, t22mintAddress);\n      should.equal((instructionsData[1] as AtaInit).params.ataAddress, destinationTokenAccount);\n      should.equal((instructionsData[1] as AtaInit).params.ownerAddress, testData.keys.destinationPubKey);\n      should.equal((instructionsData[1] as AtaInit).params.tokenName, 'tsol:t22mint');\n      should.equal((instructionsData[1] as AtaInit).params.payerAddress, testData.wrwUser.walletAddress0);\n\n      const sourceTokenAccount = await getAssociatedTokenAccountAddress(\n        t22mintAddress,\n        testData.wrwUser.walletAddress0\n      );\n      should.equal(instructionsData[2].type, 'TokenTransfer');\n      should.equal((instructionsData[2] as TokenTransfer).params.fromAddress, testData.wrwUser.walletAddress0);\n      should.equal((instructionsData[2] as TokenTransfer).params.toAddress, destinationTokenAccount);\n      should.equal((instructionsData[2] as TokenTransfer).params.amount, '2000000000');\n      should.equal((instructionsData[2] as TokenTransfer).params.tokenName, 'tsol:t22mint');\n      should.equal((instructionsData[2] as TokenTransfer).params.sourceAddress, sourceTokenAccount);\n      should.equal(\n        (instructionsData[2] as TokenTransfer).params.programId,\n        'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb'\n      );\n\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 7);\n    });\n\n    it('should recover sol tokens to recovery destination with existing token accounts', async function () {\n      const tokenTxn = await basecoin.recover({\n        userKey: testData.wrwUser.userKey,\n        backupKey: testData.wrwUser.backupKey,\n        bitgoKey: testData.wrwUser.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey2,\n        tokenContractAddress: usdtMintAddress,\n        walletPassphrase: testData.wrwUser.walletPassphrase,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n      });\n\n      tokenTxn.should.not.be.empty();\n      tokenTxn.should.hasOwnProperty('serializedTx');\n      tokenTxn.should.hasOwnProperty('scanIndex');\n      should.equal((tokenTxn as MPCTx).scanIndex, 0);\n\n      const tokenTxnDeserialize = new Transaction(coin);\n      tokenTxnDeserialize.fromRawTransaction((tokenTxn as MPCTx).serializedTx);\n      const tokenTxnJson = tokenTxnDeserialize.toJson();\n\n      should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);\n      should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);\n\n      const instructionsData = tokenTxnJson.instructionsData as TokenTransfer[];\n      should.equal(instructionsData.length, 2);\n      should.equal(instructionsData[0].type, 'NonceAdvance');\n\n      const sourceUSDTTokenAccount = await getAssociatedTokenAccountAddress(\n        usdtMintAddress,\n        testData.wrwUser.walletAddress0\n      );\n      const destinationUSDTTokenAccount = await getAssociatedTokenAccountAddress(\n        usdtMintAddress,\n        testData.keys.destinationPubKey2\n      );\n      should.equal(instructionsData[1].type, 'TokenTransfer');\n      should.equal(instructionsData[1].params.fromAddress, testData.wrwUser.walletAddress0);\n      should.equal(instructionsData[1].params.toAddress, destinationUSDTTokenAccount);\n      should.equal(instructionsData[1].params.amount, '2000000000');\n      should.equal(instructionsData[1].params.tokenName, 'tsol:usdt');\n      should.equal(instructionsData[1].params.sourceAddress, sourceUSDTTokenAccount);\n\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 7);\n    });\n\n    it('should recover sol 2022 tokens to recovery destination with existing token accounts', async function () {\n      const tokenTxn = await basecoin.recover({\n        userKey: testData.wrwUser.userKey,\n        backupKey: testData.wrwUser.backupKey,\n        bitgoKey: testData.wrwUser.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey2,\n        tokenContractAddress: t22mintAddress,\n        walletPassphrase: testData.wrwUser.walletPassphrase,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n        programId: TOKEN_2022_PROGRAM_ID.toString(),\n      });\n\n      tokenTxn.should.not.be.empty();\n      tokenTxn.should.hasOwnProperty('serializedTx');\n      tokenTxn.should.hasOwnProperty('scanIndex');\n      should.equal((tokenTxn as MPCTx).scanIndex, 0);\n\n      const tokenTxnDeserialize = new Transaction(coin);\n      tokenTxnDeserialize.fromRawTransaction((tokenTxn as MPCTx).serializedTx);\n      const tokenTxnJson = tokenTxnDeserialize.toJson();\n      console.log(tokenTxnJson);\n      should.equal(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      should.equal(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);\n      should.equal(tokenTxnJson.numSignatures, testData.SolInputData.durableNonceSignatures);\n\n      const instructionsData = tokenTxnJson.instructionsData as TokenTransfer[];\n      should.equal(instructionsData.length, 2);\n      should.equal(instructionsData[0].type, 'NonceAdvance');\n\n      const source2022TokenAccount = await getAssociatedTokenAccountAddress(\n        t22mintAddress,\n        testData.wrwUser.walletAddress0,\n        false,\n        TOKEN_2022_PROGRAM_ID.toString()\n      );\n      const destination2022TokenAccount = await getAssociatedTokenAccountAddress(\n        t22mintAddress,\n        testData.keys.destinationPubKey2,\n        false,\n        TOKEN_2022_PROGRAM_ID.toString()\n      );\n      should.equal(instructionsData[1].type, 'TokenTransfer');\n      should.equal(instructionsData[1].params.fromAddress, testData.wrwUser.walletAddress0);\n      should.equal(instructionsData[1].params.toAddress, destination2022TokenAccount);\n      should.equal(instructionsData[1].params.amount, '2000000000');\n      should.equal(instructionsData[1].params.tokenName, 'tsol:t22mint');\n      should.equal(instructionsData[1].params.sourceAddress, source2022TokenAccount);\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 7);\n    });\n\n    it('should recover sol tokens to recovery destination with existing token accounts for unsigned sweep recoveries', async function () {\n      const feeResponse = testData.SolResponses.getFeesForMessageResponse;\n      feeResponse.body.result.value = 10000;\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getFeeForMessage',\n            params: [\n              sinon.match.string,\n              {\n                commitment: 'finalized',\n              },\n            ],\n          },\n        })\n        .resolves(feeResponse);\n\n      const tokenTxn = (await basecoin.recover({\n        bitgoKey: testData.wrwUser.bitgoKey,\n        recoveryDestination: testData.keys.destinationPubKey2,\n        durableNonce: {\n          publicKey: testData.keys.durableNoncePubKey,\n          secretKey: testData.keys.durableNoncePrivKey,\n        },\n        tokenContractAddress: testData.tokenAddress.TestUSDC,\n      })) as MPCSweepTxs;\n\n      // 2 signatures and no rent exemption fee since the destination already has token accounts\n      const expectedFee = 5000 + 5000;\n\n      const { serializedTx, scanIndex, feeInfo, parsedTx } = tokenTxn.txRequests[0].transactions[0].unsignedTx;\n      assert.ok(serializedTx);\n      assert.strictEqual(scanIndex, 0);\n      assert.ok(feeInfo);\n      assert.strictEqual(feeInfo.feeString, expectedFee.toString());\n      assert.strictEqual(feeInfo.fee, expectedFee);\n      assert.ok(parsedTx);\n      assert.ok(parsedTx.inputs instanceof Array && parsedTx.inputs.length === 1);\n      assert.ok(parsedTx.outputs instanceof Array && parsedTx.outputs.length === 1);\n\n      const tokenTxnDeserialize = new Transaction(coin);\n      tokenTxnDeserialize.fromRawTransaction(tokenTxn.txRequests[0].transactions[0].unsignedTx.serializedTx);\n      const tokenTxnJson = tokenTxnDeserialize.toJson();\n\n      assert.strictEqual(tokenTxnJson.nonce, testData.SolInputData.durableNonceBlockhash);\n      assert.strictEqual(tokenTxnJson.feePayer, testData.wrwUser.walletAddress0);\n      assert.strictEqual(tokenTxnJson.numSignatures, testData.SolInputData.unsignedSweepSignatures);\n      const solCoin = basecoin as any;\n      sandBox.assert.callCount(solCoin.getDataFromNode, 7);\n    });\n\n    it('should recover sol funds from ATA address for non-bitgo recoveries', async function () {\n      // close ATA address instruction type txn\n      const closeATATxns = await basecoin.recoverCloseATA({\n        userKey: testData.closeATAkeys.userKey,\n        backupKey: testData.closeATAkeys.backupKey,\n        bitgoKey: testData.closeATAkeys.bitgoKey,\n        recoveryDestination: testData.closeATAkeys.destinationPubKey,\n        walletPassphrase: testData.closeATAkeys.walletPassword,\n        closeAtaAddress: testData.closeATAkeys.closeAtaAddress,\n        recoveryDestinationAtaAddress: testData.closeATAkeys.recoveryDestinationAtaAddress,\n      });\n      closeATATxns.should.not.be.empty();\n      should.equal(\n        closeATATxns[0].txId,\n        '2id3YC2jK9G5Wo2phDx4gJVAew8DcY5NAojnVuao8rkxwPYPe8cSwE5GzhEgJA2y8fVjDEo6iR6ykBvDxrTQrtpb'\n      );\n      should.equal(\n        closeATATxns[1].txId,\n        '5oUBgXX4enGmFEspG64goy3PRysjfrekZGg3rZNkBHUCQFd482vrVWbfDcRYMBEJt65JXymfEPm8M6d89X4xV79n'\n      );\n    });\n  });\n\n  describe('Build Consolidation Recoveries:', () => {\n    const sandBox = sinon.createSandbox();\n    const coin = coins.get('tsol');\n    const usdtMintAddress = '9cgpBeNZ2HnLda7NWaaU1i3NyTstk2c4zCMUcoAGsi9C';\n    const durableNonces = {\n      publicKeys: [\n        testData.keys.durableNoncePubKey,\n        testData.keys.durableNoncePubKey2,\n        testData.keys.durableNoncePubKey3,\n      ],\n      secretKey: testData.keys.durableNoncePrivKey,\n    };\n\n    beforeEach(() => {\n      const callBack = sandBox.stub(Sol.prototype, 'getDataFromNode' as keyof Sol);\n\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getLatestBlockhash',\n            params: [\n              {\n                commitment: 'finalized',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getBlockhashResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getFeeForMessage',\n            params: [\n              sinon.match.string,\n              {\n                commitment: 'finalized',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getFeesForMessageResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.wrwUser.walletAddress1],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponseNoFunds);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.wrwUser.walletAddress2],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.wrwUser.walletAddress3],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getBalance',\n            params: [testData.wrwUser.walletAddress5],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountBalanceResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getMinimumBalanceForRentExemption',\n            params: [165],\n          },\n        })\n        .resolves(testData.SolResponses.getMinimumBalanceForRentExemptionResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getAccountInfo',\n            params: [\n              testData.keys.durableNoncePubKey,\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountInfoResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getAccountInfo',\n            params: [\n              testData.keys.durableNoncePubKey2,\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getAccountInfoResponse2);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress1,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress2,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress3,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponseNoAccounts);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress5,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponse);\n      callBack\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'getTokenAccountsByOwner',\n            params: [\n              testData.wrwUser.walletAddress0,\n              {\n                programId: 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA',\n              },\n              {\n                encoding: 'jsonParsed',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.getTokenAccountsByOwnerResponse);\n    });\n\n    afterEach(() => {\n      sandBox.restore();\n    });\n\n    it('should build signed consolidation recoveries', async function () {\n      const res = (await basecoin.recoverConsolidations({\n        userKey: testData.wrwUser.userKey,\n        backupKey: testData.wrwUser.backupKey,\n        bitgoKey: testData.wrwUser.bitgoKey,\n        walletPassphrase: testData.wrwUser.walletPassphrase,\n        startingScanIndex: 1,\n        endingScanIndex: 4,\n        durableNonces: durableNonces,\n      })) as MPCTxs;\n      res.should.not.be.empty();\n      res.transactions.length.should.equal(2);\n      (res.lastScanIndex ?? 0).should.equal(3);\n\n      const txn1 = res.transactions[0];\n      const latestBlockhashTxnDeserialize1 = new Transaction(coin);\n      latestBlockhashTxnDeserialize1.fromRawTransaction((txn1 as MPCTx).serializedTx);\n      const latestBlockhashTxnJson1 = latestBlockhashTxnDeserialize1.toJson();\n\n      const nonce1 = testData.SolResponses.getAccountInfoResponse.body.result.value.data.parsed.info.blockhash;\n      should.equal(latestBlockhashTxnJson1.nonce, nonce1);\n      should.equal(latestBlockhashTxnJson1.feePayer, testData.wrwUser.walletAddress2);\n      should.equal(latestBlockhashTxnJson1.numSignatures, testData.SolInputData.durableNonceSignatures);\n\n      const txn2 = res.transactions[1];\n      const latestBlockhashTxnDeserialize2 = new Transaction(coin);\n      latestBlockhashTxnDeserialize2.fromRawTransaction((txn2 as MPCTx).serializedTx);\n      const latestBlockhashTxnJson2 = latestBlockhashTxnDeserialize2.toJson();\n\n      const nonce2 = testData.SolResponses.getAccountInfoResponse2.body.result.value.data.parsed.info.blockhash;\n      should.equal(latestBlockhashTxnJson2.nonce, nonce2);\n      should.equal(latestBlockhashTxnJson2.feePayer, testData.wrwUser.walletAddress3);\n      should.equal(latestBlockhashTxnJson2.numSignatures, testData.SolInputData.durableNonceSignatures);\n    });\n\n    it('should build unsigned consolidation recoveries', async function () {\n      const res = (await basecoin.recoverConsolidations({\n        bitgoKey: testData.wrwUser.bitgoKey,\n        startingScanIndex: 1,\n        endingScanIndex: 4,\n        durableNonces: durableNonces,\n      })) as MPCSweepTxs;\n      res.should.not.be.empty();\n      res.txRequests.length.should.equal(2);\n\n      const txn1 = res.txRequests[0].transactions[0].unsignedTx;\n      txn1.should.hasOwnProperty('serializedTx');\n      txn1.should.hasOwnProperty('signableHex');\n      txn1.should.hasOwnProperty('scanIndex');\n      (txn1.scanIndex ?? 0).should.equal(2);\n      txn1.should.hasOwnProperty('coin');\n      txn1.coin?.should.equal('tsol');\n      txn1.should.hasOwnProperty('derivationPath');\n      txn1.derivationPath?.should.equal('m/2');\n\n      txn1.should.hasOwnProperty('coinSpecific');\n      const coinSpecific1 = txn1.coinSpecific;\n      coinSpecific1?.should.hasOwnProperty('commonKeychain');\n\n      const latestBlockhashTxnDeserialize1 = new Transaction(coin);\n      latestBlockhashTxnDeserialize1.fromRawTransaction((txn1 as MPCTx).serializedTx);\n      const latestBlockhashTxnJson1 = latestBlockhashTxnDeserialize1.toJson();\n\n      const nonce1 = testData.SolResponses.getAccountInfoResponse.body.result.value.data.parsed.info.blockhash;\n      should.equal(latestBlockhashTxnJson1.nonce, nonce1);\n      should.equal(latestBlockhashTxnJson1.feePayer, testData.wrwUser.walletAddress2);\n      should.equal(latestBlockhashTxnJson1.numSignatures, testData.SolInputData.unsignedSweepSignatures);\n\n      const txn2 = res.txRequests[1].transactions[0].unsignedTx;\n      txn2.should.hasOwnProperty('serializedTx');\n      txn2.should.hasOwnProperty('signableHex');\n      txn2.should.hasOwnProperty('scanIndex');\n      (txn2.scanIndex ?? 0).should.equal(3);\n      txn2.should.hasOwnProperty('coin');\n      txn2.coin?.should.equal('tsol');\n      txn2.should.hasOwnProperty('derivationPath');\n      txn2.derivationPath?.should.equal('m/3');\n\n      txn2.should.hasOwnProperty('coinSpecific');\n      const coinSpecific2 = txn2.coinSpecific;\n      coinSpecific2?.should.hasOwnProperty('commonKeychain');\n      coinSpecific2?.should.hasOwnProperty('lastScanIndex');\n      coinSpecific2?.lastScanIndex?.should.equal(3);\n\n      const latestBlockhashTxnDeserialize2 = new Transaction(coin);\n      latestBlockhashTxnDeserialize2.fromRawTransaction((txn2 as MPCTx).serializedTx);\n      const latestBlockhashTxnJson2 = latestBlockhashTxnDeserialize2.toJson();\n\n      const nonce2 = testData.SolResponses.getAccountInfoResponse2.body.result.value.data.parsed.info.blockhash;\n      should.equal(latestBlockhashTxnJson2.nonce, nonce2);\n      should.equal(latestBlockhashTxnJson2.feePayer, testData.wrwUser.walletAddress3);\n      should.equal(latestBlockhashTxnJson2.numSignatures, testData.SolInputData.unsignedSweepSignatures);\n    });\n\n    it('should build unsigned token consolidation recoveries', async function () {\n      const res = (await basecoin.recoverConsolidations({\n        bitgoKey: testData.wrwUser.bitgoKey,\n        startingScanIndex: 3,\n        endingScanIndex: 5,\n        tokenContractAddress: usdtMintAddress,\n        durableNonces: durableNonces,\n      })) as MPCSweepTxs;\n      res.should.not.be.empty();\n      res.txRequests.length.should.equal(1);\n\n      const txn1 = res.txRequests[0].transactions[0].unsignedTx;\n      txn1.should.hasOwnProperty('serializedTx');\n      txn1.should.hasOwnProperty('signableHex');\n      txn1.should.hasOwnProperty('scanIndex');\n      (txn1.scanIndex ?? 0).should.equal(4);\n      txn1.should.hasOwnProperty('coin');\n      txn1.coin?.should.equal('tsol');\n      txn1.should.hasOwnProperty('derivationPath');\n      txn1.derivationPath?.should.equal('m/4');\n\n      txn1.should.hasOwnProperty('coinSpecific');\n      const coinSpecific1 = txn1.coinSpecific;\n      coinSpecific1?.should.hasOwnProperty('commonKeychain');\n\n      const latestBlockhashTxnDeserialize1 = new Transaction(coin);\n      latestBlockhashTxnDeserialize1.fromRawTransaction((txn1 as MPCTx).serializedTx);\n      const latestBlockhashTxnJson1 = latestBlockhashTxnDeserialize1.toJson();\n\n      const nonce1 = testData.SolResponses.getAccountInfoResponse.body.result.value.data.parsed.info.blockhash;\n      should.equal(latestBlockhashTxnJson1.nonce, nonce1);\n      should.equal(latestBlockhashTxnJson1.feePayer, testData.wrwUser.walletAddress5);\n      should.equal(latestBlockhashTxnJson1.numSignatures, testData.SolInputData.unsignedSweepSignatures);\n    });\n\n    it('should skip building consolidate transaction if balance is equal to zero', async function () {\n      await basecoin\n        .recoverConsolidations({\n          userKey: testData.wrwUser.userKey,\n          backupKey: testData.wrwUser.backupKey,\n          bitgoKey: testData.wrwUser.bitgoKey,\n          walletPassphrase: testData.wrwUser.walletPassphrase,\n          startingScanIndex: 1,\n          endingScanIndex: 2,\n          durableNonces: durableNonces,\n        })\n        .should.rejectedWith('Did not find an address with funds to recover');\n    });\n\n    it('should throw if startingScanIndex is not ge to 1', async () => {\n      await basecoin\n        .recoverConsolidations({\n          userKey: testData.wrwUser.userKey,\n          backupKey: testData.wrwUser.backupKey,\n          bitgoKey: testData.wrwUser.bitgoKey,\n          startingScanIndex: -1,\n          durableNonces: durableNonces,\n        })\n        .should.be.rejectedWith(\n          'Invalid starting or ending index to scan for addresses. startingScanIndex: -1, endingScanIndex: 19.'\n        );\n    });\n\n    it('should throw if scan factor is too high', async () => {\n      await basecoin\n        .recoverConsolidations({\n          userKey: testData.wrwUser.userKey,\n          backupKey: testData.wrwUser.backupKey,\n          bitgoKey: testData.wrwUser.bitgoKey,\n          startingScanIndex: 1,\n          endingScanIndex: 300,\n          durableNonces: durableNonces,\n        })\n        .should.be.rejectedWith(\n          'Invalid starting or ending index to scan for addresses. startingScanIndex: 1, endingScanIndex: 300.'\n        );\n    });\n  });\n\n  describe('broadcastTransaction', function () {\n    const sandBox = sinon.createSandbox();\n\n    afterEach(() => {\n      sandBox.restore();\n    });\n\n    it('should broadcast a transaction succesfully', async function () {\n      const serializedSignedTransaction = testData.rawTransactions.transfer.signed;\n      const broadcastStub = sandBox\n        .stub(Sol.prototype, 'getDataFromNode' as keyof Sol)\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'sendTransaction',\n            params: [\n              serializedSignedTransaction,\n              {\n                encoding: 'base64',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.broadcastTransactionResponse);\n\n      const broadcastTxn = await basecoin.broadcastTransaction({ serializedSignedTransaction });\n      assert.ok(broadcastTxn);\n      assert.ok(broadcastTxn.txId);\n      assert.strictEqual(\n        broadcastTxn.txId,\n        '2id3YC2jK9G5Wo2phDx4gJVAew8DcY5NAojnVuao8rkxwPYPe8cSwE5GzhEgJA2y8fVjDEo6iR6ykBvDxrTQrtpb'\n      );\n      assert.strictEqual(broadcastStub.callCount, 1);\n    });\n\n    it('should throw if got an error from the node', async function () {\n      const serializedSignedTransaction = testData.rawTransactions.transfer.signed;\n      const broadcastStub = sandBox\n        .stub(Sol.prototype, 'getDataFromNode' as keyof Sol)\n        .withArgs({\n          payload: {\n            id: '1',\n            jsonrpc: '2.0',\n            method: 'sendTransaction',\n            params: [\n              serializedSignedTransaction,\n              {\n                encoding: 'base64',\n              },\n            ],\n          },\n        })\n        .resolves(testData.SolResponses.broadcastTransactionResponseError);\n\n      await assert.rejects(\n        async () => {\n          await basecoin.broadcastTransaction({ serializedSignedTransaction });\n        },\n        { message: 'Error broadcasting transaction: Transaction simulation failed: Blockhash not found' }\n      );\n      assert.strictEqual(broadcastStub.callCount, 1);\n    });\n\n    it('should throw if is not a valid transaction', async function () {\n      const serializedSignedTransaction = 'randomstring';\n\n      await assert.rejects(\n        async () => {\n          await basecoin.broadcastTransaction({ serializedSignedTransaction });\n        },\n        { message: 'Invalid raw transaction' }\n      );\n    });\n\n    it('should throw if is not a signed transaction', async function () {\n      const serializedSignedTransaction = testData.rawTransactions.transfer.unsigned;\n\n      await assert.rejects(\n        async () => {\n          await basecoin.broadcastTransaction({ serializedSignedTransaction });\n        },\n        { message: 'Invalid raw transaction' }\n      );\n    });\n  });\n\n  describe('AuditKey', () => {\n    const { key: keyString, commonKeychain } = solBackupKey;\n    const key = keyString.replace(/\\s/g, '');\n    const walletPassphrase = 'kAm[EFQ6o=SxlcLFDw%,';\n    const multiSigType = 'tss';\n\n    it('should return for valid inputs', () => {\n      basecoin.assertIsValidKey({\n        encryptedPrv: key,\n        publicKey: commonKeychain,\n        walletPassphrase,\n        multiSigType,\n      });\n    });\n\n    it('should throw error if the commonKeychain is invalid', () => {\n      const alteredCommonKeychain = generateRandomPassword(10);\n      assert.throws(\n        () =>\n          basecoin.assertIsValidKey({\n            encryptedPrv: key,\n            publicKey: alteredCommonKeychain,\n            walletPassphrase,\n            multiSigType,\n          }),\n        {\n          message: 'Invalid common keychain',\n        }\n      );\n    });\n\n    it('should throw error if the walletPassphrase is incorrect', () => {\n      const incorrectPassphrase = 'foo';\n      assert.throws(\n        () =>\n          basecoin.assertIsValidKey({\n            encryptedPrv: key,\n            publicKey: commonKeychain,\n            walletPassphrase: incorrectPassphrase,\n            multiSigType,\n          }),\n        {\n          message: \"failed to decrypt prv: ccm: tag doesn't match\",\n        }\n      );\n    });\n\n    it('should throw error if the key is altered', () => {\n      const alteredKey = key.replace(/[0-9]/g, '0');\n      assert.throws(\n        () =>\n          basecoin.assertIsValidKey({\n            encryptedPrv: alteredKey,\n            publicKey: commonKeychain,\n            walletPassphrase,\n            multiSigType,\n          }),\n        {\n          message: 'failed to decrypt prv: json decrypt: invalid parameters',\n        }\n      );\n    });\n\n    it('should verify consolidation transaction', async function () {\n      // Set up wallet data\n      const walletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {\n          rootAddress: wallet.pub,\n        },\n        multisigType: 'tss',\n      };\n      const fakePrv = encrypt('password', 'prv');\n\n      const walletObj = new Wallet(bitgo, basecoin, walletData);\n      const bgUrl = common.Environments['mock'].uri;\n\n      nock(bgUrl)\n        .get('/api/v2/tsol/key/5b3424f91bf349930e34017500000000')\n        .reply(200, [\n          {\n            encryptedPrv: fakePrv,\n          },\n        ]);\n\n      // Mock the API response for buildAccountConsolidations\n      nock(bgUrl)\n        .post('/api/v2/tsol/wallet/5b34252f1bf349930e34020a00000000/consolidateAccount/build')\n        .reply(200, [\n          {\n            txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',\n            walletId: '63068ed4efa63a000877f02fd4b0fa6d',\n            txHex:\n              '02000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006f50130abf8d0d8943c5b9a51a574886a7d7b3d8db18f0ddb4ab8b6d3ec27e3c2f36f3339bb92d4296af6ae4d3abfbb07877f77d0033c883de08fa4a2eea670d0201020674a9df2b94aa4b4ada1202dc2891be366501d0acb4a01ca3e02e7fd6c1f505a71c96172044f1217c3784e8f02f49e2c8fc3591e81294ab54394f9d22fd7b7a8f8401c3f67cfa52518b34a09b08f4ea77e1c4fb9d89bfaccdc33cf8b8a9cf8d7bf0e04c89c50428e4eda5cbb759427c370f0a29a50bb0d1407e57924b0cc5b36f000000000000000000000000000000000000000000000000000000000000000006a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000040c9568195d67eb6b396fdbe97ba2e276622ef0023af09f23e6e0292abb678d90204030305010404000000040200020c020000000100000000000000',\n            feeInfo: {\n              fee: 10000,\n              feeString: '10000',\n            },\n            txInfo: {\n              minerFee: '10000',\n              spendAmount: '1547864',\n              spendAmounts: [\n                {\n                  coinName: 'tsol',\n                  amountString: '1547864',\n                },\n              ],\n              payGoFee: '0',\n              outputs: [\n                {\n                  address: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',\n                  value: 1547864,\n                  wallet: '63068ed4efa63a000877f02f',\n                  wallets: ['63068ed4efa63a000877f02f'],\n                  enterprise: '62d71a6b86068f0008f029fd',\n                  enterprises: ['62d71a6b86068f0008f029fd'],\n                  valueString: '1547864',\n                  coinName: 'tsol',\n                  walletType: 'hot',\n                  walletTypes: ['hot'],\n                },\n              ],\n              inputs: [\n                {\n                  value: 1547864,\n                  address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',\n                  valueString: '1547864',\n                },\n                {\n                  value: 10000,\n                  address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',\n                  valueString: '10000',\n                },\n              ],\n              type: 'Send',\n            },\n            consolidateId: '68a7d5d0c66e74e216b97173bd558c6d',\n            coin: 'tsol',\n          },\n        ]);\n\n      // Call the function to test\n      await assert.rejects(\n        async () => {\n          await walletObj.sendAccountConsolidations({\n            walletPassphrase: 'password',\n          });\n        },\n        {\n          message: 'tx outputs does not match with expected address',\n        }\n      );\n    });\n\n    it('should verify valid a consolidation transaction', async () => {\n      const consolidationTx = {\n        txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',\n        walletId: '63068ed4efa63a000877f02fd4b0fa6d',\n        txHex:\n          '02000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002010206aeda253a331de489838246df93879440af8c62ac4967658edc2bb5d52b9759d91c96172044f1217c3784e8f02f49e2c8fc3591e81294ab54394f9d22fd7b7a8f74a9df2b94aa4b4ada1202dc2891be366501d0acb4a01ca3e02e7fd6c1f505a7d2734f3952f3eb4aefcf6c7a6092e979dd3fe5563ccfaca1cc92652a15ddd393000000000000000000000000000000000000000000000000000000000000000006a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea9400000428dad41bbedeb38018379e4ceeb7e80757d74dd00ae8c47c38d597477339ad80204030305010404000000040200020c02000000589e170000000000',\n        feeInfo: {\n          fee: 10000,\n          feeString: '10000',\n        },\n        txInfo: {\n          minerFee: '10000',\n          spendAmount: '1547864',\n          spendAmounts: [\n            {\n              coinName: 'tsol',\n              amountString: '1547864',\n            },\n          ],\n          payGoFee: '0',\n          outputs: [\n            {\n              address: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',\n              value: 1547864,\n              wallet: '63068ed4efa63a000877f02f',\n              wallets: ['63068ed4efa63a000877f02f'],\n              enterprise: '62d71a6b86068f0008f029fd',\n              enterprises: ['62d71a6b86068f0008f029fd'],\n              valueString: '1547864',\n              coinName: 'tsol',\n              walletType: 'hot',\n              walletTypes: ['hot'],\n            },\n          ],\n          inputs: [\n            {\n              value: 1547864,\n              address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',\n              valueString: '1547864',\n            },\n            {\n              value: 10000,\n              address: 'CmYsN3f8bcm4BDkFJWNsvYgjRxMTLH6vbJWNfYdmH7GU',\n              valueString: '10000',\n            },\n          ],\n          type: 'Send',\n        },\n        consolidateId: '68a7d5d0c66e74e216b97173bd558c6d',\n        coin: 'tsol',\n      };\n\n      const mockedWallet: Partial<IWallet> = {\n        coinSpecific: () => {\n          const cs = {\n            rootAddress: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',\n          } as WalletCoinSpecific;\n          return cs;\n        },\n      };\n\n      try {\n        if (\n          !(await basecoin.verifyTransaction({\n            blockhash: '',\n            feePayer: '',\n            txParams: {},\n            txPrebuild: consolidationTx as unknown as TransactionPrebuild,\n            walletType: 'tss',\n            wallet: mockedWallet as IWallet,\n            verification: {\n              consolidationToBaseAddress: true,\n            },\n          }))\n        ) {\n          assert.fail('Transaction should pass verification');\n        }\n      } catch (e) {\n        assert.fail('Transaction should pass verification');\n      }\n    });\n\n    it('should verify a token consolidation transaction', async () => {\n      const consolidationTx = {\n        txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',\n        walletId: '63068ed4efa63a000877f02fd4b0fa6d',\n        txHex:\n          '02b7c2c7829eded4e8f947c90ed3b9afce71f616eb47dfcfbf4b765778149060013acb33b9f67fdd9c2512f48fac4e4c049eab93829b69404f3bd166fe3242c90700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000020104096da690bd558fd8634ac14f1645d8095afd0caea5953578c596e3c0dea38305ee7298bfac55101f177735659d42ed6be890ef3a1d204d9e33f32e24c5635327ca66d1fe00826e5a4f759f87b279c1aee19cce5301af4ed66ae17db48b201ed6c2a0e8a28bf565627f1ab8a34b1a95ee1b0a2a39084f1f0e2acb1c394b20185d8e0b22657c8d9c4ce5f6495efb6410c199011530f90e3ab9d8d1e4206f9ae0ffeb0000000000000000000000000000000000000000000000000000000000000000c5f9fb32f49111ab20c33f2598fc836c113e291881ac21ee29169394011244e406a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000006ddf6e1d765a193d9cbe146ceeb79ac1cb485ed5f5b37913a8cf5857eff00a9de13c74d2b4d948e1608ea6eebdafe75bc2f995aad11b21d1e2c94f2a2d12f6802050303070004040000000804020604010a0ce0076bb20400000006',\n        feeInfo: {\n          fee: 10000,\n          feeString: '10000',\n        },\n        txInfo: {\n          inputs: [\n            {\n              address: '8iLa26KSbdpBUzNK7uYq8FvyuyA5h4k4erDHsDcPbHus',\n              value: 2.0173228e10,\n              valueString: '20173228000',\n            },\n            {\n              address: '8P2kX7Tyh9eS3RKdaBhqbEtQGAX58DXzL7mhDQABGX2d',\n              value: 10000,\n              valueString: '10000',\n            },\n          ],\n          minerFee: '10000',\n          outputs: [\n            {\n              address: 'HBxZShcE86UMmF93KUM8eWJKqeEXi5cqWCLYLMMhqMYm',\n              coinName: 'sol:wif',\n              enterprise: {\n                $oid: '5553ba8ae7a5c77006719661',\n              },\n              enterprises: [\n                {\n                  $oid: '5553ba8ae7a5c77006719661',\n                },\n              ],\n              value: 2.0173228e10,\n              valueString: '20173228000',\n              wallet: {\n                $oid: '62f4c3720d92c50008257eb5',\n              },\n              walletType: 'hot',\n              wallets: [\n                {\n                  $oid: '62f4c3720d92c50008257eb5',\n                },\n              ],\n            },\n          ],\n          payGoFee: '0',\n          spendAmount: '20173228000',\n          spendAmounts: [\n            {\n              amountString: '20173228000',\n              coinName: 'sol:wif',\n            },\n          ],\n          type: 'Send',\n        },\n        consolidateId: '6712d7fda6de4906d658c04aebbf8f9b',\n        coin: 'tsol',\n      };\n\n      const mockedWallet: Partial<IWallet> = {\n        coinSpecific: () => {\n          const cs = {\n            rootAddress: 'HBxZShcE86UMmF93KUM8eWJKqeEXi5cqWCLYLMMhqMYm',\n          } as WalletCoinSpecific;\n          return cs;\n        },\n      };\n\n      try {\n        if (\n          !(await basecoin.verifyTransaction({\n            blockhash: '',\n            feePayer: '',\n            txParams: {},\n            txPrebuild: consolidationTx as unknown as TransactionPrebuild,\n            walletType: 'tss',\n            wallet: mockedWallet as IWallet,\n            verification: {\n              consolidationToBaseAddress: true,\n            },\n          }))\n        ) {\n          assert.fail('Transaction should pass verification');\n        }\n      } catch (e) {\n        assert.fail('Transaction should pass verification');\n      }\n    });\n\n    it('should verify a spoofed token consolidation transaction', async () => {\n      const consolidationTx = {\n        txRequestId: '4fdd0cae-2563-43b1-b5cf-94865158ca10',\n        walletId: '63068ed4efa63a000877f02fd4b0fa6d',\n        txHex:\n          '02b7c2c7829eded4e8f947c90ed3b9afce71f616eb47dfcfbf4b765778149060013acb33b9f67fdd9c2512f48fac4e4c049eab93829b69404f3bd166fe3242c90700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000020104096da690bd558fd8634ac14f1645d8095afd0caea5953578c596e3c0dea38305ee7298bfac55101f177735659d42ed6be890ef3a1d204d9e33f32e24c5635327ca66d1fe00826e5a4f759f87b279c1aee19cce5301af4ed66ae17db48b201ed6c2a0e8a28bf565627f1ab8a34b1a95ee1b0a2a39084f1f0e2acb1c394b20185d8e0b22657c8d9c4ce5f6495efb6410c199011530f90e3ab9d8d1e4206f9ae0ffeb0000000000000000000000000000000000000000000000000000000000000000c5f9fb32f49111ab20c33f2598fc836c113e291881ac21ee29169394011244e406a7d517192c568ee08a845f73d29788cf035c3145b21ab344d8062ea940000006ddf6e1d765a193d9cbe146ceeb79ac1cb485ed5f5b37913a8cf5857eff00a9de13c74d2b4d948e1608ea6eebdafe75bc2f995aad11b21d1e2c94f2a2d12f6802050303070004040000000804020604010a0ce0076bb20400000006',\n        feeInfo: {\n          fee: 10000,\n          feeString: '10000',\n        },\n        txInfo: {\n          inputs: [\n            {\n              address: '8iLa26KSbdpBUzNK7uYq8FvyuyA5h4k4erDHsDcPbHus',\n              value: 2.0173228e10,\n              valueString: '20173228000',\n            },\n            {\n              address: '8P2kX7Tyh9eS3RKdaBhqbEtQGAX58DXzL7mhDQABGX2d',\n              value: 10000,\n              valueString: '10000',\n            },\n          ],\n          minerFee: '10000',\n          outputs: [\n            {\n              address: 'HBxZShcE86UMmF93KUM8eWJKqeEXi5cqWCLYLMMhqMYm',\n              coinName: 'sol:wif',\n              enterprise: {\n                $oid: '5553ba8ae7a5c77006719661',\n              },\n              enterprises: [\n                {\n                  $oid: '5553ba8ae7a5c77006719661',\n                },\n              ],\n              value: 2.0173228e10,\n              valueString: '20173228000',\n              wallet: {\n                $oid: '62f4c3720d92c50008257eb5',\n              },\n              walletType: 'hot',\n              wallets: [\n                {\n                  $oid: '62f4c3720d92c50008257eb5',\n                },\n              ],\n            },\n          ],\n          payGoFee: '0',\n          spendAmount: '20173228000',\n          spendAmounts: [\n            {\n              amountString: '20173228000',\n              coinName: 'sol:wif',\n            },\n          ],\n          type: 'Send',\n        },\n        consolidateId: '6712d7fda6de4906d658c04aebbf8f9b',\n        coin: 'tsol',\n      };\n\n      const mockedWallet: Partial<IWallet> = {\n        coinSpecific: () => {\n          const cs = {\n            rootAddress: '8rQXeVEMrKvtWCEJirEM6cKYnbZuTqVTbqRPiMMAJ8R4',\n          } as WalletCoinSpecific;\n          return cs;\n        },\n      };\n\n      await assert.rejects(\n        async () =>\n          basecoin.verifyTransaction({\n            blockhash: '',\n            feePayer: '',\n            txParams: {},\n            txPrebuild: consolidationTx as unknown as TransactionPrebuild,\n            walletType: 'tss',\n            wallet: mockedWallet as IWallet,\n            verification: {\n              consolidationToBaseAddress: true,\n            },\n          }),\n        {\n          message: 'tx outputs does not match with expected address',\n        }\n      );\n    });\n  });\n\n  describe('blind signing token enablement protection', () => {\n    it('should verify as valid the enabletoken intent when prebuild tx matchs user intent ', async function () {\n      const { txParams, txPrebuildRaw, walletData } = testData.enableTokenFixtures;\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n      const sameIntentTx = await basecoin.verifyTransaction({\n        txParams,\n        txPrebuild: txPrebuildRaw,\n        wallet,\n        verification: { verifyTokenEnablement: true },\n      } as unknown as SolVerifyTransactionOptions);\n\n      sameIntentTx.should.equal(true);\n    });\n\n    it('should thrown an error when tampered prebuild tx type ', async function () {\n      const { txParams, txPrebuildRaw, sendTxHex, walletData } = testData.enableTokenFixtures;\n      const tamperedTxPrebuild = { ...txPrebuildRaw, txHex: sendTxHex };\n\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      await assert.rejects(\n        async () =>\n          await basecoin.verifyTransaction({\n            txParams,\n            txPrebuild: tamperedTxPrebuild,\n            wallet,\n            verification: { verifyTokenEnablement: true },\n          } as unknown as SolVerifyTransactionOptions),\n        {\n          message:\n            'Invalid transaction type on token enablement: expected \"AssociatedTokenAccountInitialization\", got \"Send\".',\n        }\n      );\n    });\n\n    it('should verify that tokenName matches between user intent and hex', async function () {\n      const { txParams, txPrebuildRaw, wrongTokenNameTxHex, walletData } = testData.enableTokenFixtures;\n      const tamperedTxPrebuild = { ...txPrebuildRaw, txHex: wrongTokenNameTxHex };\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      await assert.rejects(\n        async () =>\n          basecoin.verifyTransaction({\n            txParams,\n            txPrebuild: tamperedTxPrebuild,\n            wallet,\n            verification: { verifyTokenEnablement: true },\n          } as unknown as SolVerifyTransactionOptions),\n        { message: 'Invalid token name: expected tsol:ray, got tsol:t22mint on token enablement tx' }\n      );\n    });\n\n    it('should verify that tokenAddr matches between user intent and hex', async function () {\n      const { txParams, txPrebuildRaw, wrongAddrTxHex, walletData } = testData.enableTokenFixtures;\n      const tamperedTxPrebuild = { ...txPrebuildRaw, txHex: wrongAddrTxHex };\n\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n      await assert.rejects(\n        async () =>\n          basecoin.verifyTransaction({\n            txParams,\n            txPrebuild: tamperedTxPrebuild,\n            wallet,\n            verification: { verifyTokenEnablement: true },\n          } as unknown as SolVerifyTransactionOptions),\n        {\n          message:\n            'Invalid token address: expected 4bTYvvv2Hk4v2kQW8HZFFS4SzYPztQshw9Gm1suXmaBj, got G1LEgANAwKo7b8NfxTsMzrbBYDkXqi5REVJY8thrMRQm on token enablement tx',\n        }\n      );\n    });\n\n    it('should fail sendTokenEnablement call on spoofed data', async function () {\n      const { sendTokenEnablementPayload, walletData, wrongTokenNameTxHex } = testData.enableTokenFixtures;\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n      nock('https://bitgo.fakeurl').get('/api/v2/tsol/key/68bafed588671cf94ed8a5dbba882ad3').reply(200, {});\n      await assert.rejects(\n        async () =>\n          wallet.sendTokenEnablement({\n            verification: { verifyTokenEnablement: true },\n            ...sendTokenEnablementPayload,\n            prebuildTx: { ...sendTokenEnablementPayload.prebuildTx, txHex: wrongTokenNameTxHex },\n          } as unknown as PrebuildAndSignTransactionOptions),\n        {\n          message: 'Invalid token name: expected tsol:ray, got tsol:t22mint on token enablement tx',\n        }\n      );\n    });\n  });\n\n  describe('isWalletAddress', () => {\n    it('should verify valid wallet address with correct keychain and index', async function () {\n      const address = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';\n      const commonKeychain =\n        '8ea32ecacfc83effbd2e2790ee44fa7c59b4d86c29a12f09fb613d8195f93f4e21875cad3b98adada40c040c54c3569467df41a020881a6184096378701862bd';\n      const index = '1';\n      const keychains = [{ id: '1', type: 'tss' as const, commonKeychain }];\n\n      const result = await basecoin.isWalletAddress({ keychains, address, index });\n      result.should.equal(true);\n    });\n\n    it('should return false for address with incorrect keychain', async function () {\n      const address = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';\n      const wrongKeychain =\n        '0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000';\n      const index = '1';\n      const keychains = [{ id: '1', type: 'tss' as const, commonKeychain: wrongKeychain }];\n\n      const result = await basecoin.isWalletAddress({ keychains, address, index });\n      result.should.equal(false);\n    });\n\n    it('should return false for address with incorrect index', async function () {\n      const address = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';\n      const commonKeychain =\n        '8ea32ecacfc83effbd2e2790ee44fa7c59b4d86c29a12f09fb613d8195f93f4e21875cad3b98adada40c040c54c3569467df41a020881a6184096378701862bd';\n      const wrongIndex = '999';\n      const keychains = [{ id: '1', type: 'tss' as const, commonKeychain }];\n\n      const result = await basecoin.isWalletAddress({ keychains, address, index: wrongIndex });\n      result.should.equal(false);\n    });\n\n    it('should throw error for invalid address', async function () {\n      const invalidAddress = 'invalidaddress';\n      const commonKeychain =\n        '8ea32ecacfc83effbd2e2790ee44fa7c59b4d86c29a12f09fb613d8195f93f4e21875cad3b98adada40c040c54c3569467df41a020881a6184096378701862bd';\n      const index = '1';\n      const keychains = [{ id: '1', type: 'tss' as const, commonKeychain }];\n\n      await assert.rejects(async () => await basecoin.isWalletAddress({ keychains, address: invalidAddress, index }), {\n        message: `invalid address: ${invalidAddress}`,\n      });\n    });\n  });\n\n  describe('getAddressFromPublicKey', () => {\n    it('should convert public key to base58 address', function () {\n      const publicKey = '61220a9394802b1d1df37b35f7a3197970f48081092cee011fc98f7b71b2bd43';\n      const expectedAddress = '7YAesfwPk41VChUgr65bm8FEep7ymWqLSW5rpYB5zZPY';\n\n      const address = basecoin.getAddressFromPublicKey(publicKey);\n      address.should.equal(expectedAddress);\n    });\n  });\n});\n"]}
|