@bitgo-beta/utxo-bin 2.8.3-beta.5 → 2.8.3-beta.50
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/README.md +42 -80
- package/dist/src/InputParser.d.ts +44 -0
- package/dist/src/InputParser.d.ts.map +1 -0
- package/dist/src/InputParser.js +247 -0
- package/dist/src/OutputParser.d.ts +24 -0
- package/dist/src/OutputParser.d.ts.map +1 -0
- package/dist/src/OutputParser.js +52 -0
- package/dist/src/Parser.d.ts +7 -1
- package/dist/src/Parser.d.ts.map +1 -1
- package/dist/src/Parser.js +26 -2
- package/dist/src/ParserTx.d.ts +28 -0
- package/dist/src/ParserTx.d.ts.map +1 -0
- package/dist/src/ParserTx.js +94 -0
- package/dist/src/TxParser.d.ts +9 -6
- package/dist/src/TxParser.d.ts.map +1 -1
- package/dist/src/TxParser.js +21 -23
- package/dist/src/commands.d.ts +6 -1
- package/dist/src/commands.d.ts.map +1 -1
- package/dist/src/commands.js +78 -17
- package/dist/src/fetch.d.ts +4 -3
- package/dist/src/fetch.d.ts.map +1 -1
- package/dist/src/fetch.js +14 -3
- package/dist/src/format.d.ts +1 -1
- package/dist/src/format.d.ts.map +1 -1
- package/dist/src/format.js +4 -3
- package/dist/src/parseUnknown.d.ts +3 -0
- package/dist/src/parseUnknown.d.ts.map +1 -0
- package/dist/src/parseUnknown.js +50 -0
- package/dist/src/readStdin.d.ts +5 -0
- package/dist/src/readStdin.d.ts.map +1 -0
- package/dist/src/readStdin.js +42 -0
- package/dist/test/fixtures.d.ts +24 -0
- package/dist/test/fixtures.d.ts.map +1 -0
- package/dist/test/fixtures.js +88 -0
- package/dist/test/parseAddress.d.ts +10 -0
- package/dist/test/parseAddress.d.ts.map +1 -0
- package/dist/test/parseAddress.js +85 -0
- package/dist/test/parseTransaction.d.ts +2 -0
- package/dist/test/parseTransaction.d.ts.map +1 -0
- package/dist/test/parseTransaction.js +63 -0
- package/dist/tsconfig.tsbuildinfo +299 -144
- package/package.json +6 -5
- package/dist/src/InputOutputParser.d.ts +0 -35
- package/dist/src/InputOutputParser.d.ts.map +0 -1
- package/dist/src/InputOutputParser.js +0 -196
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getFixtureString = exports.getTransactionWithSpendType = exports.getPsbt = exports.formatTreeNoColor = void 0;
|
|
4
|
+
const fs_1 = require("fs");
|
|
5
|
+
const chalk_1 = require("chalk");
|
|
6
|
+
const utxolib = require("@bitgo-beta/utxo-lib");
|
|
7
|
+
const format_1 = require("../src/format");
|
|
8
|
+
function formatTreeNoColor(n, { showAll }) {
|
|
9
|
+
return format_1.formatTree(n, { hide: showAll ? [] : undefined, chalk: new chalk_1.Instance({ level: 0 }) });
|
|
10
|
+
}
|
|
11
|
+
exports.formatTreeNoColor = formatTreeNoColor;
|
|
12
|
+
const walletKeys = utxolib.testutil.getDefaultWalletKeys();
|
|
13
|
+
async function getPsbt(network, params, { writeFixture } = {}) {
|
|
14
|
+
const inputs = [
|
|
15
|
+
{ scriptType: params.spendType === 'keyPath' ? 'taprootKeyPathSpend' : params.scriptType, value: BigInt(10000) },
|
|
16
|
+
];
|
|
17
|
+
const outputs = [{ scriptType: params.scriptType, value: BigInt(9000) }];
|
|
18
|
+
let stage = params.fixtureType;
|
|
19
|
+
if (stage.startsWith('psbt')) {
|
|
20
|
+
stage = stage.slice('psbt'.length).toLowerCase();
|
|
21
|
+
}
|
|
22
|
+
if (stage !== 'unsigned' && stage !== 'halfsigned' && stage !== 'fullsigned') {
|
|
23
|
+
throw new Error(`invalid stage ${stage}`);
|
|
24
|
+
}
|
|
25
|
+
if (params.spendType === 'keyPath' && writeFixture === undefined) {
|
|
26
|
+
// because we currently cannot create deterministic signatures for taprootKeyPathSpend, we
|
|
27
|
+
// store a copy in fixtures/psbt and use that instead of creating a new one
|
|
28
|
+
const filename = `test/fixtures/psbt/${params.scriptType}.${params.spendType}.${stage}.json`;
|
|
29
|
+
try {
|
|
30
|
+
const psbtHex = JSON.parse(await fs_1.promises.readFile(filename, 'utf8'));
|
|
31
|
+
const transaction = utxolib.bitgo.createPsbtFromHex(psbtHex, network);
|
|
32
|
+
return { transaction, prevOutputs: undefined };
|
|
33
|
+
}
|
|
34
|
+
catch (e) {
|
|
35
|
+
if (e.code === 'ENOENT') {
|
|
36
|
+
return await getPsbt(network, params, { writeFixture: filename });
|
|
37
|
+
}
|
|
38
|
+
throw e;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
const transaction = utxolib.testutil.constructPsbt(inputs, outputs, network, walletKeys, stage);
|
|
42
|
+
if (writeFixture) {
|
|
43
|
+
await fs_1.promises.writeFile(writeFixture, JSON.stringify(transaction.toHex()), 'utf8');
|
|
44
|
+
}
|
|
45
|
+
return {
|
|
46
|
+
transaction,
|
|
47
|
+
prevOutputs: undefined,
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
exports.getPsbt = getPsbt;
|
|
51
|
+
async function getTransactionWithSpendType(network, params) {
|
|
52
|
+
if (params.fixtureType !== 'networkFullSigned') {
|
|
53
|
+
return await getPsbt(network, params);
|
|
54
|
+
}
|
|
55
|
+
if (params.scriptType === 'p2trMusig2' && !params.spendType) {
|
|
56
|
+
throw new Error('p2trMusig2 requires params');
|
|
57
|
+
}
|
|
58
|
+
const filename = `spend_${params.scriptType}${params.spendType ? `${params.spendType}` : ''}.json`;
|
|
59
|
+
const f = await JSON.parse(await fs_1.promises.readFile(`../utxo-lib/test/integration_local_rpc/fixtures/${utxolib.getNetworkName(network)}/v1/${filename}`, 'utf8'));
|
|
60
|
+
function getPrevOut(i) {
|
|
61
|
+
for (const t of inputTxs) {
|
|
62
|
+
if (t.getHash().equals(i.hash) && i.index in t.outs) {
|
|
63
|
+
return t.outs[i.index];
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
throw new Error(`missing input ${utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(i))}`);
|
|
67
|
+
}
|
|
68
|
+
const inputTxs = f.inputs.map((i) => utxolib.bitgo.createTransactionFromBuffer(Buffer.from(i.hex, 'hex'), network, { amountType: 'bigint' }));
|
|
69
|
+
const tx = utxolib.bitgo.createTransactionFromBuffer(Buffer.from(f.transaction.hex, 'hex'), network, {
|
|
70
|
+
amountType: 'bigint',
|
|
71
|
+
});
|
|
72
|
+
return { transaction: tx, prevOutputs: tx.ins.map((i) => getPrevOut(i)) };
|
|
73
|
+
}
|
|
74
|
+
exports.getTransactionWithSpendType = getTransactionWithSpendType;
|
|
75
|
+
async function getFixtureString(path, defaultValue) {
|
|
76
|
+
try {
|
|
77
|
+
return await fs_1.promises.readFile(path, 'utf8');
|
|
78
|
+
}
|
|
79
|
+
catch (e) {
|
|
80
|
+
if (e.code === 'ENOENT') {
|
|
81
|
+
await fs_1.promises.writeFile(path, defaultValue, 'utf8');
|
|
82
|
+
throw new Error(`wrote default value for ${path}`);
|
|
83
|
+
}
|
|
84
|
+
throw e;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
exports.getFixtureString = getFixtureString;
|
|
88
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fixtures.js","sourceRoot":"","sources":["../../test/fixtures.ts"],"names":[],"mappings":"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","sourcesContent":["import { promises as fs } from 'fs';\n\nimport { Instance } from 'chalk';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\n\nimport { formatTree } from '../src/format';\nimport { ParserNode } from '../src/Parser';\n\nexport function formatTreeNoColor(n: ParserNode, { showAll }: { showAll: boolean }): string {\n  return formatTree(n, { hide: showAll ? [] : undefined, chalk: new Instance({ level: 0 }) });\n}\n\nexport type ParsedFixture =\n  | {\n      transaction: utxolib.bitgo.UtxoTransaction<bigint>;\n      prevOutputs: utxolib.TxOutput<bigint>[];\n    }\n  | {\n      transaction: utxolib.bitgo.UtxoPsbt;\n      prevOutputs: undefined;\n    };\n\ntype FixtureParams = {\n  fixtureType: 'psbtUnsigned' | 'psbtHalfSigned' | 'psbtFullSigned' | 'networkFullSigned';\n  scriptType: utxolib.bitgo.outputScripts.ScriptType2Of3;\n  spendType?: 'keyPath' | 'scriptPath';\n};\n\nconst walletKeys = utxolib.testutil.getDefaultWalletKeys();\n\nexport async function getPsbt(\n  network: utxolib.Network,\n  params: FixtureParams,\n  { writeFixture }: { writeFixture?: string } = {}\n): Promise<ParsedFixture> {\n  const inputs: utxolib.testutil.Input[] = [\n    { scriptType: params.spendType === 'keyPath' ? 'taprootKeyPathSpend' : params.scriptType, value: BigInt(10_000) },\n  ];\n  const outputs = [{ scriptType: params.scriptType, value: BigInt(9_000) }];\n  let stage: string = params.fixtureType;\n  if (stage.startsWith('psbt')) {\n    stage = stage.slice('psbt'.length).toLowerCase();\n  }\n  if (stage !== 'unsigned' && stage !== 'halfsigned' && stage !== 'fullsigned') {\n    throw new Error(`invalid stage ${stage}`);\n  }\n  if (params.spendType === 'keyPath' && writeFixture === undefined) {\n    // because we currently cannot create deterministic signatures for taprootKeyPathSpend, we\n    // store a copy in fixtures/psbt and use that instead of creating a new one\n    const filename = `test/fixtures/psbt/${params.scriptType}.${params.spendType}.${stage}.json`;\n    try {\n      const psbtHex = JSON.parse(await fs.readFile(filename, 'utf8'));\n      const transaction = utxolib.bitgo.createPsbtFromHex(psbtHex, network);\n      return { transaction, prevOutputs: undefined };\n    } catch (e) {\n      if (e.code === 'ENOENT') {\n        return await getPsbt(network, params, { writeFixture: filename });\n      }\n      throw e;\n    }\n  }\n  const transaction = utxolib.testutil.constructPsbt(inputs, outputs, network, walletKeys, stage);\n  if (writeFixture) {\n    await fs.writeFile(writeFixture, JSON.stringify(transaction.toHex()), 'utf8');\n  }\n  return {\n    transaction,\n    prevOutputs: undefined,\n  };\n}\n\nexport async function getTransactionWithSpendType(\n  network: utxolib.Network,\n  params: FixtureParams\n): Promise<ParsedFixture> {\n  if (params.fixtureType !== 'networkFullSigned') {\n    return await getPsbt(network, params);\n  }\n\n  if (params.scriptType === 'p2trMusig2' && !params.spendType) {\n    throw new Error('p2trMusig2 requires params');\n  }\n\n  const filename = `spend_${params.scriptType}${params.spendType ? `${params.spendType}` : ''}.json`;\n  type Fixture = {\n    transaction: { hex: string };\n    inputs: { hex: string }[];\n  };\n\n  const f: Fixture = await JSON.parse(\n    await fs.readFile(\n      `../utxo-lib/test/integration_local_rpc/fixtures/${utxolib.getNetworkName(network)}/v1/${filename}`,\n      'utf8'\n    )\n  );\n\n  function getPrevOut(i: utxolib.TxInput): utxolib.TxOutput<bigint> {\n    for (const t of inputTxs) {\n      if (t.getHash().equals(i.hash) && i.index in t.outs) {\n        return t.outs[i.index];\n      }\n    }\n    throw new Error(`missing input ${utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(i))}`);\n  }\n\n  const inputTxs = f.inputs.map((i) =>\n    utxolib.bitgo.createTransactionFromBuffer(Buffer.from(i.hex, 'hex'), network, { amountType: 'bigint' })\n  );\n  const tx = utxolib.bitgo.createTransactionFromBuffer(Buffer.from(f.transaction.hex, 'hex'), network, {\n    amountType: 'bigint',\n  });\n  return { transaction: tx, prevOutputs: tx.ins.map((i) => getPrevOut(i)) };\n}\n\nexport async function getFixtureString(path: string, defaultValue: string): Promise<string> {\n  try {\n    return await fs.readFile(path, 'utf8');\n  } catch (e) {\n    if ((e as any).code === 'ENOENT') {\n      await fs.writeFile(path, defaultValue, 'utf8');\n      throw new Error(`wrote default value for ${path}`);\n    }\n    throw e;\n  }\n}\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { BIP32Interface } from 'bip32';
|
|
2
|
+
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
3
|
+
declare type Triple<T> = [T, T, T];
|
|
4
|
+
declare type KeyTriple = Triple<BIP32Interface>;
|
|
5
|
+
declare const scriptTypes: readonly ["p2sh", "p2shP2wsh", "p2wsh", "p2tr", "p2trMusig2", "p2pkh", "p2wkh"];
|
|
6
|
+
declare type ScriptType = (typeof scriptTypes)[number];
|
|
7
|
+
export declare function getKeyTriple(seed: string): KeyTriple;
|
|
8
|
+
export declare function isSupportedDepositType(network: utxolib.Network, scriptType: ScriptType): boolean;
|
|
9
|
+
export {};
|
|
10
|
+
//# sourceMappingURL=parseAddress.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parseAddress.d.ts","sourceRoot":"","sources":["../../test/parseAddress.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,MAAM,OAAO,CAAC;AACvC,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAKhD,aAAK,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;AAE3B,aAAK,SAAS,GAAG,MAAM,CAAC,cAAc,CAAC,CAAC;AAGxC,QAAA,MAAM,WAAW,iFAAqF,CAAC;AACvG,aAAK,UAAU,GAAG,CAAC,OAAO,WAAW,CAAC,CAAC,MAAM,CAAC,CAAC;AAK/C,wBAAgB,YAAY,CAAC,IAAI,EAAE,MAAM,GAAG,SAAS,CAEpD;AAED,wBAAgB,sBAAsB,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,EAAE,UAAU,EAAE,UAAU,GAAG,OAAO,CAUhG"}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.isSupportedDepositType = exports.getKeyTriple = void 0;
|
|
4
|
+
const assert = require("assert");
|
|
5
|
+
const crypto = require("crypto");
|
|
6
|
+
const yargs = require("yargs");
|
|
7
|
+
const utxolib = require("@bitgo-beta/utxo-lib");
|
|
8
|
+
const commands_1 = require("../src/commands");
|
|
9
|
+
const fixtures_1 = require("./fixtures");
|
|
10
|
+
const scriptTypesSingleSig = ['p2pkh', 'p2wkh'];
|
|
11
|
+
const scriptTypes = [...utxolib.bitgo.outputScripts.scriptTypes2Of3, ...scriptTypesSingleSig];
|
|
12
|
+
function getKey(seed) {
|
|
13
|
+
return utxolib.bip32.fromSeed(crypto.createHash('sha256').update(seed).digest());
|
|
14
|
+
}
|
|
15
|
+
function getKeyTriple(seed) {
|
|
16
|
+
return [getKey(seed + '.0'), getKey(seed + '.1'), getKey(seed + '.2')];
|
|
17
|
+
}
|
|
18
|
+
exports.getKeyTriple = getKeyTriple;
|
|
19
|
+
function isSupportedDepositType(network, scriptType) {
|
|
20
|
+
if (scriptType === 'p2pkh') {
|
|
21
|
+
return true;
|
|
22
|
+
}
|
|
23
|
+
if (scriptType === 'p2wkh') {
|
|
24
|
+
return utxolib.supportsSegwit(network);
|
|
25
|
+
}
|
|
26
|
+
return utxolib.bitgo.outputScripts.isSupportedScriptType(network, scriptType);
|
|
27
|
+
}
|
|
28
|
+
exports.isSupportedDepositType = isSupportedDepositType;
|
|
29
|
+
/**
|
|
30
|
+
*
|
|
31
|
+
* @param keys - Pubkeys to use for generating the address.
|
|
32
|
+
* If scriptType is single-sig, the first key will be used.
|
|
33
|
+
* @param scriptType
|
|
34
|
+
* @param network
|
|
35
|
+
* @return {Buffer} scriptPubKey
|
|
36
|
+
*/
|
|
37
|
+
function createScriptPubKey(keys, scriptType, network) {
|
|
38
|
+
const pubkeys = keys.map((k) => k.publicKey);
|
|
39
|
+
switch (scriptType) {
|
|
40
|
+
case 'p2sh':
|
|
41
|
+
case 'p2shP2wsh':
|
|
42
|
+
case 'p2wsh':
|
|
43
|
+
case 'p2tr':
|
|
44
|
+
case 'p2trMusig2':
|
|
45
|
+
return utxolib.bitgo.outputScripts.createOutputScript2of3(pubkeys, scriptType).scriptPubKey;
|
|
46
|
+
case 'p2pkh':
|
|
47
|
+
return utxolib.payments.p2pkh({ pubkey: keys[0].publicKey }).output;
|
|
48
|
+
case 'p2wkh':
|
|
49
|
+
return utxolib.payments.p2wpkh({ pubkey: keys[0].publicKey }).output;
|
|
50
|
+
default:
|
|
51
|
+
throw new Error(`unsupported output type ${scriptType}`);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
function getAddresses(n) {
|
|
55
|
+
const keys = getKeyTriple('parseAddress');
|
|
56
|
+
return scriptTypes
|
|
57
|
+
.filter((t) => isSupportedDepositType(n, t))
|
|
58
|
+
.flatMap((t) => utxolib.addressFormat.addressFormats
|
|
59
|
+
.filter((format) => utxolib.addressFormat.isSupportedAddressFormat(format, n))
|
|
60
|
+
.map((format) => [
|
|
61
|
+
format,
|
|
62
|
+
utxolib.addressFormat.fromOutputScriptWithFormat(createScriptPubKey(keys, t, n), format, n),
|
|
63
|
+
]));
|
|
64
|
+
}
|
|
65
|
+
function parse(address, args) {
|
|
66
|
+
return commands_1.getAddressParser(yargs.command(commands_1.cmdParseAddress).parseSync(args)).parse(address);
|
|
67
|
+
}
|
|
68
|
+
function testParseAddress(network, addressFormat, address, args, suffix) {
|
|
69
|
+
describe(`parse address ${address} with arguments ${args.join(' ')}`, function () {
|
|
70
|
+
it(`formats address`, async function () {
|
|
71
|
+
const formatted = fixtures_1.formatTreeNoColor(parse(address, args), { showAll: true });
|
|
72
|
+
const addrNoColon = address.replace(':', '_');
|
|
73
|
+
assert.strictEqual(await fixtures_1.getFixtureString(`test/fixtures/formatAddress_${utxolib.getNetworkName(network)}_${addressFormat}_${addrNoColon}${suffix}`, formatted), formatted);
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
utxolib.getNetworkList().forEach((n) => {
|
|
78
|
+
getAddresses(n).forEach(([addressFormat, address], i) => {
|
|
79
|
+
testParseAddress(n, addressFormat, address, [], '.txt');
|
|
80
|
+
if ([utxolib.networks.bitcoin, utxolib.networks.bitcoincash, utxolib.networks.ecash].includes(n) && i === 0) {
|
|
81
|
+
testParseAddress(n, addressFormat, address, ['--all'], '.all.txt');
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseAddress.js","sourceRoot":"","sources":["../../test/parseAddress.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport * as crypto from 'crypto';\nimport * as yargs from 'yargs';\nimport { BIP32Interface } from 'bip32';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\n\nimport { cmdParseAddress, getAddressParser } from '../src/commands';\nimport { getFixtureString, formatTreeNoColor } from './fixtures';\n\ntype Triple<T> = [T, T, T];\n\ntype KeyTriple = Triple<BIP32Interface>;\n\nconst scriptTypesSingleSig = ['p2pkh', 'p2wkh'] as const;\nconst scriptTypes = [...utxolib.bitgo.outputScripts.scriptTypes2Of3, ...scriptTypesSingleSig] as const;\ntype ScriptType = (typeof scriptTypes)[number];\n\nfunction getKey(seed: string): BIP32Interface {\n  return utxolib.bip32.fromSeed(crypto.createHash('sha256').update(seed).digest());\n}\nexport function getKeyTriple(seed: string): KeyTriple {\n  return [getKey(seed + '.0'), getKey(seed + '.1'), getKey(seed + '.2')];\n}\n\nexport function isSupportedDepositType(network: utxolib.Network, scriptType: ScriptType): boolean {\n  if (scriptType === 'p2pkh') {\n    return true;\n  }\n\n  if (scriptType === 'p2wkh') {\n    return utxolib.supportsSegwit(network);\n  }\n\n  return utxolib.bitgo.outputScripts.isSupportedScriptType(network, scriptType);\n}\n\n/**\n *\n * @param keys - Pubkeys to use for generating the address.\n *               If scriptType is single-sig, the first key will be used.\n * @param scriptType\n * @param network\n * @return {Buffer} scriptPubKey\n */\nfunction createScriptPubKey(keys: KeyTriple, scriptType: ScriptType, network: utxolib.Network): Buffer {\n  const pubkeys = keys.map((k) => k.publicKey);\n\n  switch (scriptType) {\n    case 'p2sh':\n    case 'p2shP2wsh':\n    case 'p2wsh':\n    case 'p2tr':\n    case 'p2trMusig2':\n      return utxolib.bitgo.outputScripts.createOutputScript2of3(pubkeys, scriptType).scriptPubKey;\n    case 'p2pkh':\n      return utxolib.payments.p2pkh({ pubkey: keys[0].publicKey }).output as Buffer;\n    case 'p2wkh':\n      return utxolib.payments.p2wpkh({ pubkey: keys[0].publicKey }).output as Buffer;\n    default:\n      throw new Error(`unsupported output type ${scriptType}`);\n  }\n}\n\nfunction getAddresses(n: utxolib.Network): [format: string, address: string][] {\n  const keys = getKeyTriple('parseAddress');\n  return scriptTypes\n    .filter((t) => isSupportedDepositType(n, t))\n    .flatMap((t) =>\n      utxolib.addressFormat.addressFormats\n        .filter((format) => utxolib.addressFormat.isSupportedAddressFormat(format, n))\n        .map((format): [string, string] => [\n          format,\n          utxolib.addressFormat.fromOutputScriptWithFormat(createScriptPubKey(keys, t, n), format, n),\n        ])\n    );\n}\n\nfunction parse(address: string, args: string[]) {\n  return getAddressParser(yargs.command(cmdParseAddress).parseSync(args)).parse(address);\n}\n\nfunction testParseAddress(\n  network: utxolib.Network,\n  addressFormat: string,\n  address: string,\n  args: string[],\n  suffix: string\n) {\n  describe(`parse address ${address} with arguments ${args.join(' ')}`, function () {\n    it(`formats address`, async function () {\n      const formatted = formatTreeNoColor(parse(address, args), { showAll: true });\n      const addrNoColon = address.replace(':', '_');\n      assert.strictEqual(\n        await getFixtureString(\n          `test/fixtures/formatAddress_${utxolib.getNetworkName(network)}_${addressFormat}_${addrNoColon}${suffix}`,\n          formatted\n        ),\n        formatted\n      );\n    });\n  });\n}\n\nutxolib.getNetworkList().forEach((n) => {\n  getAddresses(n).forEach(([addressFormat, address], i) => {\n    testParseAddress(n, addressFormat, address, [], '.txt');\n    if ([utxolib.networks.bitcoin, utxolib.networks.bitcoincash, utxolib.networks.ecash].includes(n) && i === 0) {\n      testParseAddress(n, addressFormat, address, ['--all'], '.all.txt');\n    }\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parseTransaction.d.ts","sourceRoot":"","sources":["../../test/parseTransaction.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const yargs = require("yargs");
|
|
4
|
+
const assert = require("assert");
|
|
5
|
+
const utxolib = require("@bitgo-beta/utxo-lib");
|
|
6
|
+
const fixtures_1 = require("./fixtures");
|
|
7
|
+
const commands_1 = require("../src/commands");
|
|
8
|
+
function getArgs({ showAll }) {
|
|
9
|
+
return showAll ? ['--all'] : [];
|
|
10
|
+
}
|
|
11
|
+
function getParams() {
|
|
12
|
+
return ['psbtUnsigned', 'psbtHalfSigned', 'psbtFullSigned', 'networkFullSigned'].flatMap((fixtureType) => {
|
|
13
|
+
return [false, true].flatMap((showAll) => {
|
|
14
|
+
return utxolib.bitgo.outputScripts.scriptTypes2Of3.flatMap((scriptType) => {
|
|
15
|
+
if (scriptType === 'p2trMusig2') {
|
|
16
|
+
return [
|
|
17
|
+
{ scriptType, spendType: 'keyPath', showAll, fixtureType },
|
|
18
|
+
{ scriptType, spendType: 'scriptPath', showAll, fixtureType },
|
|
19
|
+
];
|
|
20
|
+
}
|
|
21
|
+
return [{ scriptType, spendType: undefined, showAll, fixtureType }];
|
|
22
|
+
});
|
|
23
|
+
});
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
getParams().forEach(({ scriptType, spendType, fixtureType, showAll }) => {
|
|
27
|
+
describe(`parse ${fixtureType} ${scriptType} ${spendType ? spendType : 'default'} spend [args=${getArgs({
|
|
28
|
+
showAll,
|
|
29
|
+
})}`, function () {
|
|
30
|
+
function parse(tx, prevOutputs) {
|
|
31
|
+
return commands_1.getTxParser(yargs.command(commands_1.cmdParseTx).parse([...getArgs({ showAll }), '--parseError=throw'])).parse(tx, {
|
|
32
|
+
prevOutputs,
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
let fixture;
|
|
36
|
+
before(async function () {
|
|
37
|
+
fixture = await fixtures_1.getTransactionWithSpendType(utxolib.networks.testnet, {
|
|
38
|
+
scriptType,
|
|
39
|
+
spendType,
|
|
40
|
+
fixtureType,
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
it(`parses`, function () {
|
|
44
|
+
parse(fixture.transaction);
|
|
45
|
+
if (fixture.prevOutputs) {
|
|
46
|
+
parse(fixture.transaction, fixture.prevOutputs);
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
[false, true].forEach((usePrevOuts) => {
|
|
50
|
+
it(`formats [usePrevOuts=${usePrevOuts}]`, async function () {
|
|
51
|
+
if (usePrevOuts && !fixture.prevOutputs) {
|
|
52
|
+
this.skip();
|
|
53
|
+
}
|
|
54
|
+
const formatted = fixtures_1.formatTreeNoColor(parse(fixture.transaction, usePrevOuts ? fixture.prevOutputs : undefined), {
|
|
55
|
+
showAll,
|
|
56
|
+
});
|
|
57
|
+
const fixtureName = spendType ? `${scriptType}_${spendType}` : scriptType;
|
|
58
|
+
assert.strictEqual(formatted, await fixtures_1.getFixtureString(`test/fixtures/format_${fixtureName}_${fixtureType}${showAll ? '_all' : ''}${usePrevOuts ? '_prevOuts' : ''}.txt`, formatted));
|
|
59
|
+
});
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFyc2VUcmFuc2FjdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3Rlc3QvcGFyc2VUcmFuc2FjdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUNBLCtCQUErQjtBQUMvQixpQ0FBaUM7QUFDakMsZ0RBQWdEO0FBQ2hELHlDQUE2RztBQUU3Ryw4Q0FBMEQ7QUFVMUQsU0FBUyxPQUFPLENBQUMsRUFBRSxPQUFPLEVBQXdCO0lBQ2hELE9BQU8sT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7QUFDbEMsQ0FBQztBQUVELFNBQVMsU0FBUztJQUNoQixPQUFRLENBQUMsY0FBYyxFQUFFLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLG1CQUFtQixDQUFXLENBQUMsT0FBTyxDQUFDLENBQUMsV0FBVyxFQUFFLEVBQUU7UUFDbEgsT0FBTyxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsRUFBRTtZQUN2QyxPQUFPLE9BQU8sQ0FBQyxLQUFLLENBQUMsYUFBYSxDQUFDLGVBQWUsQ0FBQyxPQUFPLENBQUMsQ0FBQyxVQUFVLEVBQWdCLEVBQUU7Z0JBQ3RGLElBQUksVUFBVSxLQUFLLFlBQVksRUFBRTtvQkFDL0IsT0FBTzt3QkFDTCxFQUFFLFVBQVUsRUFBRSxTQUFTLEVBQUUsU0FBUyxFQUFFLE9BQU8sRUFBRSxXQUFXLEVBQUU7d0JBQzFELEVBQUUsVUFBVSxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsT0FBTyxFQUFFLFdBQVcsRUFBRTtxQkFDOUQsQ0FBQztpQkFDSDtnQkFDRCxPQUFPLENBQUMsRUFBRSxVQUFVLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxPQUFPLEVBQUUsV0FBVyxFQUFFLENBQUMsQ0FBQztZQUN0RSxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDO0FBRUQsU0FBUyxFQUFFLENBQUMsT0FBTyxDQUFDLENBQUMsRUFBRSxVQUFVLEVBQUUsU0FBUyxFQUFFLFdBQVcsRUFBRSxPQUFPLEVBQUUsRUFBRSxFQUFFO0lBQ3RFLFFBQVEsQ0FBQyxTQUFTLFdBQVcsSUFBSSxVQUFVLElBQUksU0FBUyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLFNBQVMsZ0JBQWdCLE9BQU8sQ0FBQztRQUN0RyxPQUFPO0tBQ1IsQ0FBQyxFQUFFLEVBQUU7UUFDSixTQUFTLEtBQUssQ0FBQyxFQUFZLEVBQUUsV0FBd0M7WUFDbkUsT0FBTyxzQkFBVyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMscUJBQVUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLEdBQUcsT0FBTyxDQUFDLEVBQUUsT0FBTyxFQUFFLENBQUMsRUFBRSxvQkFBb0IsQ0FBQyxDQUFRLENBQUMsQ0FBQyxLQUFLLENBQy9HLEVBQUUsRUFDRjtnQkFDRSxXQUFXO2FBQ1osQ0FDRixDQUFDO1FBQ0osQ0FBQztRQUVELElBQUksT0FBc0IsQ0FBQztRQUMzQixNQUFNLENBQUMsS0FBSztZQUNWLE9BQU8sR0FBRyxNQUFNLHNDQUEyQixDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTyxFQUFFO2dCQUNwRSxVQUFVO2dCQUNWLFNBQVM7Z0JBQ1QsV0FBVzthQUNaLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO1FBRUgsRUFBRSxDQUFDLFFBQVEsRUFBRTtZQUNYLEtBQUssQ0FBQyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUM7WUFDM0IsSUFBSSxPQUFPLENBQUMsV0FBVyxFQUFFO2dCQUN2QixLQUFLLENBQUMsT0FBTyxDQUFDLFdBQVcsRUFBRSxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUM7YUFDakQ7UUFDSCxDQUFDLENBQUMsQ0FBQztRQUVILENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDLFdBQVcsRUFBRSxFQUFFO1lBQ3BDLEVBQUUsQ0FBQyx3QkFBd0IsV0FBVyxHQUFHLEVBQUUsS0FBSztnQkFDOUMsSUFBSSxXQUFXLElBQUksQ0FBQyxPQUFPLENBQUMsV0FBVyxFQUFFO29CQUN2QyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7aUJBQ2I7Z0JBQ0QsTUFBTSxTQUFTLEdBQUcsNEJBQWlCLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxXQUFXLEVBQUUsV0FBVyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsRUFBRTtvQkFDN0csT0FBTztpQkFDUixDQUFDLENBQUM7Z0JBQ0gsTUFBTSxXQUFXLEdBQUcsU0FBUyxDQUFDLENBQUMsQ0FBQyxHQUFHLFVBQVUsSUFBSSxTQUFTLEVBQUUsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDO2dCQUMxRSxNQUFNLENBQUMsV0FBVyxDQUNoQixTQUFTLEVBQ1QsTUFBTSwyQkFBZ0IsQ0FDcEIsd0JBQXdCLFdBQVcsSUFBSSxXQUFXLEdBQUcsT0FBTyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsR0FDeEUsV0FBVyxDQUFDLENBQUMsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLEVBQzlCLE1BQU0sRUFDTixTQUFTLENBQ1YsQ0FDRixDQUFDO1lBQ0osQ0FBQyxDQUFDLENBQUM7UUFDTCxDQUFDLENBQUMsQ0FBQztJQUNMLENBQUMsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBtb2NoYSBmcm9tICdtb2NoYSc7XG5pbXBvcnQgKiBhcyB5YXJncyBmcm9tICd5YXJncyc7XG5pbXBvcnQgKiBhcyBhc3NlcnQgZnJvbSAnYXNzZXJ0JztcbmltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuaW1wb3J0IHsgZ2V0Rml4dHVyZVN0cmluZywgZ2V0VHJhbnNhY3Rpb25XaXRoU3BlbmRUeXBlLCBmb3JtYXRUcmVlTm9Db2xvciwgUGFyc2VkRml4dHVyZSB9IGZyb20gJy4vZml4dHVyZXMnO1xuaW1wb3J0IHsgUGFyc2VyTm9kZSB9IGZyb20gJy4uL3NyYy9QYXJzZXInO1xuaW1wb3J0IHsgY21kUGFyc2VUeCwgZ2V0VHhQYXJzZXIgfSBmcm9tICcuLi9zcmMvY29tbWFuZHMnO1xuaW1wb3J0IHsgUGFyc2VyVHggfSBmcm9tICcuLi9zcmMvUGFyc2VyVHgnO1xuXG50eXBlIFRlc3RQYXJhbXMgPSB7XG4gIHNjcmlwdFR5cGU6IHV0eG9saWIuYml0Z28ub3V0cHV0U2NyaXB0cy5TY3JpcHRUeXBlMk9mMztcbiAgc3BlbmRUeXBlOiAna2V5UGF0aCcgfCAnc2NyaXB0UGF0aCcgfCB1bmRlZmluZWQ7XG4gIGZpeHR1cmVUeXBlOiAncHNidFVuc2lnbmVkJyB8ICdwc2J0SGFsZlNpZ25lZCcgfCAncHNidEZ1bGxTaWduZWQnIHwgJ25ldHdvcmtGdWxsU2lnbmVkJztcbiAgc2hvd0FsbDogYm9vbGVhbjtcbn07XG5cbmZ1bmN0aW9uIGdldEFyZ3MoeyBzaG93QWxsIH06IHsgc2hvd0FsbDogYm9vbGVhbiB9KTogc3RyaW5nW10ge1xuICByZXR1cm4gc2hvd0FsbCA/IFsnLS1hbGwnXSA6IFtdO1xufVxuXG5mdW5jdGlvbiBnZXRQYXJhbXMoKTogVGVzdFBhcmFtc1tdIHtcbiAgcmV0dXJuIChbJ3BzYnRVbnNpZ25lZCcsICdwc2J0SGFsZlNpZ25lZCcsICdwc2J0RnVsbFNpZ25lZCcsICduZXR3b3JrRnVsbFNpZ25lZCddIGFzIGNvbnN0KS5mbGF0TWFwKChmaXh0dXJlVHlwZSkgPT4ge1xuICAgIHJldHVybiBbZmFsc2UsIHRydWVdLmZsYXRNYXAoKHNob3dBbGwpID0+IHtcbiAgICAgIHJldHVybiB1dHhvbGliLmJpdGdvLm91dHB1dFNjcmlwdHMuc2NyaXB0VHlwZXMyT2YzLmZsYXRNYXAoKHNjcmlwdFR5cGUpOiBUZXN0UGFyYW1zW10gPT4ge1xuICAgICAgICBpZiAoc2NyaXB0VHlwZSA9PT0gJ3AydHJNdXNpZzInKSB7XG4gICAgICAgICAgcmV0dXJuIFtcbiAgICAgICAgICAgIHsgc2NyaXB0VHlwZSwgc3BlbmRUeXBlOiAna2V5UGF0aCcsIHNob3dBbGwsIGZpeHR1cmVUeXBlIH0sXG4gICAgICAgICAgICB7IHNjcmlwdFR5cGUsIHNwZW5kVHlwZTogJ3NjcmlwdFBhdGgnLCBzaG93QWxsLCBmaXh0dXJlVHlwZSB9LFxuICAgICAgICAgIF07XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIFt7IHNjcmlwdFR5cGUsIHNwZW5kVHlwZTogdW5kZWZpbmVkLCBzaG93QWxsLCBmaXh0dXJlVHlwZSB9XTtcbiAgICAgIH0pO1xuICAgIH0pO1xuICB9KTtcbn1cblxuZ2V0UGFyYW1zKCkuZm9yRWFjaCgoeyBzY3JpcHRUeXBlLCBzcGVuZFR5cGUsIGZpeHR1cmVUeXBlLCBzaG93QWxsIH0pID0+IHtcbiAgZGVzY3JpYmUoYHBhcnNlICR7Zml4dHVyZVR5cGV9ICR7c2NyaXB0VHlwZX0gJHtzcGVuZFR5cGUgPyBzcGVuZFR5cGUgOiAnZGVmYXVsdCd9IHNwZW5kIFthcmdzPSR7Z2V0QXJncyh7XG4gICAgc2hvd0FsbCxcbiAgfSl9YCwgZnVuY3Rpb24gKCkge1xuICAgIGZ1bmN0aW9uIHBhcnNlKHR4OiBQYXJzZXJUeCwgcHJldk91dHB1dHM/OiB1dHhvbGliLlR4T3V0cHV0PGJpZ2ludD5bXSk6IFBhcnNlck5vZGUge1xuICAgICAgcmV0dXJuIGdldFR4UGFyc2VyKHlhcmdzLmNvbW1hbmQoY21kUGFyc2VUeCkucGFyc2UoWy4uLmdldEFyZ3MoeyBzaG93QWxsIH0pLCAnLS1wYXJzZUVycm9yPXRocm93J10pIGFzIGFueSkucGFyc2UoXG4gICAgICAgIHR4LFxuICAgICAgICB7XG4gICAgICAgICAgcHJldk91dHB1dHMsXG4gICAgICAgIH1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgbGV0IGZpeHR1cmU6IFBhcnNlZEZpeHR1cmU7XG4gICAgYmVmb3JlKGFzeW5jIGZ1bmN0aW9uICgpIHtcbiAgICAgIGZpeHR1cmUgPSBhd2FpdCBnZXRUcmFuc2FjdGlvbldpdGhTcGVuZFR5cGUodXR4b2xpYi5uZXR3b3Jrcy50ZXN0bmV0LCB7XG4gICAgICAgIHNjcmlwdFR5cGUsXG4gICAgICAgIHNwZW5kVHlwZSxcbiAgICAgICAgZml4dHVyZVR5cGUsXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KGBwYXJzZXNgLCBmdW5jdGlvbiAoKSB7XG4gICAgICBwYXJzZShmaXh0dXJlLnRyYW5zYWN0aW9uKTtcbiAgICAgIGlmIChmaXh0dXJlLnByZXZPdXRwdXRzKSB7XG4gICAgICAgIHBhcnNlKGZpeHR1cmUudHJhbnNhY3Rpb24sIGZpeHR1cmUucHJldk91dHB1dHMpO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgW2ZhbHNlLCB0cnVlXS5mb3JFYWNoKCh1c2VQcmV2T3V0cykgPT4ge1xuICAgICAgaXQoYGZvcm1hdHMgW3VzZVByZXZPdXRzPSR7dXNlUHJldk91dHN9XWAsIGFzeW5jIGZ1bmN0aW9uICh0aGlzOiBtb2NoYS5Db250ZXh0KSB7XG4gICAgICAgIGlmICh1c2VQcmV2T3V0cyAmJiAhZml4dHVyZS5wcmV2T3V0cHV0cykge1xuICAgICAgICAgIHRoaXMuc2tpcCgpO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IGZvcm1hdHRlZCA9IGZvcm1hdFRyZWVOb0NvbG9yKHBhcnNlKGZpeHR1cmUudHJhbnNhY3Rpb24sIHVzZVByZXZPdXRzID8gZml4dHVyZS5wcmV2T3V0cHV0cyA6IHVuZGVmaW5lZCksIHtcbiAgICAgICAgICBzaG93QWxsLFxuICAgICAgICB9KTtcbiAgICAgICAgY29uc3QgZml4dHVyZU5hbWUgPSBzcGVuZFR5cGUgPyBgJHtzY3JpcHRUeXBlfV8ke3NwZW5kVHlwZX1gIDogc2NyaXB0VHlwZTtcbiAgICAgICAgYXNzZXJ0LnN0cmljdEVxdWFsKFxuICAgICAgICAgIGZvcm1hdHRlZCxcbiAgICAgICAgICBhd2FpdCBnZXRGaXh0dXJlU3RyaW5nKFxuICAgICAgICAgICAgYHRlc3QvZml4dHVyZXMvZm9ybWF0XyR7Zml4dHVyZU5hbWV9XyR7Zml4dHVyZVR5cGV9JHtzaG93QWxsID8gJ19hbGwnIDogJyd9JHtcbiAgICAgICAgICAgICAgdXNlUHJldk91dHMgPyAnX3ByZXZPdXRzJyA6ICcnXG4gICAgICAgICAgICB9LnR4dGAsXG4gICAgICAgICAgICBmb3JtYXR0ZWRcbiAgICAgICAgICApXG4gICAgICAgICk7XG4gICAgICB9KTtcbiAgICB9KTtcbiAgfSk7XG59KTtcbiJdfQ==
|