UNPKG

@polkadot/types

Version:
305 lines (304 loc) • 9.71 kB
import { AbstractBase } from '@polkadot/types-codec'; import { compactAddLength, compactFromU8a, compactToU8a, isHex, isU8a, objectProperty, objectSpread, u8aConcat, u8aToHex, u8aToU8a } from '@polkadot/util'; import { EXTRINSIC_VERSION as LATEST_EXTRINSIC_VERSION } from './v4/Extrinsic.js'; import { BIT_SIGNED, BIT_UNSIGNED, DEFAULT_VERSION, UNMASK_VERSION } from './constants.js'; const VERSIONS = [ 'ExtrinsicUnknown', // v0 is unknown 'ExtrinsicUnknown', 'ExtrinsicUnknown', 'ExtrinsicUnknown', 'ExtrinsicV4' ]; export { LATEST_EXTRINSIC_VERSION }; /** @internal */ function newFromValue(registry, value, version) { if (value instanceof GenericExtrinsic) { return value.unwrap(); } const isSigned = (version & BIT_SIGNED) === BIT_SIGNED; const type = VERSIONS[version & UNMASK_VERSION] || VERSIONS[0]; // we cast here since the VERSION definition is incredibly broad - we don't have a // slice for "only add extrinsic types", and more string definitions become unwieldy return registry.createTypeUnsafe(type, [value, { isSigned, version }]); } /** @internal */ function decodeExtrinsic(registry, value, version = DEFAULT_VERSION) { if (isU8a(value) || Array.isArray(value) || isHex(value)) { return decodeU8a(registry, u8aToU8a(value), version); } else if (value instanceof registry.createClassUnsafe('Call')) { return newFromValue(registry, { method: value }, version); } return newFromValue(registry, value, version); } /** @internal */ function decodeU8a(registry, value, version) { if (!value.length) { return newFromValue(registry, new Uint8Array(), version); } const [offset, length] = compactFromU8a(value); const total = offset + length.toNumber(); if (total > value.length) { throw new Error(`Extrinsic: length less than remainder, expected at least ${total}, found ${value.length}`); } const data = value.subarray(offset, total); return newFromValue(registry, data.subarray(1), data[0]); } class ExtrinsicBase extends AbstractBase { constructor(registry, value, initialU8aLength) { super(registry, value, initialU8aLength); const signKeys = Object.keys(registry.getSignedExtensionTypes()); const getter = (key) => this.inner.signature[key]; // This is on the abstract class, ensuring that hasOwnProperty operates // correctly, i.e. it needs to be on the base class exposing it for (let i = 0, count = signKeys.length; i < count; i++) { objectProperty(this, signKeys[i], getter); } } /** * @description The arguments passed to for the call, exposes args so it is compatible with [[Call]] */ get args() { return this.method.args; } /** * @description The argument definitions, compatible with [[Call]] */ get argsDef() { return this.method.argsDef; } /** * @description The actual `[sectionIndex, methodIndex]` as used in the Call */ get callIndex() { return this.method.callIndex; } /** * @description The actual data for the Call */ get data() { return this.method.data; } /** * @description The era for this extrinsic */ get era() { return this.inner.signature.era; } /** * @description The length of the value when encoded as a Uint8Array */ get encodedLength() { return this.toU8a().length; } /** * @description `true` id the extrinsic is signed */ get isSigned() { return this.inner.signature.isSigned; } /** * @description The length of the actual data, excluding prefix */ get length() { return this.toU8a(true).length; } /** * @description The [[FunctionMetadataLatest]] that describes the extrinsic */ get meta() { return this.method.meta; } /** * @description The [[Call]] this extrinsic wraps */ get method() { return this.inner.method; } /** * @description The nonce for this extrinsic */ get nonce() { return this.inner.signature.nonce; } /** * @description The actual [[EcdsaSignature]], [[Ed25519Signature]] or [[Sr25519Signature]] */ get signature() { return this.inner.signature.signature; } /** * @description The [[Address]] that signed */ get signer() { return this.inner.signature.signer; } /** * @description Forwards compat */ get tip() { return this.inner.signature.tip; } /** * @description Forward compat */ get assetId() { return this.inner.signature.assetId; } /** * @description Forward compat */ get metadataHash() { return this.inner.signature.metadataHash; } /** * @description Forward compat */ get mode() { return this.inner.signature.mode; } /** * @description Returns the raw transaction version (not flagged with signing information) */ get type() { return this.inner.version; } get inner() { return this.unwrap(); } /** * @description Returns the encoded version flag */ get version() { return this.type | (this.isSigned ? BIT_SIGNED : BIT_UNSIGNED); } /** * @description Checks if the source matches this in type */ is(other) { return this.method.is(other); } unwrap() { return super.unwrap(); } } /** * @name GenericExtrinsic * @description * Representation of an Extrinsic in the system. It contains the actual call, * (optional) signature and encodes with an actual length prefix * * {@link https://github.com/paritytech/wiki/blob/master/Extrinsic.md#the-extrinsic-format-for-node}. * * Can be: * - signed, to create a transaction * - left as is, to create an inherent */ export class GenericExtrinsic extends ExtrinsicBase { __internal__hashCache; static LATEST_EXTRINSIC_VERSION = LATEST_EXTRINSIC_VERSION; constructor(registry, value, { version } = {}) { super(registry, decodeExtrinsic(registry, value, version)); } /** * @description returns a hash of the contents */ get hash() { if (!this.__internal__hashCache) { this.__internal__hashCache = super.hash; } return this.__internal__hashCache; } /** * @description Injects an already-generated signature into the extrinsic */ addSignature(signer, signature, payload) { this.inner.addSignature(signer, signature, payload); this.__internal__hashCache = undefined; return this; } /** * @description Returns a breakdown of the hex encoding for this Codec */ inspect() { const encoded = u8aConcat(...this.toU8aInner()); return { inner: this.isSigned ? this.inner.inspect().inner : this.inner.method.inspect().inner, outer: [compactToU8a(encoded.length), new Uint8Array([this.version])] }; } /** * @description Sign the extrinsic with a specific keypair */ sign(account, options) { this.inner.sign(account, options); this.__internal__hashCache = undefined; return this; } /** * @describe Adds a fake signature to the extrinsic */ signFake(signer, options) { this.inner.signFake(signer, options); this.__internal__hashCache = undefined; return this; } /** * @description Returns a hex string representation of the value */ toHex(isBare) { return u8aToHex(this.toU8a(isBare)); } /** * @description Converts the Object to to a human-friendly JSON, with additional fields, expansion and formatting of information */ toHuman(isExpanded, disableAscii) { return objectSpread({}, { isSigned: this.isSigned, method: this.method.toHuman(isExpanded, disableAscii) }, this.isSigned ? { assetId: this.assetId ? this.assetId.toHuman(isExpanded, disableAscii) : null, era: this.era.toHuman(isExpanded, disableAscii), metadataHash: this.metadataHash ? this.metadataHash.toHex() : null, mode: this.mode ? this.mode.toHuman() : null, nonce: this.nonce.toHuman(isExpanded, disableAscii), signature: this.signature.toHex(), signer: this.signer.toHuman(isExpanded, disableAscii), tip: this.tip.toHuman(isExpanded, disableAscii) } : null); } /** * @description Converts the Object to JSON, typically used for RPC transfers */ toJSON() { return this.toHex(); } /** * @description Returns the base runtime type name for this instance */ toRawType() { return 'Extrinsic'; } /** * @description Encodes the value as a Uint8Array as per the SCALE specifications * @param isBare true when the value is not length-prefixed */ toU8a(isBare) { const encoded = u8aConcat(...this.toU8aInner()); return isBare ? encoded : compactAddLength(encoded); } toU8aInner() { // we do not apply bare to the internal values, rather this only determines out length addition, // where we strip all lengths this creates an extrinsic that cannot be decoded return [ new Uint8Array([this.version]), this.inner.toU8a() ]; } }