PHP WebShell

Текущая директория: /opt/BitGoJS/modules/blockapis/dist/src/impl

Просмотр файла: BlockchairApi.js

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.BlockchairApi = void 0;
const utxo_lib_1 = require("@bitgo/utxo-lib");
const BaseHttpClient_1 = require("../BaseHttpClient");
const ApiBuilder_1 = require("../ApiBuilder");
const formatOutputId = utxo_lib_1.bitgo.formatOutputId;
class ErrorKeyNotInResponse extends Error {
    constructor(key) {
        super(`key ${key} not in response`);
    }
}
function unwrapRecord(body, key) {
    if (!(key in body.data)) {
        throw new ErrorKeyNotInResponse(key);
    }
    return body.data[key];
}
class BlockchairApi {
    static forCoin(coinName, params = {}) {
        // https://blockchair.com/api/docs#link_M0
        let blockchain;
        switch (coinName) {
            case 'btc':
                blockchain = 'bitcoin';
                break;
            case 'tbtc':
                blockchain = 'bitcoin/testnet';
                break;
            case 'bsv':
                blockchain = 'bitcoin-sv';
                break;
            case 'bch':
                blockchain = 'bitcoin-cash';
                break;
            case 'bcha':
                blockchain = 'ecash';
                break;
            case 'ltc':
                blockchain = 'litecoin';
                break;
            case 'dash':
                blockchain = 'dash';
                break;
            case 'doge':
                blockchain = 'dogecoin';
                break;
            case 'zec':
                blockchain = 'zcash';
                break;
            default:
                throw new ApiBuilder_1.ApiNotImplementedError(coinName);
        }
        const { httpClient = new BaseHttpClient_1.BaseHttpClient() } = params;
        return new BlockchairApi(httpClient.withBaseUrl(`https://api.blockchair.com/${blockchain}`), params.apiToken);
    }
    constructor(client, apiToken) {
        this.client = client;
        this.apiToken = apiToken ?? process.env.BLOCKCHAIR_TOKEN;
    }
    get(path) {
        return this.client.get(path + (this.apiToken ? `?key=${this.apiToken}` : ''));
    }
    async getAddressInfo(address) {
        if (!address || address.length === 0) {
            throw new Error('invalid address');
        }
        // https://blockchair.com/api/docs#link_300
        return (await this.get(`/dashboards/address/${address}`)).map((body) => {
            return {
                txCount: body.data[address].address.transaction_count,
                balance: body.data[address].address.balance,
            };
        });
    }
    async getUnspentsForAddresses(addr) {
        if (addr.length > 100) {
            throw new Error(`invalid size`);
        }
        const response = await this.get(`/dashboards/addresses/${addr.join(',')}`);
        return response.map((body) => {
            return body.data.utxo
                .flatMap((unspent) => {
                if (addr.includes(unspent.address)) {
                    return {
                        id: formatOutputId({ txid: unspent.transaction_hash, vout: unspent.index }),
                        address: unspent.address,
                        value: unspent.value,
                    };
                }
                return undefined;
            })
                .filter((unspent) => unspent !== undefined);
        });
    }
    async getTransaction(txid) {
        return (await this.get(`/dashboards/transaction/${txid}`)).map((body) => {
            return unwrapRecord(body, txid);
        });
    }
    async getTransactionStatus(txid) {
        let transaction;
        try {
            transaction = (await this.getTransaction(txid)).transaction;
        }
        catch (e) {
            if (e instanceof ErrorKeyNotInResponse) {
                return { found: false };
            }
            throw e;
        }
        const { block_id, time } = transaction;
        const date = new Date(Date.parse(time.replace(' ', 'T') + '.000Z' /* force UTC parsing */));
        return block_id === -1
            ? { found: true, confirmed: false }
            : {
                found: true,
                confirmed: true,
                blockHeight: block_id,
                date,
            };
    }
    async getTransactionInputs(txid) {
        return (await this.getTransaction(txid)).inputs.map((i) => {
            return {
                id: formatOutputId({ txid: i.transaction_hash, vout: i.index }),
                address: i.recipient,
                value: i.value,
            };
        });
    }
    async getTransactionIO(txid) {
        const tx = await this.getTransaction(txid);
        const inputs = tx.inputs.map((input) => {
            return {
                address: input.recipient,
            };
        });
        const outputs = tx.outputs.map((output) => {
            return {
                address: output.recipient,
            };
        });
        return {
            inputs,
            outputs,
        };
    }
    async getTransactionSpends(txid) {
        return (await this.getTransaction(txid)).outputs.map((o) => o.spending_transaction_hash
            ? {
                txid: o.spending_transaction_hash,
                vin: o.spending_index,
            }
            : { txid: undefined, vin: undefined });
    }
    async getTransactionHex(txid) {
        return (await this.get(`/raw/transaction/${txid}`)).map((body) => {
            return unwrapRecord(body, txid).raw_transaction;
        });
    }
}
exports.BlockchairApi = BlockchairApi;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BlockchairApi.js","sourceRoot":"","sources":["../../../src/impl/BlockchairApi.ts"],"names":[],"mappings":"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","sourcesContent":["import { bitgo } from '@bitgo/utxo-lib';\nimport { BaseHttpClient, HttpClient, Response } from '../BaseHttpClient';\nimport { ApiNotImplementedError } from '../ApiBuilder';\nimport { AddressApi, AddressInfo } from '../AddressApi';\nimport { OutputSpend, TransactionIO, UtxoApi } from '../UtxoApi';\nimport { TransactionStatus } from '../TransactionApi';\n\ntype Unspent = bitgo.Unspent;\nconst formatOutputId = bitgo.formatOutputId;\n\ntype BlockchairResponse<T> = {\n  data: T;\n};\n\ntype BlockchairRecordResponse<T> = BlockchairResponse<Record<string, T>>;\n\nclass ErrorKeyNotInResponse extends Error {\n  constructor(key: string) {\n    super(`key ${key} not in response`);\n  }\n}\n\nfunction unwrapRecord<T>(body: BlockchairRecordResponse<T>, key: string): T {\n  if (!(key in body.data)) {\n    throw new ErrorKeyNotInResponse(key);\n  }\n  return body.data[key];\n}\n\n// https://blockchair.com/api/docs#link_300\ntype BlockchairUnspent = {\n  transaction_hash: string;\n  index: number;\n  recipient: string;\n  value: number;\n  block_id: number;\n  script_hex: string;\n  spending_transaction_hash: string;\n  spending_index: number;\n  address: string;\n};\n\n// https://blockchair.com/api/docs#link_300\ntype BlockchairAddress = {\n  address: {\n    transaction_hash: string;\n    // vout\n    index: number;\n    value: number;\n    block_id: number;\n    transaction_count: number;\n    balance: number;\n  };\n  utxo: BlockchairUnspent[];\n};\n\n// https://blockchair.com/api/docs#link_200\ntype BlockchairTransaction = {\n  transaction: {\n    /**\n      The block number it's included in.\n      If the transaction is in the mempool, data.{:hash}ᵢ.transaction.block_id yields -1\n    */\n    block_id: number;\n    /**\n     * Like https://tc39.es/ecma262/#sec-date-time-string-format but with a space instead of 'T'\n     * YYYY-MM-DD HH:mm:ss\n     */\n    time: string;\n  };\n  inputs: BlockchairUnspent[];\n  outputs: BlockchairUnspent[];\n};\n\n// https://blockchair.com/api/docs#link_201\ntype BlockchairRawTransaction = {\n  raw_transaction: string;\n};\n\nexport class BlockchairApi implements AddressApi, UtxoApi {\n  protected readonly apiToken?: string;\n\n  static forCoin(coinName: string, params: { apiToken?: string; httpClient?: HttpClient } = {}): BlockchairApi {\n    // https://blockchair.com/api/docs#link_M0\n    let blockchain;\n    switch (coinName) {\n      case 'btc':\n        blockchain = 'bitcoin';\n        break;\n      case 'tbtc':\n        blockchain = 'bitcoin/testnet';\n        break;\n      case 'bsv':\n        blockchain = 'bitcoin-sv';\n        break;\n      case 'bch':\n        blockchain = 'bitcoin-cash';\n        break;\n      case 'bcha':\n        blockchain = 'ecash';\n        break;\n      case 'ltc':\n        blockchain = 'litecoin';\n        break;\n      case 'dash':\n        blockchain = 'dash';\n        break;\n      case 'doge':\n        blockchain = 'dogecoin';\n        break;\n      case 'zec':\n        blockchain = 'zcash';\n        break;\n      default:\n        throw new ApiNotImplementedError(coinName);\n    }\n    const { httpClient = new BaseHttpClient() } = params;\n    return new BlockchairApi(httpClient.withBaseUrl(`https://api.blockchair.com/${blockchain}`), params.apiToken);\n  }\n\n  constructor(public client: HttpClient, apiToken?: string) {\n    this.apiToken = apiToken ?? process.env.BLOCKCHAIR_TOKEN;\n  }\n\n  get<T>(path: string): Promise<Response<T>> {\n    return this.client.get(path + (this.apiToken ? `?key=${this.apiToken}` : ''));\n  }\n\n  async getAddressInfo(address: string): Promise<AddressInfo> {\n    if (!address || address.length === 0) {\n      throw new Error('invalid address');\n    }\n    // https://blockchair.com/api/docs#link_300\n    return (await this.get<BlockchairRecordResponse<BlockchairAddress>>(`/dashboards/address/${address}`)).map(\n      (body) => {\n        return {\n          txCount: body.data[address].address.transaction_count,\n          balance: body.data[address].address.balance,\n        };\n      }\n    );\n  }\n\n  async getUnspentsForAddresses(addr: string[]): Promise<Unspent[]> {\n    if (addr.length > 100) {\n      throw new Error(`invalid size`);\n    }\n\n    const response = await this.get<BlockchairResponse<{ utxo: BlockchairUnspent[] }>>(\n      `/dashboards/addresses/${addr.join(',')}`\n    );\n\n    return response.map((body) => {\n      return body.data.utxo\n        .flatMap((unspent): Unspent | undefined => {\n          if (addr.includes(unspent.address)) {\n            return {\n              id: formatOutputId({ txid: unspent.transaction_hash, vout: unspent.index }),\n              address: unspent.address,\n              value: unspent.value,\n            };\n          }\n          return undefined;\n        })\n        .filter((unspent): unspent is Unspent => unspent !== undefined);\n    });\n  }\n\n  async getTransaction(txid: string): Promise<BlockchairTransaction> {\n    return (await this.get<BlockchairRecordResponse<BlockchairTransaction>>(`/dashboards/transaction/${txid}`)).map(\n      (body) => {\n        return unwrapRecord(body, txid);\n      }\n    );\n  }\n\n  async getTransactionStatus(txid: string): Promise<TransactionStatus> {\n    let transaction;\n    try {\n      transaction = (await this.getTransaction(txid)).transaction;\n    } catch (e) {\n      if (e instanceof ErrorKeyNotInResponse) {\n        return { found: false };\n      }\n      throw e;\n    }\n    const { block_id, time } = transaction;\n    const date = new Date(Date.parse(time.replace(' ', 'T') + '.000Z' /* force UTC parsing */));\n    return block_id === -1\n      ? { found: true, confirmed: false }\n      : {\n          found: true,\n          confirmed: true,\n          blockHeight: block_id,\n          date,\n        };\n  }\n\n  async getTransactionInputs(txid: string): Promise<Unspent[]> {\n    return (await this.getTransaction(txid)).inputs.map((i) => {\n      return {\n        id: formatOutputId({ txid: i.transaction_hash, vout: i.index }),\n        address: i.recipient,\n        value: i.value,\n      };\n    });\n  }\n\n  async getTransactionIO(txid: string): Promise<TransactionIO> {\n    const tx = await this.getTransaction(txid);\n    const inputs = tx.inputs.map((input) => {\n      return {\n        address: input.recipient,\n      };\n    });\n    const outputs = tx.outputs.map((output) => {\n      return {\n        address: output.recipient,\n      };\n    });\n    return {\n      inputs,\n      outputs,\n    };\n  }\n\n  async getTransactionSpends(txid: string): Promise<OutputSpend[]> {\n    return (await this.getTransaction(txid)).outputs.map((o) =>\n      o.spending_transaction_hash\n        ? {\n            txid: o.spending_transaction_hash,\n            vin: o.spending_index,\n          }\n        : { txid: undefined, vin: undefined }\n    );\n  }\n\n  async getTransactionHex(txid: string): Promise<string> {\n    return (await this.get<BlockchairRecordResponse<BlockchairRawTransaction>>(`/raw/transaction/${txid}`)).map(\n      (body) => {\n        return unwrapRecord(body, txid).raw_transaction;\n      }\n    );\n  }\n}\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!