@rosen-bridge/rosen-extractor 6.3.0 → 6.3.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/getRosenData/abstract/AbstractRosenDataExtractor.d.ts +14 -20
- package/dist/lib/getRosenData/abstract/types.d.ts +13 -13
- package/dist/lib/getRosenData/bitcoin/BitcoinEsploraRosenExtractor.d.ts +15 -24
- package/dist/lib/getRosenData/bitcoin/BitcoinRosenExtractor.d.ts +15 -22
- package/dist/lib/getRosenData/bitcoin/BitcoinRpcRosenExtractor.d.ts +15 -22
- package/dist/lib/getRosenData/bitcoin/types.d.ts +66 -66
- package/dist/lib/getRosenData/bitcoin/utils.d.ts +1 -1
- package/dist/lib/getRosenData/cardano/CardanoBlockFrostRosenExtractor.d.ts +13 -16
- package/dist/lib/getRosenData/cardano/CardanoGraphQLRosenExtractor.d.ts +13 -16
- package/dist/lib/getRosenData/cardano/CardanoKoiosRosenExtractor.d.ts +13 -16
- package/dist/lib/getRosenData/cardano/CardanoOgmiosRosenExtractor.d.ts +13 -16
- package/dist/lib/getRosenData/cardano/CardanoRosenExtractor.d.ts +13 -16
- package/dist/lib/getRosenData/cardano/types.d.ts +90 -110
- package/dist/lib/getRosenData/cardano/utils.d.ts +8 -10
- package/dist/lib/getRosenData/doge/DogeEsploraRosenExtractor.d.ts +22 -0
- package/dist/lib/getRosenData/doge/DogeEsploraRosenExtractor.d.ts.map +1 -0
- package/dist/lib/getRosenData/doge/DogeEsploraRosenExtractor.js +104 -0
- package/dist/lib/getRosenData/doge/DogeRosenExtractor.d.ts +22 -0
- package/dist/lib/getRosenData/doge/DogeRosenExtractor.d.ts.map +1 -0
- package/dist/lib/getRosenData/doge/DogeRosenExtractor.js +112 -0
- package/dist/lib/getRosenData/doge/types.d.ts +59 -0
- package/dist/lib/getRosenData/doge/types.d.ts.map +1 -0
- package/dist/lib/getRosenData/doge/types.js +2 -0
- package/dist/lib/getRosenData/doge/utils.d.ts +13 -0
- package/dist/lib/getRosenData/doge/utils.d.ts.map +1 -0
- package/dist/lib/getRosenData/doge/utils.js +58 -0
- package/dist/lib/getRosenData/ergo/ErgoNodeRosenExtractor.d.ts +15 -22
- package/dist/lib/getRosenData/ergo/ErgoRosenExtractor.d.ts +9 -13
- package/dist/lib/getRosenData/ergo/types.d.ts +24 -24
- package/dist/lib/getRosenData/evm/EvmEthersRosenExtractor.d.ts +9 -15
- package/dist/lib/getRosenData/evm/EvmRosenExtractor.d.ts +9 -15
- package/dist/lib/getRosenData/evm/EvmRpcRosenExtractor.d.ts +19 -25
- package/dist/lib/getRosenData/evm/utils.d.ts +5 -5
- package/dist/lib/index.d.ts +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +2 -2
- package/dist/lib/getRosenData/Utils.d.ts +0 -59
- package/dist/lib/getRosenData/Utils.d.ts.map +0 -1
- package/dist/lib/getRosenData/Utils.js +0 -96
|
@@ -1,146 +1,126 @@
|
|
|
1
1
|
import { components } from '@blockfrost/openapi';
|
|
2
2
|
interface PaymentAddr {
|
|
3
|
-
|
|
4
|
-
|
|
3
|
+
bech32: string;
|
|
4
|
+
cred: string;
|
|
5
5
|
}
|
|
6
6
|
interface Asset {
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
7
|
+
policy_id: string;
|
|
8
|
+
asset_name: string;
|
|
9
|
+
quantity: string;
|
|
10
10
|
}
|
|
11
11
|
interface Utxo {
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
12
|
+
payment_addr: PaymentAddr;
|
|
13
|
+
stake_addr?: string | null;
|
|
14
|
+
tx_hash: string;
|
|
15
|
+
tx_index: number;
|
|
16
|
+
value: string;
|
|
17
|
+
asset_list: Array<Asset>;
|
|
18
18
|
}
|
|
19
19
|
interface Metadata {
|
|
20
|
-
|
|
20
|
+
[key: string]: Record<string, unknown>;
|
|
21
21
|
}
|
|
22
22
|
interface KoiosTransaction {
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
23
|
+
tx_hash: string;
|
|
24
|
+
block_hash: string;
|
|
25
|
+
inputs: Array<Utxo>;
|
|
26
|
+
outputs: Array<Utxo>;
|
|
27
|
+
metadata?: Metadata;
|
|
28
28
|
}
|
|
29
29
|
interface CardanoRosenData {
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
30
|
+
toChain: string;
|
|
31
|
+
toAddress: string;
|
|
32
|
+
bridgeFee: string;
|
|
33
|
+
networkFee: string;
|
|
34
|
+
fromAddress: string;
|
|
35
35
|
}
|
|
36
36
|
interface CardanoMetadataRosenData {
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
37
|
+
to: string;
|
|
38
|
+
networkFee: string;
|
|
39
|
+
bridgeFee: string;
|
|
40
|
+
toAddress: string;
|
|
41
|
+
fromAddress: string[];
|
|
42
42
|
}
|
|
43
43
|
type ListObject = Array<MetadataObject>;
|
|
44
44
|
type NativeValue = string | undefined;
|
|
45
45
|
interface JsonObject {
|
|
46
|
-
|
|
46
|
+
[key: string]: MetadataObject;
|
|
47
47
|
}
|
|
48
48
|
type MetadataObject = JsonObject | ListObject | NativeValue;
|
|
49
49
|
interface CardanoAsset {
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
50
|
+
policy_id: string;
|
|
51
|
+
asset_name: string;
|
|
52
|
+
quantity: string;
|
|
53
53
|
}
|
|
54
54
|
interface CardanoUtxo {
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
55
|
+
txId: string;
|
|
56
|
+
index: number;
|
|
57
|
+
value: bigint;
|
|
58
|
+
assets: Array<CardanoAsset>;
|
|
59
59
|
}
|
|
60
60
|
interface CardanoBoxCandidate {
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
61
|
+
address: string;
|
|
62
|
+
value: bigint;
|
|
63
|
+
assets: Array<CardanoAsset>;
|
|
64
64
|
}
|
|
65
65
|
type CardanoMetadata = Record<string, string | Record<string, any>>;
|
|
66
66
|
interface CardanoTx {
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
67
|
+
id: string;
|
|
68
|
+
inputs: CardanoUtxo[];
|
|
69
|
+
outputs: CardanoBoxCandidate[];
|
|
70
|
+
fee: bigint;
|
|
71
|
+
metadata?: CardanoMetadata;
|
|
72
72
|
}
|
|
73
73
|
interface BlockFrostTransaction {
|
|
74
|
-
|
|
75
|
-
|
|
74
|
+
utxos: components['schemas']['tx_content_utxo'];
|
|
75
|
+
metadata: components['schemas']['tx_content_metadata'];
|
|
76
76
|
}
|
|
77
|
-
type BlockFrostOutputBox =
|
|
78
|
-
components['schemas']['tx_content_utxo']['outputs'][0];
|
|
77
|
+
type BlockFrostOutputBox = components['schemas']['tx_content_utxo']['outputs'][0];
|
|
79
78
|
type GraphQLBlockTxsQuery = {
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
79
|
+
__typename?: 'Query';
|
|
80
|
+
blocks: Array<{
|
|
81
|
+
__typename?: 'Block';
|
|
82
|
+
transactions: Array<{
|
|
83
|
+
__typename?: 'Transaction';
|
|
84
|
+
hash: any;
|
|
85
|
+
fee: any;
|
|
86
|
+
inputs: Array<{
|
|
87
|
+
__typename?: 'TransactionInput';
|
|
88
|
+
sourceTxIndex: number;
|
|
89
|
+
sourceTxHash: any;
|
|
90
|
+
value: any;
|
|
91
|
+
tokens: Array<{
|
|
92
|
+
__typename?: 'Token';
|
|
93
|
+
quantity: string;
|
|
94
|
+
asset: {
|
|
95
|
+
__typename?: 'Asset';
|
|
96
|
+
assetName?: any | null;
|
|
97
|
+
policyId: any;
|
|
98
|
+
};
|
|
99
|
+
} | null>;
|
|
100
|
+
}>;
|
|
101
|
+
outputs: Array<{
|
|
102
|
+
__typename?: 'TransactionOutput';
|
|
103
|
+
address: string;
|
|
104
|
+
value: any;
|
|
105
|
+
tokens: Array<{
|
|
106
|
+
__typename?: 'Token';
|
|
107
|
+
quantity: string;
|
|
108
|
+
asset: {
|
|
109
|
+
__typename?: 'Asset';
|
|
110
|
+
assetName?: any | null;
|
|
111
|
+
policyId: any;
|
|
112
|
+
};
|
|
113
|
+
}>;
|
|
114
|
+
} | null>;
|
|
115
|
+
metadata?: Array<{
|
|
116
|
+
__typename?: 'TransactionMetadata';
|
|
117
|
+
key: string;
|
|
118
|
+
value: any;
|
|
119
|
+
} | null> | null;
|
|
100
120
|
} | null>;
|
|
101
|
-
}>;
|
|
102
|
-
outputs: Array<{
|
|
103
|
-
__typename?: 'TransactionOutput';
|
|
104
|
-
address: string;
|
|
105
|
-
value: any;
|
|
106
|
-
tokens: Array<{
|
|
107
|
-
__typename?: 'Token';
|
|
108
|
-
quantity: string;
|
|
109
|
-
asset: {
|
|
110
|
-
__typename?: 'Asset';
|
|
111
|
-
assetName?: any | null;
|
|
112
|
-
policyId: any;
|
|
113
|
-
};
|
|
114
|
-
}>;
|
|
115
|
-
} | null>;
|
|
116
|
-
metadata?: Array<{
|
|
117
|
-
__typename?: 'TransactionMetadata';
|
|
118
|
-
key: string;
|
|
119
|
-
value: any;
|
|
120
|
-
} | null> | null;
|
|
121
121
|
} | null>;
|
|
122
|
-
} | null>;
|
|
123
122
|
};
|
|
124
|
-
type GraphQLTransaction = NonNullable<
|
|
125
|
-
NonNullable<GraphQLBlockTxsQuery['blocks'][0]>['transactions'][0]
|
|
126
|
-
>;
|
|
123
|
+
type GraphQLTransaction = NonNullable<NonNullable<GraphQLBlockTxsQuery['blocks'][0]>['transactions'][0]>;
|
|
127
124
|
type GraphQLTxOutput = NonNullable<GraphQLTransaction['outputs'][0]>;
|
|
128
|
-
export {
|
|
129
|
-
|
|
130
|
-
KoiosTransaction,
|
|
131
|
-
CardanoRosenData,
|
|
132
|
-
CardanoMetadataRosenData,
|
|
133
|
-
Metadata,
|
|
134
|
-
MetadataObject,
|
|
135
|
-
JsonObject,
|
|
136
|
-
ListObject,
|
|
137
|
-
NativeValue,
|
|
138
|
-
CardanoMetadata,
|
|
139
|
-
CardanoTx,
|
|
140
|
-
CardanoBoxCandidate,
|
|
141
|
-
BlockFrostTransaction,
|
|
142
|
-
BlockFrostOutputBox,
|
|
143
|
-
GraphQLTransaction,
|
|
144
|
-
GraphQLTxOutput,
|
|
145
|
-
};
|
|
146
|
-
//# sourceMappingURL=types.d.ts.map
|
|
125
|
+
export { Utxo, KoiosTransaction, CardanoRosenData, CardanoMetadataRosenData, Metadata, MetadataObject, JsonObject, ListObject, NativeValue, CardanoMetadata, CardanoTx, CardanoBoxCandidate, BlockFrostTransaction, BlockFrostOutputBox, GraphQLTransaction, GraphQLTxOutput, };
|
|
126
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -3,13 +3,11 @@
|
|
|
3
3
|
* @param data
|
|
4
4
|
* @returns Rosen data or undefined if metadata is invalid
|
|
5
5
|
*/
|
|
6
|
-
export declare const parseRosenData: (data: any) =>
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
| undefined;
|
|
15
|
-
//# sourceMappingURL=utils.d.ts.map
|
|
6
|
+
export declare const parseRosenData: (data: any) => {
|
|
7
|
+
toChain: any;
|
|
8
|
+
toAddress: any;
|
|
9
|
+
bridgeFee: any;
|
|
10
|
+
networkFee: any;
|
|
11
|
+
fromAddress: any;
|
|
12
|
+
} | undefined;
|
|
13
|
+
//# sourceMappingURL=utils.d.ts.map
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { RosenData, TokenTransformation } from '../abstract/types';
|
|
2
|
+
import AbstractRosenDataExtractor from '../abstract/AbstractRosenDataExtractor';
|
|
3
|
+
import { DogeEsploraTransaction, EsploraTxOutput } from './types';
|
|
4
|
+
import { TokenMap } from '@rosen-bridge/tokens';
|
|
5
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
6
|
+
export declare class DogeEsploraRosenExtractor extends AbstractRosenDataExtractor<DogeEsploraTransaction> {
|
|
7
|
+
readonly chain = "doge";
|
|
8
|
+
protected lockScriptPubKey: string;
|
|
9
|
+
constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger);
|
|
10
|
+
/**
|
|
11
|
+
* extracts RosenData from given lock transaction in Esplora format
|
|
12
|
+
* @param transaction the lock transaction in Esplora format
|
|
13
|
+
*/
|
|
14
|
+
extractRawData: (transaction: DogeEsploraTransaction) => RosenData | undefined;
|
|
15
|
+
/**
|
|
16
|
+
* extracts and builds token transformation from UTXO and tokenMap
|
|
17
|
+
* @param box transaction output
|
|
18
|
+
* @param toChain event target chain
|
|
19
|
+
*/
|
|
20
|
+
getAssetTransformation: (box: EsploraTxOutput, toChain: string) => TokenTransformation | undefined;
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=DogeEsploraRosenExtractor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DogeEsploraRosenExtractor.d.ts","sourceRoot":"","sources":["../../../../lib/getRosenData/doge/DogeEsploraRosenExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACnE,OAAO,0BAA0B,MAAM,wCAAwC,CAAC;AAEhF,OAAO,EAAE,sBAAsB,EAAE,eAAe,EAAgB,MAAM,SAAS,CAAC;AAChF,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAG/D,qBAAa,yBAA0B,SAAQ,0BAA0B,CAAC,sBAAsB,CAAC;IAC/F,QAAQ,CAAC,KAAK,UAAc;IAC5B,SAAS,CAAC,gBAAgB,EAAE,MAAM,CAAC;gBAEvB,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE,cAAc;IAK1E;;;OAGG;IACH,cAAc,gBACC,sBAAsB,KAClC,SAAS,GAAG,SAAS,CA6EtB;IAEF;;;;OAIG;IACH,sBAAsB,QACf,eAAe,WACX,MAAM,KACd,mBAAmB,GAAG,SAAS,CAehC;CACH"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import AbstractRosenDataExtractor from '../abstract/AbstractRosenDataExtractor';
|
|
2
|
+
import { DOGE_CHAIN, DOGE_NATIVE_TOKEN } from '../const';
|
|
3
|
+
import { parseRosenData, addressToOutputScript } from './utils';
|
|
4
|
+
export class DogeEsploraRosenExtractor extends AbstractRosenDataExtractor {
|
|
5
|
+
chain = DOGE_CHAIN;
|
|
6
|
+
lockScriptPubKey;
|
|
7
|
+
constructor(lockAddress, tokens, logger) {
|
|
8
|
+
super(lockAddress, tokens, logger);
|
|
9
|
+
this.lockScriptPubKey = addressToOutputScript(lockAddress);
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* extracts RosenData from given lock transaction in Esplora format
|
|
13
|
+
* @param transaction the lock transaction in Esplora format
|
|
14
|
+
*/
|
|
15
|
+
extractRawData = (transaction) => {
|
|
16
|
+
const baseError = `No rosen data found for tx [${transaction.txid}]`;
|
|
17
|
+
try {
|
|
18
|
+
const outputs = transaction.vout;
|
|
19
|
+
if (outputs.length < 2) {
|
|
20
|
+
this.logger.debug(baseError + `: Insufficient number of boxes`);
|
|
21
|
+
return undefined;
|
|
22
|
+
}
|
|
23
|
+
let validData = false; // an OP_RETURN box with valid data is found
|
|
24
|
+
let validLock = false; // a lock box is found with available asset transformation
|
|
25
|
+
// parse rosen data from OP_RETURN box
|
|
26
|
+
let opReturnData;
|
|
27
|
+
for (let i = 0; i < outputs.length; i++) {
|
|
28
|
+
const output = outputs[i];
|
|
29
|
+
if (output.scriptpubkey.slice(0, 2) !== '6a')
|
|
30
|
+
continue; // not an OP_RETURN utxo
|
|
31
|
+
try {
|
|
32
|
+
opReturnData = parseRosenData(output.scriptpubkey);
|
|
33
|
+
validData = true;
|
|
34
|
+
break;
|
|
35
|
+
}
|
|
36
|
+
catch (e) {
|
|
37
|
+
this.logger.debug(`Failed to extract data from OP_RETURN box [${transaction.txid}.${i}]: ${e}`);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
if (!validData || !opReturnData) {
|
|
41
|
+
this.logger.debug(baseError + `: No OP_RETURN box with valid data is found`);
|
|
42
|
+
return undefined;
|
|
43
|
+
}
|
|
44
|
+
// find target chain token id
|
|
45
|
+
let assetTransformation;
|
|
46
|
+
for (let i = 0; i < outputs.length; i++) {
|
|
47
|
+
const output = outputs[i];
|
|
48
|
+
if (output.scriptpubkey !== this.lockScriptPubKey)
|
|
49
|
+
continue; // utxo address is not lock address
|
|
50
|
+
assetTransformation = this.getAssetTransformation(output, opReturnData.toChain);
|
|
51
|
+
if (assetTransformation) {
|
|
52
|
+
validLock = true;
|
|
53
|
+
break;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
if (!validLock || !assetTransformation) {
|
|
57
|
+
this.logger.debug(baseError + `: Failed to find rosen asset transformation`);
|
|
58
|
+
return undefined;
|
|
59
|
+
}
|
|
60
|
+
const fromAddress = `box:${transaction.vin[0].txid}.${transaction.vin[0].vout}`;
|
|
61
|
+
return {
|
|
62
|
+
toChain: opReturnData.toChain,
|
|
63
|
+
toAddress: opReturnData.toAddress,
|
|
64
|
+
bridgeFee: opReturnData.bridgeFee,
|
|
65
|
+
networkFee: opReturnData.networkFee,
|
|
66
|
+
fromAddress: fromAddress,
|
|
67
|
+
sourceChainTokenId: assetTransformation.from,
|
|
68
|
+
amount: assetTransformation.amount,
|
|
69
|
+
targetChainTokenId: assetTransformation.to,
|
|
70
|
+
sourceTxId: transaction.txid,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
catch (e) {
|
|
74
|
+
this.logger.debug(`An error occurred while getting Doge rosen data from Esplora: ${e}`);
|
|
75
|
+
if (e instanceof Error && e.stack) {
|
|
76
|
+
this.logger.debug(e.stack);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
return undefined;
|
|
80
|
+
};
|
|
81
|
+
/**
|
|
82
|
+
* extracts and builds token transformation from UTXO and tokenMap
|
|
83
|
+
* @param box transaction output
|
|
84
|
+
* @param toChain event target chain
|
|
85
|
+
*/
|
|
86
|
+
getAssetTransformation = (box, toChain) => {
|
|
87
|
+
// try to build transformation using locked DOGE
|
|
88
|
+
const wrappedDoge = this.tokens.search(DOGE_CHAIN, {
|
|
89
|
+
tokenId: DOGE_NATIVE_TOKEN,
|
|
90
|
+
});
|
|
91
|
+
if (wrappedDoge.length > 0 && Object.hasOwn(wrappedDoge[0], toChain)) {
|
|
92
|
+
const dogeAmount = box.value;
|
|
93
|
+
return {
|
|
94
|
+
from: DOGE_NATIVE_TOKEN,
|
|
95
|
+
to: this.tokens.getID(wrappedDoge[0], toChain),
|
|
96
|
+
amount: dogeAmount.toString(),
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
return undefined;
|
|
101
|
+
}
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DogeEsploraRosenExtractor.js","sourceRoot":"","sources":["../../../../lib/getRosenData/doge/DogeEsploraRosenExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { RosenData, TokenTransformation } from '../abstract/types';\nimport AbstractRosenDataExtractor from '../abstract/AbstractRosenDataExtractor';\nimport { DOGE_CHAIN, DOGE_NATIVE_TOKEN } from '../const';\nimport { DogeEsploraTransaction, EsploraTxOutput, OpReturnData } from './types';\nimport { TokenMap } from '@rosen-bridge/tokens';\nimport { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { parseRosenData, addressToOutputScript } from './utils';\n\nexport class DogeEsploraRosenExtractor extends AbstractRosenDataExtractor<DogeEsploraTransaction> {\n  readonly chain = DOGE_CHAIN;\n  protected lockScriptPubKey: string;\n\n  constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger) {\n    super(lockAddress, tokens, logger);\n    this.lockScriptPubKey = addressToOutputScript(lockAddress);\n  }\n\n  /**\n   * extracts RosenData from given lock transaction in Esplora format\n   * @param transaction the lock transaction in Esplora format\n   */\n  extractRawData = (\n    transaction: DogeEsploraTransaction\n  ): RosenData | undefined => {\n    const baseError = `No rosen data found for tx [${transaction.txid}]`;\n    try {\n      const outputs = transaction.vout;\n      if (outputs.length < 2) {\n        this.logger.debug(baseError + `: Insufficient number of boxes`);\n        return undefined;\n      }\n\n      let validData = false; // an OP_RETURN box with valid data is found\n      let validLock = false; // a lock box is found with available asset transformation\n\n      // parse rosen data from OP_RETURN box\n      let opReturnData: OpReturnData | undefined;\n      for (let i = 0; i < outputs.length; i++) {\n        const output = outputs[i];\n        if (output.scriptpubkey.slice(0, 2) !== '6a') continue; // not an OP_RETURN utxo\n\n        try {\n          opReturnData = parseRosenData(output.scriptpubkey);\n          validData = true;\n          break;\n        } catch (e) {\n          this.logger.debug(\n            `Failed to extract data from OP_RETURN box [${transaction.txid}.${i}]: ${e}`\n          );\n        }\n      }\n      if (!validData || !opReturnData) {\n        this.logger.debug(\n          baseError + `: No OP_RETURN box with valid data is found`\n        );\n        return undefined;\n      }\n\n      // find target chain token id\n      let assetTransformation: TokenTransformation | undefined;\n      for (let i = 0; i < outputs.length; i++) {\n        const output = outputs[i];\n        if (output.scriptpubkey !== this.lockScriptPubKey) continue; // utxo address is not lock address\n        assetTransformation = this.getAssetTransformation(\n          output,\n          opReturnData.toChain\n        );\n        if (assetTransformation) {\n          validLock = true;\n          break;\n        }\n      }\n      if (!validLock || !assetTransformation) {\n        this.logger.debug(\n          baseError + `: Failed to find rosen asset transformation`\n        );\n        return undefined;\n      }\n\n      const fromAddress = `box:${transaction.vin[0].txid}.${transaction.vin[0].vout}`;\n      return {\n        toChain: opReturnData.toChain,\n        toAddress: opReturnData.toAddress,\n        bridgeFee: opReturnData.bridgeFee,\n        networkFee: opReturnData.networkFee,\n        fromAddress: fromAddress,\n        sourceChainTokenId: assetTransformation.from,\n        amount: assetTransformation.amount,\n        targetChainTokenId: assetTransformation.to,\n        sourceTxId: transaction.txid,\n      };\n    } catch (e) {\n      this.logger.debug(\n        `An error occurred while getting Doge rosen data from Esplora: ${e}`\n      );\n      if (e instanceof Error && e.stack) {\n        this.logger.debug(e.stack);\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * extracts and builds token transformation from UTXO and tokenMap\n   * @param box transaction output\n   * @param toChain event target chain\n   */\n  getAssetTransformation = (\n    box: EsploraTxOutput,\n    toChain: string\n  ): TokenTransformation | undefined => {\n    // try to build transformation using locked DOGE\n    const wrappedDoge = this.tokens.search(DOGE_CHAIN, {\n      tokenId: DOGE_NATIVE_TOKEN,\n    });\n    if (wrappedDoge.length > 0 && Object.hasOwn(wrappedDoge[0], toChain)) {\n      const dogeAmount = box.value;\n      return {\n        from: DOGE_NATIVE_TOKEN,\n        to: this.tokens.getID(wrappedDoge[0], toChain),\n        amount: dogeAmount.toString(),\n      };\n    } else {\n      return undefined;\n    }\n  };\n}\n"]}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { RosenData, TokenTransformation } from '../abstract/types';
|
|
2
|
+
import AbstractRosenDataExtractor from '../abstract/AbstractRosenDataExtractor';
|
|
3
|
+
import { DogeTxOutput } from './types';
|
|
4
|
+
import { TokenMap } from '@rosen-bridge/tokens';
|
|
5
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
6
|
+
export declare class DogeRosenExtractor extends AbstractRosenDataExtractor<string> {
|
|
7
|
+
readonly chain = "doge";
|
|
8
|
+
protected lockScriptPubKey: string;
|
|
9
|
+
constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger);
|
|
10
|
+
/**
|
|
11
|
+
* extracts RosenData from given lock transaction in DogeTx format
|
|
12
|
+
* @param serializedTransaction stringified transaction in DogeTx format
|
|
13
|
+
*/
|
|
14
|
+
extractRawData: (serializedTransaction: string) => RosenData | undefined;
|
|
15
|
+
/**
|
|
16
|
+
* extracts and builds token transformation from UTXO and tokenMap
|
|
17
|
+
* @param box transaction output
|
|
18
|
+
* @param toChain event target chain
|
|
19
|
+
*/
|
|
20
|
+
getAssetTransformation: (box: DogeTxOutput, toChain: string) => TokenTransformation | undefined;
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=DogeRosenExtractor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DogeRosenExtractor.d.ts","sourceRoot":"","sources":["../../../../lib/getRosenData/doge/DogeRosenExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACnE,OAAO,0BAA0B,MAAM,wCAAwC,CAAC;AAEhF,OAAO,EAAU,YAAY,EAAgB,MAAM,SAAS,CAAC;AAC7D,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAI/D,qBAAa,kBAAmB,SAAQ,0BAA0B,CAAC,MAAM,CAAC;IACxE,QAAQ,CAAC,KAAK,UAAc;IAC5B,SAAS,CAAC,gBAAgB,EAAE,MAAM,CAAC;gBAEvB,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE,cAAc;IAK1E;;;OAGG;IACH,cAAc,0BAA2B,MAAM,KAAG,SAAS,GAAG,SAAS,CAqFrE;IAEF;;;;OAIG;IACH,sBAAsB,QACf,YAAY,WACR,MAAM,KACd,mBAAmB,GAAG,SAAS,CAehC;CACH"}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import AbstractRosenDataExtractor from '../abstract/AbstractRosenDataExtractor';
|
|
2
|
+
import { DOGE_CHAIN, DOGE_NATIVE_TOKEN } from '../const';
|
|
3
|
+
import { parseRosenData, addressToOutputScript } from './utils';
|
|
4
|
+
import JsonBigInt from '@rosen-bridge/json-bigint';
|
|
5
|
+
export class DogeRosenExtractor extends AbstractRosenDataExtractor {
|
|
6
|
+
chain = DOGE_CHAIN;
|
|
7
|
+
lockScriptPubKey;
|
|
8
|
+
constructor(lockAddress, tokens, logger) {
|
|
9
|
+
super(lockAddress, tokens, logger);
|
|
10
|
+
this.lockScriptPubKey = addressToOutputScript(lockAddress);
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* extracts RosenData from given lock transaction in DogeTx format
|
|
14
|
+
* @param serializedTransaction stringified transaction in DogeTx format
|
|
15
|
+
*/
|
|
16
|
+
extractRawData = (serializedTransaction) => {
|
|
17
|
+
let transaction;
|
|
18
|
+
try {
|
|
19
|
+
transaction = JsonBigInt.parse(serializedTransaction);
|
|
20
|
+
}
|
|
21
|
+
catch (e) {
|
|
22
|
+
throw new Error(`Failed to parse transaction json to DogeTx format while extracting rosen data: ${e}`);
|
|
23
|
+
}
|
|
24
|
+
const baseError = `No rosen data found for tx [${transaction.id}]`;
|
|
25
|
+
try {
|
|
26
|
+
const outputs = transaction.outputs;
|
|
27
|
+
if (outputs.length < 2) {
|
|
28
|
+
this.logger.debug(baseError + `: Insufficient number of boxes`);
|
|
29
|
+
return undefined;
|
|
30
|
+
}
|
|
31
|
+
let validData = false; // an OP_RETURN box with valid data is found
|
|
32
|
+
let validLock = false; // a lock box is found with available asset transformation
|
|
33
|
+
// parse rosen data from OP_RETURN box
|
|
34
|
+
let opReturnData;
|
|
35
|
+
for (let i = 0; i < outputs.length; i++) {
|
|
36
|
+
const output = outputs[i];
|
|
37
|
+
if (output.scriptPubKey.slice(0, 2) !== '6a')
|
|
38
|
+
continue; // not an OP_RETURN utxo
|
|
39
|
+
try {
|
|
40
|
+
opReturnData = parseRosenData(output.scriptPubKey);
|
|
41
|
+
validData = true;
|
|
42
|
+
break;
|
|
43
|
+
}
|
|
44
|
+
catch (e) {
|
|
45
|
+
this.logger.debug(`Failed to extract data from OP_RETURN box [${transaction.id}.${i}]: ${e}`);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
if (!validData || !opReturnData) {
|
|
49
|
+
this.logger.debug(baseError + `: No OP_RETURN box with valid data is found`);
|
|
50
|
+
return undefined;
|
|
51
|
+
}
|
|
52
|
+
// find target chain token id
|
|
53
|
+
let assetTransformation;
|
|
54
|
+
for (let i = 0; i < outputs.length; i++) {
|
|
55
|
+
const output = outputs[i];
|
|
56
|
+
if (output.scriptPubKey !== this.lockScriptPubKey)
|
|
57
|
+
continue; // utxo address is not lock address
|
|
58
|
+
assetTransformation = this.getAssetTransformation(output, opReturnData.toChain);
|
|
59
|
+
if (assetTransformation) {
|
|
60
|
+
validLock = true;
|
|
61
|
+
break;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
if (!validLock || !assetTransformation) {
|
|
65
|
+
this.logger.debug(baseError + `: Failed to find rosen asset transformation`);
|
|
66
|
+
return undefined;
|
|
67
|
+
}
|
|
68
|
+
const fromAddress = `box:${transaction.inputs[0].txId}.${transaction.inputs[0].index}`;
|
|
69
|
+
return {
|
|
70
|
+
toChain: opReturnData.toChain,
|
|
71
|
+
toAddress: opReturnData.toAddress,
|
|
72
|
+
bridgeFee: opReturnData.bridgeFee,
|
|
73
|
+
networkFee: opReturnData.networkFee,
|
|
74
|
+
fromAddress: fromAddress,
|
|
75
|
+
sourceChainTokenId: assetTransformation.from,
|
|
76
|
+
amount: assetTransformation.amount,
|
|
77
|
+
targetChainTokenId: assetTransformation.to,
|
|
78
|
+
sourceTxId: transaction.id,
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
catch (e) {
|
|
82
|
+
this.logger.debug(`An error occurred while getting Doge rosen data: ${e}`);
|
|
83
|
+
if (e instanceof Error && e.stack) {
|
|
84
|
+
this.logger.debug(e.stack);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
return undefined;
|
|
88
|
+
};
|
|
89
|
+
/**
|
|
90
|
+
* extracts and builds token transformation from UTXO and tokenMap
|
|
91
|
+
* @param box transaction output
|
|
92
|
+
* @param toChain event target chain
|
|
93
|
+
*/
|
|
94
|
+
getAssetTransformation = (box, toChain) => {
|
|
95
|
+
// try to build transformation using locked DOGE
|
|
96
|
+
const wrappedDoge = this.tokens.search(DOGE_CHAIN, {
|
|
97
|
+
tokenId: DOGE_NATIVE_TOKEN,
|
|
98
|
+
});
|
|
99
|
+
if (wrappedDoge.length > 0 && Object.hasOwn(wrappedDoge[0], toChain)) {
|
|
100
|
+
const satoshiAmount = box.value;
|
|
101
|
+
return {
|
|
102
|
+
from: DOGE_NATIVE_TOKEN,
|
|
103
|
+
to: this.tokens.getID(wrappedDoge[0], toChain),
|
|
104
|
+
amount: satoshiAmount.toString(),
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
else {
|
|
108
|
+
return undefined;
|
|
109
|
+
}
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DogeRosenExtractor.js","sourceRoot":"","sources":["../../../../lib/getRosenData/doge/DogeRosenExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { RosenData, TokenTransformation } from '../abstract/types';\nimport AbstractRosenDataExtractor from '../abstract/AbstractRosenDataExtractor';\nimport { DOGE_CHAIN, DOGE_NATIVE_TOKEN } from '../const';\nimport { DogeTx, DogeTxOutput, OpReturnData } from './types';\nimport { TokenMap } from '@rosen-bridge/tokens';\nimport { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { parseRosenData, addressToOutputScript } from './utils';\nimport JsonBigInt from '@rosen-bridge/json-bigint';\n\nexport class DogeRosenExtractor extends AbstractRosenDataExtractor<string> {\n  readonly chain = DOGE_CHAIN;\n  protected lockScriptPubKey: string;\n\n  constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger) {\n    super(lockAddress, tokens, logger);\n    this.lockScriptPubKey = addressToOutputScript(lockAddress);\n  }\n\n  /**\n   * extracts RosenData from given lock transaction in DogeTx format\n   * @param serializedTransaction stringified transaction in DogeTx format\n   */\n  extractRawData = (serializedTransaction: string): RosenData | undefined => {\n    let transaction: DogeTx;\n    try {\n      transaction = JsonBigInt.parse(serializedTransaction);\n    } catch (e) {\n      throw new Error(\n        `Failed to parse transaction json to DogeTx format while extracting rosen data: ${e}`\n      );\n    }\n    const baseError = `No rosen data found for tx [${transaction.id}]`;\n    try {\n      const outputs = transaction.outputs;\n      if (outputs.length < 2) {\n        this.logger.debug(baseError + `: Insufficient number of boxes`);\n        return undefined;\n      }\n\n      let validData = false; // an OP_RETURN box with valid data is found\n      let validLock = false; // a lock box is found with available asset transformation\n\n      // parse rosen data from OP_RETURN box\n      let opReturnData: OpReturnData | undefined;\n      for (let i = 0; i < outputs.length; i++) {\n        const output = outputs[i];\n        if (output.scriptPubKey.slice(0, 2) !== '6a') continue; // not an OP_RETURN utxo\n\n        try {\n          opReturnData = parseRosenData(output.scriptPubKey);\n          validData = true;\n          break;\n        } catch (e) {\n          this.logger.debug(\n            `Failed to extract data from OP_RETURN box [${transaction.id}.${i}]: ${e}`\n          );\n        }\n      }\n      if (!validData || !opReturnData) {\n        this.logger.debug(\n          baseError + `: No OP_RETURN box with valid data is found`\n        );\n        return undefined;\n      }\n\n      // find target chain token id\n      let assetTransformation: TokenTransformation | undefined;\n      for (let i = 0; i < outputs.length; i++) {\n        const output = outputs[i];\n        if (output.scriptPubKey !== this.lockScriptPubKey) continue; // utxo address is not lock address\n        assetTransformation = this.getAssetTransformation(\n          output,\n          opReturnData.toChain\n        );\n        if (assetTransformation) {\n          validLock = true;\n          break;\n        }\n      }\n      if (!validLock || !assetTransformation) {\n        this.logger.debug(\n          baseError + `: Failed to find rosen asset transformation`\n        );\n        return undefined;\n      }\n\n      const fromAddress = `box:${transaction.inputs[0].txId}.${transaction.inputs[0].index}`;\n      return {\n        toChain: opReturnData.toChain,\n        toAddress: opReturnData.toAddress,\n        bridgeFee: opReturnData.bridgeFee,\n        networkFee: opReturnData.networkFee,\n        fromAddress: fromAddress,\n        sourceChainTokenId: assetTransformation.from,\n        amount: assetTransformation.amount,\n        targetChainTokenId: assetTransformation.to,\n        sourceTxId: transaction.id,\n      };\n    } catch (e) {\n      this.logger.debug(\n        `An error occurred while getting Doge rosen data: ${e}`\n      );\n      if (e instanceof Error && e.stack) {\n        this.logger.debug(e.stack);\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * extracts and builds token transformation from UTXO and tokenMap\n   * @param box transaction output\n   * @param toChain event target chain\n   */\n  getAssetTransformation = (\n    box: DogeTxOutput,\n    toChain: string\n  ): TokenTransformation | undefined => {\n    // try to build transformation using locked DOGE\n    const wrappedDoge = this.tokens.search(DOGE_CHAIN, {\n      tokenId: DOGE_NATIVE_TOKEN,\n    });\n    if (wrappedDoge.length > 0 && Object.hasOwn(wrappedDoge[0], toChain)) {\n      const satoshiAmount = box.value;\n      return {\n        from: DOGE_NATIVE_TOKEN,\n        to: this.tokens.getID(wrappedDoge[0], toChain),\n        amount: satoshiAmount.toString(),\n      };\n    } else {\n      return undefined;\n    }\n  };\n}\n"]}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
export interface OpReturnData {
|
|
2
|
+
toChain: string;
|
|
3
|
+
toAddress: string;
|
|
4
|
+
bridgeFee: string;
|
|
5
|
+
networkFee: string;
|
|
6
|
+
}
|
|
7
|
+
export interface EsploraTxInput {
|
|
8
|
+
txid: string;
|
|
9
|
+
vout: number;
|
|
10
|
+
prevout: {
|
|
11
|
+
scriptpubkey: string;
|
|
12
|
+
scriptpubkey_asm: string;
|
|
13
|
+
scriptpubkey_type: string;
|
|
14
|
+
scriptpubkey_address: string;
|
|
15
|
+
value: number;
|
|
16
|
+
};
|
|
17
|
+
scriptsig: string;
|
|
18
|
+
scriptsig_asm: string;
|
|
19
|
+
is_coinbase: false;
|
|
20
|
+
sequence: number;
|
|
21
|
+
}
|
|
22
|
+
export interface EsploraTxOutput {
|
|
23
|
+
scriptpubkey: string;
|
|
24
|
+
scriptpubkey_asm: string;
|
|
25
|
+
scriptpubkey_type: string;
|
|
26
|
+
scriptpubkey_address: string;
|
|
27
|
+
value: number;
|
|
28
|
+
}
|
|
29
|
+
export interface DogeEsploraTransaction {
|
|
30
|
+
txid: string;
|
|
31
|
+
version: number;
|
|
32
|
+
locktime: number;
|
|
33
|
+
vin: Array<EsploraTxInput>;
|
|
34
|
+
vout: Array<EsploraTxOutput>;
|
|
35
|
+
size: number;
|
|
36
|
+
weight: number;
|
|
37
|
+
fee: number;
|
|
38
|
+
status: {
|
|
39
|
+
confirmed: true;
|
|
40
|
+
block_height: number;
|
|
41
|
+
block_hash: string;
|
|
42
|
+
block_time: number;
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
export interface DogeTxInput {
|
|
46
|
+
txId: string;
|
|
47
|
+
index: number;
|
|
48
|
+
scriptPubKey: string;
|
|
49
|
+
}
|
|
50
|
+
export interface DogeTxOutput {
|
|
51
|
+
scriptPubKey: string;
|
|
52
|
+
value: bigint;
|
|
53
|
+
}
|
|
54
|
+
export interface DogeTx {
|
|
55
|
+
id: string;
|
|
56
|
+
inputs: DogeTxInput[];
|
|
57
|
+
outputs: DogeTxOutput[];
|
|
58
|
+
}
|
|
59
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../../lib/getRosenData/doge/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE;QACP,YAAY,EAAE,MAAM,CAAC;QACrB,gBAAgB,EAAE,MAAM,CAAC;QACzB,iBAAiB,EAAE,MAAM,CAAC;QAC1B,oBAAoB,EAAE,MAAM,CAAC;QAC7B,KAAK,EAAE,MAAM,CAAC;KACf,CAAC;IACF,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,EAAE,MAAM,CAAC;IACtB,WAAW,EAAE,KAAK,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,eAAe;IAC9B,YAAY,EAAE,MAAM,CAAC;IACrB,gBAAgB,EAAE,MAAM,CAAC;IACzB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,oBAAoB,EAAE,MAAM,CAAC;IAC7B,KAAK,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,sBAAsB;IACrC,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,GAAG,EAAE,KAAK,CAAC,cAAc,CAAC,CAAC;IAC3B,IAAI,EAAE,KAAK,CAAC,eAAe,CAAC,CAAC;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,GAAG,EAAE,MAAM,CAAC;IACZ,MAAM,EAAE;QACN,SAAS,EAAE,IAAI,CAAC;QAChB,YAAY,EAAE,MAAM,CAAC;QACrB,UAAU,EAAE,MAAM,CAAC;QACnB,UAAU,EAAE,MAAM,CAAC;KACpB,CAAC;CACH;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,YAAY;IAC3B,YAAY,EAAE,MAAM,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,MAAM;IACrB,EAAE,EAAE,MAAM,CAAC;IACX,MAAM,EAAE,WAAW,EAAE,CAAC;IACtB,OAAO,EAAE,YAAY,EAAE,CAAC;CACzB"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWIvZ2V0Um9zZW5EYXRhL2RvZ2UvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgT3BSZXR1cm5EYXRhIHtcbiAgdG9DaGFpbjogc3RyaW5nO1xuICB0b0FkZHJlc3M6IHN0cmluZztcbiAgYnJpZGdlRmVlOiBzdHJpbmc7XG4gIG5ldHdvcmtGZWU6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBFc3Bsb3JhVHhJbnB1dCB7XG4gIHR4aWQ6IHN0cmluZztcbiAgdm91dDogbnVtYmVyO1xuICBwcmV2b3V0OiB7XG4gICAgc2NyaXB0cHVia2V5OiBzdHJpbmc7XG4gICAgc2NyaXB0cHVia2V5X2FzbTogc3RyaW5nO1xuICAgIHNjcmlwdHB1YmtleV90eXBlOiBzdHJpbmc7XG4gICAgc2NyaXB0cHVia2V5X2FkZHJlc3M6IHN0cmluZztcbiAgICB2YWx1ZTogbnVtYmVyO1xuICB9O1xuICBzY3JpcHRzaWc6IHN0cmluZztcbiAgc2NyaXB0c2lnX2FzbTogc3RyaW5nO1xuICBpc19jb2luYmFzZTogZmFsc2U7XG4gIHNlcXVlbmNlOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXNwbG9yYVR4T3V0cHV0IHtcbiAgc2NyaXB0cHVia2V5OiBzdHJpbmc7XG4gIHNjcmlwdHB1YmtleV9hc206IHN0cmluZztcbiAgc2NyaXB0cHVia2V5X3R5cGU6IHN0cmluZztcbiAgc2NyaXB0cHVia2V5X2FkZHJlc3M6IHN0cmluZztcbiAgdmFsdWU6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEb2dlRXNwbG9yYVRyYW5zYWN0aW9uIHtcbiAgdHhpZDogc3RyaW5nO1xuICB2ZXJzaW9uOiBudW1iZXI7XG4gIGxvY2t0aW1lOiBudW1iZXI7XG4gIHZpbjogQXJyYXk8RXNwbG9yYVR4SW5wdXQ+O1xuICB2b3V0OiBBcnJheTxFc3Bsb3JhVHhPdXRwdXQ+O1xuICBzaXplOiBudW1iZXI7XG4gIHdlaWdodDogbnVtYmVyO1xuICBmZWU6IG51bWJlcjtcbiAgc3RhdHVzOiB7XG4gICAgY29uZmlybWVkOiB0cnVlO1xuICAgIGJsb2NrX2hlaWdodDogbnVtYmVyO1xuICAgIGJsb2NrX2hhc2g6IHN0cmluZztcbiAgICBibG9ja190aW1lOiBudW1iZXI7XG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRG9nZVR4SW5wdXQge1xuICB0eElkOiBzdHJpbmc7XG4gIGluZGV4OiBudW1iZXI7XG4gIHNjcmlwdFB1YktleTogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERvZ2VUeE91dHB1dCB7XG4gIHNjcmlwdFB1YktleTogc3RyaW5nO1xuICB2YWx1ZTogYmlnaW50O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIERvZ2VUeCB7XG4gIGlkOiBzdHJpbmc7XG4gIGlucHV0czogRG9nZVR4SW5wdXRbXTtcbiAgb3V0cHV0czogRG9nZVR4T3V0cHV0W107XG59XG4iXX0=
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { OpReturnData } from './types';
|
|
2
|
+
/**
|
|
3
|
+
* Converts a Dogecoin address to its corresponding output script
|
|
4
|
+
* @param addr The Dogecoin address to convert
|
|
5
|
+
* @returns The output script as a hex string
|
|
6
|
+
*/
|
|
7
|
+
export declare const addressToOutputScript: (addr: string) => string;
|
|
8
|
+
/**
|
|
9
|
+
* extracts rosen data from OP_RETURN box script pub key
|
|
10
|
+
* @param scriptPubKeyHex
|
|
11
|
+
*/
|
|
12
|
+
export declare const parseRosenData: (scriptPubKeyHex: string) => OpReturnData;
|
|
13
|
+
//# sourceMappingURL=utils.d.ts.map
|