PHP WebShell

Текущая директория: /opt/BitGoJS/modules/sdk-coin-icp/dist/src/lib

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

"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.Utils = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const principal_1 = require("@dfinity/principal");
const agent = __importStar(require("@dfinity/agent"));
const crypto_1 = __importDefault(require("crypto"));
const crc_32_1 = __importDefault(require("crc-32"));
const iface_1 = require("./iface");
const keyPair_1 = require("./keyPair");
const messageCompiled = require('../../resources/messageCompiled');
const { encode, decode, Encoder } = require('cbor-x/index-no-eval'); // The "cbor-x" library is used here because it supports modern features like BigInt. do not replace it with "cbor as "cbor" is not compatible with Rust's serde_cbor when handling big numbers.
const js_sha256_1 = __importDefault(require("js-sha256"));
const bignumber_js_1 = __importDefault(require("bignumber.js"));
const secp256k1_1 = require("@noble/curves/secp256k1");
//custom encoder that avoids tagging
const encoder = new Encoder({
    structuredClone: false,
    useToJSON: false,
    mapsAsObjects: false,
    largeBigIntToFloat: false,
});
class Utils {
    constructor() {
        this.signPayload = (privateKey, payloadHex) => {
            const privateKeyBytes = Buffer.from(privateKey, 'hex');
            const payloadHash = crypto_1.default.createHash('sha256').update(Buffer.from(payloadHex, 'hex')).digest('hex');
            const signature = secp256k1_1.secp256k1.sign(payloadHash, privateKeyBytes);
            const r = Buffer.from(signature.r.toString(16).padStart(64, '0'), 'hex');
            const s = Buffer.from(signature.s.toString(16).padStart(64, '0'), 'hex');
            return Buffer.concat([r, s]).toString('hex');
        };
    }
    /** @inheritdoc */
    isValidSignature(signature) {
        throw new sdk_core_1.MethodNotImplementedError();
    }
    /**
     * gets the gas data of this transaction.
     */
    //TODO WIN-4242: to moved to a config and eventually to an API for dynamic value
    feeData() {
        return '-10000';
    }
    /**
     * Checks if the provided address is a valid ICP address.
     *
     * @param {string} address - The address to validate.
     * @returns {boolean} - Returns `true` if the address is valid, otherwise `false`.
     */
    isValidAddress(address) {
        const rootAddress = this.validateMemoAndReturnRootAddress(address);
        return rootAddress !== undefined && this.isValidHash(rootAddress);
    }
    /**
     * Validates the memo ID in the address and returns the root address.
     *
     * @param {string} address - The address to validate and extract the root address from.
     * @returns {string | undefined} - The root address if valid, otherwise `undefined`.
     */
    validateMemoAndReturnRootAddress(address) {
        if (!address) {
            return undefined;
        }
        const [rootAddress, memoId] = address.split('?memoId=');
        if (memoId && this.validateMemo(BigInt(memoId))) {
            return rootAddress;
        }
        return address;
    }
    /**
     * Checks if the provided hex string is a valid public key.
     *
     * A valid public key can be either compressed or uncompressed:
     * - Compressed public keys are 33 bytes long and start with either 0x02 or 0x03.
     * - Uncompressed public keys are 65 bytes long and start with 0x04.
     *
     * @param {string} hexStr - The hex string representation of the public key to validate.
     * @returns {boolean} - Returns `true` if the hex string is a valid public key, otherwise `false`.
     */
    isValidPublicKey(hexStr) {
        if (!this.isValidHex(hexStr) || !this.isValidLength(hexStr)) {
            return false;
        }
        const pubKeyBytes = this.hexToBytes(hexStr);
        const firstByte = pubKeyBytes[0];
        const validCompressed = pubKeyBytes.length === 33 && (firstByte === 2 || firstByte === 3);
        const validUncompressed = pubKeyBytes.length === 65 && firstByte === 4;
        return validCompressed || validUncompressed;
    }
    /**
     * Encodes a value into CBOR format and returns it as a hex string.
     *
     * @param {unknown} value - The value to encode.
     * @returns {string} - The CBOR encoded value as a hex string.
     */
    cborEncode(value) {
        if (value === undefined) {
            throw new Error('Value to encode cannot be undefined.');
        }
        const cborData = encode(value);
        return Buffer.from(cborData).toString('hex');
    }
    /**
     * Checks if the length of the given hexadecimal string is valid.
     * A valid length is either 66 characters (33 bytes) or 130 characters (65 bytes).
     *
     * @param {string} hexStr - The hexadecimal string to check.
     * @returns {boolean} - Returns `true` if the length is valid, otherwise `false`.
     */
    isValidLength(hexStr) {
        return hexStr.length / 2 === 33 || hexStr.length / 2 === 65;
    }
    /**
     * Checks if the provided string is a valid hexadecimal string.
     *
     * A valid hexadecimal string consists of pairs of hexadecimal digits (0-9, a-f, A-F).
     *
     * @param hexStr - The string to be validated as a hexadecimal string.
     * @returns True if the string is a valid hexadecimal string, false otherwise.
     */
    isValidHex(hexStr) {
        return /^([0-9a-fA-F]{2})+$/.test(hexStr);
    }
    /**
     * Converts a hexadecimal string to a Uint8Array.
     *
     * @param {string} hex - The hexadecimal string to convert.
     * @returns {Uint8Array} The resulting byte array.
     */
    hexToBytes(hex) {
        const bytes = new Uint8Array(hex.length / 2);
        for (let i = 0; i < hex.length; i += 2) {
            bytes[i / 2] = parseInt(hex.substr(i, 2), 16);
        }
        return bytes;
    }
    /** @inheritdoc */
    isValidPrivateKey(key) {
        return this.isValidKey(key);
    }
    /**
     * Validates whether the provided key is a valid ICP private key.
     *
     * This function attempts to create a new instance of `IcpKeyPair` using the provided key.
     * If the key is valid, the function returns `true`. If the key is invalid, an error is thrown,
     * and the function returns `false`.
     *
     * @param {string} key - The private key to validate.
     * @returns {boolean} - `true` if the key is valid, `false` otherwise.
     */
    isValidKey(key) {
        try {
            new keyPair_1.KeyPair({ prv: key });
            return true;
        }
        catch {
            return false;
        }
    }
    /**
     * Compresses an uncompressed public key.
     *
     * @param {string} uncompressedKey - The uncompressed public key in hexadecimal format.
     * @returns {string} - The compressed public key in hexadecimal format.
     * @throws {Error} - If the input key is not a valid uncompressed public key.
     */
    compressPublicKey(uncompressedKey) {
        if (uncompressedKey.startsWith('02') || uncompressedKey.startsWith('03')) {
            return uncompressedKey;
        }
        if (!uncompressedKey.startsWith('04') || uncompressedKey.length !== 130) {
            throw new Error('Invalid uncompressed public key format.');
        }
        const xHex = uncompressedKey.slice(2, 66);
        const yHex = uncompressedKey.slice(66);
        const y = BigInt(`0x${yHex}`);
        const prefix = y % 2n === 0n ? '02' : '03';
        return `${prefix}${xHex}`;
    }
    /**
     * Converts a public key from its hexadecimal string representation to DER format.
     *
     * @param {string} publicKeyHex - The public key in hexadecimal string format.
     * @returns The public key in DER format as a Uint8Array.
     */
    getPublicKeyInDERFormat(publicKeyHex) {
        const publicKeyBuffer = Buffer.from(publicKeyHex, 'hex');
        const ellipticKey = secp256k1_1.secp256k1.ProjectivePoint.fromHex(publicKeyBuffer.toString('hex'));
        const uncompressedPublicKeyHex = ellipticKey.toHex(false);
        const derEncodedKey = agent.wrapDER(Buffer.from(uncompressedPublicKeyHex, 'hex'), agent.SECP256K1_OID);
        return derEncodedKey;
    }
    /**
     * Converts a public key in hexadecimal format to a Dfinity Principal ID.
     *
     * @param {string} publicKeyHex - The public key in hexadecimal format.
     * @returns The corresponding Dfinity Principal ID.
     */
    getPrincipalIdFromPublicKey(publicKeyHex) {
        const derEncodedKey = this.getPublicKeyInDERFormat(publicKeyHex);
        const principalId = principal_1.Principal.selfAuthenticating(Buffer.from(derEncodedKey));
        return principalId;
    }
    /**
     * Derives a DfinityPrincipal from a given public key in hexadecimal format.
     *
     * @param {string} publicKeyHex - The public key in hexadecimal format.
     * @returns The derived DfinityPrincipal.
     * @throws Will throw an error if the principal cannot be derived from the public key.
     */
    derivePrincipalFromPublicKey(publicKeyHex) {
        try {
            const derEncodedKey = this.getPublicKeyInDERFormat(publicKeyHex);
            const principalId = principal_1.Principal.selfAuthenticating(Buffer.from(derEncodedKey));
            const principal = principal_1.Principal.fromUint8Array(principalId.toUint8Array());
            return principal;
        }
        catch (error) {
            throw new Error(`Failed to derive principal from public key: ${error.message}`);
        }
    }
    /**
     * Converts a DfinityPrincipal and an optional subAccount to a string representation of an account ID.
     *
     * @param {DfinityPrincipal} principal - The principal to convert.
     * @param {Uint8Array} [subAccount=new Uint8Array(32)] - An optional sub-account, defaults to a 32-byte array of zeros.
     * @returns {string} The hexadecimal string representation of the account ID.
     */
    fromPrincipal(principal, subAccount = new Uint8Array(32)) {
        const principalBytes = Buffer.from(principal.toUint8Array().buffer);
        return this.getAccountIdFromPrincipalBytes(this.getAccountIdPrefix(), principalBytes, subAccount);
    }
    getAccountIdFromPrincipalBytes(ACCOUNT_ID_PREFIX, principalBytes, subAccount) {
        const combinedBytes = Buffer.concat([ACCOUNT_ID_PREFIX, principalBytes, subAccount]);
        const sha224Hash = crypto_1.default.createHash('sha224').update(combinedBytes).digest();
        const checksum = Buffer.alloc(4);
        checksum.writeUInt32BE(crc_32_1.default.buf(sha224Hash) >>> 0, 0);
        const accountIdBytes = Buffer.concat([checksum, sha224Hash]);
        return accountIdBytes.toString('hex');
    }
    /**
     * Retrieves the address associated with a given hex-encoded public key.
     *
     * @param {string} hexEncodedPublicKey - The public key in hex-encoded format.
     * @returns {Promise<string>} A promise that resolves to the address derived from the provided public key.
     * @throws {Error} Throws an error if the provided public key is not in a valid hex-encoded format.
     */
    async getAddressFromPublicKey(hexEncodedPublicKey) {
        if (!this.isValidPublicKey(hexEncodedPublicKey)) {
            throw new Error('Invalid hex-encoded public key format.');
        }
        const compressedKey = this.compressPublicKey(hexEncodedPublicKey);
        const keyPair = new keyPair_1.KeyPair({ pub: compressedKey });
        return keyPair.getAddress();
    }
    /**
     * Generates a new key pair. If a seed is provided, it will be used to generate the key pair.
     *
     * @param {Buffer} [seed] - Optional seed for key generation.
     * @returns {KeyPair} - The generated key pair containing both public and private keys.
     * @throws {Error} - If the private key is missing in the generated key pair.
     */
    generateKeyPair(seed) {
        const keyPair = seed ? new keyPair_1.KeyPair({ seed }) : new keyPair_1.KeyPair();
        const { pub, prv } = keyPair.getKeys();
        if (!prv) {
            throw new Error('Private key is missing in the generated key pair.');
        }
        return { pub, prv };
    }
    /**
     * Validates the provided fee.
     *
     * @param {string} fee - The fee to validate.
     * @throws {BuildTransactionError} - If the fee is zero or invalid.
     */
    validateFee(fee) {
        const feeValue = new bignumber_js_1.default(fee);
        if (feeValue.isZero()) {
            throw new sdk_core_1.BuildTransactionError('Fee cannot be zero');
        }
        return true;
    }
    /** @inheritdoc */
    validateValue(value) {
        if (value.isLessThanOrEqualTo(0)) {
            throw new sdk_core_1.BuildTransactionError('amount cannot be less than or equal to zero');
        }
        return true;
    }
    /**
     * Validates the provided memo.
     *
     * @param {number | BigInt} memo - The memo to validate.
     * @returns {boolean} - Returns `true` if the memo is valid.
     * @throws {BuildTransactionError} - If the memo is invalid.
     */
    validateMemo(memo) {
        const memoNumber = Number(memo);
        if (memoNumber < 0 || Number.isNaN(memoNumber)) {
            throw new sdk_core_1.BuildTransactionError('Invalid memo');
        }
        return true;
    }
    validateExpireTime(expireTime) {
        if (Number(expireTime) < Date.now() * 1000000) {
            throw new sdk_core_1.BuildTransactionError('Invalid expiry time');
        }
        return true;
    }
    /**
     * Validates the raw transaction data to ensure it has a valid format in the blockchain context.
     *
     * @param {IcpTransactionData} transactionData - The transaction data to validate.
     * @throws {ParseTransactionError} If the transaction data is invalid.
     */
    validateRawTransaction(transactionData) {
        if (!transactionData) {
            throw new sdk_core_1.ParseTransactionError('Transaction data is missing.');
        }
        const { senderPublicKeyHex, senderAddress, receiverAddress } = transactionData;
        if (senderPublicKeyHex && !this.isValidPublicKey(senderPublicKeyHex)) {
            throw new sdk_core_1.ParseTransactionError('Sender public key is invalid.');
        }
        if (!this.isValidAddress(senderAddress)) {
            throw new sdk_core_1.ParseTransactionError('Sender address is invalid.');
        }
        if (!this.isValidAddress(receiverAddress)) {
            throw new sdk_core_1.ParseTransactionError('Receiver address is invalid.');
        }
        this.validateFee(transactionData.fee);
        this.validateValue(new bignumber_js_1.default(transactionData.amount));
        this.validateMemo(transactionData.memo);
        this.validateExpireTime(transactionData.expiryTime);
    }
    /**
     *
     * @param {object} update
     * @returns {Buffer}
     */
    generateHttpCanisterUpdateId(update) {
        return this.HttpCanisterUpdateRepresentationIndependentHash(update);
    }
    /**
     * Generates a representation-independent hash for an HTTP canister update.
     *
     * @param {HttpCanisterUpdate} update - The HTTP canister update object.
     * @returns {Buffer} - The hash of the update object.
     */
    HttpCanisterUpdateRepresentationIndependentHash(update) {
        const updateMap = {
            request_type: iface_1.RequestType.CALL,
            canister_id: update.canister_id,
            method_name: update.method_name,
            arg: update.arg,
            ingress_expiry: update.ingress_expiry,
            sender: update.sender,
        };
        return this.hashOfMap(updateMap);
    }
    /**
     * Generates a SHA-256 hash for a given map object.
     *
     * @param {Record<string, unknown>} map - The map object to hash.
     * @returns {Buffer} - The resulting hash as a Buffer.
     */
    hashOfMap(map) {
        const hashes = [];
        for (const key in map) {
            hashes.push(this.hashKeyVal(key, map[key]));
        }
        hashes.sort((buf0, buf1) => buf0.compare(buf1));
        return this.sha256(hashes);
    }
    /**
     * Generates a hash for a key-value pair.
     *
     * @param {string} key - The key to hash.
     * @param {string | Buffer | BigInt} val - The value to hash.
     * @returns {Buffer} - The resulting hash as a Buffer.
     */
    hashKeyVal(key, val) {
        const keyHash = this.hashString(key);
        const valHash = this.hashVal(val);
        return Buffer.concat([keyHash, valHash]);
    }
    /**
     * Generates a SHA-256 hash for a given string.
     *
     * @param {string} value - The string to hash.
     * @returns {Buffer} - The resulting hash as a Buffer.
     */
    hashString(value) {
        return this.sha256([Buffer.from(value)]);
    }
    /**
     * Generates a hash for a 64-bit unsigned integer.
     *
     * @param {bigint} n - The 64-bit unsigned integer to hash.
     * @returns {Buffer} - The resulting hash as a Buffer.
     */
    hashU64(n) {
        const buf = Buffer.allocUnsafe(10);
        let i = 0;
        while (true) {
            const byte = Number(n & BigInt(0x7f));
            n >>= BigInt(7);
            if (n === BigInt(0)) {
                buf[i] = byte;
                break;
            }
            else {
                buf[i] = byte | 0x80;
                ++i;
            }
        }
        return this.hashBytes(buf.subarray(0, i + 1));
    }
    /**
     * Generates a SHA-256 hash for an array of elements.
     *
     * @param {Array<any>} elements - The array of elements to hash.
     * @returns {Buffer} - The resulting hash as a Buffer.
     */
    hashArray(elements) {
        return this.sha256(elements.map(this.hashVal));
    }
    /**
     * Generates a hash for a given value.
     *
     * @param {string | Buffer | BigInt | number | Array<unknown>} val - The value to hash.
     * @returns {Buffer} - The resulting hash as a Buffer.
     * @throws {Error} - If the value type is unsupported.
     */
    hashVal(val) {
        if (typeof val === 'string') {
            return utils.hashString(val);
        }
        else if (Buffer.isBuffer(val) || val instanceof Uint8Array) {
            return utils.hashBytes(val);
        }
        else if (typeof val === 'bigint' || typeof val === 'number') {
            return utils.hashU64(BigInt(val));
        }
        else if (Array.isArray(val)) {
            return utils.hashArray(val);
        }
        else {
            throw new Error(`Unsupported value type for hashing: ${typeof val}`);
        }
    }
    /**
     * Computes the SHA-256 hash of the given buffer.
     *
     * @param value - The buffer to hash.
     * @returns The SHA-256 hash of the input buffer.
     */
    hashBytes(value) {
        return this.sha256([value]);
    }
    /**
     * Computes the SHA-256 hash of the provided array of Buffer chunks.
     *
     * @param {Array<Buffer>} chunks - An array of Buffer objects to be hashed.
     * @returns {Buffer} - The resulting SHA-256 hash as a Buffer.
     */
    sha256(chunks) {
        const hasher = js_sha256_1.default.sha256.create();
        chunks.forEach((chunk) => hasher.update(chunk));
        return Buffer.from(hasher.arrayBuffer());
    }
    /**
     * Converts a hexadecimal string to a Buffer.
     *
     * @param hex - The hexadecimal string to convert.
     * @returns A Buffer containing the binary data represented by the hexadecimal string.
     */
    blobFromHex(hex) {
        return Buffer.from(hex, 'hex');
    }
    /**
     * Converts a binary blob (Buffer) to a hexadecimal string.
     *
     * @param {Buffer} blob - The binary data to be converted.
     * @returns {string} The hexadecimal representation of the binary data.
     */
    blobToHex(blob) {
        return blob.toString('hex');
    }
    /**
     * Decodes a given CBOR-encoded buffer.
     *
     * @param buffer - The CBOR-encoded buffer to decode.
     * @returns The decoded data.
     */
    cborDecode(buffer) {
        const res = decode(buffer);
        return res;
    }
    /**
     * Generates a Buffer containing the domain IC request string.
     *
     * @returns {Buffer} A Buffer object initialized with the string '\x0Aic-request'.
     */
    getDomainICRequest() {
        return Buffer.from('\x0Aic-request');
    }
    /**
     * Combines the domain IC request buffer with the provided message ID buffer to create signature data.
     *
     * @param {Buffer} messageId - The buffer containing the message ID.
     * @returns {Buffer} - The concatenated buffer containing the domain IC request and the message ID.
     */
    makeSignatureData(messageId) {
        return Buffer.concat([this.getDomainICRequest(), messageId]);
    }
    /**
     * Extracts the recipient information from the provided ICP transaction data.
     *
     * @param {IcpTransactionData} icpTransactionData - The ICP transaction data containing the receiver's address and amount.
     * @returns {Recipient[]} An array containing a single recipient object with the receiver's address and amount.
     */
    getRecipients(icpTransactionData) {
        return {
            address: icpTransactionData.receiverAddress,
            amount: icpTransactionData.amount,
        };
    }
    getTransactionSignature(signatureMap, update) {
        return signatureMap.get(this.blobToHex(this.makeSignatureData(this.generateHttpCanisterUpdateId(update))));
    }
    getMetaData(memo, timestamp, ingressEnd) {
        let currentTime = Date.now() * 1000000;
        if (timestamp) {
            currentTime = Number(timestamp);
        }
        let ingressStartTime, ingressEndTime;
        if (ingressEnd) {
            ingressEndTime = Number(ingressEnd);
            ingressStartTime = ingressEndTime - iface_1.MAX_INGRESS_TTL; // 5 mins in nanoseconds
        }
        else {
            ingressStartTime = currentTime;
            ingressEndTime = ingressStartTime + iface_1.MAX_INGRESS_TTL; // 5 mins in nanoseconds
        }
        const metaData = {
            created_at_time: currentTime,
            ingress_start: ingressStartTime,
            ingress_end: ingressEndTime,
            memo: memo,
        };
        return { metaData, ingressEndTime };
    }
    convertSenderBlobToPrincipal(senderBlob) {
        const MAX_LENGTH_IN_BYTES = 29;
        if (senderBlob.length > MAX_LENGTH_IN_BYTES) {
            throw new Error('Bytes too long for a valid Principal');
        }
        const principalBytes = new Uint8Array(MAX_LENGTH_IN_BYTES);
        principalBytes.set(senderBlob.slice(0, senderBlob.length));
        return principalBytes;
    }
    fromArgs(arg) {
        const SendRequestMessage = messageCompiled.SendRequest;
        const args = SendRequestMessage.decode(arg);
        const transformedArgs = {
            payment: { receiverGets: { e8s: Number(args.payment.receiverGets.e8s) } },
            maxFee: { e8s: Number(args.maxFee.e8s) },
            to: { hash: Buffer.from(args.to.hash) },
            createdAtTime: { timestampNanos: (0, bignumber_js_1.default)(args.createdAtTime.timestampNanos.toString()).toNumber() },
            memo: { memo: Number(args.memo.memo.toString()) },
        };
        return transformedArgs;
    }
    async toArg(args) {
        const SendRequestMessage = messageCompiled.SendRequest;
        const errMsg = SendRequestMessage.verify(args);
        if (errMsg)
            throw new Error(errMsg);
        const message = SendRequestMessage.create(args);
        return SendRequestMessage.encode(message).finish();
    }
    getAccountIdPrefix() {
        return Buffer.from([0x0a, ...Buffer.from('account-id')]);
    }
    /** @inheritdoc */
    isValidBlockId(hash) {
        // ICP block hashes are 64-character hexadecimal strings
        return this.isValidHash(hash);
    }
    /**
     * Returns whether or not the string is a valid ICP hash
     *
     * @param {string} hash - string to validate
     * @returns {boolean}
     */
    isValidHash(hash) {
        return typeof hash === 'string' && /^[0-9a-fA-F]{64}$/.test(hash);
    }
    /** @inheritdoc */
    isValidTransactionId(txId) {
        return this.isValidHash(txId);
    }
    getSignatures(payloadsData, senderPublicKey, senderPrivateKey) {
        return payloadsData.payloads.map((payload) => ({
            signing_payload: payload,
            signature_type: payload.signature_type,
            public_key: {
                hex_bytes: senderPublicKey,
                curve_type: iface_1.CurveType.SECP256K1,
            },
            hex_bytes: this.signPayload(senderPrivateKey, payload.hex_bytes),
        }));
    }
    getTransactionId(unsignedTransaction, senderAddress, receiverAddress) {
        try {
            const decodedTxn = utils.cborDecode(utils.blobFromHex(unsignedTransaction));
            const updates = decodedTxn.updates;
            for (const [, update] of updates) {
                const updateArgs = update.arg;
                const sendArgs = utils.fromArgs(updateArgs);
                const transactionHash = this.generateTransactionHash(sendArgs, senderAddress, receiverAddress);
                return transactionHash;
            }
            throw new Error('No updates found in the unsigned transaction.');
        }
        catch (error) {
            throw new Error(`Unable to compute transaction ID: ${error.message}`);
        }
    }
    safeBigInt(value) {
        if (typeof value === 'bigint') {
            return value;
        }
        if (typeof value === 'number') {
            const isUnsafe = value > Number.MAX_SAFE_INTEGER || value < Number.MIN_SAFE_INTEGER;
            return isUnsafe ? BigInt(value) : value;
        }
        throw new Error(`Invalid type: expected a number or bigint, but received ${typeof value}`);
    }
    generateTransactionHash(sendArgs, senderAddress, receiverAddress) {
        const senderAccount = this.accountIdentifier(senderAddress);
        const receiverAccount = this.accountIdentifier(receiverAddress);
        const transferFields = new Map([
            [0, senderAccount],
            [1, receiverAccount],
            [2, new Map([[0, this.safeBigInt(Number(sendArgs.payment.receiverGets.e8s))]])],
            [3, new Map([[0, sendArgs.maxFee.e8s]])],
        ]);
        const operationMap = new Map([[2, transferFields]]);
        const txnFields = new Map([
            [0, operationMap],
            [1, this.safeBigInt(sendArgs.memo.memo)],
            [2, new Map([[0, BigInt(sendArgs.createdAtTime.timestampNanos)]])],
        ]);
        const processedTxn = this.getProcessedTransactionMap(txnFields);
        const serializedTxn = encoder.encode(processedTxn);
        return crypto_1.default.createHash('sha256').update(serializedTxn).digest('hex');
    }
    accountIdentifier(accountAddress) {
        const bytes = Buffer.from(accountAddress, 'hex');
        if (bytes.length === 32) {
            return { hash: bytes.slice(4) };
        }
        throw new Error(`Invalid AccountIdentifier: 64 hex chars, got ${accountAddress.length}`);
    }
    getProcessedTransactionMap(txnMap) {
        const operationMap = txnMap.get(0);
        const transferMap = operationMap.get(2);
        transferMap.set(0, this.serializeAccountIdentifier(transferMap.get(0)));
        transferMap.set(1, this.serializeAccountIdentifier(transferMap.get(1)));
        return txnMap;
    }
    serializeAccountIdentifier(accountHash) {
        if (accountHash && accountHash.hash) {
            const hashBuffer = accountHash.hash;
            const checksum = Buffer.alloc(4);
            checksum.writeUInt32BE(crc_32_1.default.buf(hashBuffer) >>> 0, 0);
            return Buffer.concat([checksum, hashBuffer]).toString('hex').toLowerCase();
        }
        throw new Error('Invalid accountHash format');
    }
}
exports.Utils = Utils;
const utils = new Utils();
exports.default = utils;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseUtils,\n  KeyPair,\n  ParseTransactionError,\n  Recipient,\n  BuildTransactionError,\n  MethodNotImplementedError,\n} from '@bitgo/sdk-core';\nimport { Principal as DfinityPrincipal } from '@dfinity/principal';\nimport * as agent from '@dfinity/agent';\nimport crypto from 'crypto';\nimport crc32 from 'crc-32';\nimport {\n  HttpCanisterUpdate,\n  IcpTransactionData,\n  RequestType,\n  Signatures,\n  MetaData,\n  SendArgs,\n  PayloadsData,\n  CurveType,\n  AccountIdentifierHash,\n  CborUnsignedTransaction,\n  MAX_INGRESS_TTL,\n} from './iface';\nimport { KeyPair as IcpKeyPair } from './keyPair';\nconst messageCompiled = require('../../resources/messageCompiled');\nconst { encode, decode, Encoder } = require('cbor-x/index-no-eval'); // The \"cbor-x\" library is used here because it supports modern features like BigInt. do not replace it with \"cbor as \"cbor\" is not compatible with Rust's serde_cbor when handling big numbers.\nimport js_sha256 from 'js-sha256';\nimport BigNumber from 'bignumber.js';\nimport { secp256k1 } from '@noble/curves/secp256k1';\n\n//custom encoder that avoids tagging\nconst encoder = new Encoder({\n  structuredClone: false,\n  useToJSON: false,\n  mapsAsObjects: false,\n  largeBigIntToFloat: false,\n});\n\nexport class Utils implements BaseUtils {\n  /** @inheritdoc */\n  isValidSignature(signature: string): boolean {\n    throw new MethodNotImplementedError();\n  }\n\n  /**\n   * gets the gas data of this transaction.\n   */\n  //TODO WIN-4242: to moved to a config and eventually to an API for dynamic value\n  feeData(): string {\n    return '-10000';\n  }\n\n  /**\n   * Checks if the provided address is a valid ICP address.\n   *\n   * @param {string} address - The address to validate.\n   * @returns {boolean} - Returns `true` if the address is valid, otherwise `false`.\n   */\n  isValidAddress(address: string): boolean {\n    const rootAddress = this.validateMemoAndReturnRootAddress(address);\n    return rootAddress !== undefined && this.isValidHash(rootAddress);\n  }\n\n  /**\n   * Validates the memo ID in the address and returns the root address.\n   *\n   * @param {string} address - The address to validate and extract the root address from.\n   * @returns {string | undefined} - The root address if valid, otherwise `undefined`.\n   */\n  validateMemoAndReturnRootAddress(address: string): string | undefined {\n    if (!address) {\n      return undefined;\n    }\n    const [rootAddress, memoId] = address.split('?memoId=');\n    if (memoId && this.validateMemo(BigInt(memoId))) {\n      return rootAddress;\n    }\n    return address;\n  }\n\n  /**\n   * Checks if the provided hex string is a valid public key.\n   *\n   * A valid public key can be either compressed or uncompressed:\n   * - Compressed public keys are 33 bytes long and start with either 0x02 or 0x03.\n   * - Uncompressed public keys are 65 bytes long and start with 0x04.\n   *\n   * @param {string} hexStr - The hex string representation of the public key to validate.\n   * @returns {boolean} - Returns `true` if the hex string is a valid public key, otherwise `false`.\n   */\n  isValidPublicKey(hexStr: string): boolean {\n    if (!this.isValidHex(hexStr) || !this.isValidLength(hexStr)) {\n      return false;\n    }\n\n    const pubKeyBytes = this.hexToBytes(hexStr);\n    const firstByte = pubKeyBytes[0];\n    const validCompressed = pubKeyBytes.length === 33 && (firstByte === 2 || firstByte === 3);\n    const validUncompressed = pubKeyBytes.length === 65 && firstByte === 4;\n\n    return validCompressed || validUncompressed;\n  }\n\n  /**\n   * Encodes a value into CBOR format and returns it as a hex string.\n   *\n   * @param {unknown} value - The value to encode.\n   * @returns {string} - The CBOR encoded value as a hex string.\n   */\n  cborEncode(value: unknown): string {\n    if (value === undefined) {\n      throw new Error('Value to encode cannot be undefined.');\n    }\n    const cborData = encode(value);\n    return Buffer.from(cborData).toString('hex');\n  }\n\n  /**\n   * Checks if the length of the given hexadecimal string is valid.\n   * A valid length is either 66 characters (33 bytes) or 130 characters (65 bytes).\n   *\n   * @param {string} hexStr - The hexadecimal string to check.\n   * @returns {boolean} - Returns `true` if the length is valid, otherwise `false`.\n   */\n  isValidLength(hexStr: string): boolean {\n    return hexStr.length / 2 === 33 || hexStr.length / 2 === 65;\n  }\n\n  /**\n   * Checks if the provided string is a valid hexadecimal string.\n   *\n   * A valid hexadecimal string consists of pairs of hexadecimal digits (0-9, a-f, A-F).\n   *\n   * @param hexStr - The string to be validated as a hexadecimal string.\n   * @returns True if the string is a valid hexadecimal string, false otherwise.\n   */\n  isValidHex(hexStr: string): boolean {\n    return /^([0-9a-fA-F]{2})+$/.test(hexStr);\n  }\n\n  /**\n   * Converts a hexadecimal string to a Uint8Array.\n   *\n   * @param {string} hex - The hexadecimal string to convert.\n   * @returns {Uint8Array} The resulting byte array.\n   */\n  hexToBytes(hex: string): Uint8Array {\n    const bytes = new Uint8Array(hex.length / 2);\n    for (let i = 0; i < hex.length; i += 2) {\n      bytes[i / 2] = parseInt(hex.substr(i, 2), 16);\n    }\n    return bytes;\n  }\n\n  /** @inheritdoc */\n  isValidPrivateKey(key: string): boolean {\n    return this.isValidKey(key);\n  }\n\n  /**\n   * Validates whether the provided key is a valid ICP private key.\n   *\n   * This function attempts to create a new instance of `IcpKeyPair` using the provided key.\n   * If the key is valid, the function returns `true`. If the key is invalid, an error is thrown,\n   * and the function returns `false`.\n   *\n   * @param {string} key - The private key to validate.\n   * @returns {boolean} - `true` if the key is valid, `false` otherwise.\n   */\n  isValidKey(key: string): boolean {\n    try {\n      new IcpKeyPair({ prv: key });\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Compresses an uncompressed public key.\n   *\n   * @param {string} uncompressedKey - The uncompressed public key in hexadecimal format.\n   * @returns {string} - The compressed public key in hexadecimal format.\n   * @throws {Error} - If the input key is not a valid uncompressed public key.\n   */\n  compressPublicKey(uncompressedKey: string): string {\n    if (uncompressedKey.startsWith('02') || uncompressedKey.startsWith('03')) {\n      return uncompressedKey;\n    }\n    if (!uncompressedKey.startsWith('04') || uncompressedKey.length !== 130) {\n      throw new Error('Invalid uncompressed public key format.');\n    }\n\n    const xHex = uncompressedKey.slice(2, 66);\n    const yHex = uncompressedKey.slice(66);\n    const y = BigInt(`0x${yHex}`);\n    const prefix = y % 2n === 0n ? '02' : '03';\n\n    return `${prefix}${xHex}`;\n  }\n\n  /**\n   * Converts a public key from its hexadecimal string representation to DER format.\n   *\n   * @param {string} publicKeyHex - The public key in hexadecimal string format.\n   * @returns The public key in DER format as a Uint8Array.\n   */\n  getPublicKeyInDERFormat(publicKeyHex: string): Uint8Array {\n    const publicKeyBuffer = Buffer.from(publicKeyHex, 'hex');\n    const ellipticKey = secp256k1.ProjectivePoint.fromHex(publicKeyBuffer.toString('hex'));\n    const uncompressedPublicKeyHex = ellipticKey.toHex(false);\n    const derEncodedKey = agent.wrapDER(Buffer.from(uncompressedPublicKeyHex, 'hex'), agent.SECP256K1_OID);\n    return derEncodedKey;\n  }\n\n  /**\n   * Converts a public key in hexadecimal format to a Dfinity Principal ID.\n   *\n   * @param {string} publicKeyHex - The public key in hexadecimal format.\n   * @returns The corresponding Dfinity Principal ID.\n   */\n  getPrincipalIdFromPublicKey(publicKeyHex: string): DfinityPrincipal {\n    const derEncodedKey = this.getPublicKeyInDERFormat(publicKeyHex);\n    const principalId = DfinityPrincipal.selfAuthenticating(Buffer.from(derEncodedKey));\n    return principalId;\n  }\n\n  /**\n   * Derives a DfinityPrincipal from a given public key in hexadecimal format.\n   *\n   * @param {string} publicKeyHex - The public key in hexadecimal format.\n   * @returns The derived DfinityPrincipal.\n   * @throws Will throw an error if the principal cannot be derived from the public key.\n   */\n  derivePrincipalFromPublicKey(publicKeyHex: string): DfinityPrincipal {\n    try {\n      const derEncodedKey = this.getPublicKeyInDERFormat(publicKeyHex);\n      const principalId = DfinityPrincipal.selfAuthenticating(Buffer.from(derEncodedKey));\n      const principal = DfinityPrincipal.fromUint8Array(principalId.toUint8Array());\n      return principal;\n    } catch (error) {\n      throw new Error(`Failed to derive principal from public key: ${error.message}`);\n    }\n  }\n\n  /**\n   * Converts a DfinityPrincipal and an optional subAccount to a string representation of an account ID.\n   *\n   * @param {DfinityPrincipal} principal - The principal to convert.\n   * @param {Uint8Array} [subAccount=new Uint8Array(32)] - An optional sub-account, defaults to a 32-byte array of zeros.\n   * @returns {string} The hexadecimal string representation of the account ID.\n   */\n  fromPrincipal(principal: DfinityPrincipal, subAccount: Uint8Array = new Uint8Array(32)): string {\n    const principalBytes = Buffer.from(principal.toUint8Array().buffer);\n    return this.getAccountIdFromPrincipalBytes(this.getAccountIdPrefix(), principalBytes, subAccount);\n  }\n\n  getAccountIdFromPrincipalBytes(\n    ACCOUNT_ID_PREFIX: Buffer<ArrayBuffer>,\n    principalBytes: Buffer<ArrayBufferLike>,\n    subAccount: Uint8Array<ArrayBufferLike>\n  ): string {\n    const combinedBytes = Buffer.concat([ACCOUNT_ID_PREFIX, principalBytes, subAccount]);\n    const sha224Hash = crypto.createHash('sha224').update(combinedBytes).digest();\n    const checksum = Buffer.alloc(4);\n    checksum.writeUInt32BE(crc32.buf(sha224Hash) >>> 0, 0);\n    const accountIdBytes = Buffer.concat([checksum, sha224Hash]);\n    return accountIdBytes.toString('hex');\n  }\n\n  /**\n   * Retrieves the address associated with a given hex-encoded public key.\n   *\n   * @param {string} hexEncodedPublicKey - The public key in hex-encoded format.\n   * @returns {Promise<string>} A promise that resolves to the address derived from the provided public key.\n   * @throws {Error} Throws an error if the provided public key is not in a valid hex-encoded format.\n   */\n  async getAddressFromPublicKey(hexEncodedPublicKey: string): Promise<string> {\n    if (!this.isValidPublicKey(hexEncodedPublicKey)) {\n      throw new Error('Invalid hex-encoded public key format.');\n    }\n    const compressedKey = this.compressPublicKey(hexEncodedPublicKey);\n    const keyPair = new IcpKeyPair({ pub: compressedKey });\n    return keyPair.getAddress();\n  }\n\n  /**\n   * Generates a new key pair. If a seed is provided, it will be used to generate the key pair.\n   *\n   * @param {Buffer} [seed] - Optional seed for key generation.\n   * @returns {KeyPair} - The generated key pair containing both public and private keys.\n   * @throws {Error} - If the private key is missing in the generated key pair.\n   */\n  public generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new IcpKeyPair({ seed }) : new IcpKeyPair();\n    const { pub, prv } = keyPair.getKeys();\n    if (!prv) {\n      throw new Error('Private key is missing in the generated key pair.');\n    }\n    return { pub, prv };\n  }\n\n  /**\n   * Validates the provided fee.\n   *\n   * @param {string} fee - The fee to validate.\n   * @throws {BuildTransactionError} - If the fee is zero or invalid.\n   */\n  validateFee(fee: string): boolean {\n    const feeValue = new BigNumber(fee);\n    if (feeValue.isZero()) {\n      throw new BuildTransactionError('Fee cannot be zero');\n    }\n    return true;\n  }\n\n  /** @inheritdoc */\n  validateValue(value: BigNumber): boolean {\n    if (value.isLessThanOrEqualTo(0)) {\n      throw new BuildTransactionError('amount cannot be less than or equal to zero');\n    }\n    return true;\n  }\n\n  /**\n   * Validates the provided memo.\n   *\n   * @param {number | BigInt} memo - The memo to validate.\n   * @returns {boolean} - Returns `true` if the memo is valid.\n   * @throws {BuildTransactionError} - If the memo is invalid.\n   */\n  validateMemo(memo: number | BigInt): boolean {\n    const memoNumber = Number(memo);\n    if (memoNumber < 0 || Number.isNaN(memoNumber)) {\n      throw new BuildTransactionError('Invalid memo');\n    }\n    return true;\n  }\n\n  validateExpireTime(expireTime: number | BigInt): boolean {\n    if (Number(expireTime) < Date.now() * 1000_000) {\n      throw new BuildTransactionError('Invalid expiry time');\n    }\n    return true;\n  }\n\n  /**\n   * Validates the raw transaction data to ensure it has a valid format in the blockchain context.\n   *\n   * @param {IcpTransactionData} transactionData - The transaction data to validate.\n   * @throws {ParseTransactionError} If the transaction data is invalid.\n   */\n  validateRawTransaction(transactionData: IcpTransactionData): void {\n    if (!transactionData) {\n      throw new ParseTransactionError('Transaction data is missing.');\n    }\n    const { senderPublicKeyHex, senderAddress, receiverAddress } = transactionData;\n    if (senderPublicKeyHex && !this.isValidPublicKey(senderPublicKeyHex)) {\n      throw new ParseTransactionError('Sender public key is invalid.');\n    }\n    if (!this.isValidAddress(senderAddress)) {\n      throw new ParseTransactionError('Sender address is invalid.');\n    }\n    if (!this.isValidAddress(receiverAddress)) {\n      throw new ParseTransactionError('Receiver address is invalid.');\n    }\n    this.validateFee(transactionData.fee);\n    this.validateValue(new BigNumber(transactionData.amount));\n    this.validateMemo(transactionData.memo);\n    this.validateExpireTime(transactionData.expiryTime);\n  }\n\n  /**\n   *\n   * @param {object} update\n   * @returns {Buffer}\n   */\n  generateHttpCanisterUpdateId(update: HttpCanisterUpdate): Buffer {\n    return this.HttpCanisterUpdateRepresentationIndependentHash(update);\n  }\n\n  /**\n   * Generates a representation-independent hash for an HTTP canister update.\n   *\n   * @param {HttpCanisterUpdate} update - The HTTP canister update object.\n   * @returns {Buffer} - The hash of the update object.\n   */\n  HttpCanisterUpdateRepresentationIndependentHash(update: HttpCanisterUpdate): Buffer {\n    const updateMap = {\n      request_type: RequestType.CALL,\n      canister_id: update.canister_id,\n      method_name: update.method_name,\n      arg: update.arg,\n      ingress_expiry: update.ingress_expiry,\n      sender: update.sender,\n    };\n    return this.hashOfMap(updateMap);\n  }\n\n  /**\n   * Generates a SHA-256 hash for a given map object.\n   *\n   * @param {Record<string, unknown>} map - The map object to hash.\n   * @returns {Buffer} - The resulting hash as a Buffer.\n   */\n  hashOfMap(map: Record<string, any>): Buffer {\n    const hashes: Buffer[] = [];\n    for (const key in map) {\n      hashes.push(this.hashKeyVal(key, map[key]));\n    }\n    hashes.sort((buf0, buf1) => buf0.compare(buf1));\n    return this.sha256(hashes);\n  }\n\n  /**\n   * Generates a hash for a key-value pair.\n   *\n   * @param {string} key - The key to hash.\n   * @param {string | Buffer | BigInt} val - The value to hash.\n   * @returns {Buffer} - The resulting hash as a Buffer.\n   */\n  hashKeyVal(key: string, val: any): Buffer {\n    const keyHash = this.hashString(key);\n    const valHash = this.hashVal(val);\n    return Buffer.concat([keyHash, valHash]);\n  }\n\n  /**\n   * Generates a SHA-256 hash for a given string.\n   *\n   * @param {string} value - The string to hash.\n   * @returns {Buffer} - The resulting hash as a Buffer.\n   */\n  hashString(value: string): Buffer {\n    return this.sha256([Buffer.from(value)]);\n  }\n\n  /**\n   * Generates a hash for a 64-bit unsigned integer.\n   *\n   * @param {bigint} n - The 64-bit unsigned integer to hash.\n   * @returns {Buffer} - The resulting hash as a Buffer.\n   */\n  hashU64(n: bigint): Buffer {\n    const buf = Buffer.allocUnsafe(10);\n    let i = 0;\n    while (true) {\n      const byte = Number(n & BigInt(0x7f));\n      n >>= BigInt(7);\n      if (n === BigInt(0)) {\n        buf[i] = byte;\n        break;\n      } else {\n        buf[i] = byte | 0x80;\n        ++i;\n      }\n    }\n    return this.hashBytes(buf.subarray(0, i + 1));\n  }\n\n  /**\n   * Generates a SHA-256 hash for an array of elements.\n   *\n   * @param {Array<any>} elements - The array of elements to hash.\n   * @returns {Buffer} - The resulting hash as a Buffer.\n   */\n  hashArray(elements: Array<any>): Buffer {\n    return this.sha256(elements.map(this.hashVal));\n  }\n\n  /**\n   * Generates a hash for a given value.\n   *\n   * @param {string | Buffer | BigInt | number | Array<unknown>} val - The value to hash.\n   * @returns {Buffer} - The resulting hash as a Buffer.\n   * @throws {Error} - If the value type is unsupported.\n   */\n  hashVal(val: string | Buffer | BigInt | number | Array<unknown>): Buffer {\n    if (typeof val === 'string') {\n      return utils.hashString(val);\n    } else if (Buffer.isBuffer(val) || val instanceof Uint8Array) {\n      return utils.hashBytes(val);\n    } else if (typeof val === 'bigint' || typeof val === 'number') {\n      return utils.hashU64(BigInt(val));\n    } else if (Array.isArray(val)) {\n      return utils.hashArray(val);\n    } else {\n      throw new Error(`Unsupported value type for hashing: ${typeof val}`);\n    }\n  }\n\n  /**\n   * Computes the SHA-256 hash of the given buffer.\n   *\n   * @param value - The buffer to hash.\n   * @returns The SHA-256 hash of the input buffer.\n   */\n  hashBytes(value: Buffer | Uint8Array): Buffer {\n    return this.sha256([value]);\n  }\n\n  /**\n   * Computes the SHA-256 hash of the provided array of Buffer chunks.\n   *\n   * @param {Array<Buffer>} chunks - An array of Buffer objects to be hashed.\n   * @returns {Buffer} - The resulting SHA-256 hash as a Buffer.\n   */\n  sha256(chunks: Array<Buffer> | Array<Uint8Array>): Buffer {\n    const hasher = js_sha256.sha256.create();\n    chunks.forEach((chunk) => hasher.update(chunk));\n    return Buffer.from(hasher.arrayBuffer());\n  }\n\n  /**\n   * Converts a hexadecimal string to a Buffer.\n   *\n   * @param hex - The hexadecimal string to convert.\n   * @returns A Buffer containing the binary data represented by the hexadecimal string.\n   */\n  blobFromHex(hex: string): Buffer {\n    return Buffer.from(hex, 'hex');\n  }\n\n  /**\n   * Converts a binary blob (Buffer) to a hexadecimal string.\n   *\n   * @param {Buffer} blob - The binary data to be converted.\n   * @returns {string} The hexadecimal representation of the binary data.\n   */\n  blobToHex(blob: Buffer): string {\n    return blob.toString('hex');\n  }\n\n  /**\n   * Decodes a given CBOR-encoded buffer.\n   *\n   * @param buffer - The CBOR-encoded buffer to decode.\n   * @returns The decoded data.\n   */\n  cborDecode(buffer: Buffer): unknown {\n    const res = decode(buffer);\n    return res;\n  }\n\n  /**\n   * Generates a Buffer containing the domain IC request string.\n   *\n   * @returns {Buffer} A Buffer object initialized with the string '\\x0Aic-request'.\n   */\n  getDomainICRequest(): Buffer {\n    return Buffer.from('\\x0Aic-request');\n  }\n\n  /**\n   * Combines the domain IC request buffer with the provided message ID buffer to create signature data.\n   *\n   * @param {Buffer} messageId - The buffer containing the message ID.\n   * @returns {Buffer} - The concatenated buffer containing the domain IC request and the message ID.\n   */\n  makeSignatureData(messageId: Buffer): Buffer {\n    return Buffer.concat([this.getDomainICRequest(), messageId]);\n  }\n\n  /**\n   * Extracts the recipient information from the provided ICP transaction data.\n   *\n   * @param {IcpTransactionData} icpTransactionData - The ICP transaction data containing the receiver's address and amount.\n   * @returns {Recipient[]} An array containing a single recipient object with the receiver's address and amount.\n   */\n  getRecipients(icpTransactionData: IcpTransactionData): Recipient {\n    return {\n      address: icpTransactionData.receiverAddress,\n      amount: icpTransactionData.amount,\n    };\n  }\n\n  getTransactionSignature(signatureMap: Map<string, Signatures>, update: HttpCanisterUpdate): Signatures | undefined {\n    return signatureMap.get(this.blobToHex(this.makeSignatureData(this.generateHttpCanisterUpdateId(update))));\n  }\n\n  getMetaData(\n    memo: number | BigInt,\n    timestamp: number | bigint | undefined,\n    ingressEnd: number | BigInt | undefined\n  ): { metaData: MetaData; ingressEndTime: number | BigInt } {\n    let currentTime = Date.now() * 1000000;\n    if (timestamp) {\n      currentTime = Number(timestamp);\n    }\n    let ingressStartTime: number, ingressEndTime: number;\n    if (ingressEnd) {\n      ingressEndTime = Number(ingressEnd);\n      ingressStartTime = ingressEndTime - MAX_INGRESS_TTL; // 5 mins in nanoseconds\n    } else {\n      ingressStartTime = currentTime;\n      ingressEndTime = ingressStartTime + MAX_INGRESS_TTL; // 5 mins in nanoseconds\n    }\n    const metaData: MetaData = {\n      created_at_time: currentTime,\n      ingress_start: ingressStartTime,\n      ingress_end: ingressEndTime,\n      memo: memo,\n    };\n\n    return { metaData, ingressEndTime };\n  }\n\n  convertSenderBlobToPrincipal(senderBlob: Uint8Array): Uint8Array {\n    const MAX_LENGTH_IN_BYTES = 29;\n    if (senderBlob.length > MAX_LENGTH_IN_BYTES) {\n      throw new Error('Bytes too long for a valid Principal');\n    }\n    const principalBytes = new Uint8Array(MAX_LENGTH_IN_BYTES);\n    principalBytes.set(senderBlob.slice(0, senderBlob.length));\n    return principalBytes;\n  }\n\n  fromArgs(arg: Uint8Array): SendArgs {\n    const SendRequestMessage = messageCompiled.SendRequest;\n    const args = SendRequestMessage.decode(arg) as unknown as SendArgs;\n    const transformedArgs: SendArgs = {\n      payment: { receiverGets: { e8s: Number(args.payment.receiverGets.e8s) } },\n      maxFee: { e8s: Number(args.maxFee.e8s) },\n      to: { hash: Buffer.from(args.to.hash) },\n      createdAtTime: { timestampNanos: BigNumber(args.createdAtTime.timestampNanos.toString()).toNumber() },\n      memo: { memo: Number(args.memo.memo.toString()) },\n    };\n    return transformedArgs;\n  }\n\n  async toArg(args: SendArgs): Promise<Uint8Array> {\n    const SendRequestMessage = messageCompiled.SendRequest;\n    const errMsg = SendRequestMessage.verify(args);\n    if (errMsg) throw new Error(errMsg);\n    const message = SendRequestMessage.create(args as any);\n    return SendRequestMessage.encode(message).finish();\n  }\n\n  getAccountIdPrefix(): Buffer<ArrayBuffer> {\n    return Buffer.from([0x0a, ...Buffer.from('account-id')]);\n  }\n\n  /** @inheritdoc */\n  isValidBlockId(hash: string): boolean {\n    // ICP block hashes are 64-character hexadecimal strings\n    return this.isValidHash(hash);\n  }\n\n  /**\n   * Returns whether or not the string is a valid ICP hash\n   *\n   * @param {string} hash - string to validate\n   * @returns {boolean}\n   */\n  isValidHash(hash: string): boolean {\n    return typeof hash === 'string' && /^[0-9a-fA-F]{64}$/.test(hash);\n  }\n\n  /** @inheritdoc */\n  isValidTransactionId(txId: string): boolean {\n    return this.isValidHash(txId);\n  }\n\n  getSignatures(payloadsData: PayloadsData, senderPublicKey: string, senderPrivateKey: string): Signatures[] {\n    return payloadsData.payloads.map((payload) => ({\n      signing_payload: payload,\n      signature_type: payload.signature_type,\n      public_key: {\n        hex_bytes: senderPublicKey,\n        curve_type: CurveType.SECP256K1,\n      },\n      hex_bytes: this.signPayload(senderPrivateKey, payload.hex_bytes),\n    }));\n  }\n\n  signPayload = (privateKey: string, payloadHex: string): string => {\n    const privateKeyBytes = Buffer.from(privateKey, 'hex');\n    const payloadHash = crypto.createHash('sha256').update(Buffer.from(payloadHex, 'hex')).digest('hex');\n    const signature = secp256k1.sign(payloadHash, privateKeyBytes);\n    const r = Buffer.from(signature.r.toString(16).padStart(64, '0'), 'hex');\n    const s = Buffer.from(signature.s.toString(16).padStart(64, '0'), 'hex');\n    return Buffer.concat([r, s]).toString('hex');\n  };\n\n  getTransactionId(unsignedTransaction: string, senderAddress: string, receiverAddress: string): string {\n    try {\n      const decodedTxn = utils.cborDecode(utils.blobFromHex(unsignedTransaction)) as CborUnsignedTransaction;\n      const updates = decodedTxn.updates as unknown as [string, HttpCanisterUpdate][];\n      for (const [, update] of updates) {\n        const updateArgs = update.arg;\n        const sendArgs = utils.fromArgs(updateArgs);\n        const transactionHash = this.generateTransactionHash(sendArgs, senderAddress, receiverAddress);\n        return transactionHash;\n      }\n      throw new Error('No updates found in the unsigned transaction.');\n    } catch (error) {\n      throw new Error(`Unable to compute transaction ID: ${error.message}`);\n    }\n  }\n\n  safeBigInt(value: unknown): number | bigint {\n    if (typeof value === 'bigint') {\n      return value;\n    }\n\n    if (typeof value === 'number') {\n      const isUnsafe = value > Number.MAX_SAFE_INTEGER || value < Number.MIN_SAFE_INTEGER;\n      return isUnsafe ? BigInt(value) : value;\n    }\n\n    throw new Error(`Invalid type: expected a number or bigint, but received ${typeof value}`);\n  }\n\n  generateTransactionHash(sendArgs: SendArgs, senderAddress: string, receiverAddress: string): string {\n    const senderAccount = this.accountIdentifier(senderAddress);\n    const receiverAccount = this.accountIdentifier(receiverAddress);\n\n    const transferFields = new Map<any, any>([\n      [0, senderAccount],\n      [1, receiverAccount],\n      [2, new Map([[0, this.safeBigInt(Number(sendArgs.payment.receiverGets.e8s))]])],\n      [3, new Map([[0, sendArgs.maxFee.e8s]])],\n    ]);\n\n    const operationMap = new Map([[2, transferFields]]);\n    const txnFields = new Map<any, any>([\n      [0, operationMap],\n      [1, this.safeBigInt(sendArgs.memo.memo)],\n      [2, new Map([[0, BigInt(sendArgs.createdAtTime.timestampNanos)]])],\n    ]);\n\n    const processedTxn = this.getProcessedTransactionMap(txnFields);\n    const serializedTxn = encoder.encode(processedTxn);\n    return crypto.createHash('sha256').update(serializedTxn).digest('hex');\n  }\n\n  accountIdentifier(accountAddress: string): AccountIdentifierHash {\n    const bytes = Buffer.from(accountAddress, 'hex');\n    if (bytes.length === 32) {\n      return { hash: bytes.slice(4) };\n    }\n    throw new Error(`Invalid AccountIdentifier: 64 hex chars, got ${accountAddress.length}`);\n  }\n\n  getProcessedTransactionMap(txnMap: Map<any, any>): Map<any, any> {\n    const operationMap = txnMap.get(0);\n    const transferMap = operationMap.get(2);\n    transferMap.set(0, this.serializeAccountIdentifier(transferMap.get(0)));\n    transferMap.set(1, this.serializeAccountIdentifier(transferMap.get(1)));\n    return txnMap;\n  }\n\n  serializeAccountIdentifier(accountHash: AccountIdentifierHash): string {\n    if (accountHash && accountHash.hash) {\n      const hashBuffer = accountHash.hash;\n      const checksum = Buffer.alloc(4);\n      checksum.writeUInt32BE(crc32.buf(hashBuffer) >>> 0, 0);\n      return Buffer.concat([checksum, hashBuffer]).toString('hex').toLowerCase();\n    }\n    throw new Error('Invalid accountHash format');\n  }\n}\n\nconst utils = new Utils();\nexport default utils;\n"]}

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


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