UNPKG

@substrate/api-sidecar

Version:

REST service that makes it easy to interact with blockchain nodes built using Substrate's FRAME framework.

126 lines 6.76 kB
"use strict"; // Copyright 2017-2025 Parity Technologies (UK) Ltd. // This file is part of Substrate API Sidecar. // // Substrate API Sidecar is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see <http://www.gnu.org/licenses/>. var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const middleware_1 = require("../../middleware"); const services_1 = require("../../services"); const AbstractController_1 = __importDefault(require("../AbstractController")); /** * GET balance information for an address. * * Paths: * - `address`: The address to query. * * Query: * - (Optional)`at`: Block at which to retrieve runtime version information at. Block * identifier, as the block height or block hash. Defaults to most recent block. * - (Optional) `useRcBlock`: When true, treats the `at` parameter as a relay chain block * to find corresponding Asset Hub blocks. Only supported for Asset Hub endpoints. * * Returns: * - When using `useRcBlock=true`: An array of response objects, one for each Asset Hub block found * in the specified relay chain block. Returns empty array `[]` if no Asset Hub blocks found. * - When using `useRcBlock=false` or omitted: A single response object. * * Response object structure: * - `at`: Block number and hash at which the call was made. * - `nonce`: Account nonce. * - `free`: Free balance of the account. Not equivalent to _spendable_ balance. This is the only * balance that matters in terms of most operations on tokens. * - `reserved`: Reserved balance of the account. * - `miscFrozen`: The amount that `free` may not drop below when withdrawing for anything except * transaction fee payment. * - `feeFrozen`: The amount that `free` may not drop below when withdrawing specifically for * transaction fee payment. * - `locks`: Array of locks on a balance. There can be many of these on an account and they * "overlap", so the same balance is frozen by multiple locks. Contains: * - `id`: An identifier for this lock. Only one lock may be in existence for each identifier. * - `amount`: The amount below which the free balance may not drop with this lock in effect. * - `reasons`: If true, then the lock remains in effect even for payment of transaction fees. * - `rcBlockNumber`: The relay chain block number used for the query. Only present when `useRcBlock=true`. * - `ahTimestamp`: The Asset Hub block timestamp. Only present when `useRcBlock=true`. * * Substrate Reference: * - FRAME System: https://crates.parity.io/frame_system/index.html * - Balances Pallet: https://crates.parity.io/pallet_balances/index.html * - `AccountInfo`: https://crates.parity.io/frame_system/struct.AccountInfo.html * - `AccountData`: https://crates.parity.io/pallet_balances/struct.AccountData.html * - `BalanceLock`: https://crates.parity.io/pallet_balances/struct.BalanceLock.html */ class AccountsBalanceController extends AbstractController_1.default { constructor(api) { super(api, '/accounts/:address/balance-info', new services_1.AccountsBalanceInfoService(api)); /** * Get the latest account balance summary of `address`. * * @param req Express Request * @param res Express Response */ this.getAccountBalanceInfo = async ({ params: { address }, query: { at, useRcBlock, token, denominated } }, res) => { if (useRcBlock === 'true') { const rcAtResults = await this.getHashFromRcAt(at); // Return empty array if no Asset Hub blocks found if (rcAtResults.length === 0) { AccountsBalanceController.sanitizedSend(res, []); return; } const tokenArg = typeof token === 'string' ? token.toUpperCase() : // We assume the first token is the native token this.api.registry.chainTokens[0].toUpperCase(); const withDenomination = denominated === 'true'; // Process each Asset Hub block found const results = []; for (const { ahHash, rcBlockNumber } of rcAtResults) { const historicApi = await this.api.at(ahHash); const result = await this.service.fetchAccountBalanceInfo(ahHash, historicApi, address, tokenArg, withDenomination); const apiAt = await this.api.at(ahHash); const ahTimestamp = await apiAt.query.timestamp.now(); const enhancedResult = { ...result, rcBlockNumber, ahTimestamp: ahTimestamp.toString(), }; results.push(enhancedResult); } AccountsBalanceController.sanitizedSend(res, results); } else { const hash = await this.getHashFromAt(at); const tokenArg = typeof token === 'string' ? token.toUpperCase() : // We assume the first token is the native token this.api.registry.chainTokens[0].toUpperCase(); const withDenomination = denominated === 'true'; const historicApi = await this.api.at(hash); const result = await this.service.fetchAccountBalanceInfo(hash, historicApi, address, tokenArg, withDenomination); AccountsBalanceController.sanitizedSend(res, result); } }; this.initRoutes(); } initRoutes() { this.router.use(this.path, middleware_1.validateAddress, (0, middleware_1.validateBoolean)(['denominated']), middleware_1.validateUseRcBlock); this.safeMountAsyncGetHandlers([['', this.getAccountBalanceInfo]]); } } AccountsBalanceController.controllerName = 'AccountsBalanceInfo'; AccountsBalanceController.requiredPallets = [['Balances', 'System']]; exports.default = AccountsBalanceController; //# sourceMappingURL=AccountsBalanceInfoController.js.map