@bitgo-beta/sdk-coin-algo 1.3.19-beta.90 → 1.3.19-beta.901
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/CHANGELOG.md +541 -0
- package/dist/src/algo.d.ts +96 -16
- package/dist/src/algo.d.ts.map +1 -1
- package/dist/src/algo.js +265 -47
- package/dist/src/algoToken.d.ts +1 -1
- package/dist/src/algoToken.d.ts.map +1 -1
- package/dist/src/algoToken.js +3 -3
- package/dist/src/index.js +23 -9
- package/dist/src/lib/assetTransferBuilder.d.ts +1 -1
- package/dist/src/lib/assetTransferBuilder.js +2 -2
- package/dist/src/lib/ifaces.d.ts +0 -1
- package/dist/src/lib/ifaces.d.ts.map +1 -1
- package/dist/src/lib/index.js +23 -9
- package/dist/src/lib/keyPair.js +24 -10
- package/dist/src/lib/keyRegistrationBuilder.js +1 -1
- package/dist/src/lib/seedEncoding.js +23 -9
- package/dist/src/lib/transaction.js +3 -4
- package/dist/src/lib/transactionBuilder.d.ts +6 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +45 -29
- package/dist/src/lib/transactionBuilderFactory.js +1 -1
- package/dist/src/lib/transferBuilder.js +1 -1
- package/dist/src/lib/txnSchema.js +1 -1
- package/dist/src/lib/utils.d.ts +6 -1
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +35 -11
- package/dist/src/seedValidator.js +23 -9
- package/package.json +8 -9
package/dist/src/algo.js
CHANGED
|
@@ -1,7 +1,11 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
3
|
if (k2 === undefined) k2 = k;
|
|
4
|
-
Object.
|
|
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);
|
|
5
9
|
}) : (function(o, m, k, k2) {
|
|
6
10
|
if (k2 === undefined) k2 = k;
|
|
7
11
|
o[k2] = m[k];
|
|
@@ -11,13 +15,23 @@ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (
|
|
|
11
15
|
}) : function(o, v) {
|
|
12
16
|
o["default"] = v;
|
|
13
17
|
});
|
|
14
|
-
var __importStar = (this && this.__importStar) || function (
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
};
|
|
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
|
+
})();
|
|
21
35
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
22
36
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
23
37
|
};
|
|
@@ -26,13 +40,17 @@ exports.Algo = void 0;
|
|
|
26
40
|
/**
|
|
27
41
|
* @prettier
|
|
28
42
|
*/
|
|
29
|
-
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
30
43
|
const _ = __importStar(require("lodash"));
|
|
31
44
|
const seedValidator_1 = require("./seedValidator");
|
|
32
45
|
const statics_1 = require("@bitgo-beta/statics");
|
|
33
46
|
const AlgoLib = __importStar(require("./lib"));
|
|
34
47
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
35
48
|
const stellar_sdk_1 = __importDefault(require("stellar-sdk"));
|
|
49
|
+
const bignumber_js_1 = __importDefault(require("bignumber.js"));
|
|
50
|
+
const utils_1 = __importDefault(require("./lib/utils"));
|
|
51
|
+
const algosdk = __importStar(require("algosdk"));
|
|
52
|
+
const transactionBuilder_1 = require("./lib/transactionBuilder");
|
|
53
|
+
const buffer_1 = require("buffer");
|
|
36
54
|
const SUPPORTED_ADDRESS_VERSION = 1;
|
|
37
55
|
const MSIG_THRESHOLD = 2; // m in m-of-n
|
|
38
56
|
class Algo extends sdk_core_1.BaseCoin {
|
|
@@ -73,12 +91,11 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
73
91
|
allowsAccountConsolidations() {
|
|
74
92
|
return true;
|
|
75
93
|
}
|
|
76
|
-
/**
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
*/
|
|
94
|
+
/** inheritdoc */
|
|
95
|
+
deriveKeyWithSeed() {
|
|
96
|
+
throw new sdk_core_1.NotSupported('method deriveKeyWithSeed not supported for eddsa curve');
|
|
97
|
+
}
|
|
98
|
+
/** inheritdoc */
|
|
82
99
|
generateKeyPair(seed) {
|
|
83
100
|
const keyPair = seed ? new AlgoLib.KeyPair({ seed }) : new AlgoLib.KeyPair();
|
|
84
101
|
const keys = keyPair.getKeys();
|
|
@@ -87,9 +104,18 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
87
104
|
}
|
|
88
105
|
return {
|
|
89
106
|
pub: keyPair.getAddress(),
|
|
90
|
-
prv: AlgoLib.algoUtils.encodeSeed(Buffer.from(keyPair.getSigningKey())),
|
|
107
|
+
prv: AlgoLib.algoUtils.encodeSeed(buffer_1.Buffer.from(keyPair.getSigningKey())),
|
|
91
108
|
};
|
|
92
109
|
}
|
|
110
|
+
/** inheritdoc */
|
|
111
|
+
generateRootKeyPair(seed) {
|
|
112
|
+
const keyPair = seed ? new AlgoLib.KeyPair({ seed }) : new AlgoLib.KeyPair();
|
|
113
|
+
const keys = keyPair.getKeys();
|
|
114
|
+
if (!keys.prv) {
|
|
115
|
+
throw new Error('Missing prv in key generation.');
|
|
116
|
+
}
|
|
117
|
+
return { prv: keys.prv + keys.pub, pub: keys.pub };
|
|
118
|
+
}
|
|
93
119
|
/**
|
|
94
120
|
* Return boolean indicating whether input is valid public key for the coin.
|
|
95
121
|
*
|
|
@@ -97,7 +123,7 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
97
123
|
* @returns {Boolean} is it valid?
|
|
98
124
|
*/
|
|
99
125
|
isValidPub(pub) {
|
|
100
|
-
return AlgoLib.algoUtils.isValidAddress(pub);
|
|
126
|
+
return AlgoLib.algoUtils.isValidAddress(pub) || AlgoLib.algoUtils.isValidPublicKey(pub);
|
|
101
127
|
}
|
|
102
128
|
/**
|
|
103
129
|
* Return boolean indicating whether input is valid seed for the coin
|
|
@@ -108,7 +134,7 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
108
134
|
* @returns {Boolean} is it valid?
|
|
109
135
|
*/
|
|
110
136
|
isValidPrv(prv) {
|
|
111
|
-
return AlgoLib.algoUtils.isValidSeed(prv);
|
|
137
|
+
return AlgoLib.algoUtils.isValidSeed(prv) || AlgoLib.algoUtils.isValidPrivateKey(prv);
|
|
112
138
|
}
|
|
113
139
|
/**
|
|
114
140
|
* Return boolean indicating whether input is valid public key for the coin
|
|
@@ -127,10 +153,10 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
127
153
|
*/
|
|
128
154
|
async signMessage(key, message) {
|
|
129
155
|
const algoKeypair = new AlgoLib.KeyPair({ prv: key.prv });
|
|
130
|
-
if (Buffer.isBuffer(message)) {
|
|
156
|
+
if (buffer_1.Buffer.isBuffer(message)) {
|
|
131
157
|
message = message.toString('base64');
|
|
132
158
|
}
|
|
133
|
-
return Buffer.from(algoKeypair.signMessage(message));
|
|
159
|
+
return buffer_1.Buffer.from(algoKeypair.signMessage(message));
|
|
134
160
|
}
|
|
135
161
|
/**
|
|
136
162
|
* Specifies what key we will need for signing` - Algorand needs the backup, bitgo pubs.
|
|
@@ -324,7 +350,7 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
324
350
|
if (!transaction) {
|
|
325
351
|
throw new Error('Invalid transaction');
|
|
326
352
|
}
|
|
327
|
-
const signedTxHex = Buffer.from(transaction.toBroadcastFormat()).toString('base64');
|
|
353
|
+
const signedTxHex = buffer_1.Buffer.from(transaction.toBroadcastFormat()).toString('base64');
|
|
328
354
|
if (numberSigners === 1) {
|
|
329
355
|
return { txHex: signedTxHex };
|
|
330
356
|
}
|
|
@@ -363,28 +389,194 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
363
389
|
async verifyTransaction(params) {
|
|
364
390
|
return true;
|
|
365
391
|
}
|
|
366
|
-
/** @inheritDoc */
|
|
367
|
-
deriveKeyWithSeed({ key, seed }) {
|
|
368
|
-
const derivationPathInput = utxolib.crypto.hash256(Buffer.from(seed, 'utf8')).toString('hex');
|
|
369
|
-
const derivationPathParts = [
|
|
370
|
-
999999,
|
|
371
|
-
parseInt(derivationPathInput.slice(0, 7), 16),
|
|
372
|
-
parseInt(derivationPathInput.slice(7, 14), 16),
|
|
373
|
-
];
|
|
374
|
-
const derivationPath = 'm/' + derivationPathParts.map((part) => `${part}'`).join('/');
|
|
375
|
-
const derivedKey = sdk_core_1.Ed25519KeyDeriver.derivePath(derivationPath, key).key;
|
|
376
|
-
const keypair = new AlgoLib.KeyPair({ seed: derivedKey });
|
|
377
|
-
return {
|
|
378
|
-
key: keypair.getAddress(),
|
|
379
|
-
derivationPath,
|
|
380
|
-
};
|
|
381
|
-
}
|
|
382
392
|
decodeTx(txn) {
|
|
383
393
|
return AlgoLib.algoUtils.decodeAlgoTxn(txn);
|
|
384
394
|
}
|
|
385
395
|
getAddressFromPublicKey(pubKey) {
|
|
386
396
|
return AlgoLib.algoUtils.publicKeyToAlgoAddress(pubKey);
|
|
387
397
|
}
|
|
398
|
+
supportsDeriveKeyWithSeed() {
|
|
399
|
+
return false;
|
|
400
|
+
}
|
|
401
|
+
/** inherited doc */
|
|
402
|
+
getDefaultMultisigType() {
|
|
403
|
+
return sdk_core_1.multisigTypes.onchain;
|
|
404
|
+
}
|
|
405
|
+
/**
|
|
406
|
+
* Gets config for how token enablements work for this coin
|
|
407
|
+
* @returns
|
|
408
|
+
* requiresTokenEnablement: True if tokens need to be enabled for this coin
|
|
409
|
+
* supportsMultipleTokenEnablements: True if multiple tokens can be enabled in one transaction
|
|
410
|
+
*/
|
|
411
|
+
getTokenEnablementConfig() {
|
|
412
|
+
return {
|
|
413
|
+
requiresTokenEnablement: true,
|
|
414
|
+
supportsMultipleTokenEnablements: false,
|
|
415
|
+
};
|
|
416
|
+
}
|
|
417
|
+
/**
|
|
418
|
+
* Gets the balance of the root address in base units of algo
|
|
419
|
+
* Eg. If balance is 1 Algo, this returns 1*10^6
|
|
420
|
+
* @param rootAddress
|
|
421
|
+
* @param client
|
|
422
|
+
*/
|
|
423
|
+
async getAccountBalance(rootAddress, client) {
|
|
424
|
+
const accountInformation = await client.accountInformation(rootAddress).do();
|
|
425
|
+
// Extract the balance from the account information
|
|
426
|
+
return accountInformation.amount;
|
|
427
|
+
}
|
|
428
|
+
/**
|
|
429
|
+
* Returns the Algo client for the given token, baseServer and port
|
|
430
|
+
* Used to interact with the Algo network
|
|
431
|
+
*/
|
|
432
|
+
getClient(token, baseServer, port) {
|
|
433
|
+
return new algosdk.Algodv2(token, baseServer, port);
|
|
434
|
+
}
|
|
435
|
+
async recover(params) {
|
|
436
|
+
const isUnsignedSweep = this.isValidPub(params.userKey) && this.isValidPub(params.backupKey);
|
|
437
|
+
if (!params.nodeParams) {
|
|
438
|
+
throw new Error('Please provide the details of an ALGO node to use for recovery');
|
|
439
|
+
}
|
|
440
|
+
// Validate the root address
|
|
441
|
+
if (!this.isValidAddress(params.rootAddress)) {
|
|
442
|
+
throw new Error('invalid rootAddress, got: ' + params.rootAddress);
|
|
443
|
+
}
|
|
444
|
+
// Validate the destination address
|
|
445
|
+
if (!this.isValidAddress(params.recoveryDestination)) {
|
|
446
|
+
throw new Error('invalid recoveryDestination, got: ' + params.recoveryDestination);
|
|
447
|
+
}
|
|
448
|
+
if (params.firstRound && new bignumber_js_1.default(params.firstRound).isNegative()) {
|
|
449
|
+
throw new Error('first round needs to be a positive value');
|
|
450
|
+
}
|
|
451
|
+
const genesisId = this.bitgo.getEnv() === 'prod' ? transactionBuilder_1.MAINNET_GENESIS_ID : transactionBuilder_1.TESTNET_GENESIS_ID;
|
|
452
|
+
const genesisHash = this.bitgo.getEnv() === 'prod' ? transactionBuilder_1.MAINNET_GENESIS_HASH : transactionBuilder_1.TESTNET_GENESIS_HASH;
|
|
453
|
+
utils_1.default.validateBase64(genesisHash);
|
|
454
|
+
if (!isUnsignedSweep && !params.walletPassphrase) {
|
|
455
|
+
throw new Error('walletPassphrase is required for non-bitgo recovery');
|
|
456
|
+
}
|
|
457
|
+
const factory = new AlgoLib.TransactionBuilderFactory(statics_1.coins.get('algo'));
|
|
458
|
+
const txBuilder = factory.getTransferBuilder();
|
|
459
|
+
let userPrv;
|
|
460
|
+
let backupPrv;
|
|
461
|
+
if (!isUnsignedSweep) {
|
|
462
|
+
if (!params.bitgoKey) {
|
|
463
|
+
throw new Error('bitgo public key from the keyCard is required for non-bitgo recovery');
|
|
464
|
+
}
|
|
465
|
+
try {
|
|
466
|
+
userPrv = this.bitgo.decrypt({ input: params.userKey, password: params.walletPassphrase });
|
|
467
|
+
backupPrv = this.bitgo.decrypt({ input: params.backupKey, password: params.walletPassphrase });
|
|
468
|
+
const userKeyAddress = utils_1.default.privateKeyToAlgoAddress(userPrv);
|
|
469
|
+
const backupKeyAddress = utils_1.default.privateKeyToAlgoAddress(backupPrv);
|
|
470
|
+
txBuilder.numberOfRequiredSigners(2).setSigners([userKeyAddress, backupKeyAddress, params.bitgoKey]);
|
|
471
|
+
}
|
|
472
|
+
catch (e) {
|
|
473
|
+
throw new Error('unable to decrypt userKey or backupKey with the walletPassphrase provided, got error: ' + e.message);
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
const client = this.getClient(params.nodeParams.token, params.nodeParams.baseServer, params.nodeParams.port);
|
|
477
|
+
const nativeBalance = await this.getAccountBalance(params.rootAddress, client);
|
|
478
|
+
// Algorand accounts require a min. balance of 1 ALGO
|
|
479
|
+
const MIN_MICROALGOS_BALANCE = 100000;
|
|
480
|
+
const spendableAmount = new bignumber_js_1.default(nativeBalance).minus(params.fee).minus(MIN_MICROALGOS_BALANCE).toNumber();
|
|
481
|
+
if (new bignumber_js_1.default(spendableAmount).isZero() || new bignumber_js_1.default(spendableAmount).isLessThanOrEqualTo(params.fee)) {
|
|
482
|
+
throw new Error('Insufficient balance to recover, got balance: ' +
|
|
483
|
+
nativeBalance +
|
|
484
|
+
' fee: ' +
|
|
485
|
+
params.fee +
|
|
486
|
+
' min account balance: ' +
|
|
487
|
+
MIN_MICROALGOS_BALANCE);
|
|
488
|
+
}
|
|
489
|
+
let latestRound;
|
|
490
|
+
if (!params.firstRound) {
|
|
491
|
+
latestRound = await client
|
|
492
|
+
.status()
|
|
493
|
+
.do()
|
|
494
|
+
.then((status) => status['last-round']);
|
|
495
|
+
}
|
|
496
|
+
const firstRound = !params.firstRound ? latestRound : params.firstRound;
|
|
497
|
+
if (!firstRound) {
|
|
498
|
+
throw new Error('Unable to fetch the latest round from the node. Please provide the firstRound or try again.');
|
|
499
|
+
}
|
|
500
|
+
const LAST_ROUND_BUFFER = 1000;
|
|
501
|
+
const lastRound = firstRound + LAST_ROUND_BUFFER;
|
|
502
|
+
txBuilder
|
|
503
|
+
.fee({ fee: params.fee.toString() })
|
|
504
|
+
.isFlatFee(true)
|
|
505
|
+
.sender({
|
|
506
|
+
address: params.rootAddress,
|
|
507
|
+
})
|
|
508
|
+
.to({
|
|
509
|
+
address: params.recoveryDestination,
|
|
510
|
+
})
|
|
511
|
+
.amount(spendableAmount)
|
|
512
|
+
.genesisId(genesisId)
|
|
513
|
+
.genesisHash(genesisHash)
|
|
514
|
+
.firstRound(new bignumber_js_1.default(firstRound).toNumber())
|
|
515
|
+
.lastRound(new bignumber_js_1.default(lastRound).toNumber());
|
|
516
|
+
if (params.note) {
|
|
517
|
+
const note = new Uint8Array(buffer_1.Buffer.from(params.note, 'utf-8'));
|
|
518
|
+
txBuilder.note(note);
|
|
519
|
+
}
|
|
520
|
+
// Cold wallet, offline vault
|
|
521
|
+
if (isUnsignedSweep) {
|
|
522
|
+
const tx = await txBuilder.build();
|
|
523
|
+
const txJson = tx.toJson();
|
|
524
|
+
return {
|
|
525
|
+
txHex: buffer_1.Buffer.from(tx.toBroadcastFormat()).toString('hex'),
|
|
526
|
+
type: txJson.type,
|
|
527
|
+
userKey: params.userKey,
|
|
528
|
+
backupKey: params.backupKey,
|
|
529
|
+
bitgoKey: params.bitgoKey,
|
|
530
|
+
address: params.rootAddress,
|
|
531
|
+
coin: this.getChain(),
|
|
532
|
+
feeInfo: txJson.fee,
|
|
533
|
+
amount: txJson.amount ?? nativeBalance.toString(),
|
|
534
|
+
firstRound: txJson.firstRound,
|
|
535
|
+
lastRound: txJson.lastRound,
|
|
536
|
+
genesisId: genesisId,
|
|
537
|
+
genesisHash: genesisHash,
|
|
538
|
+
note: txJson.note ? buffer_1.Buffer.from(txJson.note.buffer).toString('utf-8') : undefined,
|
|
539
|
+
keys: [params.userKey, params.backupKey, params.bitgoKey],
|
|
540
|
+
addressVersion: 1,
|
|
541
|
+
};
|
|
542
|
+
}
|
|
543
|
+
// Non-bitgo Recovery (Hot wallets)
|
|
544
|
+
txBuilder.sign({ key: userPrv });
|
|
545
|
+
txBuilder.sign({ key: backupPrv });
|
|
546
|
+
const tx = await txBuilder.build();
|
|
547
|
+
const txJson = tx.toJson();
|
|
548
|
+
return {
|
|
549
|
+
tx: buffer_1.Buffer.from(tx.toBroadcastFormat()).toString('base64'),
|
|
550
|
+
id: txJson.id,
|
|
551
|
+
coin: this.getChain(),
|
|
552
|
+
fee: txJson.fee,
|
|
553
|
+
firstRound: txJson.firstRound,
|
|
554
|
+
lastRound: txJson.lastRound,
|
|
555
|
+
genesisId: genesisId,
|
|
556
|
+
genesisHash: genesisHash,
|
|
557
|
+
note: txJson.note ? buffer_1.Buffer.from(txJson.note.buffer).toString('utf-8') : undefined,
|
|
558
|
+
};
|
|
559
|
+
}
|
|
560
|
+
/**
|
|
561
|
+
* Accepts a fully signed serialized base64 transaction and broadcasts it on the network.
|
|
562
|
+
* Uses the external node provided by the client
|
|
563
|
+
* @param serializedSignedTransaction
|
|
564
|
+
* @param nodeParams
|
|
565
|
+
*/
|
|
566
|
+
async broadcastTransaction({ serializedSignedTransaction, nodeParams, }) {
|
|
567
|
+
if (!nodeParams) {
|
|
568
|
+
throw new Error('Please provide the details of the algorand node');
|
|
569
|
+
}
|
|
570
|
+
try {
|
|
571
|
+
const txHex = buffer_1.Buffer.from(serializedSignedTransaction, 'base64').toString('hex');
|
|
572
|
+
const algoTx = utils_1.default.toUint8Array(txHex);
|
|
573
|
+
const client = this.getClient(nodeParams.token, nodeParams.baseServer, nodeParams.port);
|
|
574
|
+
return await client.sendRawTransaction(algoTx).do();
|
|
575
|
+
}
|
|
576
|
+
catch (e) {
|
|
577
|
+
throw new Error('Failed to broadcast transaction, error: ' + e.message);
|
|
578
|
+
}
|
|
579
|
+
}
|
|
388
580
|
/**
|
|
389
581
|
* Stellar and Algorand both use keys on the ed25519 curve, but use different encodings.
|
|
390
582
|
* As the HSM doesn't have explicit support to create Algorand addresses, we use the Stellar
|
|
@@ -396,24 +588,50 @@ class Algo extends sdk_core_1.BaseCoin {
|
|
|
396
588
|
* @return {*}
|
|
397
589
|
*/
|
|
398
590
|
stellarAddressToAlgoAddress(addressOrPubKey) {
|
|
591
|
+
// we have an Algorand address
|
|
399
592
|
if (this.isValidAddress(addressOrPubKey)) {
|
|
400
|
-
// we have an Algorand address
|
|
401
593
|
return addressOrPubKey;
|
|
402
594
|
}
|
|
403
|
-
if (!stellar_sdk_1.default.StrKey.isValidEd25519PublicKey(addressOrPubKey)) {
|
|
404
|
-
throw new sdk_core_1.UnexpectedAddressError('Neither an Algorand address nor a stellar pubkey.');
|
|
405
|
-
}
|
|
406
595
|
// we have a stellar key
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
596
|
+
if (stellar_sdk_1.default.StrKey.isValidEd25519PublicKey(addressOrPubKey)) {
|
|
597
|
+
const stellarPub = stellar_sdk_1.default.StrKey.decodeEd25519PublicKey(addressOrPubKey);
|
|
598
|
+
const algoAddress = AlgoLib.algoUtils.encodeAddress(stellarPub);
|
|
599
|
+
if (this.isValidAddress(algoAddress)) {
|
|
600
|
+
return algoAddress;
|
|
601
|
+
}
|
|
602
|
+
throw new sdk_core_1.UnexpectedAddressError('Cannot convert Stellar address to an Algorand address via stellar pubkey.');
|
|
603
|
+
// we have a root pubkey
|
|
411
604
|
}
|
|
412
|
-
|
|
605
|
+
else if (AlgoLib.algoUtils.isValidPublicKey(addressOrPubKey)) {
|
|
606
|
+
const kp = new AlgoLib.KeyPair({ pub: addressOrPubKey });
|
|
607
|
+
const algoAddress = kp.getAddress();
|
|
608
|
+
if (this.isValidAddress(algoAddress)) {
|
|
609
|
+
return algoAddress;
|
|
610
|
+
}
|
|
611
|
+
throw new sdk_core_1.UnexpectedAddressError('Invalid root pubkey.');
|
|
612
|
+
}
|
|
613
|
+
throw new sdk_core_1.UnexpectedAddressError('Neither an Algorand address, a stellar pubkey or a root public key.');
|
|
413
614
|
}
|
|
414
615
|
getBuilder() {
|
|
415
616
|
return new AlgoLib.TransactionBuilderFactory(statics_1.coins.get(this.getBaseChain()));
|
|
416
617
|
}
|
|
618
|
+
/** @inheritDoc */
|
|
619
|
+
auditDecryptedKey({ publicKey, prv, multiSigType }) {
|
|
620
|
+
if (multiSigType === 'tss') {
|
|
621
|
+
throw new Error('Unsupported multiSigType');
|
|
622
|
+
}
|
|
623
|
+
let algoKey;
|
|
624
|
+
try {
|
|
625
|
+
algoKey = new AlgoLib.KeyPair({ prv });
|
|
626
|
+
}
|
|
627
|
+
catch (e) {
|
|
628
|
+
throw new Error('Invalid private key');
|
|
629
|
+
}
|
|
630
|
+
if (publicKey && publicKey !== algoKey.getKeys().pub) {
|
|
631
|
+
throw new Error('Invalid public key');
|
|
632
|
+
}
|
|
633
|
+
return;
|
|
634
|
+
}
|
|
417
635
|
}
|
|
418
636
|
exports.Algo = Algo;
|
|
419
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"algo.js","sourceRoot":"","sources":["../../src/algo.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport * as _ from 'lodash';\nimport { SeedValidator } from './seedValidator';\nimport { coins, CoinFamily } from '@bitgo-beta/statics';\nimport * as AlgoLib from './lib';\nimport {\n  AddressCoinSpecific,\n  BaseCoin,\n  BitGoBase,\n  Ed25519KeyDeriver,\n  InvalidAddressError,\n  InvalidKey,\n  KeyIndices,\n  KeyPair,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  SignedTransaction,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  TokenManagementType,\n  TransactionExplanation,\n  TransactionRecipient,\n  TransactionType,\n  UnexpectedAddressError,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport stellar from 'stellar-sdk';\n\nconst SUPPORTED_ADDRESS_VERSION = 1;\nconst MSIG_THRESHOLD = 2; // m in m-of-n\n\nexport interface AlgoAddressCoinSpecifics extends AddressCoinSpecific {\n  rootAddress: string;\n  bitgoKey: string;\n  bitgoPubKey?: string;\n  addressVersion: number;\n  threshold: number;\n}\n\nexport interface VerifyAlgoAddressOptions extends VerifyAddressOptions {\n  chain: number;\n  index: number;\n  coin: string;\n  wallet: string;\n  coinSpecific: AlgoAddressCoinSpecifics;\n}\n\nexport interface AlgoTransactionExplanation extends TransactionExplanation {\n  memo?: string;\n  type?: string | number;\n  voteKey?: string;\n  selectionKey?: string;\n  voteFirst?: number;\n  voteLast?: number;\n  voteKeyDilution?: number;\n  tokenId?: number;\n  operations?: TransactionOperation[];\n}\n\nexport interface TransactionOperation {\n  type: string;\n  coin: string;\n}\n\nexport interface SignTransactionOptions extends BaseSignTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  prv: string;\n}\n\nexport interface TransactionPrebuild {\n  txHex: string;\n  halfSigned?: {\n    txHex: string;\n  };\n  txInfo: {\n    from: string;\n    to: string;\n    amount: string;\n    fee: number;\n    firstRound: number;\n    lastRound: number;\n    genesisID: string;\n    genesisHash: string;\n    note?: string;\n  };\n  keys: string[];\n  addressVersion: number;\n}\n\nexport interface FullySignedTransaction {\n  txHex: string;\n}\n\nexport interface HalfSignedTransaction {\n  halfSigned: {\n    txHex: string;\n  };\n}\n\nexport interface TransactionFee {\n  fee: string;\n}\nexport interface ExplainTransactionOptions {\n  txHex?: string;\n  halfSigned?: {\n    txHex: string;\n  };\n  publicKeys?: string[];\n  feeInfo: TransactionFee;\n}\n\nexport interface VerifiedTransactionParameters {\n  txHex: string;\n  addressVersion: number;\n  signers: string[];\n  prv: string;\n  isHalfSigned: boolean;\n  numberSigners: number;\n}\n\nexport class Algo extends BaseCoin {\n  readonly ENABLE_TOKEN: TokenManagementType = 'enabletoken';\n  readonly DISABLE_TOKEN: TokenManagementType = 'disabletoken';\n\n  constructor(bitgo: BitGoBase) {\n    super(bitgo);\n  }\n\n  static createInstance(bitgo: BitGoBase): BaseCoin {\n    return new Algo(bitgo);\n  }\n\n  getChain(): string {\n    return 'algo';\n  }\n\n  getBaseChain(): string {\n    return 'algo';\n  }\n\n  getFamily(): string {\n    return 'algo';\n  }\n\n  getFullName(): string {\n    return 'Algorand';\n  }\n\n  getBaseFactor(): number | string {\n    return 1e6;\n  }\n\n  /**\n   * Flag for sending value of 0\n   * @returns {boolean} True if okay to send 0 value, false otherwise\n   */\n  valuelessTransferAllowed(): boolean {\n    return true;\n  }\n\n  /**\n   * Algorand supports account consolidations. These are transfers from the receive addresses\n   * to the main address.\n   */\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  /**\n   * Generate ed25519 key pair\n   *\n   * @param seed\n   * @returns {Object} object with generated pub, prv\n   */\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new AlgoLib.KeyPair({ seed }) : new AlgoLib.KeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n\n    return {\n      pub: keyPair.getAddress(),\n      prv: AlgoLib.algoUtils.encodeSeed(Buffer.from(keyPair.getSigningKey())),\n    };\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin.\n   *\n   * @param {String} pub the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPub(pub: string): boolean {\n    return AlgoLib.algoUtils.isValidAddress(pub);\n  }\n\n  /**\n   * Return boolean indicating whether input is valid seed for the coin\n   * In Algorand, when the private key is encoded as base32 string only the first 32 bytes are taken,\n   * so the encoded value is actually the seed\n   *\n   * @param {String} prv the prv to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPrv(prv: string): boolean {\n    return AlgoLib.algoUtils.isValidSeed(prv);\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin\n   *\n   * @param {String} address the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidAddress(address: string): boolean {\n    return AlgoLib.algoUtils.isValidAddress(address);\n  }\n\n  /**\n   * Sign message with private key\n   *\n   * @param key\n   * @param message\n   */\n  async signMessage(key: KeyPair, message: string | Buffer): Promise<Buffer> {\n    const algoKeypair = new AlgoLib.KeyPair({ prv: key.prv });\n    if (Buffer.isBuffer(message)) {\n      message = message.toString('base64');\n    }\n    return Buffer.from(algoKeypair.signMessage(message));\n  }\n\n  /**\n   * Specifies what key we will need for signing` - Algorand needs the backup, bitgo pubs.\n   */\n  keyIdsForSigning(): number[] {\n    return [KeyIndices.USER, KeyIndices.BACKUP, KeyIndices.BITGO];\n  }\n\n  getTokenNameById(tokenId: number | string): string {\n    const tokenNames = coins.filter((coin) => coin.family === 'algo' && coin.isToken).map(({ name }) => name!);\n    return tokenNames.find((tokenName) => tokenName.split('-')[1] === `${tokenId}`) || 'AlgoToken unknown';\n  }\n\n  /**\n   * Explain/parse transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<AlgoTransactionExplanation | undefined> {\n    const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);\n    if (!txHex || !params.feeInfo) {\n      throw new Error('missing explain tx parameters');\n    }\n\n    const factory = this.getBuilder();\n\n    const txBuilder = factory.from(txHex);\n    const tx = await txBuilder.build();\n    const txJson = tx.toJson();\n\n    if (tx.type === TransactionType.Send) {\n      const outputs: TransactionRecipient[] = [\n        {\n          address: txJson.to,\n          amount: txJson.amount,\n          memo: txJson.note,\n        },\n      ];\n      const operations: TransactionOperation[] = [];\n\n      const isTokenTx = this.isTokenTx(txJson.type);\n      if (isTokenTx) {\n        const type = AlgoLib.algoUtils.getTokenTxType(txJson.amount, txJson.from, txJson.to, txJson.closeRemainderTo);\n        operations.push({\n          type: type,\n          coin: this.getTokenNameById(txJson.tokenId),\n        });\n      }\n\n      const displayOrder = [\n        'id',\n        'outputAmount',\n        'changeAmount',\n        'outputs',\n        'changeOutputs',\n        'fee',\n        'memo',\n        'type',\n        'operations',\n      ];\n\n      const explanationResult: AlgoTransactionExplanation = {\n        displayOrder,\n        id: txJson.id,\n        outputAmount: txJson.amount.toString(),\n        changeAmount: '0',\n        outputs,\n        changeOutputs: [],\n        fee: txJson.fee,\n        memo: txJson.note,\n        type: tx.type.toString(),\n        operations,\n      };\n\n      if (txJson.tokenId) {\n        explanationResult.tokenId = txJson.tokenId;\n      }\n\n      return explanationResult;\n    }\n\n    if (tx.type === TransactionType.WalletInitialization) {\n      const displayOrder = [\n        'id',\n        'fee',\n        'memo',\n        'type',\n        'voteKey',\n        'selectionKey',\n        'voteFirst',\n        'voteLast',\n        'voteKeyDilution',\n      ];\n\n      return {\n        displayOrder,\n        id: txJson.id,\n        outputAmount: '0',\n        changeAmount: '0',\n        outputs: [],\n        changeOutputs: [],\n        fee: txJson.fee,\n        memo: txJson.note,\n        type: tx.type,\n        voteKey: txJson.voteKey,\n        selectionKey: txJson.selectionKey,\n        voteFirst: txJson.voteFirst,\n        voteLast: txJson.voteLast,\n        voteKeyDilution: txJson.voteKeyDilution,\n      };\n    }\n  }\n\n  /**\n   * returns if a tx is a token tx\n   * @param type {string} - tx type\n   * @returns true if it's a token tx\n   */\n  isTokenTx(type: string): boolean {\n    return type === 'axfer';\n  }\n\n  /**\n   * Check if a seed is a valid stellar seed\n   *\n   * @param {String} seed the seed to check\n   * @returns {Boolean} true if the input is a Stellar seed\n   */\n  isStellarSeed(seed: string): boolean {\n    return SeedValidator.isValidEd25519SeedForCoin(seed, CoinFamily.XLM);\n  }\n\n  /**\n   * Convert a stellar seed to an algo seed\n   *\n   * @param {String} seed the seed to convert\n   * @returns {Boolean | null} seed in algo encoding\n   */\n  convertFromStellarSeed(seed: string): string | null {\n    // assume this is a trust custodial seed if its a valid ed25519 prv\n    if (!this.isStellarSeed(seed) || SeedValidator.hasCompetingSeedFormats(seed)) {\n      return null;\n    }\n\n    if (SeedValidator.isValidEd25519SeedForCoin(seed, CoinFamily.XLM)) {\n      return AlgoLib.algoUtils.convertFromStellarSeed(seed);\n    }\n\n    return null;\n  }\n\n  verifySignTransactionParams(params: SignTransactionOptions): VerifiedTransactionParameters {\n    const prv = params.prv;\n    const addressVersion = params.txPrebuild.addressVersion;\n    let isHalfSigned = false;\n\n    // it's possible this tx was already signed - take the halfSigned\n    // txHex if it is\n    let txHex = params.txPrebuild.txHex;\n    if (params.txPrebuild.halfSigned) {\n      isHalfSigned = true;\n      txHex = params.txPrebuild.halfSigned.txHex;\n    }\n\n    if (_.isUndefined(txHex)) {\n      throw new Error('missing txPrebuild parameter');\n    }\n\n    if (!_.isString(txHex)) {\n      throw new Error(`txPrebuild must be an object, got type ${typeof txHex}`);\n    }\n\n    if (_.isUndefined(prv)) {\n      throw new Error('missing prv parameter to sign transaction');\n    }\n\n    if (!_.isString(prv)) {\n      throw new Error(`prv must be a string, got type ${typeof prv}`);\n    }\n\n    if (!_.has(params.txPrebuild, 'keys')) {\n      throw new Error('missing public keys parameter to sign transaction');\n    }\n\n    if (!_.isNumber(addressVersion)) {\n      throw new Error('missing addressVersion parameter to sign transaction');\n    }\n\n    const signers = params.txPrebuild.keys.map((key) => {\n      // if we are receiving addresses do not try to convert them\n      if (!AlgoLib.algoUtils.isValidAddress(key)) {\n        return AlgoLib.algoUtils.publicKeyToAlgoAddress(AlgoLib.algoUtils.toUint8Array(key));\n      }\n      return key;\n    });\n    // TODO(https://bitgoinc.atlassian.net/browse/STLX-6067): fix the number of signers using\n    // should be similar to other coins implementation\n    // If we have a number with digits to eliminate them without taking any rounding criteria.\n    const numberSigners = Math.trunc(signers.length / 2) + 1;\n    return { txHex, addressVersion, signers, prv, isHalfSigned, numberSigners };\n  }\n\n  /**\n   * Assemble keychain and half-sign prebuilt transaction\n   *\n   * @param params\n   * @param params.txPrebuild {TransactionPrebuild} prebuild object returned by platform\n   * @param params.prv {String} user prv\n   * @returns {Promise<SignedTransaction>}\n   */\n  async signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    const { txHex, signers, prv, isHalfSigned, numberSigners } = this.verifySignTransactionParams(params);\n    const factory = this.getBuilder();\n    const txBuilder = factory.from(txHex);\n    txBuilder.numberOfRequiredSigners(numberSigners);\n    txBuilder.sign({ key: prv });\n    txBuilder.setSigners(signers);\n    const transaction = await txBuilder.build();\n    if (!transaction) {\n      throw new Error('Invalid transaction');\n    }\n    const signedTxHex = Buffer.from(transaction.toBroadcastFormat()).toString('base64');\n    if (numberSigners === 1) {\n      return { txHex: signedTxHex };\n    } else if (isHalfSigned) {\n      return { txHex: signedTxHex };\n    } else {\n      return { halfSigned: { txHex: signedTxHex } };\n    }\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n\n  /**\n   * Check if address can be used to send funds.\n   *\n   * @param params.address address to validate\n   * @param params.keychains public keys to generate the wallet\n   */\n  async isWalletAddress(params: VerifyAlgoAddressOptions): Promise<boolean> {\n    const {\n      address,\n      keychains,\n      coinSpecific: { bitgoPubKey },\n    } = params;\n\n    if (!this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n\n    if (!keychains) {\n      throw new Error('missing required param keychains');\n    }\n\n    const effectiveKeychain = bitgoPubKey ? keychains.slice(0, -1).concat([{ pub: bitgoPubKey }]) : keychains;\n    const pubKeys = effectiveKeychain.map((key) => this.stellarAddressToAlgoAddress(key.pub));\n\n    if (!pubKeys.every((pubKey) => this.isValidPub(pubKey))) {\n      throw new InvalidKey('invalid public key');\n    }\n\n    const rootAddress = AlgoLib.algoUtils.multisigAddress(SUPPORTED_ADDRESS_VERSION, MSIG_THRESHOLD, pubKeys);\n\n    return rootAddress === address;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    return true;\n  }\n\n  /** @inheritDoc */\n  deriveKeyWithSeed({ key, seed }: { key: string; seed: string }): { derivationPath: string; key: string } {\n    const derivationPathInput = utxolib.crypto.hash256(Buffer.from(seed, 'utf8')).toString('hex');\n    const derivationPathParts = [\n      999999,\n      parseInt(derivationPathInput.slice(0, 7), 16),\n      parseInt(derivationPathInput.slice(7, 14), 16),\n    ];\n    const derivationPath = 'm/' + derivationPathParts.map((part) => `${part}'`).join('/');\n    const derivedKey = Ed25519KeyDeriver.derivePath(derivationPath, key).key;\n    const keypair = new AlgoLib.KeyPair({ seed: derivedKey });\n    return {\n      key: keypair.getAddress(),\n      derivationPath,\n    };\n  }\n\n  decodeTx(txn: Buffer): unknown {\n    return AlgoLib.algoUtils.decodeAlgoTxn(txn);\n  }\n\n  getAddressFromPublicKey(pubKey: Uint8Array): string {\n    return AlgoLib.algoUtils.publicKeyToAlgoAddress(pubKey);\n  }\n\n  /**\n   * Stellar and Algorand both use keys on the ed25519 curve, but use different encodings.\n   * As the HSM doesn't have explicit support to create Algorand addresses, we use the Stellar\n   * keys and re-encode them to the Algorand encoding.\n   *\n   * This method should only be used when creating Algorand custodial wallets reusing Stellar keys.\n   *\n   * @param {string} addressOrPubKey a Stellar pubkey or Algorand address\n   * @return {*}\n   */\n  private stellarAddressToAlgoAddress(addressOrPubKey: string): string {\n    if (this.isValidAddress(addressOrPubKey)) {\n      // we have an Algorand address\n      return addressOrPubKey;\n    }\n\n    if (!stellar.StrKey.isValidEd25519PublicKey(addressOrPubKey)) {\n      throw new UnexpectedAddressError('Neither an Algorand address nor a stellar pubkey.');\n    }\n\n    // we have a stellar key\n    const stellarPub = stellar.StrKey.decodeEd25519PublicKey(addressOrPubKey);\n    const algoAddress = AlgoLib.algoUtils.encodeAddress(stellarPub);\n\n    if (!this.isValidAddress(algoAddress)) {\n      throw new UnexpectedAddressError('Cannot convert Stellar address to an Algorand address via pubkey.');\n    }\n\n    return algoAddress;\n  }\n\n  private getBuilder(): AlgoLib.TransactionBuilderFactory {\n    return new AlgoLib.TransactionBuilderFactory(coins.get(this.getBaseChain()));\n  }\n}\n"]}
|
|
637
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"algo.js","sourceRoot":"","sources":["../../src/algo.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as _ from 'lodash';\nimport { SeedValidator } from './seedValidator';\nimport { coins, CoinFamily } from '@bitgo-beta/statics';\nimport * as AlgoLib from './lib';\nimport {\n  AddressCoinSpecific,\n  BaseBroadcastTransactionOptions,\n  BaseBroadcastTransactionResult,\n  BaseCoin,\n  BitGoBase,\n  InvalidAddressError,\n  InvalidKey,\n  KeyIndices,\n  KeyPair,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  SignedTransaction,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  TokenManagementType,\n  TransactionExplanation,\n  TransactionRecipient,\n  TransactionType,\n  UnexpectedAddressError,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n  NotSupported,\n  MultisigType,\n  multisigTypes,\n  AuditDecryptedKeyParams,\n} from '@bitgo-beta/sdk-core';\nimport stellar from 'stellar-sdk';\nimport BigNumber from 'bignumber.js';\nimport Utils from './lib/utils';\nimport { TxData } from './lib/ifaces';\nimport * as algosdk from 'algosdk';\nimport {\n  MAINNET_GENESIS_HASH,\n  MAINNET_GENESIS_ID,\n  TESTNET_GENESIS_HASH,\n  TESTNET_GENESIS_ID,\n} from './lib/transactionBuilder';\nimport { Buffer } from 'buffer';\n\nconst SUPPORTED_ADDRESS_VERSION = 1;\nconst MSIG_THRESHOLD = 2; // m in m-of-n\n\nexport interface AlgoAddressCoinSpecifics extends AddressCoinSpecific {\n  rootAddress: string;\n  bitgoKey: string;\n  bitgoPubKey?: string;\n  addressVersion: number;\n  threshold: number;\n}\n\nexport interface VerifyAlgoAddressOptions extends VerifyAddressOptions {\n  chain: number;\n  index: number;\n  coin: string;\n  wallet: string;\n  coinSpecific: AlgoAddressCoinSpecifics;\n}\n\nexport interface AlgoTransactionExplanation extends TransactionExplanation {\n  memo?: string;\n  type?: string | number;\n  voteKey?: string;\n  selectionKey?: string;\n  voteFirst?: number;\n  voteLast?: number;\n  voteKeyDilution?: number;\n  tokenId?: number;\n  operations?: TransactionOperation[];\n}\n\nexport interface TransactionOperation {\n  type: string;\n  coin: string;\n}\n\nexport interface SignTransactionOptions extends BaseSignTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  prv: string;\n}\n\nexport interface TransactionPrebuild {\n  txHex: string;\n  halfSigned?: {\n    txHex: string;\n  };\n  txInfo: {\n    from: string;\n    to: string;\n    amount: string;\n    fee: number;\n    firstRound: number;\n    lastRound: number;\n    genesisID: string;\n    genesisHash: string;\n    note?: string;\n  };\n  keys: string[];\n  addressVersion: number;\n}\n\nexport interface FullySignedTransaction {\n  txHex: string;\n}\n\nexport interface HalfSignedTransaction {\n  halfSigned: {\n    txHex: string;\n  };\n}\n\nexport interface TransactionFee {\n  fee: string;\n}\nexport interface ExplainTransactionOptions {\n  txHex?: string;\n  halfSigned?: {\n    txHex: string;\n  };\n  publicKeys?: string[];\n  feeInfo: TransactionFee;\n}\n\ninterface NodeParams {\n  token: string;\n  baseServer: string;\n  port: number;\n}\n\nexport interface VerifiedTransactionParameters {\n  txHex: string;\n  addressVersion: number;\n  signers: string[];\n  prv: string;\n  isHalfSigned: boolean;\n  numberSigners: number;\n}\n\nexport interface RecoveryOptions {\n  backupKey: string;\n  userKey: string;\n  rootAddress: string;\n  recoveryDestination: string;\n  bitgoKey: string;\n  walletPassphrase?: string;\n  fee: number;\n  firstRound?: number;\n  note?: string;\n  nodeParams: NodeParams;\n}\n\ninterface RecoveryInfo {\n  id: string;\n  tx: string;\n  coin: string;\n  fee: number;\n  firstRound: number;\n  lastRound: number;\n  genesisId: string;\n  genesisHash: string;\n  note?: string;\n}\n\nexport interface OfflineVaultTxInfo {\n  txHex: string;\n  userKey: string;\n  backupKey: string;\n  bitgoKey: string;\n  type?: string;\n  address: string;\n  coin: string;\n  feeInfo: number;\n  amount: string;\n  firstRound: number;\n  lastRound: number;\n  genesisId: string;\n  genesisHash: string;\n  note?: string;\n  addressVersion: number;\n  keys: string[];\n}\n\nexport interface BroadcastTransactionOptions extends BaseBroadcastTransactionOptions {\n  nodeParams: NodeParams;\n}\n\nexport class Algo extends BaseCoin {\n  readonly ENABLE_TOKEN: TokenManagementType = 'enabletoken';\n  readonly DISABLE_TOKEN: TokenManagementType = 'disabletoken';\n\n  constructor(bitgo: BitGoBase) {\n    super(bitgo);\n  }\n\n  static createInstance(bitgo: BitGoBase): BaseCoin {\n    return new Algo(bitgo);\n  }\n\n  getChain(): string {\n    return 'algo';\n  }\n\n  getBaseChain(): string {\n    return 'algo';\n  }\n\n  getFamily(): string {\n    return 'algo';\n  }\n\n  getFullName(): string {\n    return 'Algorand';\n  }\n\n  getBaseFactor(): number | string {\n    return 1e6;\n  }\n\n  /**\n   * Flag for sending value of 0\n   * @returns {boolean} True if okay to send 0 value, false otherwise\n   */\n  valuelessTransferAllowed(): boolean {\n    return true;\n  }\n\n  /**\n   * Algorand supports account consolidations. These are transfers from the receive addresses\n   * to the main address.\n   */\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  /** inheritdoc */\n  deriveKeyWithSeed(): { derivationPath: string; key: string } {\n    throw new NotSupported('method deriveKeyWithSeed not supported for eddsa curve');\n  }\n\n  /** inheritdoc */\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new AlgoLib.KeyPair({ seed }) : new AlgoLib.KeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n\n    return {\n      pub: keyPair.getAddress(),\n      prv: AlgoLib.algoUtils.encodeSeed(Buffer.from(keyPair.getSigningKey())),\n    };\n  }\n\n  /** inheritdoc */\n  generateRootKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new AlgoLib.KeyPair({ seed }) : new AlgoLib.KeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n    return { prv: keys.prv + keys.pub, pub: keys.pub };\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin.\n   *\n   * @param {String} pub the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPub(pub: string): boolean {\n    return AlgoLib.algoUtils.isValidAddress(pub) || AlgoLib.algoUtils.isValidPublicKey(pub);\n  }\n\n  /**\n   * Return boolean indicating whether input is valid seed for the coin\n   * In Algorand, when the private key is encoded as base32 string only the first 32 bytes are taken,\n   * so the encoded value is actually the seed\n   *\n   * @param {String} prv the prv to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPrv(prv: string): boolean {\n    return AlgoLib.algoUtils.isValidSeed(prv) || AlgoLib.algoUtils.isValidPrivateKey(prv);\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin\n   *\n   * @param {String} address the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidAddress(address: string): boolean {\n    return AlgoLib.algoUtils.isValidAddress(address);\n  }\n\n  /**\n   * Sign message with private key\n   *\n   * @param key\n   * @param message\n   */\n  async signMessage(key: KeyPair, message: string | Buffer): Promise<Buffer> {\n    const algoKeypair = new AlgoLib.KeyPair({ prv: key.prv });\n    if (Buffer.isBuffer(message)) {\n      message = message.toString('base64');\n    }\n    return Buffer.from(algoKeypair.signMessage(message));\n  }\n\n  /**\n   * Specifies what key we will need for signing` - Algorand needs the backup, bitgo pubs.\n   */\n  keyIdsForSigning(): number[] {\n    return [KeyIndices.USER, KeyIndices.BACKUP, KeyIndices.BITGO];\n  }\n\n  getTokenNameById(tokenId: number | string): string {\n    const tokenNames = coins.filter((coin) => coin.family === 'algo' && coin.isToken).map(({ name }) => name!);\n    return tokenNames.find((tokenName) => tokenName.split('-')[1] === `${tokenId}`) || 'AlgoToken unknown';\n  }\n\n  /**\n   * Explain/parse transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<AlgoTransactionExplanation | undefined> {\n    const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);\n    if (!txHex || !params.feeInfo) {\n      throw new Error('missing explain tx parameters');\n    }\n\n    const factory = this.getBuilder();\n\n    const txBuilder = factory.from(txHex);\n    const tx = await txBuilder.build();\n    const txJson = tx.toJson();\n\n    if (tx.type === TransactionType.Send) {\n      const outputs: TransactionRecipient[] = [\n        {\n          address: txJson.to,\n          amount: txJson.amount,\n          memo: txJson.note,\n        },\n      ];\n      const operations: TransactionOperation[] = [];\n\n      const isTokenTx = this.isTokenTx(txJson.type);\n      if (isTokenTx) {\n        const type = AlgoLib.algoUtils.getTokenTxType(txJson.amount, txJson.from, txJson.to, txJson.closeRemainderTo);\n        operations.push({\n          type: type,\n          coin: this.getTokenNameById(txJson.tokenId),\n        });\n      }\n\n      const displayOrder = [\n        'id',\n        'outputAmount',\n        'changeAmount',\n        'outputs',\n        'changeOutputs',\n        'fee',\n        'memo',\n        'type',\n        'operations',\n      ];\n\n      const explanationResult: AlgoTransactionExplanation = {\n        displayOrder,\n        id: txJson.id,\n        outputAmount: txJson.amount.toString(),\n        changeAmount: '0',\n        outputs,\n        changeOutputs: [],\n        fee: txJson.fee,\n        memo: txJson.note,\n        type: tx.type.toString(),\n        operations,\n      };\n\n      if (txJson.tokenId) {\n        explanationResult.tokenId = txJson.tokenId;\n      }\n\n      return explanationResult;\n    }\n\n    if (tx.type === TransactionType.WalletInitialization) {\n      const displayOrder = [\n        'id',\n        'fee',\n        'memo',\n        'type',\n        'voteKey',\n        'selectionKey',\n        'voteFirst',\n        'voteLast',\n        'voteKeyDilution',\n      ];\n\n      return {\n        displayOrder,\n        id: txJson.id,\n        outputAmount: '0',\n        changeAmount: '0',\n        outputs: [],\n        changeOutputs: [],\n        fee: txJson.fee,\n        memo: txJson.note,\n        type: tx.type,\n        voteKey: txJson.voteKey,\n        selectionKey: txJson.selectionKey,\n        voteFirst: txJson.voteFirst,\n        voteLast: txJson.voteLast,\n        voteKeyDilution: txJson.voteKeyDilution,\n      };\n    }\n  }\n\n  /**\n   * returns if a tx is a token tx\n   * @param type {string} - tx type\n   * @returns true if it's a token tx\n   */\n  isTokenTx(type: string): boolean {\n    return type === 'axfer';\n  }\n\n  /**\n   * Check if a seed is a valid stellar seed\n   *\n   * @param {String} seed the seed to check\n   * @returns {Boolean} true if the input is a Stellar seed\n   */\n  isStellarSeed(seed: string): boolean {\n    return SeedValidator.isValidEd25519SeedForCoin(seed, CoinFamily.XLM);\n  }\n\n  /**\n   * Convert a stellar seed to an algo seed\n   *\n   * @param {String} seed the seed to convert\n   * @returns {Boolean | null} seed in algo encoding\n   */\n  convertFromStellarSeed(seed: string): string | null {\n    // assume this is a trust custodial seed if its a valid ed25519 prv\n    if (!this.isStellarSeed(seed) || SeedValidator.hasCompetingSeedFormats(seed)) {\n      return null;\n    }\n\n    if (SeedValidator.isValidEd25519SeedForCoin(seed, CoinFamily.XLM)) {\n      return AlgoLib.algoUtils.convertFromStellarSeed(seed);\n    }\n\n    return null;\n  }\n\n  verifySignTransactionParams(params: SignTransactionOptions): VerifiedTransactionParameters {\n    const prv = params.prv;\n    const addressVersion = params.txPrebuild.addressVersion;\n    let isHalfSigned = false;\n\n    // it's possible this tx was already signed - take the halfSigned\n    // txHex if it is\n    let txHex = params.txPrebuild.txHex;\n    if (params.txPrebuild.halfSigned) {\n      isHalfSigned = true;\n      txHex = params.txPrebuild.halfSigned.txHex;\n    }\n\n    if (_.isUndefined(txHex)) {\n      throw new Error('missing txPrebuild parameter');\n    }\n\n    if (!_.isString(txHex)) {\n      throw new Error(`txPrebuild must be an object, got type ${typeof txHex}`);\n    }\n\n    if (_.isUndefined(prv)) {\n      throw new Error('missing prv parameter to sign transaction');\n    }\n\n    if (!_.isString(prv)) {\n      throw new Error(`prv must be a string, got type ${typeof prv}`);\n    }\n\n    if (!_.has(params.txPrebuild, 'keys')) {\n      throw new Error('missing public keys parameter to sign transaction');\n    }\n\n    if (!_.isNumber(addressVersion)) {\n      throw new Error('missing addressVersion parameter to sign transaction');\n    }\n\n    const signers = params.txPrebuild.keys.map((key) => {\n      // if we are receiving addresses do not try to convert them\n      if (!AlgoLib.algoUtils.isValidAddress(key)) {\n        return AlgoLib.algoUtils.publicKeyToAlgoAddress(AlgoLib.algoUtils.toUint8Array(key));\n      }\n      return key;\n    });\n    // TODO(https://bitgoinc.atlassian.net/browse/STLX-6067): fix the number of signers using\n    // should be similar to other coins implementation\n    // If we have a number with digits to eliminate them without taking any rounding criteria.\n    const numberSigners = Math.trunc(signers.length / 2) + 1;\n    return { txHex, addressVersion, signers, prv, isHalfSigned, numberSigners };\n  }\n\n  /**\n   * Assemble keychain and half-sign prebuilt transaction\n   *\n   * @param params\n   * @param params.txPrebuild {TransactionPrebuild} prebuild object returned by platform\n   * @param params.prv {String} user prv\n   * @returns {Promise<SignedTransaction>}\n   */\n  async signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    const { txHex, signers, prv, isHalfSigned, numberSigners } = this.verifySignTransactionParams(params);\n    const factory = this.getBuilder();\n    const txBuilder = factory.from(txHex);\n    txBuilder.numberOfRequiredSigners(numberSigners);\n    txBuilder.sign({ key: prv });\n    txBuilder.setSigners(signers);\n    const transaction = await txBuilder.build();\n    if (!transaction) {\n      throw new Error('Invalid transaction');\n    }\n    const signedTxHex = Buffer.from(transaction.toBroadcastFormat()).toString('base64');\n    if (numberSigners === 1) {\n      return { txHex: signedTxHex };\n    } else if (isHalfSigned) {\n      return { txHex: signedTxHex };\n    } else {\n      return { halfSigned: { txHex: signedTxHex } };\n    }\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n\n  /**\n   * Check if address can be used to send funds.\n   *\n   * @param params.address address to validate\n   * @param params.keychains public keys to generate the wallet\n   */\n  async isWalletAddress(params: VerifyAlgoAddressOptions): Promise<boolean> {\n    const {\n      address,\n      keychains,\n      coinSpecific: { bitgoPubKey },\n    } = params;\n\n    if (!this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n\n    if (!keychains) {\n      throw new Error('missing required param keychains');\n    }\n\n    const effectiveKeychain = bitgoPubKey ? keychains.slice(0, -1).concat([{ pub: bitgoPubKey }]) : keychains;\n    const pubKeys = effectiveKeychain.map((key) => this.stellarAddressToAlgoAddress(key.pub));\n\n    if (!pubKeys.every((pubKey) => this.isValidPub(pubKey))) {\n      throw new InvalidKey('invalid public key');\n    }\n\n    const rootAddress = AlgoLib.algoUtils.multisigAddress(SUPPORTED_ADDRESS_VERSION, MSIG_THRESHOLD, pubKeys);\n\n    return rootAddress === address;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    return true;\n  }\n\n  decodeTx(txn: Buffer): unknown {\n    return AlgoLib.algoUtils.decodeAlgoTxn(txn);\n  }\n\n  getAddressFromPublicKey(pubKey: Uint8Array): string {\n    return AlgoLib.algoUtils.publicKeyToAlgoAddress(pubKey);\n  }\n\n  supportsDeriveKeyWithSeed(): boolean {\n    return false;\n  }\n\n  /** inherited doc */\n  getDefaultMultisigType(): MultisigType {\n    return multisigTypes.onchain;\n  }\n\n  /**\n   * Gets config for how token enablements work for this coin\n   * @returns\n   *    requiresTokenEnablement: True if tokens need to be enabled for this coin\n   *    supportsMultipleTokenEnablements: True if multiple tokens can be enabled in one transaction\n   */\n  getTokenEnablementConfig() {\n    return {\n      requiresTokenEnablement: true,\n      supportsMultipleTokenEnablements: false,\n    };\n  }\n\n  /**\n   * Gets the balance of the root address in base units of algo\n   * Eg. If balance is 1 Algo, this returns 1*10^6\n   * @param rootAddress\n   * @param client\n   */\n  async getAccountBalance(rootAddress: string, client: algosdk.Algodv2): Promise<number> {\n    const accountInformation = await client.accountInformation(rootAddress).do();\n    // Extract the balance from the account information\n    return accountInformation.amount;\n  }\n\n  /**\n   * Returns the Algo client for the given token, baseServer and port\n   * Used to interact with the Algo network\n   */\n  getClient(token: string, baseServer: string, port: number): algosdk.Algodv2 {\n    return new algosdk.Algodv2(token, baseServer, port);\n  }\n\n  public async recover(params: RecoveryOptions): Promise<RecoveryInfo | OfflineVaultTxInfo> {\n    const isUnsignedSweep = this.isValidPub(params.userKey) && this.isValidPub(params.backupKey);\n\n    if (!params.nodeParams) {\n      throw new Error('Please provide the details of an ALGO node to use for recovery');\n    }\n\n    // Validate the root address\n    if (!this.isValidAddress(params.rootAddress)) {\n      throw new Error('invalid rootAddress, got: ' + params.rootAddress);\n    }\n\n    // Validate the destination address\n    if (!this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination, got: ' + params.recoveryDestination);\n    }\n\n    if (params.firstRound && new BigNumber(params.firstRound).isNegative()) {\n      throw new Error('first round needs to be a positive value');\n    }\n\n    const genesisId = this.bitgo.getEnv() === 'prod' ? MAINNET_GENESIS_ID : TESTNET_GENESIS_ID;\n    const genesisHash = this.bitgo.getEnv() === 'prod' ? MAINNET_GENESIS_HASH : TESTNET_GENESIS_HASH;\n\n    Utils.validateBase64(genesisHash);\n\n    if (!isUnsignedSweep && !params.walletPassphrase) {\n      throw new Error('walletPassphrase is required for non-bitgo recovery');\n    }\n\n    const factory = new AlgoLib.TransactionBuilderFactory(coins.get('algo'));\n    const txBuilder = factory.getTransferBuilder();\n\n    let userPrv: string | undefined;\n    let backupPrv: string | undefined;\n    if (!isUnsignedSweep) {\n      if (!params.bitgoKey) {\n        throw new Error('bitgo public key from the keyCard is required for non-bitgo recovery');\n      }\n      try {\n        userPrv = this.bitgo.decrypt({ input: params.userKey, password: params.walletPassphrase });\n        backupPrv = this.bitgo.decrypt({ input: params.backupKey, password: params.walletPassphrase });\n        const userKeyAddress = Utils.privateKeyToAlgoAddress(userPrv);\n        const backupKeyAddress = Utils.privateKeyToAlgoAddress(backupPrv);\n        txBuilder.numberOfRequiredSigners(2).setSigners([userKeyAddress, backupKeyAddress, params.bitgoKey]);\n      } catch (e) {\n        throw new Error(\n          'unable to decrypt userKey or backupKey with the walletPassphrase provided, got error: ' + e.message\n        );\n      }\n    }\n\n    const client = this.getClient(params.nodeParams.token, params.nodeParams.baseServer, params.nodeParams.port);\n    const nativeBalance = await this.getAccountBalance(params.rootAddress, client);\n\n    // Algorand accounts require a min. balance of 1 ALGO\n    const MIN_MICROALGOS_BALANCE = 100000;\n    const spendableAmount = new BigNumber(nativeBalance).minus(params.fee).minus(MIN_MICROALGOS_BALANCE).toNumber();\n\n    if (new BigNumber(spendableAmount).isZero() || new BigNumber(spendableAmount).isLessThanOrEqualTo(params.fee)) {\n      throw new Error(\n        'Insufficient balance to recover, got balance: ' +\n          nativeBalance +\n          ' fee: ' +\n          params.fee +\n          ' min account balance: ' +\n          MIN_MICROALGOS_BALANCE\n      );\n    }\n\n    let latestRound: number | undefined;\n    if (!params.firstRound) {\n      latestRound = await client\n        .status()\n        .do()\n        .then((status) => status['last-round']);\n    }\n\n    const firstRound = !params.firstRound ? latestRound : params.firstRound;\n    if (!firstRound) {\n      throw new Error('Unable to fetch the latest round from the node. Please provide the firstRound or try again.');\n    }\n    const LAST_ROUND_BUFFER = 1000;\n    const lastRound = firstRound + LAST_ROUND_BUFFER;\n\n    txBuilder\n      .fee({ fee: params.fee.toString() })\n      .isFlatFee(true)\n      .sender({\n        address: params.rootAddress,\n      })\n      .to({\n        address: params.recoveryDestination,\n      })\n      .amount(spendableAmount)\n      .genesisId(genesisId)\n      .genesisHash(genesisHash)\n      .firstRound(new BigNumber(firstRound).toNumber())\n      .lastRound(new BigNumber(lastRound).toNumber());\n\n    if (params.note) {\n      const note = new Uint8Array(Buffer.from(params.note, 'utf-8'));\n      txBuilder.note(note);\n    }\n\n    // Cold wallet, offline vault\n    if (isUnsignedSweep) {\n      const tx = await txBuilder.build();\n      const txJson = tx.toJson() as TxData;\n\n      return {\n        txHex: Buffer.from(tx.toBroadcastFormat()).toString('hex'),\n        type: txJson.type,\n        userKey: params.userKey,\n        backupKey: params.backupKey,\n        bitgoKey: params.bitgoKey,\n        address: params.rootAddress,\n        coin: this.getChain(),\n        feeInfo: txJson.fee,\n        amount: txJson.amount ?? nativeBalance.toString(),\n        firstRound: txJson.firstRound,\n        lastRound: txJson.lastRound,\n        genesisId: genesisId,\n        genesisHash: genesisHash,\n        note: txJson.note ? Buffer.from(txJson.note.buffer).toString('utf-8') : undefined,\n        keys: [params.userKey, params.backupKey, params.bitgoKey],\n        addressVersion: 1,\n      };\n    }\n\n    // Non-bitgo Recovery (Hot wallets)\n    txBuilder.sign({ key: userPrv });\n    txBuilder.sign({ key: backupPrv });\n\n    const tx = await txBuilder.build();\n    const txJson = tx.toJson() as TxData;\n\n    return {\n      tx: Buffer.from(tx.toBroadcastFormat()).toString('base64'),\n      id: txJson.id,\n      coin: this.getChain(),\n      fee: txJson.fee,\n      firstRound: txJson.firstRound,\n      lastRound: txJson.lastRound,\n      genesisId: genesisId,\n      genesisHash: genesisHash,\n      note: txJson.note ? Buffer.from(txJson.note.buffer).toString('utf-8') : undefined,\n    };\n  }\n\n  /**\n   * Accepts a fully signed serialized base64 transaction and broadcasts it on the network.\n   * Uses the external node provided by the client\n   * @param serializedSignedTransaction\n   * @param nodeParams\n   */\n  async broadcastTransaction({\n    serializedSignedTransaction,\n    nodeParams,\n  }: BroadcastTransactionOptions): Promise<BaseBroadcastTransactionResult> {\n    if (!nodeParams) {\n      throw new Error('Please provide the details of the algorand node');\n    }\n    try {\n      const txHex = Buffer.from(serializedSignedTransaction, 'base64').toString('hex');\n      const algoTx = Utils.toUint8Array(txHex);\n      const client = this.getClient(nodeParams.token, nodeParams.baseServer, nodeParams.port);\n\n      return await client.sendRawTransaction(algoTx).do();\n    } catch (e) {\n      throw new Error('Failed to broadcast transaction, error: ' + e.message);\n    }\n  }\n\n  /**\n   * Stellar and Algorand both use keys on the ed25519 curve, but use different encodings.\n   * As the HSM doesn't have explicit support to create Algorand addresses, we use the Stellar\n   * keys and re-encode them to the Algorand encoding.\n   *\n   * This method should only be used when creating Algorand custodial wallets reusing Stellar keys.\n   *\n   * @param {string} addressOrPubKey a Stellar pubkey or Algorand address\n   * @return {*}\n   */\n  private stellarAddressToAlgoAddress(addressOrPubKey: string): string {\n    // we have an Algorand address\n    if (this.isValidAddress(addressOrPubKey)) {\n      return addressOrPubKey;\n    }\n\n    // we have a stellar key\n    if (stellar.StrKey.isValidEd25519PublicKey(addressOrPubKey)) {\n      const stellarPub = stellar.StrKey.decodeEd25519PublicKey(addressOrPubKey);\n      const algoAddress = AlgoLib.algoUtils.encodeAddress(stellarPub);\n      if (this.isValidAddress(algoAddress)) {\n        return algoAddress;\n      }\n      throw new UnexpectedAddressError('Cannot convert Stellar address to an Algorand address via stellar pubkey.');\n      // we have a root pubkey\n    } else if (AlgoLib.algoUtils.isValidPublicKey(addressOrPubKey)) {\n      const kp = new AlgoLib.KeyPair({ pub: addressOrPubKey });\n      const algoAddress = kp.getAddress();\n      if (this.isValidAddress(algoAddress)) {\n        return algoAddress;\n      }\n      throw new UnexpectedAddressError('Invalid root pubkey.');\n    }\n\n    throw new UnexpectedAddressError('Neither an Algorand address, a stellar pubkey or a root public key.');\n  }\n\n  private getBuilder(): AlgoLib.TransactionBuilderFactory {\n    return new AlgoLib.TransactionBuilderFactory(coins.get(this.getBaseChain()));\n  }\n\n  /** @inheritDoc */\n  auditDecryptedKey({ publicKey, prv, multiSigType }: AuditDecryptedKeyParams) {\n    if (multiSigType === 'tss') {\n      throw new Error('Unsupported multiSigType');\n    }\n\n    let algoKey;\n    try {\n      algoKey = new AlgoLib.KeyPair({ prv });\n    } catch (e) {\n      throw new Error('Invalid private key');\n    }\n    if (publicKey && publicKey !== algoKey.getKeys().pub) {\n      throw new Error('Invalid public key');\n    }\n\n    return;\n  }\n}\n"]}
|
package/dist/src/algoToken.d.ts
CHANGED
|
@@ -11,7 +11,7 @@ export declare class AlgoToken extends Algo {
|
|
|
11
11
|
private readonly _code;
|
|
12
12
|
constructor(bitgo: BitGoBase, tokenConfig: AlgoTokenConfig);
|
|
13
13
|
static createTokenConstructor(config: AlgoTokenConfig): CoinConstructor;
|
|
14
|
-
static createTokenConstructors(): NamedCoinConstructor[];
|
|
14
|
+
static createTokenConstructors(tokenConfigs?: AlgoTokenConfig[]): NamedCoinConstructor[];
|
|
15
15
|
get type(): string;
|
|
16
16
|
get name(): string;
|
|
17
17
|
get coin(): string;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"algoToken.d.ts","sourceRoot":"","sources":["../../src/algoToken.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAC9B,OAAO,EAAE,SAAS,EAAgB,eAAe,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AACtG,OAAO,EAAE,eAAe,EAAuB,MAAM,qBAAqB,CAAC;AAE3E,qBAAa,SAAU,SAAQ,IAAI;IACjC,MAAM,CAAC,QAAQ,CAAC,gBAAgB,SAAqC;IACrE,MAAM,CAAC,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAW;IAC/C,SAAgB,WAAW,EAAE,eAAe,CAAC;IAC7C,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAS;gBAEnB,KAAK,EAAE,SAAS,EAAE,WAAW,EAAE,eAAe;IAkB1D,MAAM,CAAC,sBAAsB,CAAC,MAAM,EAAE,eAAe,GAAG,eAAe;IAIvE,MAAM,CAAC,uBAAuB,
|
|
1
|
+
{"version":3,"file":"algoToken.d.ts","sourceRoot":"","sources":["../../src/algoToken.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAC9B,OAAO,EAAE,SAAS,EAAgB,eAAe,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AACtG,OAAO,EAAE,eAAe,EAAuB,MAAM,qBAAqB,CAAC;AAE3E,qBAAa,SAAU,SAAQ,IAAI;IACjC,MAAM,CAAC,QAAQ,CAAC,gBAAgB,SAAqC;IACrE,MAAM,CAAC,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAW;IAC/C,SAAgB,WAAW,EAAE,eAAe,CAAC;IAC7C,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAS;gBAEnB,KAAK,EAAE,SAAS,EAAE,WAAW,EAAE,eAAe;IAkB1D,MAAM,CAAC,sBAAsB,CAAC,MAAM,EAAE,eAAe,GAAG,eAAe;IAIvE,MAAM,CAAC,uBAAuB,CAAC,YAAY,GAAE,eAAe,EAAwB,GAAG,oBAAoB,EAAE;IAW7G,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,IAAI,OAAO,IAAI,MAAM,CAEpB;IAED,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,IAAI,MAAM,IAAI,MAAM,GAAG,SAAS,CAE/B;IAED,IAAI,aAAa,IAAI,MAAM,CAE1B;IAED,QAAQ,IAAI,MAAM;IAIlB,YAAY,IAAI,MAAM;IAItB,WAAW,IAAI,MAAM;IAIrB,aAAa,IAAI,MAAM;IAIvB;;;OAGG;IACH,wBAAwB,IAAI,OAAO;CAGpC"}
|