@bitgo/sdk-coin-rune 1.1.13 → 1.1.15
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 +10 -0
- package/dist/src/lib/constants.d.ts +3 -1
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +5 -3
- package/dist/src/rune.d.ts +11 -1
- package/dist/src/rune.d.ts.map +1 -1
- package/dist/src/rune.js +86 -2
- package/package.json +7 -7
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,16 @@
|
|
|
3
3
|
All notable changes to this project will be documented in this file.
|
|
4
4
|
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
|
|
5
5
|
|
|
6
|
+
## [1.1.15](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-coin-rune@1.1.14...@bitgo/sdk-coin-rune@1.1.15) (2025-01-09)
|
|
7
|
+
|
|
8
|
+
### Bug Fixes
|
|
9
|
+
|
|
10
|
+
- **sdk-coin-rune:** add native transaction fees for wallet recovery ([140f0ce](https://github.com/BitGo/BitGoJS/commit/140f0ce419a0a1093a463a0f955d094ee7c16334))
|
|
11
|
+
|
|
12
|
+
## [1.1.14](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-coin-rune@1.1.13...@bitgo/sdk-coin-rune@1.1.14) (2025-01-03)
|
|
13
|
+
|
|
14
|
+
**Note:** Version bump only for package @bitgo/sdk-coin-rune
|
|
15
|
+
|
|
6
16
|
## [1.1.13](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-coin-rune@1.1.12...@bitgo/sdk-coin-rune@1.1.13) (2024-12-24)
|
|
7
17
|
|
|
8
18
|
**Note:** Version bump only for package @bitgo/sdk-coin-rune
|
|
@@ -7,5 +7,7 @@ export declare const testnetAccountAddressRegex: RegExp;
|
|
|
7
7
|
export declare const testnetValidatorAddressRegex: RegExp;
|
|
8
8
|
export declare const TESTNET_ADDRESS_PREFIX = "sthor";
|
|
9
9
|
export declare const GAS_LIMIT = 200000;
|
|
10
|
-
export declare const GAS_AMOUNT = "
|
|
10
|
+
export declare const GAS_AMOUNT = "0";
|
|
11
|
+
export declare const RUNE_FEES = "2000000";
|
|
12
|
+
export declare const ROOT_PATH = "m/0";
|
|
11
13
|
//# sourceMappingURL=constants.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../../src/lib/constants.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,kBAAkB,UAAW,CAAC;AAC3C,eAAO,MAAM,0BAA0B,QAAuD,CAAC;AAC/F,eAAO,MAAM,4BAA4B,QAAuD,CAAC;AACjG,eAAO,MAAM,sBAAsB,SAAS,CAAC;AAE7C,eAAO,MAAM,kBAAkB,UAAW,CAAC;AAC3C,eAAO,MAAM,0BAA0B,QAAwD,CAAC;AAChG,eAAO,MAAM,4BAA4B,QAAwD,CAAC;AAClG,eAAO,MAAM,sBAAsB,UAAU,CAAC;AAE9C,eAAO,MAAM,SAAS,SAAS,CAAC;AAChC,eAAO,MAAM,UAAU,
|
|
1
|
+
{"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../../src/lib/constants.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,kBAAkB,UAAW,CAAC;AAC3C,eAAO,MAAM,0BAA0B,QAAuD,CAAC;AAC/F,eAAO,MAAM,4BAA4B,QAAuD,CAAC;AACjG,eAAO,MAAM,sBAAsB,SAAS,CAAC;AAE7C,eAAO,MAAM,kBAAkB,UAAW,CAAC;AAC3C,eAAO,MAAM,0BAA0B,QAAwD,CAAC;AAChG,eAAO,MAAM,4BAA4B,QAAwD,CAAC;AAClG,eAAO,MAAM,sBAAsB,UAAU,CAAC;AAE9C,eAAO,MAAM,SAAS,SAAS,CAAC;AAChC,eAAO,MAAM,UAAU,MAAM,CAAC;AAC9B,eAAO,MAAM,SAAS,YAAY,CAAC;AACnC,eAAO,MAAM,SAAS,QAAQ,CAAC"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.GAS_AMOUNT = exports.GAS_LIMIT = exports.TESTNET_ADDRESS_PREFIX = exports.testnetValidatorAddressRegex = exports.testnetAccountAddressRegex = exports.testnetValidDenoms = exports.MAINNET_ADDRESS_PREFIX = exports.mainnetValidatorAddressRegex = exports.mainnetAccountAddressRegex = exports.mainnetValidDenoms = void 0;
|
|
3
|
+
exports.ROOT_PATH = exports.RUNE_FEES = exports.GAS_AMOUNT = exports.GAS_LIMIT = exports.TESTNET_ADDRESS_PREFIX = exports.testnetValidatorAddressRegex = exports.testnetAccountAddressRegex = exports.testnetValidDenoms = exports.MAINNET_ADDRESS_PREFIX = exports.mainnetValidatorAddressRegex = exports.mainnetAccountAddressRegex = exports.mainnetValidDenoms = void 0;
|
|
4
4
|
exports.mainnetValidDenoms = ['rune'];
|
|
5
5
|
exports.mainnetAccountAddressRegex = /^(thor)1(['qpzry9x8gf2tvdw0s3jn54khce6mua7l]{38})$/;
|
|
6
6
|
exports.mainnetValidatorAddressRegex = /^(thor)1(['qpzry9x8gf2tvdw0s3jn54khce6mua7l]{38})$/;
|
|
@@ -10,5 +10,7 @@ exports.testnetAccountAddressRegex = /^(sthor)1(['qpzry9x8gf2tvdw0s3jn54khce6mua
|
|
|
10
10
|
exports.testnetValidatorAddressRegex = /^(sthor)1(['qpzry9x8gf2tvdw0s3jn54khce6mua7l]{38})$/;
|
|
11
11
|
exports.TESTNET_ADDRESS_PREFIX = 'sthor';
|
|
12
12
|
exports.GAS_LIMIT = 200000;
|
|
13
|
-
exports.GAS_AMOUNT = '
|
|
14
|
-
|
|
13
|
+
exports.GAS_AMOUNT = '0'; // Gas amount should be zero for RUNE transactions, as fees (0.02 RUNE) is cut from sender balance directly in the transaction
|
|
14
|
+
exports.RUNE_FEES = '2000000'; // https://dev.thorchain.org/concepts/fees.html#thorchain-native-rune
|
|
15
|
+
exports.ROOT_PATH = 'm/0';
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpYi9jb25zdGFudHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQWEsUUFBQSxrQkFBa0IsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQzlCLFFBQUEsMEJBQTBCLEdBQUcsb0RBQW9ELENBQUM7QUFDbEYsUUFBQSw0QkFBNEIsR0FBRyxvREFBb0QsQ0FBQztBQUNwRixRQUFBLHNCQUFzQixHQUFHLE1BQU0sQ0FBQztBQUVoQyxRQUFBLGtCQUFrQixHQUFHLENBQUMsTUFBTSxDQUFDLENBQUM7QUFDOUIsUUFBQSwwQkFBMEIsR0FBRyxxREFBcUQsQ0FBQztBQUNuRixRQUFBLDRCQUE0QixHQUFHLHFEQUFxRCxDQUFDO0FBQ3JGLFFBQUEsc0JBQXNCLEdBQUcsT0FBTyxDQUFDO0FBRWpDLFFBQUEsU0FBUyxHQUFHLE1BQU0sQ0FBQztBQUNuQixRQUFBLFVBQVUsR0FBRyxHQUFHLENBQUMsQ0FBQyw4SEFBOEg7QUFDaEosUUFBQSxTQUFTLEdBQUcsU0FBUyxDQUFDLENBQUMscUVBQXFFO0FBQzVGLFFBQUEsU0FBUyxHQUFHLEtBQUssQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBtYWlubmV0VmFsaWREZW5vbXMgPSBbJ3J1bmUnXTtcbmV4cG9ydCBjb25zdCBtYWlubmV0QWNjb3VudEFkZHJlc3NSZWdleCA9IC9eKHRob3IpMShbJ3FwenJ5OXg4Z2YydHZkdzBzM2puNTRraGNlNm11YTdsXXszOH0pJC87XG5leHBvcnQgY29uc3QgbWFpbm5ldFZhbGlkYXRvckFkZHJlc3NSZWdleCA9IC9eKHRob3IpMShbJ3FwenJ5OXg4Z2YydHZkdzBzM2puNTRraGNlNm11YTdsXXszOH0pJC87XG5leHBvcnQgY29uc3QgTUFJTk5FVF9BRERSRVNTX1BSRUZJWCA9ICd0aG9yJztcblxuZXhwb3J0IGNvbnN0IHRlc3RuZXRWYWxpZERlbm9tcyA9IFsncnVuZSddO1xuZXhwb3J0IGNvbnN0IHRlc3RuZXRBY2NvdW50QWRkcmVzc1JlZ2V4ID0gL14oc3Rob3IpMShbJ3FwenJ5OXg4Z2YydHZkdzBzM2puNTRraGNlNm11YTdsXXszOH0pJC87XG5leHBvcnQgY29uc3QgdGVzdG5ldFZhbGlkYXRvckFkZHJlc3NSZWdleCA9IC9eKHN0aG9yKTEoWydxcHpyeTl4OGdmMnR2ZHcwczNqbjU0a2hjZTZtdWE3bF17Mzh9KSQvO1xuZXhwb3J0IGNvbnN0IFRFU1RORVRfQUREUkVTU19QUkVGSVggPSAnc3Rob3InO1xuXG5leHBvcnQgY29uc3QgR0FTX0xJTUlUID0gMjAwMDAwO1xuZXhwb3J0IGNvbnN0IEdBU19BTU9VTlQgPSAnMCc7IC8vIEdhcyBhbW91bnQgc2hvdWxkIGJlIHplcm8gZm9yIFJVTkUgdHJhbnNhY3Rpb25zLCBhcyBmZWVzICgwLjAyIFJVTkUpIGlzIGN1dCBmcm9tIHNlbmRlciBiYWxhbmNlIGRpcmVjdGx5IGluIHRoZSB0cmFuc2FjdGlvblxuZXhwb3J0IGNvbnN0IFJVTkVfRkVFUyA9ICcyMDAwMDAwJzsgLy8gaHR0cHM6Ly9kZXYudGhvcmNoYWluLm9yZy9jb25jZXB0cy9mZWVzLmh0bWwjdGhvcmNoYWluLW5hdGl2ZS1ydW5lXG5leHBvcnQgY29uc3QgUk9PVF9QQVRIID0gJ20vMCc7XG4iXX0=
|
package/dist/src/rune.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { CosmosCoin, CosmosKeyPair, GasAmountDetails } from '@bitgo/abstract-cosmos';
|
|
1
|
+
import { CosmosCoin, CosmosKeyPair, CosmosLikeCoinRecoveryOutput, GasAmountDetails, RecoveryOptions } from '@bitgo/abstract-cosmos';
|
|
2
2
|
import { BaseCoin, BitGoBase, VerifyTransactionOptions } from '@bitgo/sdk-core';
|
|
3
3
|
import { BaseCoin as StaticsBaseCoin } from '@bitgo/statics';
|
|
4
4
|
import { TransactionBuilderFactory } from './lib';
|
|
@@ -26,5 +26,15 @@ export declare class Rune extends CosmosCoin {
|
|
|
26
26
|
/** @inheritDoc **/
|
|
27
27
|
getAddressFromPublicKey(publicKey: string): string;
|
|
28
28
|
verifyTransaction(params: VerifyTransactionOptions): Promise<boolean>;
|
|
29
|
+
getNativeRuneTxnFees(): string;
|
|
30
|
+
/**
|
|
31
|
+
* This function is overridden from CosmosCoin class' recover function due to the difference in fees handling in thorchain
|
|
32
|
+
* @param {RecoveryOptions} params parameters needed to construct and
|
|
33
|
+
* (maybe) sign the transaction
|
|
34
|
+
*
|
|
35
|
+
* @returns {CosmosLikeCoinRecoveryOutput} the serialized transaction hex string and index
|
|
36
|
+
* of the address being swept
|
|
37
|
+
*/
|
|
38
|
+
recover(params: RecoveryOptions): Promise<CosmosLikeCoinRecoveryOutput>;
|
|
29
39
|
}
|
|
30
40
|
//# sourceMappingURL=rune.d.ts.map
|
package/dist/src/rune.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"rune.d.ts","sourceRoot":"","sources":["../../src/rune.ts"],"names":[],"mappings":"AAAA,OAAO,
|
|
1
|
+
{"version":3,"file":"rune.d.ts","sourceRoot":"","sources":["../../src/rune.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,UAAU,EACV,aAAa,EACb,4BAA4B,EAG5B,gBAAgB,EAChB,eAAe,EAEhB,MAAM,wBAAwB,CAAC;AAChC,OAAO,EACL,QAAQ,EACR,SAAS,EAKT,wBAAwB,EACzB,MAAM,iBAAiB,CAAC;AACzB,OAAO,EAAE,QAAQ,IAAI,eAAe,EAAmB,MAAM,gBAAgB,CAAC;AAC9E,OAAO,EAAW,yBAAyB,EAAE,MAAM,OAAO,CAAC;AAE3D,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAOxC,qBAAa,IAAK,SAAQ,UAAU;IAClC,SAAS,CAAC,QAAQ,CAAC,MAAM,EAAE,SAAS,CAAC;IACrC,SAAS,CAAC,QAAQ,CAAC,YAAY,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IAC3D,SAAS,aAAa,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC;IAS/E,MAAM,CAAC,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC,GAAG,QAAQ;IAI1F,mBAAmB;IACnB,UAAU,IAAI,yBAAyB;IAIvC;;OAEG;IACI,aAAa,IAAI,MAAM;IAI9B,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAIxC,mBAAmB;IACnB,SAAS,CAAC,gBAAgB,IAAI,MAAM;IAIpC,mBAAmB;IACnB,eAAe,IAAI,MAAM;IAIzB,mBAAmB;IACnB,mBAAmB,IAAI,gBAAgB;IAOvC,mBAAmB;IACnB,UAAU,CAAC,SAAS,EAAE,MAAM,GAAG,aAAa;IAI5C,mBAAmB;IACnB,uBAAuB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM;IAI5C,iBAAiB,CAAC,MAAM,EAAE,wBAAwB,GAAG,OAAO,CAAC,OAAO,CAAC;IAuC3E,oBAAoB,IAAI,MAAM;IAI9B;;;;;;;OAOG;IACG,OAAO,CAAC,MAAM,EAAE,eAAe,GAAG,OAAO,CAAC,4BAA4B,CAAC;CA0F9E"}
|
package/dist/src/rune.js
CHANGED
|
@@ -43,6 +43,7 @@ const utils_1 = require("./lib/utils");
|
|
|
43
43
|
const bignumber_js_1 = require("bignumber.js");
|
|
44
44
|
const bech32 = require('bech32-buffer');
|
|
45
45
|
const _ = __importStar(require("lodash"));
|
|
46
|
+
const crypto_1 = require("crypto");
|
|
46
47
|
class Rune extends abstract_cosmos_1.CosmosCoin {
|
|
47
48
|
constructor(bitgo, staticsCoin) {
|
|
48
49
|
super(bitgo, staticsCoin);
|
|
@@ -70,7 +71,7 @@ class Rune extends abstract_cosmos_1.CosmosCoin {
|
|
|
70
71
|
}
|
|
71
72
|
/** @inheritDoc **/
|
|
72
73
|
getPublicNodeUrl() {
|
|
73
|
-
return sdk_core_1.Environments[this.bitgo.getEnv()].
|
|
74
|
+
return sdk_core_1.Environments[this.bitgo.getEnv()].runeNodeUrl;
|
|
74
75
|
}
|
|
75
76
|
/** @inheritDoc **/
|
|
76
77
|
getDenomination() {
|
|
@@ -126,6 +127,89 @@ class Rune extends abstract_cosmos_1.CosmosCoin {
|
|
|
126
127
|
}
|
|
127
128
|
return true;
|
|
128
129
|
}
|
|
130
|
+
getNativeRuneTxnFees() {
|
|
131
|
+
return constants_1.RUNE_FEES;
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* This function is overridden from CosmosCoin class' recover function due to the difference in fees handling in thorchain
|
|
135
|
+
* @param {RecoveryOptions} params parameters needed to construct and
|
|
136
|
+
* (maybe) sign the transaction
|
|
137
|
+
*
|
|
138
|
+
* @returns {CosmosLikeCoinRecoveryOutput} the serialized transaction hex string and index
|
|
139
|
+
* of the address being swept
|
|
140
|
+
*/
|
|
141
|
+
async recover(params) {
|
|
142
|
+
// Step 1: Check if params contains the required parameters
|
|
143
|
+
if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
|
|
144
|
+
throw new Error('invalid recoveryDestination');
|
|
145
|
+
}
|
|
146
|
+
if (!params.userKey) {
|
|
147
|
+
throw new Error('missing userKey');
|
|
148
|
+
}
|
|
149
|
+
if (!params.backupKey) {
|
|
150
|
+
throw new Error('missing backupKey');
|
|
151
|
+
}
|
|
152
|
+
if (!params.walletPassphrase) {
|
|
153
|
+
throw new Error('missing wallet passphrase');
|
|
154
|
+
}
|
|
155
|
+
// Step 2: Fetch the bitgo key from params
|
|
156
|
+
const userKey = params.userKey.replace(/\s/g, '');
|
|
157
|
+
const backupKey = params.backupKey.replace(/\s/g, '');
|
|
158
|
+
const { userKeyShare, backupKeyShare, commonKeyChain } = await sdk_core_1.ECDSAUtils.getMpcV2RecoveryKeyShares(userKey, backupKey, params.walletPassphrase); // baseAddress is not extracted
|
|
159
|
+
// Step 3: Instantiate the ECDSA signer and fetch the address details
|
|
160
|
+
const MPC = new sdk_core_1.Ecdsa();
|
|
161
|
+
const chainId = await this.getChainId();
|
|
162
|
+
const publicKey = MPC.deriveUnhardened(commonKeyChain, constants_1.ROOT_PATH).slice(0, 66);
|
|
163
|
+
const senderAddress = this.getAddressFromPublicKey(publicKey);
|
|
164
|
+
// Step 4: Fetch account details such as accountNo, balance and check for sufficient funds once gasAmount has been deducted
|
|
165
|
+
const [accountNumber, sequenceNo] = await this.getAccountDetails(senderAddress);
|
|
166
|
+
const balance = new bignumber_js_1.BigNumber(await this.getAccountBalance(senderAddress));
|
|
167
|
+
const gasBudget = {
|
|
168
|
+
amount: [{ denom: this.getDenomination(), amount: this.getGasAmountDetails().gasAmount }],
|
|
169
|
+
gasLimit: this.getGasAmountDetails().gasLimit,
|
|
170
|
+
};
|
|
171
|
+
const actualBalance = balance.minus(this.getNativeRuneTxnFees());
|
|
172
|
+
if (actualBalance.isLessThanOrEqualTo(0)) {
|
|
173
|
+
throw new Error('Did not have enough funds to recover');
|
|
174
|
+
}
|
|
175
|
+
// Step 5: Once sufficient funds are present, construct the recover tx messsage
|
|
176
|
+
const amount = [
|
|
177
|
+
{
|
|
178
|
+
denom: this.getDenomination(),
|
|
179
|
+
amount: actualBalance.toFixed(),
|
|
180
|
+
},
|
|
181
|
+
];
|
|
182
|
+
const sendMessage = [
|
|
183
|
+
{
|
|
184
|
+
fromAddress: senderAddress,
|
|
185
|
+
toAddress: params.recoveryDestination,
|
|
186
|
+
amount: amount,
|
|
187
|
+
},
|
|
188
|
+
];
|
|
189
|
+
// Step 6: Build the unsigned tx using the constructed message
|
|
190
|
+
const txnBuilder = this.getBuilder().getTransferBuilder();
|
|
191
|
+
txnBuilder
|
|
192
|
+
.messages(sendMessage)
|
|
193
|
+
.gasBudget(gasBudget)
|
|
194
|
+
.publicKey(publicKey)
|
|
195
|
+
.sequence(Number(sequenceNo))
|
|
196
|
+
.accountNumber(Number(accountNumber))
|
|
197
|
+
.chainId(chainId);
|
|
198
|
+
const unsignedTransaction = (await txnBuilder.build());
|
|
199
|
+
let serializedTx = unsignedTransaction.toBroadcastFormat();
|
|
200
|
+
const signableHex = unsignedTransaction.signablePayload.toString('hex');
|
|
201
|
+
// Step 7: Sign the tx
|
|
202
|
+
const message = unsignedTransaction.signablePayload;
|
|
203
|
+
const messageHash = (0, crypto_1.createHash)('sha256').update(message).digest();
|
|
204
|
+
const signature = await sdk_core_1.ECDSAUtils.signRecoveryMpcV2(messageHash, userKeyShare, backupKeyShare, commonKeyChain);
|
|
205
|
+
const signableBuffer = Buffer.from(signableHex, 'hex');
|
|
206
|
+
MPC.verify(signableBuffer, signature, this.getHashFunction());
|
|
207
|
+
const cosmosKeyPair = this.getKeyPair(publicKey);
|
|
208
|
+
txnBuilder.addSignature({ pub: cosmosKeyPair.getKeys().pub }, Buffer.from(signature.r + signature.s, 'hex'));
|
|
209
|
+
const signedTransaction = await txnBuilder.build();
|
|
210
|
+
serializedTx = signedTransaction.toBroadcastFormat();
|
|
211
|
+
return { serializedTx: serializedTx };
|
|
212
|
+
}
|
|
129
213
|
}
|
|
130
214
|
exports.Rune = Rune;
|
|
131
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rune.js","sourceRoot":"","sources":["../../src/rune.ts"],"names":[],"mappings":"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","sourcesContent":["import { CosmosCoin, CosmosKeyPair, GasAmountDetails } from '@bitgo/abstract-cosmos';\nimport { BaseCoin, BitGoBase, Environments, TransactionType, VerifyTransactionOptions } from '@bitgo/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, BaseUnit, coins } from '@bitgo/statics';\nimport { KeyPair, TransactionBuilderFactory } from './lib';\nimport { GAS_AMOUNT, GAS_LIMIT } from './lib/constants';\nimport { RuneUtils } from './lib/utils';\nimport { BigNumber } from 'bignumber.js';\nconst bech32 = require('bech32-buffer');\nimport * as _ from 'lodash';\n\nexport class Rune extends CosmosCoin {\n  protected readonly _utils: RuneUtils;\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo, staticsCoin);\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n    this._staticsCoin = staticsCoin;\n    this._utils = new RuneUtils();\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Rune(bitgo, staticsCoin);\n  }\n\n  /** @inheritDoc **/\n  getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e8;\n  }\n\n  isValidAddress(address: string): boolean {\n    return this._utils.isValidAddress(address) || this._utils.isValidValidatorAddress(address);\n  }\n\n  /** @inheritDoc **/\n  protected getPublicNodeUrl(): string {\n    return Environments[this.bitgo.getEnv()].coreumNodeUrl;\n  }\n\n  /** @inheritDoc **/\n  getDenomination(): string {\n    return BaseUnit.RUNE;\n  }\n\n  /** @inheritDoc **/\n  getGasAmountDetails(): GasAmountDetails {\n    return {\n      gasAmount: GAS_AMOUNT,\n      gasLimit: GAS_LIMIT,\n    };\n  }\n\n  /** @inheritDoc **/\n  getKeyPair(publicKey: string): CosmosKeyPair {\n    return new KeyPair({ pub: publicKey });\n  }\n\n  /** @inheritDoc **/\n  getAddressFromPublicKey(publicKey: string): string {\n    return new KeyPair({ pub: publicKey }).getAddress();\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const { txPrebuild, txParams } = params;\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n    const transaction = await this.getBuilder().from(rawTx).build();\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients?.map((recipient) => _.pick(recipient, ['address', 'amount']));\n      let filteredOutputs = explainedTx.outputs.map((output) => _.pick(output, ['address', 'amount']));\n\n      filteredOutputs = filteredOutputs.map((output) => {\n        const prefix = this._utils.getNetworkPrefix();\n        const convertedAddress = bech32.encode(prefix, output.address);\n        return {\n          ...output,\n          address: convertedAddress,\n        };\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      // WithdrawDelegatorRewards and ContractCall transaction don't have amount\n      if (transaction.type !== TransactionType.StakingWithdraw && transaction.type !== TransactionType.ContractCall) {\n        for (const recipients of txParams.recipients) {\n          totalAmount = totalAmount.plus(recipients.amount);\n        }\n        if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n          throw new Error('Tx total amount does not match with expected total amount field');\n        }\n      }\n    }\n    return true;\n  }\n}\n"]}
|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rune.js","sourceRoot":"","sources":["../../src/rune.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CosmosCoin,\n  CosmosKeyPair,\n  CosmosLikeCoinRecoveryOutput,\n  CosmosTransaction,\n  FeeData,\n  GasAmountDetails,\n  RecoveryOptions,\n  SendMessage,\n} from '@bitgo/abstract-cosmos';\nimport {\n  BaseCoin,\n  BitGoBase,\n  Ecdsa,\n  ECDSAUtils,\n  Environments,\n  TransactionType,\n  VerifyTransactionOptions,\n} from '@bitgo/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, BaseUnit, coins } from '@bitgo/statics';\nimport { KeyPair, TransactionBuilderFactory } from './lib';\nimport { GAS_AMOUNT, GAS_LIMIT, RUNE_FEES, ROOT_PATH } from './lib/constants';\nimport { RuneUtils } from './lib/utils';\nimport { BigNumber } from 'bignumber.js';\nconst bech32 = require('bech32-buffer');\nimport * as _ from 'lodash';\nimport { Coin } from '@cosmjs/stargate';\nimport { createHash } from 'crypto';\n\nexport class Rune extends CosmosCoin {\n  protected readonly _utils: RuneUtils;\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo, staticsCoin);\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n    this._staticsCoin = staticsCoin;\n    this._utils = new RuneUtils();\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Rune(bitgo, staticsCoin);\n  }\n\n  /** @inheritDoc **/\n  getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e8;\n  }\n\n  isValidAddress(address: string): boolean {\n    return this._utils.isValidAddress(address) || this._utils.isValidValidatorAddress(address);\n  }\n\n  /** @inheritDoc **/\n  protected getPublicNodeUrl(): string {\n    return Environments[this.bitgo.getEnv()].runeNodeUrl;\n  }\n\n  /** @inheritDoc **/\n  getDenomination(): string {\n    return BaseUnit.RUNE;\n  }\n\n  /** @inheritDoc **/\n  getGasAmountDetails(): GasAmountDetails {\n    return {\n      gasAmount: GAS_AMOUNT,\n      gasLimit: GAS_LIMIT,\n    };\n  }\n\n  /** @inheritDoc **/\n  getKeyPair(publicKey: string): CosmosKeyPair {\n    return new KeyPair({ pub: publicKey });\n  }\n\n  /** @inheritDoc **/\n  getAddressFromPublicKey(publicKey: string): string {\n    return new KeyPair({ pub: publicKey }).getAddress();\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const { txPrebuild, txParams } = params;\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n    const transaction = await this.getBuilder().from(rawTx).build();\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients?.map((recipient) => _.pick(recipient, ['address', 'amount']));\n      let filteredOutputs = explainedTx.outputs.map((output) => _.pick(output, ['address', 'amount']));\n\n      filteredOutputs = filteredOutputs.map((output) => {\n        const prefix = this._utils.getNetworkPrefix();\n        const convertedAddress = bech32.encode(prefix, output.address);\n        return {\n          ...output,\n          address: convertedAddress,\n        };\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      // WithdrawDelegatorRewards and ContractCall transaction don't have amount\n      if (transaction.type !== TransactionType.StakingWithdraw && transaction.type !== TransactionType.ContractCall) {\n        for (const recipients of txParams.recipients) {\n          totalAmount = totalAmount.plus(recipients.amount);\n        }\n        if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n          throw new Error('Tx total amount does not match with expected total amount field');\n        }\n      }\n    }\n    return true;\n  }\n\n  getNativeRuneTxnFees(): string {\n    return RUNE_FEES;\n  }\n\n  /**\n   * This function is overridden from CosmosCoin class' recover function due to the difference in fees handling in thorchain\n   * @param {RecoveryOptions} params parameters needed to construct and\n   * (maybe) sign the transaction\n   *\n   * @returns {CosmosLikeCoinRecoveryOutput} the serialized transaction hex string and index\n   * of the address being swept\n   */\n  async recover(params: RecoveryOptions): Promise<CosmosLikeCoinRecoveryOutput> {\n    // Step 1: Check if params contains the required parameters\n\n    if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n\n    if (!params.userKey) {\n      throw new Error('missing userKey');\n    }\n\n    if (!params.backupKey) {\n      throw new Error('missing backupKey');\n    }\n\n    if (!params.walletPassphrase) {\n      throw new Error('missing wallet passphrase');\n    }\n\n    // Step 2: Fetch the bitgo key from params\n    const userKey = params.userKey.replace(/\\s/g, '');\n    const backupKey = params.backupKey.replace(/\\s/g, '');\n\n    const { userKeyShare, backupKeyShare, commonKeyChain } = await ECDSAUtils.getMpcV2RecoveryKeyShares(\n      userKey,\n      backupKey,\n      params.walletPassphrase\n    ); // baseAddress is not extracted\n    // Step 3: Instantiate the ECDSA signer and fetch the address details\n    const MPC = new Ecdsa();\n    const chainId = await this.getChainId();\n    const publicKey = MPC.deriveUnhardened(commonKeyChain, ROOT_PATH).slice(0, 66);\n    const senderAddress = this.getAddressFromPublicKey(publicKey);\n\n    // Step 4: Fetch account details such as accountNo, balance and check for sufficient funds once gasAmount has been deducted\n    const [accountNumber, sequenceNo] = await this.getAccountDetails(senderAddress);\n    const balance = new BigNumber(await this.getAccountBalance(senderAddress));\n    const gasBudget: FeeData = {\n      amount: [{ denom: this.getDenomination(), amount: this.getGasAmountDetails().gasAmount }],\n      gasLimit: this.getGasAmountDetails().gasLimit,\n    };\n    const actualBalance = balance.minus(this.getNativeRuneTxnFees());\n\n    if (actualBalance.isLessThanOrEqualTo(0)) {\n      throw new Error('Did not have enough funds to recover');\n    }\n\n    // Step 5: Once sufficient funds are present, construct the recover tx messsage\n    const amount: Coin[] = [\n      {\n        denom: this.getDenomination(),\n        amount: actualBalance.toFixed(),\n      },\n    ];\n    const sendMessage: SendMessage[] = [\n      {\n        fromAddress: senderAddress,\n        toAddress: params.recoveryDestination,\n        amount: amount,\n      },\n    ];\n\n    // Step 6: Build the unsigned tx using the constructed message\n    const txnBuilder = this.getBuilder().getTransferBuilder();\n    txnBuilder\n      .messages(sendMessage)\n      .gasBudget(gasBudget)\n      .publicKey(publicKey)\n      .sequence(Number(sequenceNo))\n      .accountNumber(Number(accountNumber))\n      .chainId(chainId);\n    const unsignedTransaction = (await txnBuilder.build()) as CosmosTransaction;\n    let serializedTx = unsignedTransaction.toBroadcastFormat();\n    const signableHex = unsignedTransaction.signablePayload.toString('hex');\n\n    // Step 7: Sign the tx\n    const message = unsignedTransaction.signablePayload;\n    const messageHash = createHash('sha256').update(message).digest();\n\n    const signature = await ECDSAUtils.signRecoveryMpcV2(messageHash, userKeyShare, backupKeyShare, commonKeyChain);\n\n    const signableBuffer = Buffer.from(signableHex, 'hex');\n    MPC.verify(signableBuffer, signature, this.getHashFunction());\n    const cosmosKeyPair = this.getKeyPair(publicKey);\n    txnBuilder.addSignature({ pub: cosmosKeyPair.getKeys().pub }, Buffer.from(signature.r + signature.s, 'hex'));\n    const signedTransaction = await txnBuilder.build();\n    serializedTx = signedTransaction.toBroadcastFormat();\n\n    return { serializedTx: serializedTx };\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@bitgo/sdk-coin-rune",
|
|
3
|
-
"version": "1.1.
|
|
3
|
+
"version": "1.1.15",
|
|
4
4
|
"description": "BitGo SDK coin library for Thorchain",
|
|
5
5
|
"main": "./dist/src/index.js",
|
|
6
6
|
"types": "./dist/src/index.d.ts",
|
|
@@ -40,9 +40,9 @@
|
|
|
40
40
|
]
|
|
41
41
|
},
|
|
42
42
|
"dependencies": {
|
|
43
|
-
"@bitgo/abstract-cosmos": "^11.2.
|
|
44
|
-
"@bitgo/sdk-core": "^28.
|
|
45
|
-
"@bitgo/statics": "^50.
|
|
43
|
+
"@bitgo/abstract-cosmos": "^11.2.14",
|
|
44
|
+
"@bitgo/sdk-core": "^28.19.0",
|
|
45
|
+
"@bitgo/statics": "^50.18.0",
|
|
46
46
|
"@cosmjs/amino": "^0.29.5",
|
|
47
47
|
"@cosmjs/encoding": "^0.29.5",
|
|
48
48
|
"@cosmjs/proto-signing": "^0.29.5",
|
|
@@ -52,8 +52,8 @@
|
|
|
52
52
|
"lodash": "^4.17.21"
|
|
53
53
|
},
|
|
54
54
|
"devDependencies": {
|
|
55
|
-
"@bitgo/sdk-api": "^1.
|
|
56
|
-
"@bitgo/sdk-test": "^8.0.
|
|
55
|
+
"@bitgo/sdk-api": "^1.58.0",
|
|
56
|
+
"@bitgo/sdk-test": "^8.0.63"
|
|
57
57
|
},
|
|
58
|
-
"gitHead": "
|
|
58
|
+
"gitHead": "a67d6d6aefb10365ec158c3cf23814b29d085bbc"
|
|
59
59
|
}
|