PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/abstract-cosmos/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.CosmosUtils = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const amino_1 = require("@cosmjs/amino");
const encoding_1 = require("@cosmjs/encoding");
const proto_signing_1 = require("@cosmjs/proto-signing");
const stargate_1 = require("@cosmjs/stargate");
const statics_1 = require("@bitgo/statics");
const bignumber_js_1 = __importDefault(require("bignumber.js"));
const tx_1 = require("cosmjs-types/cosmos/tx/v1beta1/tx");
const tx_2 = require("cosmjs-types/cosmwasm/wasm/v1/tx");
const crypto_1 = require("crypto");
const constants = __importStar(require("./constants"));
const keyPair_1 = require("./keyPair");
const { MsgSend } = require('../../resources/MsgCompiled').types;
class CosmosUtils {
    constructor() {
        this.registry = new proto_signing_1.Registry([...stargate_1.defaultRegistryTypes]);
        this.registry.register(constants.executeContractMsgTypeUrl, tx_2.MsgExecuteContract);
        this.registry.register('/types.MsgSend', MsgSend);
    }
    /** @inheritdoc */
    isValidBlockId(hash) {
        return this.validateBlake2b(hash);
    }
    /** @inheritdoc */
    isValidPrivateKey(key) {
        try {
            new keyPair_1.CosmosKeyPair({ prv: key });
            return true;
        }
        catch {
            return false;
        }
    }
    /** @inheritdoc */
    isValidPublicKey(key) {
        try {
            new keyPair_1.CosmosKeyPair({ pub: key });
            return true;
        }
        catch {
            return false;
        }
    }
    /** @inheritdoc */
    isValidSignature(signature) {
        throw new sdk_core_1.NotImplementedError('isValidSignature not implemented');
    }
    /** @inheritdoc */
    isValidTransactionId(txId) {
        return this.validateBlake2b(txId);
    }
    /**
     * Checks if transaction hash is in valid black2b format
     */
    validateBlake2b(hash) {
        if (hash?.length !== 64) {
            return false;
        }
        return hash.match(/^[a-zA-Z0-9]+$/) !== null;
    }
    /**
     * Validates whether amounts are in range
     *
     * @param {number[]} amounts - the amounts to validate
     * @returns {boolean} - the validation result
     */
    isValidAmounts(amounts) {
        for (const amount of amounts) {
            if (!this.isValidAmount(amount)) {
                return false;
            }
        }
        return true;
    }
    /**
     * Validates whether amount is in range
     * @param {number} amount
     * @returns {boolean} the validation result
     */
    isValidAmount(amount) {
        const bigNumberAmount = new bignumber_js_1.default(amount);
        if (!bigNumberAmount.isInteger() || bigNumberAmount.isLessThanOrEqualTo(0)) {
            return false;
        }
        return true;
    }
    /**
     * Decodes raw tx data into messages, signing info, and fee data
     * @param {string} txHex - raw base64 tx
     * @returns {DecodedTxRaw} Decoded transaction
     */
    getDecodedTxFromRawBase64(txRaw) {
        try {
            return (0, proto_signing_1.decodeTxRaw)((0, encoding_1.fromBase64)(txRaw));
        }
        catch (e) {
            throw new sdk_core_1.ParseTransactionError('Error decoding TxRaw base64 encoded string: ' + e.message);
        }
    }
    /**
     * Returns the array of messages in the body of the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {EncodeObject[]} messages along with type url
     */
    getEncodedMessagesFromDecodedTx(decodedTx) {
        return decodedTx.body.messages;
    }
    /**
     * Checks the txn sequence is valid or not
     * @param {number} sequence
     */
    validateSequence(sequence) {
        if (sequence < 0) {
            throw new sdk_core_1.InvalidTransactionError('Invalid sequence: less than zero');
        }
    }
    /**
     * Pulls the sequence number from a DecodedTxRaw AuthInfo property
     * @param {DecodedTxRaw} decodedTx
     * @returns {number} sequence
     */
    getSequenceFromDecodedTx(decodedTx) {
        return Number(decodedTx.authInfo.signerInfos[0].sequence);
    }
    /**
     * Pulls the typeUrl from the encoded message of a DecodedTxRaw
     * @param {DecodedTxRaw} decodedTx
     * @returns {string} cosmos proto type url
     */
    getTypeUrlFromDecodedTx(decodedTx) {
        const encodedMessage = this.getEncodedMessagesFromDecodedTx(decodedTx)[0];
        return encodedMessage.typeUrl;
    }
    /**
     * Returns the fee data from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {FeeData} fee data
     */
    getGasBudgetFromDecodedTx(decodedTx) {
        return {
            amount: decodedTx.authInfo.fee?.amount,
            gasLimit: Number(decodedTx.authInfo.fee?.gasLimit),
        };
    }
    /**
     * Returns the publicKey from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {string | undefined} publicKey in hex format if it exists, undefined otherwise
     */
    getPublicKeyFromDecodedTx(decodedTx) {
        const publicKeyUInt8Array = decodedTx.authInfo.signerInfos?.[0].publicKey?.value;
        if (publicKeyUInt8Array) {
            return (0, encoding_1.toHex)((0, encoding_1.fromBase64)((0, proto_signing_1.decodePubkey)(decodedTx.authInfo.signerInfos?.[0].publicKey)?.value));
        }
        return undefined;
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} Send transaction message data
     */
    getSendMessageDataFromDecodedTx(decodedTx) {
        return decodedTx.body.messages.map((message) => {
            const value = this.registry.decode(message);
            return {
                value: {
                    fromAddress: value.fromAddress,
                    toAddress: value.toAddress,
                    amount: value.amount,
                },
                typeUrl: message.typeUrl,
            };
        });
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} Delegate of undelegate transaction message data
     */
    getDelegateOrUndelegateMessageDataFromDecodedTx(decodedTx) {
        return decodedTx.body.messages.map((message) => {
            const value = this.registry.decode(message);
            return {
                value: {
                    delegatorAddress: value.delegatorAddress,
                    validatorAddress: value.validatorAddress,
                    amount: value.amount,
                },
                typeUrl: message.typeUrl,
            };
        });
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} Redelegate transaction message data
     */
    getRedelegateMessageDataFromDecodedTx(decodedTx) {
        return decodedTx.body.messages.map((message) => {
            const value = this.registry.decode(message);
            return {
                value: {
                    delegatorAddress: value.delegatorAddress,
                    validatorSrcAddress: value.validatorSrcAddress,
                    validatorDstAddress: value.validatorDstAddress,
                    amount: value.amount,
                },
                typeUrl: message.typeUrl,
            };
        });
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} WithdrawDelegatorRewards transaction message data
     */
    getWithdrawRewardsMessageDataFromDecodedTx(decodedTx) {
        return decodedTx.body.messages.map((message) => {
            const value = this.registry.decode(message);
            return {
                value: {
                    delegatorAddress: value.delegatorAddress,
                    validatorAddress: value.validatorAddress,
                },
                typeUrl: message.typeUrl,
            };
        });
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} Delegate of undelegate transaction message data
     */
    getWithdrawDelegatorRewardsMessageDataFromDecodedTx(decodedTx) {
        return decodedTx.body.messages.map((message) => {
            const value = this.registry.decode(message);
            return {
                value: {
                    delegatorAddress: value.delegatorAddress,
                    validatorAddress: value.validatorAddress,
                },
                typeUrl: message.typeUrl,
            };
        });
    }
    /**
     * Get a cosmos chain address from its equivalent hex
     * @param {string} prefix
     * @param {string} addressHex
     * @returns {string}
     */
    getCosmosLikeAddressFromHex(prefix, addressHex) {
        if (addressHex.indexOf('0x') === 0) {
            addressHex = addressHex.slice(2);
        }
        return (0, encoding_1.toBech32)(prefix, (0, encoding_1.fromHex)(addressHex));
    }
    /**
     * Get a EVM chain address from its equivalent hex
     * @param {string} prefix
     * @param {string} addressHex
     * @returns {string}
     */
    getEvmLikeAddressFromCosmos(cosmosLikeAddress) {
        return '0x' + (0, encoding_1.toHex)((0, encoding_1.fromBech32)(cosmosLikeAddress).data);
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} Execute contract transaction message data
     */
    getExecuteContractMessageDataFromDecodedTx(decodedTx) {
        return decodedTx.body.messages.map((message) => {
            const value = this.registry.decode(message);
            return {
                value: {
                    sender: value.sender,
                    contract: value.contract,
                    msg: value.msg,
                    funds: value.funds,
                },
                typeUrl: message.typeUrl,
            };
        });
    }
    /**
     * Returns the array of MessageData[] from the decoded transaction
     * @param {DecodedTxRaw} decodedTx
     * @returns {MessageData[]} Custom transaction message data
     */
    getCustomMessageDataFromDecodedTx(decodedTx) {
        throw new sdk_core_1.NotSupported('Custom message data not supported');
    }
    /**
     * Determines bitgo transaction type based on cosmos proto type url
     * @param {string} typeUrl
     * @returns {TransactionType | undefined} TransactionType if url is supported else undefined
     */
    getTransactionTypeFromTypeUrl(typeUrl) {
        switch (typeUrl) {
            case constants.sendMsgTypeUrl:
                return sdk_core_1.TransactionType.Send;
            case constants.sendMsgType:
                return sdk_core_1.TransactionType.Send;
            case constants.delegateMsgTypeUrl:
                return sdk_core_1.TransactionType.StakingActivate;
            case constants.undelegateMsgTypeUrl:
                return sdk_core_1.TransactionType.StakingDeactivate;
            case constants.withdrawDelegatorRewardMsgTypeUrl:
                return sdk_core_1.TransactionType.StakingWithdraw;
            case constants.executeContractMsgTypeUrl:
                return sdk_core_1.TransactionType.ContractCall;
            case constants.redelegateTypeUrl:
                return sdk_core_1.TransactionType.StakingRedelegate;
            default:
                return undefined;
        }
    }
    /**
     * Takes a hex encoded pubkey, converts it to the Amino JSON representation (type/value wrapper)
     * and returns it as protobuf `Any`
     * @param {string} pubkey hex encoded compressed secp256k1 public key
     * @returns {Any} pubkey encoded as protobuf `Any`
     */
    getEncodedPubkey(pubkey) {
        return (0, proto_signing_1.encodePubkey)((0, amino_1.encodeSecp256k1Pubkey)((0, encoding_1.fromHex)(pubkey)));
    }
    /**
     * Gets the send messages used in the final step of encoding a transaction. This allows for any final processing needed.
     * @param {CosmosLikeTransaction} cosmosLikeTransaction transaction to get send messages from
     * @returns {Any[]} processed send messages
     */
    getSendMessagesForEncodingTx(cosmosLikeTransaction) {
        return cosmosLikeTransaction.sendMessages;
    }
    /**
     * Creates a txRaw from an cosmos like transaction @see CosmosLikeTransaction
     * @Precondition cosmosLikeTransaction.publicKey must be defined
     * @param {CosmosLikeTransaction} cosmosLikeTransaction
     * @returns {TxRaw} Unsigned raw transaction
     */
    createTxRawFromCosmosLikeTransaction(cosmosLikeTransaction) {
        if (!cosmosLikeTransaction.publicKey) {
            throw new Error('publicKey is required to create a txRaw');
        }
        const encodedPublicKey = this.getEncodedPubkey(cosmosLikeTransaction.publicKey);
        const messages = this.getSendMessagesForEncodingTx(cosmosLikeTransaction);
        let txBodyValue;
        if (cosmosLikeTransaction.memo) {
            txBodyValue = {
                memo: cosmosLikeTransaction.memo,
                messages: messages,
            };
        }
        else {
            txBodyValue = {
                messages: messages,
            };
        }
        const txBodyBytes = this.registry.encodeTxBody(txBodyValue);
        const sequence = cosmosLikeTransaction.sequence;
        const authInfoBytes = (0, proto_signing_1.makeAuthInfoBytes)([{ pubkey: encodedPublicKey, sequence }], cosmosLikeTransaction.gasBudget.amount, cosmosLikeTransaction.gasBudget.gasLimit, undefined, undefined, undefined);
        return tx_1.TxRaw.fromPartial({
            bodyBytes: txBodyBytes,
            authInfoBytes: authInfoBytes,
        });
    }
    /**
     * Encodes a signature into a txRaw
     * @param {string} publicKeyHex publicKey in hex encoded string format
     * @param {string} signatureHex signature in hex encoded string format
     * @param {TxRaw} unsignedTx raw transaction
     * @returns {TxRaw} Signed raw transaction
     */
    createSignedTxRaw(publicKeyHex, signatureHex, unsignedTx) {
        const stdSignature = (0, amino_1.encodeSecp256k1Signature)((0, encoding_1.fromHex)(publicKeyHex), (0, encoding_1.fromHex)(signatureHex));
        return tx_1.TxRaw.fromPartial({
            bodyBytes: unsignedTx.bodyBytes,
            authInfoBytes: unsignedTx.authInfoBytes,
            signatures: [(0, encoding_1.fromBase64)(stdSignature.signature)],
        });
    }
    /**
     * Decodes a raw transaction into a DecodedTxRaw and checks if it has non empty signatures
     * @param {string} rawTransaction
     * @returns {boolean} true if transaction is signed else false
     */
    isSignedRawTx(rawTransaction) {
        const decodedTx = this.getDecodedTxFromRawBase64(rawTransaction);
        if (decodedTx.signatures.length > 0) {
            return true;
        }
        return false;
    }
    /**
     * Returns whether or not the string is a valid protocol public key
     * @param {string | undefined} publicKey - the  public key to be validated
     */
    validatePublicKey(publicKey) {
        if (publicKey !== undefined) {
            try {
                new keyPair_1.CosmosKeyPair({ pub: publicKey });
            }
            catch {
                throw new sdk_core_1.InvalidTransactionError(`Invalid Public Key`);
            }
        }
    }
    /**
     * Creates a sign doc from an cosmos like transaction @see CosmosLikeTransaction
     * @Precondition cosmosLikeTransaction.accountNumber and cosmosLikeTransaction.chainId must be defined
     * @param {CosmosLikeTransaction} cosmosLikeTransaction
     * @returns {SignDoc} sign doc
     */
    createSignDoc(cosmosLikeTransaction, accountNumber, chainId) {
        if (!accountNumber) {
            throw new Error('accountNumber is required to create a sign doc');
        }
        if (!chainId) {
            throw new Error('chainId is required to create a sign doc');
        }
        if (!cosmosLikeTransaction) {
            throw new Error('cosmosLikeTransaction is required to create a sign doc');
        }
        const txRaw = this.createTxRawFromCosmosLikeTransaction(cosmosLikeTransaction);
        return (0, proto_signing_1.makeSignDoc)(txRaw.bodyBytes, txRaw.authInfoBytes, chainId, accountNumber);
    }
    /**
     * Returns whether or not the string is a valid hex
     * @param hexString - hex string format
     * @returns {boolean} true if string is hex else false
     */
    isValidHexString(hexString) {
        return /^[0-9A-Fa-f]*$/.test(hexString);
    }
    /**
     * Validates the WithdrawDelegatorRewardsMessage
     * @param {WithdrawDelegatorRewardsMessage} withdrawRewardsMessage - The WithdrawDelegatorRewardsMessage to validate.
     * @throws {InvalidTransactionError} Throws an error if the validatorAddress or delegatorAddress is invalid or missing.
     */
    validateWithdrawRewardsMessage(withdrawRewardsMessage) {
        if (!withdrawRewardsMessage.validatorAddress ||
            !this.isValidValidatorAddress(withdrawRewardsMessage.validatorAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid WithdrawDelegatorRewardsMessage validatorAddress: ` + withdrawRewardsMessage.validatorAddress);
        }
        if (!withdrawRewardsMessage.delegatorAddress || !this.isValidAddress(withdrawRewardsMessage.delegatorAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid WithdrawDelegatorRewardsMessage delegatorAddress: ` + withdrawRewardsMessage.delegatorAddress);
        }
    }
    /**
     * Helper method to check if the specified properties in an object are missing or null.
     * @param {Object} obj - The object to check.
     * @param {string[]} keys - An array of property keys to check.
     * @throws {Error} Throws an error if any of the specified properties are missing or null.
     */
    isObjPropertyNull(obj, keys) {
        for (const key of keys) {
            if (obj[key] == null) {
                throw new Error(`Missing or null value for property ${key}`);
            }
        }
    }
    /**
     * Validates the DelegateOrUndelegeteMessage
     * @param {DelegateOrUndelegeteMessage} delegateMessage - The DelegateOrUndelegeteMessage to validate.
     * @throws {InvalidTransactionError} Throws an error if the validatorAddress, delegatorAddress, or amount is invalid or missing.
     */
    validateDelegateOrUndelegateMessage(delegateMessage) {
        this.isObjPropertyNull(delegateMessage, ['validatorAddress', 'delegatorAddress']);
        if (!this.isValidValidatorAddress(delegateMessage.validatorAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid DelegateOrUndelegeteMessage validatorAddress: ` + delegateMessage.validatorAddress);
        }
        if (!this.isValidAddress(delegateMessage.delegatorAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid DelegateOrUndelegeteMessage delegatorAddress: ` + delegateMessage.delegatorAddress);
        }
        this.validateAmount(delegateMessage.amount);
    }
    /**
     * Validates the RedelegateMessage
     * @param {DelegateOrUndelegeteMessage} redelegateMessage - The RedelegateMessage to validate.
     * @throws {InvalidTransactionError} Throws an error if the validatorSrcAddress, validatorDstAddress, delegatorAddress, or amount is invalid or missing.
     */
    validateRedelegateMessage(redelegateMessage) {
        this.isObjPropertyNull(redelegateMessage, ['validatorSrcAddress', 'validatorDstAddress', 'delegatorAddress']);
        if (!this.isValidValidatorAddress(redelegateMessage.validatorSrcAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid RedelegateMessage validatorSrcAddress: ` + redelegateMessage.validatorSrcAddress);
        }
        if (!this.isValidValidatorAddress(redelegateMessage.validatorDstAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid RedelegateMessage validatorDstAddress: ` + redelegateMessage.validatorDstAddress);
        }
        if (!this.isValidAddress(redelegateMessage.delegatorAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid DelegateOrUndelegeteMessage delegatorAddress: ` + redelegateMessage.delegatorAddress);
        }
        this.validateAmount(redelegateMessage.amount);
    }
    /**
     * Validates the CustomMessage
     * @param {CustomMessage} customMessage - The CustomMessage to validate.
     * @throws {InvalidTransactionError} Throws an error if the custom message is invalid or missing required fields.
     * @throws {NotSupported} Throws an error if the custom message data is not supported.
     */
    validateCustomMessage(customMessage) {
        throw new sdk_core_1.NotSupported('Custom message data not supported');
    }
    /**
     * Validates the MessageData
     * @param {MessageData} messageData - The MessageData to validate.
     * @throws {InvalidTransactionError} Throws an error if the messageData is invalid or missing required fields.
     */
    validateMessageData(messageData) {
        if (messageData == null) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid MessageData: undefined`);
        }
        if (messageData.typeUrl == null || this.getTransactionTypeFromTypeUrl(messageData.typeUrl) == null) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid MessageData typeurl: ` + messageData.typeUrl);
        }
        const type = this.getTransactionTypeFromTypeUrl(messageData.typeUrl);
        switch (type) {
            case sdk_core_1.TransactionType.Send: {
                const value = messageData.value;
                this.validateSendMessage(value);
                break;
            }
            case sdk_core_1.TransactionType.StakingActivate:
            case sdk_core_1.TransactionType.StakingDeactivate: {
                const value = messageData.value;
                this.validateDelegateOrUndelegateMessage(value);
                break;
            }
            case sdk_core_1.TransactionType.StakingWithdraw: {
                const value = messageData.value;
                this.validateWithdrawRewardsMessage(value);
                break;
            }
            case sdk_core_1.TransactionType.ContractCall: {
                const value = messageData.value;
                this.validateExecuteContractMessage(value, sdk_core_1.TransactionType.ContractCall);
                break;
            }
            case sdk_core_1.TransactionType.StakingRedelegate: {
                const value = messageData.value;
                this.validateRedelegateMessage(value);
                break;
            }
            case sdk_core_1.TransactionType.CustomTx: {
                const value = messageData.value;
                this.validateCustomMessage(value);
                break;
            }
            default:
                throw new sdk_core_1.InvalidTransactionError(`Invalid MessageData TypeUrl is not supported: ` + messageData.typeUrl);
        }
    }
    /**
     * Validates the Cosmos-like transaction.
     * @param {CosmosLikeTransaction} tx - The transaction to validate.
     * @throws {InvalidTransactionError} Throws an error if the transaction is invalid or missing required fields.
     */
    validateTransaction(tx) {
        this.validateSequence(tx.sequence);
        this.validateGasBudget(tx.gasBudget);
        this.validatePublicKey(tx.publicKey);
        if (tx.sendMessages === undefined || tx.sendMessages.length === 0) {
            throw new sdk_core_1.InvalidTransactionError('Invalid transaction: messages is required');
        }
        else {
            tx.sendMessages.forEach((message) => this.validateMessageData(message));
        }
    }
    /**
     * Creates a Cosmos-like transaction.
     * @param {number} sequence - The sender address sequence number for the transaction.
     * @param {MessageData[]} messages - The array of message data for the transaction.
     * @param {FeeData} gasBudget - The fee data for the transaction.
     * @param {string} [publicKey] - The public key associated with the sender.
     * @param {string} [memo] - The memo for the transaction.
     * @returns {CosmosLikeTransaction} Returns the created Cosmos-like transaction.
     * @throws {InvalidTransactionError} Throws an error if the created transaction is invalid.
     */
    createTransaction(sequence, messages, gasBudget, publicKey, memo) {
        const cosmosLikeTxn = {
            sequence: sequence,
            sendMessages: messages,
            gasBudget: gasBudget,
            publicKey: publicKey,
            memo: memo,
        };
        this.validateTransaction(cosmosLikeTxn);
        return cosmosLikeTxn;
    }
    /**
     * Creates a Cosmos-like transaction with a hash.
     * @param {number} sequence - The sender address sequence number for the transaction.
     * @param {MessageData[]} messages - The array of message data for the transaction.
     * @param {FeeData} gasBudget - The fee data for the transaction.
     * @param {string} [publicKey] - The public key associated with the transaction.
     * @param {Buffer} [signature] - The signature for the transaction.
     * @param {string} [memo] - The memo for the transaction.
     * @returns {CosmosLikeTransaction} Returns the created Cosmos-like transaction with the hash and signature if provided.
     */
    createTransactionWithHash(sequence, messages, gasBudget, publicKey, signature, memo) {
        const cosmosLikeTxn = this.createTransaction(sequence, messages, gasBudget, publicKey, memo);
        let hash = constants.UNAVAILABLE_TEXT;
        if (signature !== undefined) {
            const unsignedTx = this.createTxRawFromCosmosLikeTransaction(cosmosLikeTxn);
            const signedTx = tx_1.TxRaw.fromPartial({
                bodyBytes: unsignedTx.bodyBytes,
                authInfoBytes: unsignedTx.authInfoBytes,
                signatures: [signature],
            });
            hash = (0, crypto_1.createHash)('sha256')
                .update(tx_1.TxRaw.encode(signedTx).finish())
                .digest()
                .toString('hex')
                .toLocaleUpperCase('en-US');
            return { ...cosmosLikeTxn, hash: hash, signature: signature };
        }
        return { ...cosmosLikeTxn, hash: hash };
    }
    /**
     * Deserializes base64 enocded raw transaction string into @see CosmosLikeTransaction
     * @param {string} rawTx base64 enocded raw transaction string
     * @returns {CosmosLikeTransaction} Deserialized cosmosLikeTransaction
     */
    deserializeTransaction(rawTx) {
        const decodedTx = this.getDecodedTxFromRawBase64(rawTx);
        const typeUrl = this.getTypeUrlFromDecodedTx(decodedTx);
        const type = this.getTransactionTypeFromTypeUrl(typeUrl);
        let sendMessageData;
        if (type === sdk_core_1.TransactionType.Send) {
            sendMessageData = this.getSendMessageDataFromDecodedTx(decodedTx);
        }
        else if (type === sdk_core_1.TransactionType.StakingActivate || type === sdk_core_1.TransactionType.StakingDeactivate) {
            sendMessageData = this.getDelegateOrUndelegateMessageDataFromDecodedTx(decodedTx);
        }
        else if (type === sdk_core_1.TransactionType.StakingWithdraw) {
            sendMessageData = this.getWithdrawRewardsMessageDataFromDecodedTx(decodedTx);
        }
        else if (type === sdk_core_1.TransactionType.ContractCall) {
            sendMessageData = this.getExecuteContractMessageDataFromDecodedTx(decodedTx);
        }
        else if (type === sdk_core_1.TransactionType.StakingRedelegate) {
            sendMessageData = this.getRedelegateMessageDataFromDecodedTx(decodedTx);
        }
        else if (type === sdk_core_1.TransactionType.CustomTx) {
            sendMessageData = this.getCustomMessageDataFromDecodedTx(decodedTx);
        }
        else {
            throw new Error('Transaction type not supported: ' + typeUrl);
        }
        const sequence = this.getSequenceFromDecodedTx(decodedTx);
        const gasBudget = this.getGasBudgetFromDecodedTx(decodedTx);
        const publicKey = this.getPublicKeyFromDecodedTx(decodedTx);
        const signature = decodedTx.signatures?.[0] !== undefined ? Buffer.from(decodedTx.signatures[0]) : undefined;
        return this.createTransactionWithHash(sequence, sendMessageData, gasBudget, publicKey, signature, decodedTx.body?.memo);
    }
    /**
     * Validates an array of coin amounts.
     * @param {Coin[]} amountArray - The array of coin amounts to validate.
     * @param {TransactionType} transactionType - optional field for transaction type
     */
    validateAmountData(amountArray, transactionType) {
        amountArray.forEach((coinAmount) => {
            this.validateAmount(coinAmount, transactionType);
        });
    }
    /**
     * Validates the gas limit and gas amount for a transaction.
     * @param {FeeData} gasBudget - The gas budget to validate.
     * @throws {InvalidTransactionError} Throws an error if the gas budget is invalid.
     */
    validateGasBudget(gasBudget) {
        if (gasBudget.gasLimit <= 0) {
            throw new sdk_core_1.InvalidTransactionError('Invalid gas limit ' + gasBudget.gasLimit);
        }
        this.validateAmountData(gasBudget.amount);
    }
    /**
     * Validates a send message for a transaction.
     * @param {SendMessage} sendMessage - The send message to validate.
     * @throws {InvalidTransactionError} Throws an error if the send message is invalid.
     */
    validateSendMessage(sendMessage) {
        if (!sendMessage.toAddress || !this.isValidAddress(sendMessage.toAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid SendMessage toAddress: ` + sendMessage.toAddress);
        }
        if (!sendMessage.fromAddress || !this.isValidAddress(sendMessage.fromAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid SendMessage fromAddress: ` + sendMessage.fromAddress);
        }
        this.validateAmountData(sendMessage.amount);
    }
    /**
     * Validates a coin amount.
     * @param {Coin} amount - The coin amount to validate.
     * @param {TransactionType} transactionType - optional field for transaction type
     * @throws {InvalidTransactionError} Throws an error if the coin amount is invalid.
     */
    validateAmount(amount, transactionType) {
        throw new sdk_core_1.NotImplementedError('validateAmount not implemented');
    }
    /**
     * Checks if a cosmos like Bech32 address matches given regular expression and
     * validates memoId if present
     * @param {string} address
     * @param {RegExp} regExp Regular expression to validate the root address against after trimming the memoId
     * @returns {boolean} true if address is valid
     */
    isValidCosmosLikeAddressWithMemoId(address, regExp) {
        if (typeof address !== 'string')
            return false;
        const addressArray = address.split('?memoId=');
        if (![1, 2].includes(addressArray.length) || // should have at most one occurrence of 'memoId='
            !this.isValidBech32AddressMatchingRegex(addressArray[0], regExp) ||
            (addressArray[1] && !this.isValidMemoId(addressArray[1]))) {
            return false;
        }
        return true;
    }
    /**
     * Checks if address is valid Bech32 and matches given regular expression
     * @param {string} address
     * @param {RegExp} regExp Regular expression to validate the address against
     * @returns {boolean} true if address is valid
     */
    isValidBech32AddressMatchingRegex(address, regExp) {
        try {
            (0, encoding_1.fromBech32)(address);
        }
        catch (e) {
            return false;
        }
        return regExp.test(address);
    }
    /**
     * Return boolean indicating whether a memo id is valid
     *
     * @param memoId memo id
     * @returns true if memo id is valid
     */
    isValidMemoId(memoId) {
        // Allow alphanumeric memo IDs (including uppercase and lowercase letters)
        const alphanumericRegex = /^[0-9a-zA-Z]+$/;
        // Check if the memoId is alphanumeric
        if (!alphanumericRegex.test(memoId)) {
            return false;
        }
        // If the memoId is purely numeric, ensure it is a positive integer
        if (/^\d+$/.test(memoId)) {
            const memoIdNumber = new bignumber_js_1.default(memoId);
            return memoIdNumber.gte(0) && memoIdNumber.isInteger();
        }
        return true;
    }
    /**
     * Validates if the address matches with regex @see accountAddressRegex
     * @param {string} address
     * @returns {boolean} - the validation result
     */
    isValidValidatorAddress(address) {
        throw new sdk_core_1.NotImplementedError('isValidValidatorAddress not implemented');
    }
    /**
     * Validates if the address matches with regex @see accountAddressRegex
     * @param {string} address
     * @returns {boolean} - the validation result
     */
    isValidAddress(address) {
        throw new sdk_core_1.NotImplementedError('isValidAddress not implemented');
    }
    /**
     * Validates if the address matches with regex @see contractAddressRegex
     * @param {string} address
     * @returns {boolean} - the validation result
     */
    isValidContractAddress(address) {
        throw new sdk_core_1.NotImplementedError('isValidContractAddress not implemented');
    }
    /**
     * Validates a execute contract message
     * @param {ExecuteContractMessage} message - The execute contract message to validate
     * @param {TransactionType} transactionType - optional field for transaction type
     * @throws {InvalidTransactionError} Throws an error if the message is invalid
     */
    validateExecuteContractMessage(message, transactionType) {
        if (!message.contract || !this.isValidContractAddress(message.contract)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid ExecuteContractMessage contract address: ` + message.contract);
        }
        if (!message.sender || !this.isValidAddress(message.sender)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid ExecuteContractMessage sender address: ` + message.sender);
        }
        if (!message.msg) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid ExecuteContractMessage msg: ` + message.msg);
        }
        if (message.funds) {
            this.validateAmountData(message.funds, transactionType);
        }
    }
    /**
     * Get coin specific hash function
     * @returns {Hash} The hash function
     */
    getHashFunction() {
        return (0, crypto_1.createHash)('sha256');
    }
    getTokenDenomsUsingCoinFamily(coinFamily) {
        // using set to remove duplicates as denom can be same on testnet and mainnet for a few tokens
        return [
            ...new Set(statics_1.coins
                .filter((coin) => coin.family.toLowerCase() === coinFamily.toLowerCase() && coin.isToken && coin.denom !== undefined)
                .map((coin) => coin.denom)),
        ];
    }
}
exports.CosmosUtils = CosmosUtils;
const utils = new CosmosUtils();
exports.default = utils;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,8CAOyB;AACzB,yCAAgF;AAChF,+CAAoF;AACpF,yDAS+B;AAC/B,+CAA8D;AAC9D,4CAAuC;AACvC,gEAAqC;AACrC,0DAAmE;AAEnE,yDAAsE;AACtE,mCAA0C;AAC1C,uDAAyC;AAWzC,uCAAqD;AAErD,MAAM,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC,6BAA6B,CAAC,CAAC,KAAK,CAAC;AAEjE,MAAa,WAAW;IAGtB;QACE,IAAI,CAAC,QAAQ,GAAG,IAAI,wBAAQ,CAAC,CAAC,GAAG,+BAAoB,CAAC,CAAC,CAAC;QACxD,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC,yBAAyB,EAAE,uBAAkB,CAAC,CAAC;QAChF,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,gBAAgB,EAAE,OAAO,CAAC,CAAC;IACpD,CAAC;IAED,kBAAkB;IAClB,cAAc,CAAC,IAAY;QACzB,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAED,kBAAkB;IAClB,iBAAiB,CAAC,GAAW;QAC3B,IAAI,CAAC;YACH,IAAI,uBAAO,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;YAC1B,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED,kBAAkB;IAClB,gBAAgB,CAAC,GAAW;QAC1B,IAAI,CAAC;YACH,IAAI,uBAAO,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;YAC1B,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED,kBAAkB;IAClB,gBAAgB,CAAC,SAAiB;QAChC,MAAM,IAAI,8BAAmB,CAAC,kCAAkC,CAAC,CAAC;IACpE,CAAC;IAED,kBAAkB;IAClB,oBAAoB,CAAC,IAAY;QAC/B,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,IAAY;QAC1B,IAAI,IAAI,EAAE,MAAM,KAAK,EAAE,EAAE,CAAC;YACxB,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,KAAK,IAAI,CAAC;IAC/C,CAAC;IAED;;;;;OAKG;IACH,cAAc,CAAC,OAAiB;QAC9B,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,EAAE,CAAC;gBAChC,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,aAAa,CAAC,MAAc;QAC1B,MAAM,eAAe,GAAG,IAAI,sBAAS,CAAC,MAAM,CAAC,CAAC;QAC9C,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,IAAI,eAAe,CAAC,mBAAmB,CAAC,CAAC,CAAC,EAAE,CAAC;YAC3E,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,yBAAyB,CAAC,KAAa;QACrC,IAAI,CAAC;YACH,OAAO,IAAA,2BAAW,EAAC,IAAA,qBAAU,EAAC,KAAK,CAAC,CAAC,CAAC;QACxC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,IAAI,gCAAqB,CAAC,8CAA8C,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC;QAC9F,CAAC;IACH,CAAC;IAED;;;;OAIG;IACK,+BAA+B,CAAC,SAAuB;QAC7D,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC;IACjC,CAAC;IAED;;;OAGG;IACH,gBAAgB,CAAC,QAAgB;QAC/B,IAAI,QAAQ,GAAG,CAAC,EAAE,CAAC;YACjB,MAAM,IAAI,kCAAuB,CAAC,kCAAkC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,wBAAwB,CAAC,SAAuB;QAC9C,OAAO,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;IAC5D,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CAAC,SAAuB;QAC7C,MAAM,cAAc,GAAG,IAAI,CAAC,+BAA+B,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1E,OAAO,cAAc,CAAC,OAAO,CAAC;IAChC,CAAC;IAED;;;;OAIG;IACH,yBAAyB,CAAC,SAAuB;QAC/C,OAAO;YACL,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,GAAG,EAAE,MAAgB;YAChD,QAAQ,EAAE,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG,EAAE,QAAQ,CAAC;SACnD,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,yBAAyB,CAAC,SAAuB;QAC/C,MAAM,mBAAmB,GAAG,SAAS,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,KAAK,CAAC;QACjF,IAAI,mBAAmB,EAAE,CAAC;YACxB,OAAO,IAAA,gBAAK,EAAC,IAAA,qBAAU,EAAC,IAAA,4BAAY,EAAC,SAAS,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC;QAC/F,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACO,+BAA+B,CAAC,SAAuB;QAC/D,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO;gBACL,KAAK,EAAE;oBACL,WAAW,EAAE,KAAK,CAAC,WAAW;oBAC9B,SAAS,EAAE,KAAK,CAAC,SAAS;oBAC1B,MAAM,EAAE,KAAK,CAAC,MAAM;iBACrB;gBACD,OAAO,EAAE,OAAO,CAAC,OAAO;aACzB,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,+CAA+C,CAAC,SAAuB;QACrE,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO;gBACL,KAAK,EAAE;oBACL,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;oBACxC,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;oBACxC,MAAM,EAAE,KAAK,CAAC,MAAM;iBACrB;gBACD,OAAO,EAAE,OAAO,CAAC,OAAO;aACzB,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,qCAAqC,CAAC,SAAuB;QAC3D,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO;gBACL,KAAK,EAAE;oBACL,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;oBACxC,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;oBAC9C,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;oBAC9C,MAAM,EAAE,KAAK,CAAC,MAAM;iBACrB;gBACD,OAAO,EAAE,OAAO,CAAC,OAAO;aACzB,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,0CAA0C,CAAC,SAAuB;QAChE,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO;gBACL,KAAK,EAAE;oBACL,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;oBACxC,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;iBACzC;gBACD,OAAO,EAAE,OAAO,CAAC,OAAO;aACzB,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,mDAAmD,CAAC,SAAuB;QACzE,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO;gBACL,KAAK,EAAE;oBACL,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;oBACxC,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;iBACzC;gBACD,OAAO,EAAE,OAAO,CAAC,OAAO;aACzB,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,2BAA2B,CAAC,MAAc,EAAE,UAAkB;QAC5D,IAAI,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACnC,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACnC,CAAC;QACD,OAAO,IAAA,mBAAQ,EAAC,MAAM,EAAE,IAAA,kBAAO,EAAC,UAAU,CAAC,CAAC,CAAC;IAC/C,CAAC;IAED;;;;;OAKG;IACH,2BAA2B,CAAC,iBAAyB;QACnD,OAAO,IAAI,GAAG,IAAA,gBAAK,EAAC,IAAA,qBAAU,EAAC,iBAAiB,CAAC,CAAC,IAAI,CAAC,CAAC;IAC1D,CAAC;IAED;;;;OAIG;IACH,0CAA0C,CAAC,SAAuB;QAChE,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO;gBACL,KAAK,EAAE;oBACL,MAAM,EAAE,KAAK,CAAC,MAAM;oBACpB,QAAQ,EAAE,KAAK,CAAC,QAAQ;oBACxB,GAAG,EAAE,KAAK,CAAC,GAAG;oBACd,KAAK,EAAE,KAAK,CAAC,KAAK;iBACnB;gBACD,OAAO,EAAE,OAAO,CAAC,OAAO;aACzB,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,iCAAiC,CAAC,SAAuB;QACvD,MAAM,IAAI,uBAAY,CAAC,mCAAmC,CAAC,CAAC;IAC9D,CAAC;IAED;;;;OAIG;IACH,6BAA6B,CAAC,OAAe;QAC3C,QAAQ,OAAO,EAAE,CAAC;YAChB,KAAK,SAAS,CAAC,cAAc;gBAC3B,OAAO,0BAAe,CAAC,IAAI,CAAC;YAC9B,KAAK,SAAS,CAAC,WAAW;gBACxB,OAAO,0BAAe,CAAC,IAAI,CAAC;YAC9B,KAAK,SAAS,CAAC,kBAAkB;gBAC/B,OAAO,0BAAe,CAAC,eAAe,CAAC;YACzC,KAAK,SAAS,CAAC,oBAAoB;gBACjC,OAAO,0BAAe,CAAC,iBAAiB,CAAC;YAC3C,KAAK,SAAS,CAAC,iCAAiC;gBAC9C,OAAO,0BAAe,CAAC,eAAe,CAAC;YACzC,KAAK,SAAS,CAAC,yBAAyB;gBACtC,OAAO,0BAAe,CAAC,YAAY,CAAC;YACtC,KAAK,SAAS,CAAC,iBAAiB;gBAC9B,OAAO,0BAAe,CAAC,iBAAiB,CAAC;YAC3C;gBACE,OAAO,SAAS,CAAC;QACrB,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,gBAAgB,CAAC,MAAc;QAC7B,OAAO,IAAA,4BAAY,EAAC,IAAA,6BAAqB,EAAC,IAAA,kBAAO,EAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IAC9D,CAAC;IAED;;;;OAIG;IACH,4BAA4B,CAAC,qBAA2D;QACtF,OAAO,qBAAqB,CAAC,YAAgC,CAAC;IAChE,CAAC;IAED;;;;;OAKG;IACH,oCAAoC,CAAC,qBAA2D;QAC9F,IAAI,CAAC,qBAAqB,CAAC,SAAS,EAAE,CAAC;YACrC,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;QAC7D,CAAC;QACD,MAAM,gBAAgB,GAAQ,IAAI,CAAC,gBAAgB,CAAC,qBAAqB,CAAC,SAAS,CAAC,CAAC;QACrF,MAAM,QAAQ,GAAG,IAAI,CAAC,4BAA4B,CAAC,qBAAqB,CAAC,CAAC;QAC1E,IAAI,WAAW,CAAC;QAChB,IAAI,qBAAqB,CAAC,IAAI,EAAE,CAAC;YAC/B,WAAW,GAAG;gBACZ,IAAI,EAAE,qBAAqB,CAAC,IAAI;gBAChC,QAAQ,EAAE,QAAQ;aACnB,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,WAAW,GAAG;gBACZ,QAAQ,EAAE,QAAQ;aACnB,CAAC;QACJ,CAAC;QAED,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;QAC5D,MAAM,QAAQ,GAAG,qBAAqB,CAAC,QAAQ,CAAC;QAChD,MAAM,aAAa,GAAG,IAAA,iCAAiB,EACrC,CAAC,EAAE,MAAM,EAAE,gBAAgB,EAAE,QAAQ,EAAE,CAAC,EACxC,qBAAqB,CAAC,SAAS,CAAC,MAAM,EACtC,qBAAqB,CAAC,SAAS,CAAC,QAAQ,EACxC,SAAS,EACT,SAAS,EACT,SAAS,CACV,CAAC;QACF,OAAO,UAAK,CAAC,WAAW,CAAC;YACvB,SAAS,EAAE,WAAW;YACtB,aAAa,EAAE,aAAa;SAC7B,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,iBAAiB,CACf,YAAoB,EACpB,YAAoB,EACpB,UAAgE;QAEhE,MAAM,YAAY,GAAG,IAAA,gCAAwB,EAAC,IAAA,kBAAO,EAAC,YAAY,CAAC,EAAE,IAAA,kBAAO,EAAC,YAAY,CAAC,CAAC,CAAC;QAC5F,OAAO,UAAK,CAAC,WAAW,CAAC;YACvB,SAAS,EAAE,UAAU,CAAC,SAAS;YAC/B,aAAa,EAAE,UAAU,CAAC,aAAa;YACvC,UAAU,EAAE,CAAC,IAAA,qBAAU,EAAC,YAAY,CAAC,SAAS,CAAC,CAAC;SACjD,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,aAAa,CAAC,cAAsB;QAClC,MAAM,SAAS,GAAG,IAAI,CAAC,yBAAyB,CAAC,cAAc,CAAC,CAAC;QACjE,IAAI,SAAS,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpC,OAAO,IAAI,CAAC;QACd,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;OAGG;IACH,iBAAiB,CAAC,SAA6B;QAC7C,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC5B,IAAI,CAAC;gBACH,IAAI,uBAAO,CAAC,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC;YAClC,CAAC;YAAC,MAAM,CAAC;gBACP,MAAM,IAAI,kCAAuB,CAAC,oBAAoB,CAAC,CAAC;YAC1D,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,aAAa,CACX,qBAA2D,EAC3D,aAAiC,EACjC,OAA2B;QAE3B,IAAI,CAAC,aAAa,EAAE,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,gDAAgD,CAAC,CAAC;QACpE,CAAC;QACD,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;QAC9D,CAAC;QACD,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;QAC5E,CAAC;QACD,MAAM,KAAK,GAAG,IAAI,CAAC,oCAAoC,CAAC,qBAAqB,CAAC,CAAC;QAC/E,OAAO,IAAA,2BAAW,EAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,aAAa,EAAE,OAAO,EAAE,aAAa,CAAC,CAAC;IACnF,CAAC;IAED;;;;OAIG;IACH,gBAAgB,CAAC,SAAiB;QAChC,OAAO,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC1C,CAAC;IAED;;;;OAIG;IACH,8BAA8B,CAAC,sBAAuD;QACpF,IACE,CAAC,sBAAsB,CAAC,gBAAgB;YACxC,CAAC,IAAI,CAAC,uBAAuB,CAAC,sBAAsB,CAAC,gBAAgB,CAAC,EACtE,CAAC;YACD,MAAM,IAAI,kCAAuB,CAC/B,4DAA4D,GAAG,sBAAsB,CAAC,gBAAgB,CACvG,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,sBAAsB,CAAC,gBAAgB,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,sBAAsB,CAAC,gBAAgB,CAAC,EAAE,CAAC;YAC9G,MAAM,IAAI,kCAAuB,CAC/B,4DAA4D,GAAG,sBAAsB,CAAC,gBAAgB,CACvG,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,iBAAiB,CAAC,GAA2B,EAAE,IAAmB;QAChE,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACvB,IAAI,GAAG,CAAC,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC;gBACrB,MAAM,IAAI,KAAK,CAAC,sCAAsC,GAAG,EAAE,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,mCAAmC,CAAC,eAA4C;QAC9E,IAAI,CAAC,iBAAiB,CAAC,eAAe,EAAE,CAAC,kBAAkB,EAAE,kBAAkB,CAAC,CAAC,CAAC;QAElF,IAAI,CAAC,IAAI,CAAC,uBAAuB,CAAC,eAAe,CAAC,gBAAgB,CAAC,EAAE,CAAC;YACpE,MAAM,IAAI,kCAAuB,CAC/B,wDAAwD,GAAG,eAAe,CAAC,gBAAgB,CAC5F,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,gBAAgB,CAAC,EAAE,CAAC;YAC3D,MAAM,IAAI,kCAAuB,CAC/B,wDAAwD,GAAG,eAAe,CAAC,gBAAgB,CAC5F,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC;IAED;;;;OAIG;IACH,yBAAyB,CAAC,iBAAoC;QAC5D,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,EAAE,CAAC,qBAAqB,EAAE,qBAAqB,EAAE,kBAAkB,CAAC,CAAC,CAAC;QAE9G,IAAI,CAAC,IAAI,CAAC,uBAAuB,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,EAAE,CAAC;YACzE,MAAM,IAAI,kCAAuB,CAC/B,iDAAiD,GAAG,iBAAiB,CAAC,mBAAmB,CAC1F,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,uBAAuB,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,EAAE,CAAC;YACzE,MAAM,IAAI,kCAAuB,CAC/B,iDAAiD,GAAG,iBAAiB,CAAC,mBAAmB,CAC1F,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,gBAAgB,CAAC,EAAE,CAAC;YAC7D,MAAM,IAAI,kCAAuB,CAC/B,wDAAwD,GAAG,iBAAiB,CAAC,gBAAgB,CAC9F,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;IAChD,CAAC;IAED;;;;;OAKG;IACH,qBAAqB,CAAC,aAA4B;QAChD,MAAM,IAAI,uBAAY,CAAC,mCAAmC,CAAC,CAAC;IAC9D,CAAC;IAED;;;;OAIG;IACH,mBAAmB,CAAC,WAAuC;QACzD,IAAI,WAAW,IAAI,IAAI,EAAE,CAAC;YACxB,MAAM,IAAI,kCAAuB,CAAC,gCAAgC,CAAC,CAAC;QACtE,CAAC;QACD,IAAI,WAAW,CAAC,OAAO,IAAI,IAAI,IAAI,IAAI,CAAC,6BAA6B,CAAC,WAAW,CAAC,OAAO,CAAC,IAAI,IAAI,EAAE,CAAC;YACnG,MAAM,IAAI,kCAAuB,CAAC,+BAA+B,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;QAC3F,CAAC;QAED,MAAM,IAAI,GAAG,IAAI,CAAC,6BAA6B,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACrE,QAAQ,IAAI,EAAE,CAAC;YACb,KAAK,0BAAe,CAAC,IAAI,CAAC,CAAC,CAAC;gBAC1B,MAAM,KAAK,GAAG,WAAW,CAAC,KAAoB,CAAC;gBAC/C,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAChC,MAAM;YACR,CAAC;YACD,KAAK,0BAAe,CAAC,eAAe,CAAC;YACrC,KAAK,0BAAe,CAAC,iBAAiB,CAAC,CAAC,CAAC;gBACvC,MAAM,KAAK,GAAG,WAAW,CAAC,KAAoC,CAAC;gBAC/D,IAAI,CAAC,mCAAmC,CAAC,KAAK,CAAC,CAAC;gBAChD,MAAM;YACR,CAAC;YACD,KAAK,0BAAe,CAAC,eAAe,CAAC,CAAC,CAAC;gBACrC,MAAM,KAAK,GAAG,WAAW,CAAC,KAAwC,CAAC;gBACnE,IAAI,CAAC,8BAA8B,CAAC,KAAK,CAAC,CAAC;gBAC3C,MAAM;YACR,CAAC;YACD,KAAK,0BAAe,CAAC,YAAY,CAAC,CAAC,CAAC;gBAClC,MAAM,KAAK,GAAG,WAAW,CAAC,KAA+B,CAAC;gBAC1D,IAAI,CAAC,8BAA8B,CAAC,KAAK,EAAE,0BAAe,CAAC,YAAY,CAAC,CAAC;gBACzE,MAAM;YACR,CAAC;YACD,KAAK,0BAAe,CAAC,iBAAiB,CAAC,CAAC,CAAC;gBACvC,MAAM,KAAK,GAAG,WAAW,CAAC,KAA0B,CAAC;gBACrD,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;gBACtC,MAAM;YACR,CAAC;YACD,KAAK,0BAAe,CAAC,QAAQ,CAAC,CAAC,CAAC;gBAC9B,MAAM,KAAK,GAAG,WAAW,CAAC,KAAsB,CAAC;gBACjD,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;gBAClC,MAAM;YACR,CAAC;YACD;gBACE,MAAM,IAAI,kCAAuB,CAAC,gDAAgD,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;QAC9G,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,mBAAmB,CAAC,EAAwC;QAC1D,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC;QACnC,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC;QACrC,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC;QACrC,IAAI,EAAE,CAAC,YAAY,KAAK,SAAS,IAAI,EAAE,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAClE,MAAM,IAAI,kCAAuB,CAAC,2CAA2C,CAAC,CAAC;QACjF,CAAC;aAAM,CAAC;YACN,EAAE,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC;QAC1E,CAAC;IACH,CAAC;IAED;;;;;;;;;OASG;IACH,iBAAiB,CACf,QAAgB,EAChB,QAAsC,EACtC,SAAkB,EAClB,SAAkB,EAClB,IAAa;QAEb,MAAM,aAAa,GAAG;YACpB,QAAQ,EAAE,QAAQ;YAClB,YAAY,EAAE,QAAQ;YACtB,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;YACpB,IAAI,EAAE,IAAI;SACX,CAAC;QACF,IAAI,CAAC,mBAAmB,CAAC,aAAa,CAAC,CAAC;QACxC,OAAO,aAAa,CAAC;IACvB,CAAC;IAED;;;;;;;;;OASG;IACH,yBAAyB,CACvB,QAAgB,EAChB,QAAsC,EACtC,SAAkB,EAClB,SAAkB,EAClB,SAAkB,EAClB,IAAa;QAEb,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,QAAQ,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,IAAI,CAAC,CAAC;QAC7F,IAAI,IAAI,GAAG,SAAS,CAAC,gBAAgB,CAAC;QACtC,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC5B,MAAM,UAAU,GAAG,IAAI,CAAC,oCAAoC,CAAC,aAAa,CAAC,CAAC;YAC5E,MAAM,QAAQ,GAAG,UAAK,CAAC,WAAW,CAAC;gBACjC,SAAS,EAAE,UAAU,CAAC,SAAS;gBAC/B,aAAa,EAAE,UAAU,CAAC,aAAa;gBACvC,UAAU,EAAE,CAAC,SAAS,CAAC;aACxB,CAAC,CAAC;YACH,IAAI,GAAG,IAAA,mBAAU,EAAC,QAAQ,CAAC;iBACxB,MAAM,CAAC,UAAK,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,MAAM,EAAE,CAAC;iBACvC,MAAM,EAAE;iBACR,QAAQ,CAAC,KAAK,CAAC;iBACf,iBAAiB,CAAC,OAAO,CAAC,CAAC;YAC9B,OAAO,EAAE,GAAG,aAAa,EAAE,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;QAChE,CAAC;QACD,OAAO,EAAE,GAAG,aAAa,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC;IAC1C,CAAC;IAED;;;;OAIG;IACH,sBAAsB,CAAC,KAAa;QAClC,MAAM,SAAS,GAAG,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;QACxD,MAAM,OAAO,GAAG,IAAI,CAAC,uBAAuB,CAAC,SAAS,CAAC,CAAC;QACxD,MAAM,IAAI,GAAgC,IAAI,CAAC,6BAA6B,CAAC,OAAO,CAAC,CAAC;QACtF,IAAI,eAA6C,CAAC;QAClD,IAAI,IAAI,KAAK,0BAAe,CAAC,IAAI,EAAE,CAAC;YAClC,eAAe,GAAG,IAAI,CAAC,+BAA+B,CAAC,SAAS,CAAC,CAAC;QACpE,CAAC;aAAM,IAAI,IAAI,KAAK,0BAAe,CAAC,eAAe,IAAI,IAAI,KAAK,0BAAe,CAAC,iBAAiB,EAAE,CAAC;YAClG,eAAe,GAAG,IAAI,CAAC,+CAA+C,CAAC,SAAS,CAAC,CAAC;QACpF,CAAC;aAAM,IAAI,IAAI,KAAK,0BAAe,CAAC,eAAe,EAAE,CAAC;YACpD,eAAe,GAAG,IAAI,CAAC,0CAA0C,CAAC,SAAS,CAAC,CAAC;QAC/E,CAAC;aAAM,IAAI,IAAI,KAAK,0BAAe,CAAC,YAAY,EAAE,CAAC;YACjD,eAAe,GAAG,IAAI,CAAC,0CAA0C,CAAC,SAAS,CAAC,CAAC;QAC/E,CAAC;aAAM,IAAI,IAAI,KAAK,0BAAe,CAAC,iBAAiB,EAAE,CAAC;YACtD,eAAe,GAAG,IAAI,CAAC,qCAAqC,CAAC,SAAS,CAAC,CAAC;QAC1E,CAAC;aAAM,IAAI,IAAI,KAAK,0BAAe,CAAC,QAAQ,EAAE,CAAC;YAC7C,eAAe,GAAG,IAAI,CAAC,iCAAiC,CAAC,SAAS,CAAC,CAAC;QACtE,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,kCAAkC,GAAG,OAAO,CAAC,CAAC;QAChE,CAAC;QACD,MAAM,QAAQ,GAAG,IAAI,CAAC,wBAAwB,CAAC,SAAS,CAAC,CAAC;QAC1D,MAAM,SAAS,GAAG,IAAI,CAAC,yBAAyB,CAAC,SAAS,CAAC,CAAC;QAC5D,MAAM,SAAS,GAAG,IAAI,CAAC,yBAAyB,CAAC,SAAS,CAAC,CAAC;QAC5D,MAAM,SAAS,GAAG,SAAS,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAC7G,OAAO,IAAI,CAAC,yBAAyB,CACnC,QAAQ,EACR,eAAe,EACf,SAAS,EACT,SAAS,EACT,SAAS,EACT,SAAS,CAAC,IAAI,EAAE,IAAI,CACrB,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,kBAAkB,CAAC,WAAmB,EAAE,eAAiC;QACvE,WAAW,CAAC,OAAO,CAAC,CAAC,UAAU,EAAE,EAAE;YACjC,IAAI,CAAC,cAAc,CAAC,UAAU,EAAE,eAAe,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CAAC,SAAkB;QAClC,IAAI,SAAS,CAAC,QAAQ,IAAI,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,kCAAuB,CAAC,oBAAoB,GAAG,SAAS,CAAC,QAAQ,CAAC,CAAC;QAC/E,CAAC;QACD,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;IAC5C,CAAC;IAED;;;;OAIG;IACH,mBAAmB,CAAC,WAAwB;QAC1C,IAAI,CAAC,WAAW,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;YAC1E,MAAM,IAAI,kCAAuB,CAAC,iCAAiC,GAAG,WAAW,CAAC,SAAS,CAAC,CAAC;QAC/F,CAAC;QACD,IAAI,CAAC,WAAW,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,WAAW,CAAC,EAAE,CAAC;YAC9E,MAAM,IAAI,kCAAuB,CAAC,mCAAmC,GAAG,WAAW,CAAC,WAAW,CAAC,CAAC;QACnG,CAAC;QACD,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;OAKG;IACH,cAAc,CAAC,MAAY,EAAE,eAAiC;QAC5D,MAAM,IAAI,8BAAmB,CAAC,gCAAgC,CAAC,CAAC;IAClE,CAAC;IAED;;;;;;OAMG;IACO,kCAAkC,CAAC,OAAe,EAAE,MAAc;QAC1E,IAAI,OAAO,OAAO,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC;QAC9C,MAAM,YAAY,GAAG,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAC/C,IACE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,kDAAkD;YAC3F,CAAC,IAAI,CAAC,iCAAiC,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,MAAM,CAAC;YAChE,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,EACzD,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;OAKG;IACO,iCAAiC,CAAC,OAAe,EAAE,MAAc;QACzE,IAAI,CAAC;YACH,IAAA,qBAAU,EAAC,OAAO,CAAC,CAAC;QACtB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC9B,CAAC;IAED;;;;;OAKG;IACH,aAAa,CAAC,MAAc;QAC1B,0EAA0E;QAC1E,MAAM,iBAAiB,GAAG,gBAAgB,CAAC;QAE3C,sCAAsC;QACtC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;YACpC,OAAO,KAAK,CAAC;QACf,CAAC;QAED,mEAAmE;QACnE,IAAI,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;YACzB,MAAM,YAAY,GAAG,IAAI,sBAAS,CAAC,MAAM,CAAC,CAAC;YAC3C,OAAO,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,YAAY,CAAC,SAAS,EAAE,CAAC;QACzD,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CAAC,OAAe;QACrC,MAAM,IAAI,8BAAmB,CAAC,yCAAyC,CAAC,CAAC;IAC3E,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,OAAe;QAC5B,MAAM,IAAI,8BAAmB,CAAC,gCAAgC,CAAC,CAAC;IAClE,CAAC;IAED;;;;OAIG;IACH,sBAAsB,CAAC,OAAe;QACpC,MAAM,IAAI,8BAAmB,CAAC,wCAAwC,CAAC,CAAC;IAC1E,CAAC;IAED;;;;;OAKG;IACH,8BAA8B,CAAC,OAA+B,EAAE,eAAiC;QAC/F,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YACxE,MAAM,IAAI,kCAAuB,CAAC,mDAAmD,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC5G,CAAC;QACD,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YAC5D,MAAM,IAAI,kCAAuB,CAAC,iDAAiD,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;QACxG,CAAC;QACD,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;YACjB,MAAM,IAAI,kCAAuB,CAAC,sCAAsC,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;QAC1F,CAAC;QACD,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,IAAI,CAAC,kBAAkB,CAAC,OAAO,CAAC,KAAK,EAAE,eAAe,CAAC,CAAC;QAC1D,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,OAAO,IAAA,mBAAU,EAAC,QAAQ,CAAC,CAAC;IAC9B,CAAC;IAED,6BAA6B,CAAC,UAAkB;QAC9C,8FAA8F;QAC9F,OAAO;YACL,GAAG,IAAI,GAAG,CACR,eAAK;iBACF,MAAM,CACL,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,KAAK,UAAU,CAAC,WAAW,EAAE,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,KAAK,KAAK,SAAS,CAC7G;iBACA,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAe,CAAC,CACvC;SACF,CAAC;IACJ,CAAC;CACF;AA34BD,kCA24BC;AAED,MAAM,KAAK,GAAG,IAAI,WAAW,EAAE,CAAC;AAEhC,kBAAe,KAAK,CAAC","sourcesContent":["import {\n  BaseUtils,\n  InvalidTransactionError,\n  NotImplementedError,\n  NotSupported,\n  ParseTransactionError,\n  TransactionType,\n} from '@bitgo/sdk-core';\nimport { encodeSecp256k1Pubkey, encodeSecp256k1Signature } from '@cosmjs/amino';\nimport { fromBase64, fromBech32, fromHex, toBech32, toHex } from '@cosmjs/encoding';\nimport {\n  DecodedTxRaw,\n  decodePubkey,\n  decodeTxRaw,\n  EncodeObject,\n  encodePubkey,\n  makeAuthInfoBytes,\n  makeSignDoc,\n  Registry,\n} from '@cosmjs/proto-signing';\nimport { Coin, defaultRegistryTypes } from '@cosmjs/stargate';\nimport { coins } from '@bitgo/statics';\nimport BigNumber from 'bignumber.js';\nimport { SignDoc, TxRaw } from 'cosmjs-types/cosmos/tx/v1beta1/tx';\nimport { Any } from 'cosmjs-types/google/protobuf/any';\nimport { MsgExecuteContract } from 'cosmjs-types/cosmwasm/wasm/v1/tx';\nimport { createHash, Hash } from 'crypto';\nimport * as constants from './constants';\nimport {\n  CosmosLikeTransaction,\n  DelegateOrUndelegeteMessage,\n  ExecuteContractMessage,\n  FeeData,\n  MessageData,\n  RedelegateMessage,\n  SendMessage,\n  WithdrawDelegatorRewardsMessage,\n} from './iface';\nimport { CosmosKeyPair as KeyPair } from './keyPair';\n\nconst { MsgSend } = require('../../resources/MsgCompiled').types;\n\nexport class CosmosUtils<CustomMessage = never> implements BaseUtils {\n  protected registry;\n\n  constructor() {\n    this.registry = new Registry([...defaultRegistryTypes]);\n    this.registry.register(constants.executeContractMsgTypeUrl, MsgExecuteContract);\n    this.registry.register('/types.MsgSend', MsgSend);\n  }\n\n  /** @inheritdoc */\n  isValidBlockId(hash: string): boolean {\n    return this.validateBlake2b(hash);\n  }\n\n  /** @inheritdoc */\n  isValidPrivateKey(key: string): boolean {\n    try {\n      new KeyPair({ prv: key });\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /** @inheritdoc */\n  isValidPublicKey(key: string): boolean {\n    try {\n      new KeyPair({ pub: key });\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /** @inheritdoc */\n  isValidSignature(signature: string): boolean {\n    throw new NotImplementedError('isValidSignature not implemented');\n  }\n\n  /** @inheritdoc */\n  isValidTransactionId(txId: string): boolean {\n    return this.validateBlake2b(txId);\n  }\n\n  /**\n   * Checks if transaction hash is in valid black2b format\n   */\n  validateBlake2b(hash: string): boolean {\n    if (hash?.length !== 64) {\n      return false;\n    }\n    return hash.match(/^[a-zA-Z0-9]+$/) !== null;\n  }\n\n  /**\n   * Validates whether amounts are in range\n   *\n   * @param {number[]} amounts - the amounts to validate\n   * @returns {boolean} - the validation result\n   */\n  isValidAmounts(amounts: number[]): boolean {\n    for (const amount of amounts) {\n      if (!this.isValidAmount(amount)) {\n        return false;\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Validates whether amount is in range\n   * @param {number} amount\n   * @returns {boolean} the validation result\n   */\n  isValidAmount(amount: number): boolean {\n    const bigNumberAmount = new BigNumber(amount);\n    if (!bigNumberAmount.isInteger() || bigNumberAmount.isLessThanOrEqualTo(0)) {\n      return false;\n    }\n    return true;\n  }\n\n  /**\n   * Decodes raw tx data into messages, signing info, and fee data\n   * @param {string} txHex - raw base64 tx\n   * @returns {DecodedTxRaw} Decoded transaction\n   */\n  getDecodedTxFromRawBase64(txRaw: string): DecodedTxRaw {\n    try {\n      return decodeTxRaw(fromBase64(txRaw));\n    } catch (e) {\n      throw new ParseTransactionError('Error decoding TxRaw base64 encoded string: ' + e.message);\n    }\n  }\n\n  /**\n   * Returns the array of messages in the body of the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {EncodeObject[]} messages along with type url\n   */\n  private getEncodedMessagesFromDecodedTx(decodedTx: DecodedTxRaw): EncodeObject[] {\n    return decodedTx.body.messages;\n  }\n\n  /**\n   * Checks the txn sequence is valid or not\n   * @param {number} sequence\n   */\n  validateSequence(sequence: number) {\n    if (sequence < 0) {\n      throw new InvalidTransactionError('Invalid sequence: less than zero');\n    }\n  }\n\n  /**\n   * Pulls the sequence number from a DecodedTxRaw AuthInfo property\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {number} sequence\n   */\n  getSequenceFromDecodedTx(decodedTx: DecodedTxRaw): number {\n    return Number(decodedTx.authInfo.signerInfos[0].sequence);\n  }\n\n  /**\n   * Pulls the typeUrl from the encoded message of a DecodedTxRaw\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {string} cosmos proto type url\n   */\n  getTypeUrlFromDecodedTx(decodedTx: DecodedTxRaw): string {\n    const encodedMessage = this.getEncodedMessagesFromDecodedTx(decodedTx)[0];\n    return encodedMessage.typeUrl;\n  }\n\n  /**\n   * Returns the fee data from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {FeeData} fee data\n   */\n  getGasBudgetFromDecodedTx(decodedTx: DecodedTxRaw): FeeData {\n    return {\n      amount: decodedTx.authInfo.fee?.amount as Coin[],\n      gasLimit: Number(decodedTx.authInfo.fee?.gasLimit),\n    };\n  }\n\n  /**\n   * Returns the publicKey from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {string | undefined} publicKey in hex format if it exists, undefined otherwise\n   */\n  getPublicKeyFromDecodedTx(decodedTx: DecodedTxRaw): string | undefined {\n    const publicKeyUInt8Array = decodedTx.authInfo.signerInfos?.[0].publicKey?.value;\n    if (publicKeyUInt8Array) {\n      return toHex(fromBase64(decodePubkey(decodedTx.authInfo.signerInfos?.[0].publicKey)?.value));\n    }\n    return undefined;\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} Send transaction message data\n   */\n  protected getSendMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    return decodedTx.body.messages.map((message) => {\n      const value = this.registry.decode(message);\n      return {\n        value: {\n          fromAddress: value.fromAddress,\n          toAddress: value.toAddress,\n          amount: value.amount,\n        },\n        typeUrl: message.typeUrl,\n      };\n    });\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} Delegate of undelegate transaction message data\n   */\n  getDelegateOrUndelegateMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    return decodedTx.body.messages.map((message) => {\n      const value = this.registry.decode(message);\n      return {\n        value: {\n          delegatorAddress: value.delegatorAddress,\n          validatorAddress: value.validatorAddress,\n          amount: value.amount,\n        },\n        typeUrl: message.typeUrl,\n      };\n    });\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} Redelegate transaction message data\n   */\n  getRedelegateMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    return decodedTx.body.messages.map((message) => {\n      const value = this.registry.decode(message);\n      return {\n        value: {\n          delegatorAddress: value.delegatorAddress,\n          validatorSrcAddress: value.validatorSrcAddress,\n          validatorDstAddress: value.validatorDstAddress,\n          amount: value.amount,\n        },\n        typeUrl: message.typeUrl,\n      };\n    });\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} WithdrawDelegatorRewards transaction message data\n   */\n  getWithdrawRewardsMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    return decodedTx.body.messages.map((message) => {\n      const value = this.registry.decode(message);\n      return {\n        value: {\n          delegatorAddress: value.delegatorAddress,\n          validatorAddress: value.validatorAddress,\n        },\n        typeUrl: message.typeUrl,\n      };\n    });\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} Delegate of undelegate transaction message data\n   */\n  getWithdrawDelegatorRewardsMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    return decodedTx.body.messages.map((message) => {\n      const value = this.registry.decode(message);\n      return {\n        value: {\n          delegatorAddress: value.delegatorAddress,\n          validatorAddress: value.validatorAddress,\n        },\n        typeUrl: message.typeUrl,\n      };\n    });\n  }\n\n  /**\n   * Get a cosmos chain address from its equivalent hex\n   * @param {string} prefix\n   * @param {string} addressHex\n   * @returns {string}\n   */\n  getCosmosLikeAddressFromHex(prefix: string, addressHex: string): string {\n    if (addressHex.indexOf('0x') === 0) {\n      addressHex = addressHex.slice(2);\n    }\n    return toBech32(prefix, fromHex(addressHex));\n  }\n\n  /**\n   * Get a EVM chain address from its equivalent hex\n   * @param {string} prefix\n   * @param {string} addressHex\n   * @returns {string}\n   */\n  getEvmLikeAddressFromCosmos(cosmosLikeAddress: string): string {\n    return '0x' + toHex(fromBech32(cosmosLikeAddress).data);\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} Execute contract transaction message data\n   */\n  getExecuteContractMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    return decodedTx.body.messages.map((message) => {\n      const value = this.registry.decode(message);\n      return {\n        value: {\n          sender: value.sender,\n          contract: value.contract,\n          msg: value.msg,\n          funds: value.funds,\n        },\n        typeUrl: message.typeUrl,\n      };\n    });\n  }\n\n  /**\n   * Returns the array of MessageData[] from the decoded transaction\n   * @param {DecodedTxRaw} decodedTx\n   * @returns {MessageData[]} Custom transaction message data\n   */\n  getCustomMessageDataFromDecodedTx(decodedTx: DecodedTxRaw): MessageData<CustomMessage>[] {\n    throw new NotSupported('Custom message data not supported');\n  }\n\n  /**\n   * Determines bitgo transaction type based on cosmos proto type url\n   * @param {string} typeUrl\n   * @returns {TransactionType | undefined} TransactionType if url is supported else undefined\n   */\n  getTransactionTypeFromTypeUrl(typeUrl: string): TransactionType | undefined {\n    switch (typeUrl) {\n      case constants.sendMsgTypeUrl:\n        return TransactionType.Send;\n      case constants.sendMsgType:\n        return TransactionType.Send;\n      case constants.delegateMsgTypeUrl:\n        return TransactionType.StakingActivate;\n      case constants.undelegateMsgTypeUrl:\n        return TransactionType.StakingDeactivate;\n      case constants.withdrawDelegatorRewardMsgTypeUrl:\n        return TransactionType.StakingWithdraw;\n      case constants.executeContractMsgTypeUrl:\n        return TransactionType.ContractCall;\n      case constants.redelegateTypeUrl:\n        return TransactionType.StakingRedelegate;\n      default:\n        return undefined;\n    }\n  }\n\n  /**\n   * Takes a hex encoded pubkey, converts it to the Amino JSON representation (type/value wrapper)\n   * and returns it as protobuf `Any`\n   * @param {string} pubkey hex encoded compressed secp256k1 public key\n   * @returns {Any} pubkey encoded as protobuf `Any`\n   */\n  getEncodedPubkey(pubkey: string): Any {\n    return encodePubkey(encodeSecp256k1Pubkey(fromHex(pubkey)));\n  }\n\n  /**\n   * Gets the send messages used in the final step of encoding a transaction. This allows for any final processing needed.\n   * @param {CosmosLikeTransaction} cosmosLikeTransaction transaction to get send messages from\n   * @returns {Any[]} processed send messages\n   */\n  getSendMessagesForEncodingTx(cosmosLikeTransaction: CosmosLikeTransaction<CustomMessage>): Any[] {\n    return cosmosLikeTransaction.sendMessages as unknown as Any[];\n  }\n\n  /**\n   * Creates a txRaw from an cosmos like transaction @see CosmosLikeTransaction\n   * @Precondition cosmosLikeTransaction.publicKey must be defined\n   * @param {CosmosLikeTransaction} cosmosLikeTransaction\n   * @returns {TxRaw} Unsigned raw transaction\n   */\n  createTxRawFromCosmosLikeTransaction(cosmosLikeTransaction: CosmosLikeTransaction<CustomMessage>): TxRaw {\n    if (!cosmosLikeTransaction.publicKey) {\n      throw new Error('publicKey is required to create a txRaw');\n    }\n    const encodedPublicKey: Any = this.getEncodedPubkey(cosmosLikeTransaction.publicKey);\n    const messages = this.getSendMessagesForEncodingTx(cosmosLikeTransaction);\n    let txBodyValue;\n    if (cosmosLikeTransaction.memo) {\n      txBodyValue = {\n        memo: cosmosLikeTransaction.memo,\n        messages: messages,\n      };\n    } else {\n      txBodyValue = {\n        messages: messages,\n      };\n    }\n\n    const txBodyBytes = this.registry.encodeTxBody(txBodyValue);\n    const sequence = cosmosLikeTransaction.sequence;\n    const authInfoBytes = makeAuthInfoBytes(\n      [{ pubkey: encodedPublicKey, sequence }],\n      cosmosLikeTransaction.gasBudget.amount,\n      cosmosLikeTransaction.gasBudget.gasLimit,\n      undefined,\n      undefined,\n      undefined\n    );\n    return TxRaw.fromPartial({\n      bodyBytes: txBodyBytes,\n      authInfoBytes: authInfoBytes,\n    });\n  }\n\n  /**\n   * Encodes a signature into a txRaw\n   * @param {string} publicKeyHex publicKey in hex encoded string format\n   * @param {string} signatureHex signature in hex encoded string format\n   * @param {TxRaw} unsignedTx raw transaction\n   * @returns {TxRaw} Signed raw transaction\n   */\n  createSignedTxRaw(\n    publicKeyHex: string,\n    signatureHex: string,\n    unsignedTx: { bodyBytes: Uint8Array; authInfoBytes: Uint8Array }\n  ): TxRaw {\n    const stdSignature = encodeSecp256k1Signature(fromHex(publicKeyHex), fromHex(signatureHex));\n    return TxRaw.fromPartial({\n      bodyBytes: unsignedTx.bodyBytes,\n      authInfoBytes: unsignedTx.authInfoBytes,\n      signatures: [fromBase64(stdSignature.signature)],\n    });\n  }\n\n  /**\n   * Decodes a raw transaction into a DecodedTxRaw and checks if it has non empty signatures\n   * @param {string} rawTransaction\n   * @returns {boolean} true if transaction is signed else false\n   */\n  isSignedRawTx(rawTransaction: string): boolean {\n    const decodedTx = this.getDecodedTxFromRawBase64(rawTransaction);\n    if (decodedTx.signatures.length > 0) {\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * Returns whether or not the string is a valid protocol public key\n   * @param {string | undefined} publicKey - the  public key to be validated\n   */\n  validatePublicKey(publicKey: string | undefined) {\n    if (publicKey !== undefined) {\n      try {\n        new KeyPair({ pub: publicKey });\n      } catch {\n        throw new InvalidTransactionError(`Invalid Public Key`);\n      }\n    }\n  }\n\n  /**\n   * Creates a sign doc from an cosmos like transaction @see CosmosLikeTransaction\n   * @Precondition cosmosLikeTransaction.accountNumber and cosmosLikeTransaction.chainId must be defined\n   * @param {CosmosLikeTransaction} cosmosLikeTransaction\n   * @returns {SignDoc} sign doc\n   */\n  createSignDoc(\n    cosmosLikeTransaction: CosmosLikeTransaction<CustomMessage>,\n    accountNumber: number | undefined,\n    chainId: string | undefined\n  ): SignDoc {\n    if (!accountNumber) {\n      throw new Error('accountNumber is required to create a sign doc');\n    }\n    if (!chainId) {\n      throw new Error('chainId is required to create a sign doc');\n    }\n    if (!cosmosLikeTransaction) {\n      throw new Error('cosmosLikeTransaction is required to create a sign doc');\n    }\n    const txRaw = this.createTxRawFromCosmosLikeTransaction(cosmosLikeTransaction);\n    return makeSignDoc(txRaw.bodyBytes, txRaw.authInfoBytes, chainId, accountNumber);\n  }\n\n  /**\n   * Returns whether or not the string is a valid hex\n   * @param hexString - hex string format\n   * @returns {boolean} true if string is hex else false\n   */\n  isValidHexString(hexString: string): boolean {\n    return /^[0-9A-Fa-f]*$/.test(hexString);\n  }\n\n  /**\n   * Validates the WithdrawDelegatorRewardsMessage\n   * @param {WithdrawDelegatorRewardsMessage} withdrawRewardsMessage - The WithdrawDelegatorRewardsMessage to validate.\n   * @throws {InvalidTransactionError} Throws an error if the validatorAddress or delegatorAddress is invalid or missing.\n   */\n  validateWithdrawRewardsMessage(withdrawRewardsMessage: WithdrawDelegatorRewardsMessage) {\n    if (\n      !withdrawRewardsMessage.validatorAddress ||\n      !this.isValidValidatorAddress(withdrawRewardsMessage.validatorAddress)\n    ) {\n      throw new InvalidTransactionError(\n        `Invalid WithdrawDelegatorRewardsMessage validatorAddress: ` + withdrawRewardsMessage.validatorAddress\n      );\n    }\n    if (!withdrawRewardsMessage.delegatorAddress || !this.isValidAddress(withdrawRewardsMessage.delegatorAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid WithdrawDelegatorRewardsMessage delegatorAddress: ` + withdrawRewardsMessage.delegatorAddress\n      );\n    }\n  }\n\n  /**\n   * Helper method to check if the specified properties in an object are missing or null.\n   * @param {Object} obj - The object to check.\n   * @param {string[]} keys - An array of property keys to check.\n   * @throws {Error} Throws an error if any of the specified properties are missing or null.\n   */\n  isObjPropertyNull(obj: { [key: string]: any }, keys: Array<string>) {\n    for (const key of keys) {\n      if (obj[key] == null) {\n        throw new Error(`Missing or null value for property ${key}`);\n      }\n    }\n  }\n\n  /**\n   * Validates the DelegateOrUndelegeteMessage\n   * @param {DelegateOrUndelegeteMessage} delegateMessage - The DelegateOrUndelegeteMessage to validate.\n   * @throws {InvalidTransactionError} Throws an error if the validatorAddress, delegatorAddress, or amount is invalid or missing.\n   */\n  validateDelegateOrUndelegateMessage(delegateMessage: DelegateOrUndelegeteMessage) {\n    this.isObjPropertyNull(delegateMessage, ['validatorAddress', 'delegatorAddress']);\n\n    if (!this.isValidValidatorAddress(delegateMessage.validatorAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid DelegateOrUndelegeteMessage validatorAddress: ` + delegateMessage.validatorAddress\n      );\n    }\n    if (!this.isValidAddress(delegateMessage.delegatorAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid DelegateOrUndelegeteMessage delegatorAddress: ` + delegateMessage.delegatorAddress\n      );\n    }\n    this.validateAmount(delegateMessage.amount);\n  }\n\n  /**\n   * Validates the RedelegateMessage\n   * @param {DelegateOrUndelegeteMessage} redelegateMessage - The RedelegateMessage to validate.\n   * @throws {InvalidTransactionError} Throws an error if the validatorSrcAddress, validatorDstAddress, delegatorAddress, or amount is invalid or missing.\n   */\n  validateRedelegateMessage(redelegateMessage: RedelegateMessage) {\n    this.isObjPropertyNull(redelegateMessage, ['validatorSrcAddress', 'validatorDstAddress', 'delegatorAddress']);\n\n    if (!this.isValidValidatorAddress(redelegateMessage.validatorSrcAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid RedelegateMessage validatorSrcAddress: ` + redelegateMessage.validatorSrcAddress\n      );\n    }\n    if (!this.isValidValidatorAddress(redelegateMessage.validatorDstAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid RedelegateMessage validatorDstAddress: ` + redelegateMessage.validatorDstAddress\n      );\n    }\n    if (!this.isValidAddress(redelegateMessage.delegatorAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid DelegateOrUndelegeteMessage delegatorAddress: ` + redelegateMessage.delegatorAddress\n      );\n    }\n    this.validateAmount(redelegateMessage.amount);\n  }\n\n  /**\n   * Validates the CustomMessage\n   * @param {CustomMessage} customMessage - The CustomMessage to validate.\n   * @throws {InvalidTransactionError} Throws an error if the custom message is invalid or missing required fields.\n   * @throws {NotSupported} Throws an error if the custom message data is not supported.\n   */\n  validateCustomMessage(customMessage: CustomMessage) {\n    throw new NotSupported('Custom message data not supported');\n  }\n\n  /**\n   * Validates the MessageData\n   * @param {MessageData} messageData - The MessageData to validate.\n   * @throws {InvalidTransactionError} Throws an error if the messageData is invalid or missing required fields.\n   */\n  validateMessageData(messageData: MessageData<CustomMessage>): void {\n    if (messageData == null) {\n      throw new InvalidTransactionError(`Invalid MessageData: undefined`);\n    }\n    if (messageData.typeUrl == null || this.getTransactionTypeFromTypeUrl(messageData.typeUrl) == null) {\n      throw new InvalidTransactionError(`Invalid MessageData typeurl: ` + messageData.typeUrl);\n    }\n\n    const type = this.getTransactionTypeFromTypeUrl(messageData.typeUrl);\n    switch (type) {\n      case TransactionType.Send: {\n        const value = messageData.value as SendMessage;\n        this.validateSendMessage(value);\n        break;\n      }\n      case TransactionType.StakingActivate:\n      case TransactionType.StakingDeactivate: {\n        const value = messageData.value as DelegateOrUndelegeteMessage;\n        this.validateDelegateOrUndelegateMessage(value);\n        break;\n      }\n      case TransactionType.StakingWithdraw: {\n        const value = messageData.value as WithdrawDelegatorRewardsMessage;\n        this.validateWithdrawRewardsMessage(value);\n        break;\n      }\n      case TransactionType.ContractCall: {\n        const value = messageData.value as ExecuteContractMessage;\n        this.validateExecuteContractMessage(value, TransactionType.ContractCall);\n        break;\n      }\n      case TransactionType.StakingRedelegate: {\n        const value = messageData.value as RedelegateMessage;\n        this.validateRedelegateMessage(value);\n        break;\n      }\n      case TransactionType.CustomTx: {\n        const value = messageData.value as CustomMessage;\n        this.validateCustomMessage(value);\n        break;\n      }\n      default:\n        throw new InvalidTransactionError(`Invalid MessageData TypeUrl is not supported: ` + messageData.typeUrl);\n    }\n  }\n\n  /**\n   * Validates the Cosmos-like transaction.\n   * @param {CosmosLikeTransaction} tx - The transaction to validate.\n   * @throws {InvalidTransactionError} Throws an error if the transaction is invalid or missing required fields.\n   */\n  validateTransaction(tx: CosmosLikeTransaction<CustomMessage>): void {\n    this.validateSequence(tx.sequence);\n    this.validateGasBudget(tx.gasBudget);\n    this.validatePublicKey(tx.publicKey);\n    if (tx.sendMessages === undefined || tx.sendMessages.length === 0) {\n      throw new InvalidTransactionError('Invalid transaction: messages is required');\n    } else {\n      tx.sendMessages.forEach((message) => this.validateMessageData(message));\n    }\n  }\n\n  /**\n   * Creates a Cosmos-like transaction.\n   * @param {number} sequence - The sender address sequence number for the transaction.\n   * @param {MessageData[]} messages - The array of message data for the transaction.\n   * @param {FeeData} gasBudget - The fee data for the transaction.\n   * @param {string} [publicKey] - The public key associated with the sender.\n   * @param {string} [memo] - The memo for the transaction.\n   * @returns {CosmosLikeTransaction} Returns the created Cosmos-like transaction.\n   * @throws {InvalidTransactionError} Throws an error if the created transaction is invalid.\n   */\n  createTransaction(\n    sequence: number,\n    messages: MessageData<CustomMessage>[],\n    gasBudget: FeeData,\n    publicKey?: string,\n    memo?: string\n  ): CosmosLikeTransaction<CustomMessage> {\n    const cosmosLikeTxn = {\n      sequence: sequence,\n      sendMessages: messages,\n      gasBudget: gasBudget,\n      publicKey: publicKey,\n      memo: memo,\n    };\n    this.validateTransaction(cosmosLikeTxn);\n    return cosmosLikeTxn;\n  }\n\n  /**\n   * Creates a Cosmos-like transaction with a hash.\n   * @param {number} sequence - The sender address sequence number for the transaction.\n   * @param {MessageData[]} messages - The array of message data for the transaction.\n   * @param {FeeData} gasBudget - The fee data for the transaction.\n   * @param {string} [publicKey] - The public key associated with the transaction.\n   * @param {Buffer} [signature] - The signature for the transaction.\n   * @param {string} [memo] - The memo for the transaction.\n   * @returns {CosmosLikeTransaction} Returns the created Cosmos-like transaction with the hash and signature if provided.\n   */\n  createTransactionWithHash(\n    sequence: number,\n    messages: MessageData<CustomMessage>[],\n    gasBudget: FeeData,\n    publicKey?: string,\n    signature?: Buffer,\n    memo?: string\n  ): CosmosLikeTransaction<CustomMessage> {\n    const cosmosLikeTxn = this.createTransaction(sequence, messages, gasBudget, publicKey, memo);\n    let hash = constants.UNAVAILABLE_TEXT;\n    if (signature !== undefined) {\n      const unsignedTx = this.createTxRawFromCosmosLikeTransaction(cosmosLikeTxn);\n      const signedTx = TxRaw.fromPartial({\n        bodyBytes: unsignedTx.bodyBytes,\n        authInfoBytes: unsignedTx.authInfoBytes,\n        signatures: [signature],\n      });\n      hash = createHash('sha256')\n        .update(TxRaw.encode(signedTx).finish())\n        .digest()\n        .toString('hex')\n        .toLocaleUpperCase('en-US');\n      return { ...cosmosLikeTxn, hash: hash, signature: signature };\n    }\n    return { ...cosmosLikeTxn, hash: hash };\n  }\n\n  /**\n   * Deserializes base64 enocded raw transaction string into @see CosmosLikeTransaction\n   * @param {string} rawTx base64 enocded raw transaction string\n   * @returns {CosmosLikeTransaction} Deserialized cosmosLikeTransaction\n   */\n  deserializeTransaction(rawTx: string): CosmosLikeTransaction<CustomMessage> {\n    const decodedTx = this.getDecodedTxFromRawBase64(rawTx);\n    const typeUrl = this.getTypeUrlFromDecodedTx(decodedTx);\n    const type: TransactionType | undefined = this.getTransactionTypeFromTypeUrl(typeUrl);\n    let sendMessageData: MessageData<CustomMessage>[];\n    if (type === TransactionType.Send) {\n      sendMessageData = this.getSendMessageDataFromDecodedTx(decodedTx);\n    } else if (type === TransactionType.StakingActivate || type === TransactionType.StakingDeactivate) {\n      sendMessageData = this.getDelegateOrUndelegateMessageDataFromDecodedTx(decodedTx);\n    } else if (type === TransactionType.StakingWithdraw) {\n      sendMessageData = this.getWithdrawRewardsMessageDataFromDecodedTx(decodedTx);\n    } else if (type === TransactionType.ContractCall) {\n      sendMessageData = this.getExecuteContractMessageDataFromDecodedTx(decodedTx);\n    } else if (type === TransactionType.StakingRedelegate) {\n      sendMessageData = this.getRedelegateMessageDataFromDecodedTx(decodedTx);\n    } else if (type === TransactionType.CustomTx) {\n      sendMessageData = this.getCustomMessageDataFromDecodedTx(decodedTx);\n    } else {\n      throw new Error('Transaction type not supported: ' + typeUrl);\n    }\n    const sequence = this.getSequenceFromDecodedTx(decodedTx);\n    const gasBudget = this.getGasBudgetFromDecodedTx(decodedTx);\n    const publicKey = this.getPublicKeyFromDecodedTx(decodedTx);\n    const signature = decodedTx.signatures?.[0] !== undefined ? Buffer.from(decodedTx.signatures[0]) : undefined;\n    return this.createTransactionWithHash(\n      sequence,\n      sendMessageData,\n      gasBudget,\n      publicKey,\n      signature,\n      decodedTx.body?.memo\n    );\n  }\n\n  /**\n   * Validates an array of coin amounts.\n   * @param {Coin[]} amountArray - The array of coin amounts to validate.\n   * @param {TransactionType} transactionType - optional field for transaction type\n   */\n  validateAmountData(amountArray: Coin[], transactionType?: TransactionType): void {\n    amountArray.forEach((coinAmount) => {\n      this.validateAmount(coinAmount, transactionType);\n    });\n  }\n\n  /**\n   * Validates the gas limit and gas amount for a transaction.\n   * @param {FeeData} gasBudget - The gas budget to validate.\n   * @throws {InvalidTransactionError} Throws an error if the gas budget is invalid.\n   */\n  validateGasBudget(gasBudget: FeeData): void {\n    if (gasBudget.gasLimit <= 0) {\n      throw new InvalidTransactionError('Invalid gas limit ' + gasBudget.gasLimit);\n    }\n    this.validateAmountData(gasBudget.amount);\n  }\n\n  /**\n   * Validates a send message for a transaction.\n   * @param {SendMessage} sendMessage - The send message to validate.\n   * @throws {InvalidTransactionError} Throws an error if the send message is invalid.\n   */\n  validateSendMessage(sendMessage: SendMessage) {\n    if (!sendMessage.toAddress || !this.isValidAddress(sendMessage.toAddress)) {\n      throw new InvalidTransactionError(`Invalid SendMessage toAddress: ` + sendMessage.toAddress);\n    }\n    if (!sendMessage.fromAddress || !this.isValidAddress(sendMessage.fromAddress)) {\n      throw new InvalidTransactionError(`Invalid SendMessage fromAddress: ` + sendMessage.fromAddress);\n    }\n    this.validateAmountData(sendMessage.amount);\n  }\n\n  /**\n   * Validates a coin amount.\n   * @param {Coin} amount - The coin amount to validate.\n   * @param {TransactionType} transactionType - optional field for transaction type\n   * @throws {InvalidTransactionError} Throws an error if the coin amount is invalid.\n   */\n  validateAmount(amount: Coin, transactionType?: TransactionType): void {\n    throw new NotImplementedError('validateAmount not implemented');\n  }\n\n  /**\n   * Checks if a cosmos like Bech32 address matches given regular expression and\n   * validates memoId if present\n   * @param {string} address\n   * @param {RegExp} regExp Regular expression to validate the root address against after trimming the memoId\n   * @returns {boolean} true if address is valid\n   */\n  protected isValidCosmosLikeAddressWithMemoId(address: string, regExp: RegExp): boolean {\n    if (typeof address !== 'string') return false;\n    const addressArray = address.split('?memoId=');\n    if (\n      ![1, 2].includes(addressArray.length) || // should have at most one occurrence of 'memoId='\n      !this.isValidBech32AddressMatchingRegex(addressArray[0], regExp) ||\n      (addressArray[1] && !this.isValidMemoId(addressArray[1]))\n    ) {\n      return false;\n    }\n    return true;\n  }\n\n  /**\n   * Checks if address is valid Bech32 and matches given regular expression\n   * @param {string} address\n   * @param {RegExp} regExp Regular expression to validate the address against\n   * @returns {boolean} true if address is valid\n   */\n  protected isValidBech32AddressMatchingRegex(address: string, regExp: RegExp): boolean {\n    try {\n      fromBech32(address);\n    } catch (e) {\n      return false;\n    }\n    return regExp.test(address);\n  }\n\n  /**\n   * Return boolean indicating whether a memo id is valid\n   *\n   * @param memoId memo id\n   * @returns true if memo id is valid\n   */\n  isValidMemoId(memoId: string): boolean {\n    // Allow alphanumeric memo IDs (including uppercase and lowercase letters)\n    const alphanumericRegex = /^[0-9a-zA-Z]+$/;\n\n    // Check if the memoId is alphanumeric\n    if (!alphanumericRegex.test(memoId)) {\n      return false;\n    }\n\n    // If the memoId is purely numeric, ensure it is a positive integer\n    if (/^\\d+$/.test(memoId)) {\n      const memoIdNumber = new BigNumber(memoId);\n      return memoIdNumber.gte(0) && memoIdNumber.isInteger();\n    }\n\n    return true;\n  }\n\n  /**\n   * Validates if the address matches with regex @see accountAddressRegex\n   * @param {string} address\n   * @returns {boolean} - the validation result\n   */\n  isValidValidatorAddress(address: string): boolean {\n    throw new NotImplementedError('isValidValidatorAddress not implemented');\n  }\n\n  /**\n   * Validates if the address matches with regex @see accountAddressRegex\n   * @param {string} address\n   * @returns {boolean} - the validation result\n   */\n  isValidAddress(address: string): boolean {\n    throw new NotImplementedError('isValidAddress not implemented');\n  }\n\n  /**\n   * Validates if the address matches with regex @see contractAddressRegex\n   * @param {string} address\n   * @returns {boolean} - the validation result\n   */\n  isValidContractAddress(address: string): boolean {\n    throw new NotImplementedError('isValidContractAddress not implemented');\n  }\n\n  /**\n   * Validates a execute contract message\n   * @param {ExecuteContractMessage} message - The execute contract message to validate\n   * @param {TransactionType} transactionType - optional field for transaction type\n   * @throws {InvalidTransactionError} Throws an error if the message is invalid\n   */\n  validateExecuteContractMessage(message: ExecuteContractMessage, transactionType?: TransactionType) {\n    if (!message.contract || !this.isValidContractAddress(message.contract)) {\n      throw new InvalidTransactionError(`Invalid ExecuteContractMessage contract address: ` + message.contract);\n    }\n    if (!message.sender || !this.isValidAddress(message.sender)) {\n      throw new InvalidTransactionError(`Invalid ExecuteContractMessage sender address: ` + message.sender);\n    }\n    if (!message.msg) {\n      throw new InvalidTransactionError(`Invalid ExecuteContractMessage msg: ` + message.msg);\n    }\n    if (message.funds) {\n      this.validateAmountData(message.funds, transactionType);\n    }\n  }\n\n  /**\n   * Get coin specific hash function\n   * @returns {Hash} The hash function\n   */\n  getHashFunction(): Hash {\n    return createHash('sha256');\n  }\n\n  getTokenDenomsUsingCoinFamily(coinFamily: string): string[] {\n    // using set to remove duplicates as denom can be same on testnet and mainnet for a few tokens\n    return [\n      ...new Set(\n        coins\n          .filter(\n            (coin) => coin.family.toLowerCase() === coinFamily.toLowerCase() && coin.isToken && coin.denom !== undefined\n          )\n          .map((coin) => coin.denom as string)\n      ),\n    ];\n  }\n}\n\nconst utils = new CosmosUtils();\n\nexport default utils;\n"]}

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


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