diff --git a/README.md b/README.md index ee955dab..83d3a5a4 100644 --- a/README.md +++ b/README.md @@ -4,7 +4,7 @@ MultiversX SDK for JavaScript and TypeScript (written in TypeScript). ## Documentation - - [Cookbook](https://docs.multiversx.com/sdk-and-tools/erdjs/erdjs-cookbook/) + - [Cookbook](https://docs.multiversx.com/sdk-and-tools/sdk-js/sdk-js-cookbook/) ## Distribution diff --git a/package-lock.json b/package-lock.json index f25dd1bf..266b424f 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "@multiversx/sdk-core", - "version": "12.13.0", + "version": "12.14.0", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "@multiversx/sdk-core", - "version": "12.13.0", + "version": "12.14.0", "license": "MIT", "dependencies": { "@multiversx/sdk-transaction-decoder": "1.0.2", diff --git a/package.json b/package.json index 29314cf9..03a9351f 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@multiversx/sdk-core", - "version": "12.13.0", + "version": "12.14.0", "description": "MultiversX SDK for JavaScript and TypeScript", "main": "out/index.js", "types": "out/index.d.js", diff --git a/src/interfaceOfNetwork.ts b/src/interfaceOfNetwork.ts index 2236f10f..07edeb9e 100644 --- a/src/interfaceOfNetwork.ts +++ b/src/interfaceOfNetwork.ts @@ -66,7 +66,15 @@ export interface ITransactionLogs { events: ITransactionEvent[]; findSingleOrNoneEvent(identifier: string, predicate?: (event: ITransactionEvent) => boolean): ITransactionEvent | undefined; + + /** + * @deprecated Will be removed from the interface (with no replacement). Not used in "sdk-core". + */ findFirstOrNoneEvent(identifier: string, predicate?: (event: ITransactionEvent) => boolean): ITransactionEvent | undefined; + + /** + * @deprecated Will be removed from the interface (with no replacement). Not used in "sdk-core". + */ findEvents(identifier: string, predicate?: (event: ITransactionEvent) => boolean): ITransactionEvent[]; } diff --git a/src/smartcontracts/argSerializer.ts b/src/smartcontracts/argSerializer.ts index 164ed9dc..d4dbeafb 100644 --- a/src/smartcontracts/argSerializer.ts +++ b/src/smartcontracts/argSerializer.ts @@ -1,6 +1,6 @@ import { ARGUMENTS_SEPARATOR } from "../constants"; import { BinaryCodec } from "./codec"; -import { EndpointParameterDefinition, Type, TypedValue, U32Type, U32Value } from "./typesystem"; +import { Type, TypedValue, U32Type, U32Value } from "./typesystem"; import { OptionalType, OptionalValue } from "./typesystem/algebraic"; import { CompositeType, CompositeValue } from "./typesystem/composite"; import { VariadicType, VariadicValue } from "./typesystem/variadic"; @@ -15,9 +15,13 @@ interface ICodec { encodeTopLevel(typedValue: TypedValue): Buffer; } +interface IParameterDefinition { + type: Type; +} + // TODO: perhaps move default construction options to a factory (ArgSerializerFactory), instead of referencing them in the constructor // (postpone as much as possible, breaking change) -const defaultArgSerializerrOptions: IArgSerializerOptions = { +const defaultArgSerializerOptions: IArgSerializerOptions = { codec: new BinaryCodec() }; @@ -25,14 +29,14 @@ export class ArgSerializer { codec: ICodec; constructor(options?: IArgSerializerOptions) { - options = { ...defaultArgSerializerrOptions, ...options }; + options = { ...defaultArgSerializerOptions, ...options }; this.codec = options.codec; } /** * Reads typed values from an arguments string (e.g. aa@bb@@cc), given parameter definitions. */ - stringToValues(joinedString: string, parameters: EndpointParameterDefinition[]): TypedValue[] { + stringToValues(joinedString: string, parameters: IParameterDefinition[]): TypedValue[] { let buffers = this.stringToBuffers(joinedString); let values = this.buffersToValues(buffers, parameters); return values; @@ -49,7 +53,7 @@ export class ArgSerializer { /** * Decodes a set of buffers into a set of typed values, given parameter definitions. */ - buffersToValues(buffers: Buffer[], parameters: EndpointParameterDefinition[]): TypedValue[] { + buffersToValues(buffers: Buffer[], parameters: IParameterDefinition[]): TypedValue[] { // TODO: Refactor, split (function is quite complex). const self = this; diff --git a/src/smartcontracts/resultsParser.spec.ts b/src/smartcontracts/resultsParser.spec.ts index 25c0576b..e257cc07 100644 --- a/src/smartcontracts/resultsParser.spec.ts +++ b/src/smartcontracts/resultsParser.spec.ts @@ -1,14 +1,18 @@ import { ContractQueryResponse, ContractResultItem, ContractResults, TransactionEvent, TransactionEventTopic, TransactionLogs, TransactionOnNetwork } from "@multiversx/sdk-network-providers"; +import { TransactionEventData } from "@multiversx/sdk-network-providers/out/transactionEvents"; +import BigNumber from "bignumber.js"; import { assert } from "chai"; import * as fs from "fs"; import path from "path"; import { Address } from "../address"; +import { IAddress } from "../interface"; import { ITransactionOnNetwork } from "../interfaceOfNetwork"; import { LogLevel, Logger } from "../logger"; +import { loadAbiRegistry } from "../testutils"; import { ArgSerializer } from "./argSerializer"; import { ResultsParser } from "./resultsParser"; import { ReturnCode } from "./returnCode"; -import { BigUIntType, BigUIntValue, EndpointDefinition, EndpointModifiers, EndpointParameterDefinition, StringType, StringValue, TypedValue, U32Type, U32Value, U64Type, U64Value, VariadicType, VariadicValue } from "./typesystem"; +import { AbiRegistry, BigUIntType, BigUIntValue, EndpointDefinition, EndpointModifiers, EndpointParameterDefinition, StringType, StringValue, TypedValue, U32Type, U32Value, U64Type, U64Value, VariadicType, VariadicValue } from "./typesystem"; import { BytesType, BytesValue } from "./typesystem/bytes"; const KnownReturnCodes: string[] = [ @@ -235,6 +239,125 @@ describe("test smart contract results parser", () => { assert.deepEqual(bundle.values, []); }); + it("should parse contract event", async () => { + const abiRegistry = await loadAbiRegistry("src/testdata/esdt-safe.abi.json"); + const eventDefinition = abiRegistry.getEvent("deposit"); + + const event = new TransactionEvent({ + topics: [ + new TransactionEventTopic("ZGVwb3NpdA=="), + new TransactionEventTopic("cmzC1LRt1r10pMhNAnFb+FyudjGMq4G8CefCYdQUmmc="), + new TransactionEventTopic("AAAADFdFR0xELTAxZTQ5ZAAAAAAAAAAAAAAAAWQ="), + ], + dataPayload: new TransactionEventData(Buffer.from("AAAAAAAAA9sAAAA=", "base64")) + }); + + const bundle = parser.parseEvent(event, eventDefinition); + + assert.equal((bundle.dest_address).bech32(), "erd1wfkv9495dhtt6a9yepxsyu2mlpw2ua333j4cr0qfulpxr4q5nfnshgyqun"); + assert.equal(bundle.tokens[0].token_identifier, "WEGLD-01e49d"); + assert.deepEqual(bundle.tokens[0].token_nonce, new BigNumber(0)); + assert.deepEqual(bundle.tokens[0].amount, new BigNumber(100)); + assert.deepEqual(bundle.event_data.tx_nonce, new BigNumber(987)); + assert.isNull(bundle.event_data.opt_function); + assert.isNull(bundle.event_data.opt_arguments); + assert.isNull(bundle.event_data.opt_gas_limit); + }); + + it("should parse contract event (with multi-values)", async () => { + const abiRegistry = AbiRegistry.create({ + "events": [ + { + "identifier": "foobar", + "inputs": [ + { + "name": "a", + "type": "multi", + "indexed": true + }, + { + "name": "b", + "type": "multi", + "indexed": true + }, + { + "name": "c", + "type": "u8", + "indexed": false + } + ] + } + ] + }); + + const eventDefinition = abiRegistry.getEvent("foobar"); + + const event = { + topics: [ + new TransactionEventTopic(Buffer.from("not used").toString("base64")), + new TransactionEventTopic(Buffer.from([42]).toString("base64")), + new TransactionEventTopic(Buffer.from("test").toString("base64")), + new TransactionEventTopic(Buffer.from([43]).toString("base64")), + new TransactionEventTopic(Buffer.from("test").toString("base64")), + new TransactionEventTopic(Buffer.from("test").toString("base64")), + new TransactionEventTopic(Buffer.from([44]).toString("base64")), + ], + dataPayload: new TransactionEventData(Buffer.from([42])) + }; + + const bundle = parser.parseEvent(event, eventDefinition); + assert.deepEqual(bundle.a, [new BigNumber(42), "test", new BigNumber(43), "test"]); + assert.deepEqual(bundle.b, ["test", new BigNumber(44)]); + assert.deepEqual(bundle.c, new BigNumber(42)); + }); + + it("should parse contract event (Sirius)", async () => { + const abiRegistry = AbiRegistry.create({ + "events": [ + { + "identifier": "foobar", + "inputs": [ + { + "name": "a", + "type": "u8", + "indexed": true + }, + { + "name": "b", + "type": "u8", + "indexed": false + }, + { + "name": "c", + "type": "u8", + "indexed": false + } + ] + } + ] + }); + + const eventDefinition = abiRegistry.getEvent("foobar"); + + const event = { + topics: [ + new TransactionEventTopic(Buffer.from("not used").toString("base64")), + new TransactionEventTopic(Buffer.from([42]).toString("base64")), + ], + additionalData: [ + new TransactionEventData(Buffer.from([43])), + new TransactionEventData(Buffer.from([44])) + ], + // Will be ignored. + dataPayload: new TransactionEventData(Buffer.from([43])), + }; + + const bundle = parser.parseEvent(event, eventDefinition); + assert.deepEqual(bundle.a, new BigNumber(42)); + assert.deepEqual(bundle.b, new BigNumber(43)); + assert.deepEqual(bundle.c, new BigNumber(44)); + }); + // This test should be enabled manually and run against a set of sample transactions. // 2022-04-03: test ran against ~1800 transactions sampled from devnet. it.skip("should parse real-world contract outcomes", async () => { diff --git a/src/smartcontracts/resultsParser.ts b/src/smartcontracts/resultsParser.ts index b9122a38..6186e8e6 100644 --- a/src/smartcontracts/resultsParser.ts +++ b/src/smartcontracts/resultsParser.ts @@ -7,7 +7,7 @@ import { Logger } from "../logger"; import { ArgSerializer } from "./argSerializer"; import { TypedOutcomeBundle, UntypedOutcomeBundle } from "./interface"; import { ReturnCode } from "./returnCode"; -import { EndpointDefinition, EndpointParameterDefinition, TypedValue } from "./typesystem"; +import { Type, TypedValue } from "./typesystem"; enum WellKnownEvents { OnTransactionCompleted = "completedTxEvent", @@ -23,8 +23,24 @@ interface IResultsParserOptions { argsSerializer: IArgsSerializer; } +interface IParameterDefinition { + type: Type; +} + +interface IEventInputDefinition { + name: string; + type: Type; + indexed: boolean; +} + +interface ITransactionEvent { + readonly topics: { valueOf(): Uint8Array }[]; + readonly dataPayload?: { valueOf(): Uint8Array }; + readonly additionalData?: { valueOf(): Uint8Array }[]; +} + interface IArgsSerializer { - buffersToValues(buffers: Buffer[], parameters: EndpointParameterDefinition[]): TypedValue[]; + buffersToValues(buffers: Buffer[], parameters: IParameterDefinition[]): TypedValue[]; stringToBuffers(joinedString: string): Buffer[]; } @@ -46,7 +62,7 @@ export class ResultsParser { this.argsSerializer = options.argsSerializer; } - parseQueryResponse(queryResponse: IContractQueryResponse, endpoint: EndpointDefinition): TypedOutcomeBundle { + parseQueryResponse(queryResponse: IContractQueryResponse, endpoint: { output: IParameterDefinition[] }): TypedOutcomeBundle { let parts = queryResponse.getReturnDataParts(); let values = this.argsSerializer.buffersToValues(parts, endpoint.output); let returnCode = new ReturnCode(queryResponse.returnCode.toString()); @@ -72,7 +88,7 @@ export class ResultsParser { }; } - parseOutcome(transaction: ITransactionOnNetwork, endpoint: EndpointDefinition): TypedOutcomeBundle { + parseOutcome(transaction: ITransactionOnNetwork, endpoint: { output: IParameterDefinition[] }): TypedOutcomeBundle { let untypedBundle = this.parseUntypedOutcome(transaction); let values = this.argsSerializer.buffersToValues(untypedBundle.values, endpoint.output); @@ -315,4 +331,40 @@ export class ResultsParser { let returnCode = ReturnCode.fromBuffer(returnCodePart); return { returnCode, returnDataParts }; } + + parseEvent(transactionEvent: ITransactionEvent, eventDefinition: { inputs: IEventInputDefinition[] }): any { + const result: any = {}; + + // We skip the first topic, because that's the event identifier. + const topics = transactionEvent.topics.map(topic => Buffer.from(topic.valueOf())).slice(1); + // < Sirius. + const legacyData = transactionEvent.dataPayload?.valueOf() || Buffer.from([]); + // >= Sirius. + const additionalData = transactionEvent.additionalData?.map(data => Buffer.from(data.valueOf())) || []; + + // < Sirius. + if (additionalData.length == 0) { + if (legacyData.length > 0) { + additionalData.push(Buffer.from(legacyData)); + } + } + + // "Indexed" ABI "event.inputs" correspond to "event.topics[1:]": + const indexedInputs = eventDefinition.inputs.filter(input => input.indexed); + const decodedTopics = this.argsSerializer.buffersToValues(topics, indexedInputs); + + for (let i = 0; i < indexedInputs.length; i++) { + result[indexedInputs[i].name] = decodedTopics[i].valueOf(); + } + + // "Non-indexed" ABI "event.inputs" correspond to "event.data": + const nonIndexedInputs = eventDefinition.inputs.filter(input => !input.indexed); + const decodedDataParts = this.argsSerializer.buffersToValues(additionalData, nonIndexedInputs); + + for (let i = 0; i < nonIndexedInputs.length; i++) { + result[nonIndexedInputs[i].name] = decodedDataParts[i].valueOf(); + } + + return result; + } } diff --git a/src/smartcontracts/typesystem/abiRegistry.spec.ts b/src/smartcontracts/typesystem/abiRegistry.spec.ts index 1b8ccd85..55910ace 100644 --- a/src/smartcontracts/typesystem/abiRegistry.spec.ts +++ b/src/smartcontracts/typesystem/abiRegistry.spec.ts @@ -140,4 +140,20 @@ describe("test abi registry", () => { assert.deepEqual(registry.getEndpoint("bar").output[0].type, new VariadicType(new U32Type(), true)); assert.deepEqual(registry.getEndpoint("bar").output[1].type, new VariadicType(new BytesType(), true)); }); + + it("should load ABI wih events", async () => { + const registry = await loadAbiRegistry("src/testdata/esdt-safe.abi.json"); + + assert.lengthOf(registry.events, 8); + + const depositEvent = registry.getEvent("deposit"); + assert.deepEqual(depositEvent.inputs[0].type, new AddressType()); + assert.deepEqual(depositEvent.inputs[1].type, new ListType(registry.getCustomType("EsdtTokenPayment"))); + assert.deepEqual(depositEvent.inputs[2].type, registry.getCustomType("DepositEvent")); + + const setStatusEvent = registry.getEvent("setStatusEvent"); + assert.deepEqual(setStatusEvent.inputs[0].type, new U64Type()); + assert.deepEqual(setStatusEvent.inputs[1].type, new U64Type()); + assert.deepEqual(setStatusEvent.inputs[2].type, registry.getCustomType("TransactionStatus")); + }); }); diff --git a/src/smartcontracts/typesystem/abiRegistry.ts b/src/smartcontracts/typesystem/abiRegistry.ts index 2f4679a4..41976255 100644 --- a/src/smartcontracts/typesystem/abiRegistry.ts +++ b/src/smartcontracts/typesystem/abiRegistry.ts @@ -2,6 +2,7 @@ import * as errors from "../../errors"; import { guardValueIsSetWithMessage } from "../../utils"; import { EndpointDefinition, EndpointParameterDefinition } from "./endpoint"; import { EnumType } from "./enum"; +import { EventDefinition, EventTopicDefinition } from "./event"; import { StructType } from "./struct"; import { TypeMapper } from "./typeMapper"; import { CustomType } from "./types"; @@ -13,17 +14,20 @@ export class AbiRegistry { readonly constructorDefinition: EndpointDefinition; readonly endpoints: EndpointDefinition[] = []; readonly customTypes: CustomType[] = []; + readonly events: EventDefinition[] = []; private constructor(options: { name: string; constructorDefinition: EndpointDefinition; endpoints: EndpointDefinition[]; customTypes: CustomType[], + events?: EventDefinition[] }) { this.name = options.name; this.constructorDefinition = options.constructorDefinition; this.endpoints = options.endpoints; this.customTypes = options.customTypes; + this.events = options.events || []; } static create(options: { @@ -31,11 +35,13 @@ export class AbiRegistry { constructor?: any, endpoints?: any[]; types?: Record + events?: any[] }): AbiRegistry { const name = options.name || interfaceNamePlaceholder; const constructor = options.constructor || {}; const endpoints = options.endpoints || []; const types = options.types || {}; + const events = options.events || []; // Load arbitrary input parameters into properly-defined objects (e.g. EndpointDefinition and CustomType). const constructorDefinition = EndpointDefinition.fromJSON({ name: "constructor", ...constructor }); @@ -54,17 +60,26 @@ export class AbiRegistry { } } + const eventDefinitions = events.map(item => EventDefinition.fromJSON(item)); + const registry = new AbiRegistry({ name: name, constructorDefinition: constructorDefinition, endpoints: endpointDefinitions, customTypes: customTypes, + events: eventDefinitions }); const remappedRegistry = registry.remapToKnownTypes(); return remappedRegistry; } + getCustomType(name: string): CustomType { + const result = this.customTypes.find((e) => e.getName() == name); + guardValueIsSetWithMessage(`custom type [${name}] not found`, result); + return result!; + } + getStruct(name: string): StructType { const result = this.customTypes.find((e) => e.getName() == name && e.hasExactClass(StructType.ClassName)); guardValueIsSetWithMessage(`struct [${name}] not found`, result); @@ -95,6 +110,12 @@ export class AbiRegistry { return result!; } + getEvent(name: string): EventDefinition { + const result = this.events.find((e) => e.identifier == name); + guardValueIsSetWithMessage(`event [${name}] not found`, result); + return result!; + } + /** * Right after loading ABI definitions into a registry, the endpoints and the custom types (structs, enums) * use raw types for their I/O parameters (in the case of endpoints), or for their fields (in the case of structs). @@ -129,12 +150,15 @@ export class AbiRegistry { newEndpoints.push(mapEndpoint(endpoint, mapper)); } + const newEvents: EventDefinition[] = this.events.map((event) => mapEvent(event, mapper)); + // Now return the new registry, with all types remapped to known types const newRegistry = new AbiRegistry({ name: this.name, constructorDefinition: newConstructor, endpoints: newEndpoints, customTypes: newCustomTypes, + events: newEvents }); return newRegistry; @@ -172,3 +196,15 @@ function mapEndpoint(endpoint: EndpointDefinition, mapper: TypeMapper): Endpoint return new EndpointDefinition(endpoint.name, newInput, newOutput, endpoint.modifiers); } + +function mapEvent(event: EventDefinition, mapper: TypeMapper): EventDefinition { + const newInputs = event.inputs.map( + (e) => new EventTopicDefinition({ + name: e.name, + type: mapper.mapType(e.type), + indexed: e.indexed + }) + ); + + return new EventDefinition(event.identifier, newInputs); +} diff --git a/src/smartcontracts/typesystem/event.ts b/src/smartcontracts/typesystem/event.ts new file mode 100644 index 00000000..5b4dcac6 --- /dev/null +++ b/src/smartcontracts/typesystem/event.ts @@ -0,0 +1,47 @@ +import { TypeExpressionParser } from "./typeExpressionParser"; +import { Type } from "./types"; + +const NamePlaceholder = "?"; + +export class EventDefinition { + readonly identifier: string; + readonly inputs: EventTopicDefinition[] = []; + + constructor(identifier: string, inputs: EventTopicDefinition[]) { + this.identifier = identifier; + this.inputs = inputs || []; + } + + static fromJSON(json: { + identifier: string, + inputs: any[] + }): EventDefinition { + json.identifier = json.identifier == null ? NamePlaceholder : json.identifier; + json.inputs = json.inputs || []; + + const inputs = json.inputs.map(param => EventTopicDefinition.fromJSON(param)); + return new EventDefinition(json.identifier, inputs); + } +} + +export class EventTopicDefinition { + readonly name: string; + readonly type: Type; + readonly indexed: boolean; + + constructor(options: { name: string, type: Type, indexed: boolean }) { + this.name = options.name; + this.type = options.type; + this.indexed = options.indexed; + } + + static fromJSON(json: { name?: string, type: string, indexed: boolean }): EventTopicDefinition { + const parsedType = new TypeExpressionParser().parse(json.type); + + return new EventTopicDefinition({ + name: json.name || NamePlaceholder, + type: parsedType, + indexed: json.indexed + }); + } +} diff --git a/src/testdata/esdt-safe.abi.json b/src/testdata/esdt-safe.abi.json new file mode 100644 index 00000000..ee17c3f3 --- /dev/null +++ b/src/testdata/esdt-safe.abi.json @@ -0,0 +1,725 @@ +{ + "buildInfo": { + "rustc": { + "version": "1.75.0-nightly", + "commitHash": "42b1224e9eb37177f608d3f6a6f2be2ee13902e4", + "commitDate": "2023-10-15", + "channel": "Nightly", + "short": "rustc 1.75.0-nightly (42b1224e9 2023-10-15)" + }, + "contractCrate": { + "name": "esdt-safe", + "version": "0.0.0" + }, + "framework": { + "name": "multiversx-sc", + "version": "0.43.5" + } + }, + "name": "EsdtSafe", + "constructor": { + "inputs": [ + { + "name": "min_valid_signers", + "type": "u32" + }, + { + "name": "signers", + "type": "variadic
", + "multi_arg": true + } + ], + "outputs": [] + }, + "endpoints": [ + { + "name": "upgrade", + "mutability": "mutable", + "inputs": [], + "outputs": [] + }, + { + "docs": [ + "Create an Elrond -> Sovereign transaction." + ], + "name": "deposit", + "mutability": "mutable", + "payableInTokens": [ + "*" + ], + "inputs": [ + { + "name": "to", + "type": "Address" + }, + { + "name": "opt_transfer_data", + "type": "optional", + "multi_arg": true + } + ], + "outputs": [] + }, + { + "docs": [ + "Claim funds for failed Elrond -> Sovereign transactions.", + "These are not sent automatically to prevent the contract getting stuck.", + "For example, if the receiver is a SC, a frozen account, etc." + ], + "name": "claimRefund", + "mutability": "mutable", + "inputs": [ + { + "name": "token_id", + "type": "TokenIdentifier" + } + ], + "outputs": [ + { + "type": "List" + } + ] + }, + { + "docs": [ + "Sets the statuses for the transactions, after they were executed on the Sovereign side.", + "", + "Only TransactionStatus::Executed (3) and TransactionStatus::Rejected (4) values are allowed.", + "Number of provided statuses must be equal to number of transactions in the batch." + ], + "name": "setTransactionBatchStatus", + "mutability": "mutable", + "inputs": [ + { + "name": "batch_id", + "type": "u64" + }, + { + "name": "signature", + "type": "array48" + }, + { + "name": "tx_statuses", + "type": "variadic", + "multi_arg": true + } + ], + "outputs": [] + }, + { + "name": "setMinValidSigners", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "new_value", + "type": "u32" + } + ], + "outputs": [] + }, + { + "name": "addSigners", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "signers", + "type": "variadic
", + "multi_arg": true + } + ], + "outputs": [] + }, + { + "name": "removeSigners", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "signers", + "type": "variadic
", + "multi_arg": true + } + ], + "outputs": [] + }, + { + "name": "registerToken", + "mutability": "mutable", + "payableInTokens": [ + "EGLD" + ], + "inputs": [ + { + "name": "sov_token_id", + "type": "TokenIdentifier" + }, + { + "name": "token_type", + "type": "EsdtTokenType" + }, + { + "name": "token_display_name", + "type": "bytes" + }, + { + "name": "token_ticker", + "type": "bytes" + }, + { + "name": "num_decimals", + "type": "u32" + }, + { + "name": "bls_multisig", + "type": "array48" + } + ], + "outputs": [] + }, + { + "name": "clearRegisteredToken", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "sov_token_id", + "type": "TokenIdentifier" + } + ], + "outputs": [] + }, + { + "name": "batchTransferEsdtToken", + "mutability": "mutable", + "inputs": [ + { + "name": "batch_id", + "type": "u64" + }, + { + "name": "signature", + "type": "array48" + }, + { + "name": "transfers", + "type": "variadic", + "multi_arg": true + } + ], + "outputs": [] + }, + { + "name": "setMaxTxBatchSize", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "new_max_tx_batch_size", + "type": "u32" + } + ], + "outputs": [] + }, + { + "name": "setMaxTxBatchBlockDuration", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "new_max_tx_batch_block_duration", + "type": "u64" + } + ], + "outputs": [] + }, + { + "name": "getCurrentTxBatch", + "mutability": "readonly", + "inputs": [], + "outputs": [ + { + "type": "optional,List,Option>>>>", + "multi_result": true + } + ] + }, + { + "name": "getFirstBatchAnyStatus", + "mutability": "readonly", + "inputs": [], + "outputs": [ + { + "type": "optional,List,Option>>>>", + "multi_result": true + } + ] + }, + { + "name": "getBatch", + "mutability": "readonly", + "inputs": [ + { + "name": "batch_id", + "type": "u64" + } + ], + "outputs": [ + { + "type": "optional,List,Option>>>>", + "multi_result": true + } + ] + }, + { + "name": "getBatchStatus", + "mutability": "readonly", + "inputs": [ + { + "name": "batch_id", + "type": "u64" + } + ], + "outputs": [ + { + "type": "BatchStatus" + } + ] + }, + { + "name": "getFirstBatchId", + "mutability": "readonly", + "inputs": [], + "outputs": [ + { + "type": "u64" + } + ] + }, + { + "name": "getLastBatchId", + "mutability": "readonly", + "inputs": [], + "outputs": [ + { + "type": "u64" + } + ] + }, + { + "name": "setMaxBridgedAmount", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [ + { + "name": "token_id", + "type": "TokenIdentifier" + }, + { + "name": "max_amount", + "type": "BigUint" + } + ], + "outputs": [] + }, + { + "name": "getMaxBridgedAmount", + "mutability": "readonly", + "inputs": [ + { + "name": "token_id", + "type": "TokenIdentifier" + } + ], + "outputs": [ + { + "type": "BigUint" + } + ] + }, + { + "name": "pause", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [], + "outputs": [] + }, + { + "name": "unpause", + "onlyOwner": true, + "mutability": "mutable", + "inputs": [], + "outputs": [] + }, + { + "name": "isPaused", + "mutability": "readonly", + "inputs": [], + "outputs": [ + { + "type": "bool" + } + ] + } + ], + "promisesCallbackNames": [ + "transfer_callback" + ], + "events": [ + { + "identifier": "deposit", + "inputs": [ + { + "name": "dest_address", + "type": "Address", + "indexed": true + }, + { + "name": "tokens", + "type": "List", + "indexed": true + }, + { + "name": "event_data", + "type": "DepositEvent" + } + ] + }, + { + "identifier": "setStatusEvent", + "inputs": [ + { + "name": "batch_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_status", + "type": "TransactionStatus", + "indexed": true + } + ] + }, + { + "identifier": "addRefundTransactionEvent", + "inputs": [ + { + "name": "tx_id", + "type": "u64", + "indexed": true + }, + { + "name": "original_tx_id", + "type": "u64", + "indexed": true + } + ] + }, + { + "identifier": "transferPerformedEvent", + "inputs": [ + { + "name": "batch_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx", + "type": "Transaction" + } + ] + }, + { + "identifier": "transferFailedInvalidToken", + "inputs": [ + { + "name": "batch_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_id", + "type": "u64", + "indexed": true + } + ] + }, + { + "identifier": "transferFailedFrozenDestinationAccount", + "inputs": [ + { + "name": "batch_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_id", + "type": "u64", + "indexed": true + } + ] + }, + { + "identifier": "transferOverMaxAmount", + "inputs": [ + { + "name": "batch_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_id", + "type": "u64", + "indexed": true + } + ] + }, + { + "identifier": "transferFailedExecutionFailed", + "inputs": [ + { + "name": "batch_id", + "type": "u64", + "indexed": true + }, + { + "name": "tx_id", + "type": "u64", + "indexed": true + } + ] + } + ], + "hasCallback": true, + "types": { + "BatchStatus": { + "type": "enum", + "variants": [ + { + "name": "AlreadyProcessed", + "discriminant": 0 + }, + { + "name": "Empty", + "discriminant": 1 + }, + { + "name": "PartiallyFull", + "discriminant": 2, + "fields": [ + { + "name": "end_block_nonce", + "type": "u64" + }, + { + "name": "tx_ids", + "type": "List" + } + ] + }, + { + "name": "Full", + "discriminant": 3 + }, + { + "name": "WaitingForSignatures", + "discriminant": 4 + } + ] + }, + "DepositEvent": { + "type": "struct", + "fields": [ + { + "name": "tx_nonce", + "type": "u64" + }, + { + "name": "opt_function", + "type": "Option" + }, + { + "name": "opt_arguments", + "type": "Option>" + }, + { + "name": "opt_gas_limit", + "type": "Option" + } + ] + }, + "EsdtTokenPayment": { + "type": "struct", + "fields": [ + { + "name": "token_identifier", + "type": "TokenIdentifier" + }, + { + "name": "token_nonce", + "type": "u64" + }, + { + "name": "amount", + "type": "BigUint" + } + ] + }, + "EsdtTokenType": { + "type": "enum", + "variants": [ + { + "name": "Fungible", + "discriminant": 0 + }, + { + "name": "NonFungible", + "discriminant": 1 + }, + { + "name": "SemiFungible", + "discriminant": 2 + }, + { + "name": "Meta", + "discriminant": 3 + }, + { + "name": "Invalid", + "discriminant": 4 + } + ] + }, + "StolenFromFrameworkEsdtTokenData": { + "type": "struct", + "fields": [ + { + "name": "token_type", + "type": "EsdtTokenType" + }, + { + "name": "amount", + "type": "BigUint" + }, + { + "name": "frozen", + "type": "bool" + }, + { + "name": "hash", + "type": "bytes" + }, + { + "name": "name", + "type": "bytes" + }, + { + "name": "attributes", + "type": "bytes" + }, + { + "name": "creator", + "type": "Address" + }, + { + "name": "royalties", + "type": "BigUint" + }, + { + "name": "uris", + "type": "List" + } + ] + }, + "Transaction": { + "type": "struct", + "fields": [ + { + "name": "block_nonce", + "type": "u64" + }, + { + "name": "nonce", + "type": "u64" + }, + { + "name": "from", + "type": "Address" + }, + { + "name": "to", + "type": "Address" + }, + { + "name": "tokens", + "type": "List" + }, + { + "name": "token_data", + "type": "List" + }, + { + "name": "opt_transfer_data", + "type": "Option" + }, + { + "name": "is_refund_tx", + "type": "bool" + } + ] + }, + "TransactionStatus": { + "type": "enum", + "variants": [ + { + "name": "None", + "discriminant": 0 + }, + { + "name": "Pending", + "discriminant": 1 + }, + { + "name": "InProgress", + "discriminant": 2 + }, + { + "name": "Executed", + "discriminant": 3 + }, + { + "name": "Rejected", + "discriminant": 4 + } + ] + }, + "TransferData": { + "type": "struct", + "fields": [ + { + "name": "gas_limit", + "type": "u64" + }, + { + "name": "function", + "type": "bytes" + }, + { + "name": "args", + "type": "List" + } + ] + } + } +}