1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | const client_common_1 = require("@neo-one/client-common");
|
4 | const errors_1 = require("../errors");
|
5 | const utils_1 = require("../utils");
|
6 | const common_1 = require("./common");
|
7 | const Input_1 = require("./Input");
|
8 | const TransactionBase_1 = require("./TransactionBase");
|
9 | const TransactionType_1 = require("./TransactionType");
|
10 | class ClaimTransaction extends TransactionBase_1.TransactionBase(client_common_1.ClaimTransactionModel) {
|
11 | constructor(add) {
|
12 | super(add);
|
13 | this.sizeExclusive = utils_1.utils.lazy(() => client_common_1.IOHelper.sizeOfUInt8 + client_common_1.IOHelper.sizeOfArray(this.claims, (claim) => claim.size));
|
14 | const getScriptHashesForVerifying = super.getScriptHashesForVerifying.bind(this);
|
15 | this.claimGetScriptHashesForVerifyingInternal = utils_1.utils.lazyAsync(async (options) => {
|
16 | const { getOutput } = options;
|
17 | const [hashesSet, hashes] = await Promise.all([
|
18 | getScriptHashesForVerifying(options),
|
19 | Promise.all(this.claims.map(async (claim) => {
|
20 | const output = await getOutput(claim);
|
21 | return client_common_1.common.uInt160ToHex(output.address);
|
22 | })),
|
23 | ]);
|
24 | return new Set([...hashesSet, ...hashes]);
|
25 | });
|
26 | }
|
27 | static deserializeWireBase(options) {
|
28 | const { reader } = options;
|
29 | const { type, version } = super.deserializeTransactionBaseStartWireBase(options);
|
30 | if (type !== TransactionType_1.TransactionType.Claim) {
|
31 | throw new client_common_1.InvalidFormatError(`Expected transaction type to be ${TransactionType_1.TransactionType.Claim}. Received: ${type}`);
|
32 | }
|
33 | const claims = reader.readArray(() => Input_1.Input.deserializeWireBase(options));
|
34 | const { attributes, inputs, outputs, scripts } = super.deserializeTransactionBaseEndWireBase(options);
|
35 | return new this({
|
36 | version,
|
37 | claims,
|
38 | attributes,
|
39 | inputs,
|
40 | outputs,
|
41 | scripts,
|
42 | });
|
43 | }
|
44 | async getNetworkFee(_context) {
|
45 | return utils_1.utils.ZERO;
|
46 | }
|
47 | async getClaimReferences({ getOutput }) {
|
48 | return Promise.all(this.claims.map(async (input) => getOutput(input)));
|
49 | }
|
50 | async getScriptHashesForVerifying(options) {
|
51 | return this.claimGetScriptHashesForVerifyingInternal(options);
|
52 | }
|
53 | async verify(options) {
|
54 | const [results] = await Promise.all([super.verify(options), this.verifyInternal(options)]);
|
55 | return results;
|
56 | }
|
57 | async serializeJSON(context) {
|
58 | const transactionBaseJSON = await super.serializeTransactionBaseJSON(context);
|
59 | return Object.assign({}, transactionBaseJSON, { type: 'ClaimTransaction', claims: this.claims.map((claim) => claim.serializeJSON(context)) });
|
60 | }
|
61 | async verifyInternal(options) {
|
62 | const { calculateClaimAmount, getOutput, utilityToken, memPool = [] } = options;
|
63 | if (common_1.hasDuplicateInputs(this.claims)) {
|
64 | throw new errors_1.VerifyError('Duplicate claims');
|
65 | }
|
66 | if (memPool.some((transaction) => transaction instanceof ClaimTransaction &&
|
67 | transaction.type === TransactionType_1.TransactionType.Claim &&
|
68 | common_1.hasIntersectingInputs(this.claims, transaction.claims))) {
|
69 | throw new errors_1.VerifyError('Duplicate claims in mempool');
|
70 | }
|
71 | const [results, claimAmount] = await Promise.all([
|
72 | this.getTransactionResults({ getOutput }),
|
73 | calculateClaimAmount(this.claims).catch((error) => {
|
74 | throw new errors_1.VerifyError(`Invalid claims: ${error.message}`);
|
75 | }),
|
76 | ]);
|
77 | const result = Object.entries(results).find(([assetHex, _value]) => client_common_1.common.uInt256Equal(client_common_1.common.hexToUInt256(assetHex), utilityToken.hash));
|
78 | if (result === undefined || result[1].gt(utils_1.utils.ZERO)) {
|
79 | throw new errors_1.VerifyError('Invalid claim value');
|
80 | }
|
81 | if (!claimAmount.eq(result[1].neg())) {
|
82 | throw new errors_1.VerifyError('Invalid claim value');
|
83 | }
|
84 | }
|
85 | }
|
86 | exports.ClaimTransaction = ClaimTransaction;
|
87 |
|
88 | //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["ClaimTransaction.ts"],"names":[],"mappings":"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","file":"neo-one-node-core/src/transaction/ClaimTransaction.js","sourcesContent":["import {\n  ClaimTransactionJSON,\n  ClaimTransactionModel,\n  ClaimTransactionModelAdd,\n  common,\n  InvalidFormatError,\n  IOHelper,\n  UInt160Hex,\n} from '@neo-one/client-common';\nimport { Constructor } from '@neo-one/utils';\nimport { BN } from 'bn.js';\nimport { VerifyError } from '../errors';\nimport { DeserializeWireBaseOptions, SerializeJSONContext } from '../Serializable';\nimport { utils } from '../utils';\nimport { VerifyScriptResult } from '../vm';\nimport { Witness } from '../Witness';\nimport { Attribute } from './attribute';\nimport { hasDuplicateInputs, hasIntersectingInputs } from './common';\nimport { Input } from './Input';\nimport { Output } from './Output';\nimport {\n  FeeContext,\n  GetReferencesOptions,\n  TransactionBase,\n  TransactionGetScriptHashesForVerifyingOptions,\n  TransactionVerifyOptions,\n} from './TransactionBase';\nimport { TransactionType } from './TransactionType';\n\nexport interface ClaimTransactionAdd extends ClaimTransactionModelAdd<Attribute, Input, Output, Witness> {}\n\nexport class ClaimTransaction extends TransactionBase<\n  TransactionType.Claim,\n  ClaimTransactionJSON,\n  Constructor<ClaimTransactionModel<Attribute, Input, Output, Witness>>\n>(ClaimTransactionModel) {\n  public static deserializeWireBase(options: DeserializeWireBaseOptions): ClaimTransaction {\n    const { reader } = options;\n\n    const { type, version } = super.deserializeTransactionBaseStartWireBase(options);\n\n    if (type !== TransactionType.Claim) {\n      throw new InvalidFormatError(`Expected transaction type to be ${TransactionType.Claim}. Received: ${type}`);\n    }\n\n    const claims = reader.readArray(() => Input.deserializeWireBase(options));\n\n    const { attributes, inputs, outputs, scripts } = super.deserializeTransactionBaseEndWireBase(options);\n\n    return new this({\n      version,\n      claims,\n      attributes,\n      inputs,\n      outputs,\n      scripts,\n    });\n  }\n\n  public readonly claims!: readonly Input[];\n  protected readonly sizeExclusive: () => number = utils.lazy(\n    () => IOHelper.sizeOfUInt8 + IOHelper.sizeOfArray(this.claims, (claim) => claim.size),\n  );\n  private readonly claimGetScriptHashesForVerifyingInternal: (\n    options: TransactionGetScriptHashesForVerifyingOptions,\n  ) => Promise<Set<UInt160Hex>>;\n\n  public constructor(add: ClaimTransactionAdd) {\n    super(add);\n\n    const getScriptHashesForVerifying = super.getScriptHashesForVerifying.bind(this);\n    this.claimGetScriptHashesForVerifyingInternal = utils.lazyAsync(\n      async (options: TransactionGetScriptHashesForVerifyingOptions) => {\n        const { getOutput } = options;\n        const [hashesSet, hashes] = await Promise.all([\n          getScriptHashesForVerifying(options),\n          Promise.all(\n            this.claims.map(async (claim) => {\n              const output = await getOutput(claim);\n\n              return common.uInt160ToHex(output.address);\n            }),\n          ),\n        ]);\n\n        return new Set([...hashesSet, ...hashes]);\n      },\n    );\n  }\n\n  public async getNetworkFee(_context: FeeContext): Promise<BN> {\n    return utils.ZERO;\n  }\n\n  public async getClaimReferences({ getOutput }: GetReferencesOptions): Promise<readonly Output[]> {\n    return Promise.all(this.claims.map(async (input) => getOutput(input)));\n  }\n\n  public async getScriptHashesForVerifying(\n    options: TransactionGetScriptHashesForVerifyingOptions,\n  ): Promise<Set<UInt160Hex>> {\n    return this.claimGetScriptHashesForVerifyingInternal(options);\n  }\n\n  public async verify(options: TransactionVerifyOptions): Promise<readonly VerifyScriptResult[]> {\n    const [results] = await Promise.all([super.verify(options), this.verifyInternal(options)]);\n\n    return results;\n  }\n\n  public async serializeJSON(context: SerializeJSONContext): Promise<ClaimTransactionJSON> {\n    const transactionBaseJSON = await super.serializeTransactionBaseJSON(context);\n\n    return {\n      ...transactionBaseJSON,\n      type: 'ClaimTransaction',\n      claims: this.claims.map((claim) => claim.serializeJSON(context)),\n    };\n  }\n\n  private async verifyInternal(options: TransactionVerifyOptions): Promise<void> {\n    const { calculateClaimAmount, getOutput, utilityToken, memPool = [] } = options;\n    if (hasDuplicateInputs(this.claims)) {\n      throw new VerifyError('Duplicate claims');\n    }\n\n    if (\n      memPool.some(\n        (transaction) =>\n          transaction instanceof ClaimTransaction &&\n          transaction.type === TransactionType.Claim &&\n          hasIntersectingInputs(this.claims, transaction.claims),\n      )\n    ) {\n      throw new VerifyError('Duplicate claims in mempool');\n    }\n    const [results, claimAmount] = await Promise.all([\n      this.getTransactionResults({ getOutput }),\n      calculateClaimAmount(this.claims).catch((error) => {\n        throw new VerifyError(`Invalid claims: ${error.message}`);\n      }),\n    ]);\n\n    // tslint:disable-next-line no-unused\n    const result = Object.entries(results).find(([assetHex, _value]) =>\n      common.uInt256Equal(common.hexToUInt256(assetHex), utilityToken.hash),\n    );\n\n    if (result === undefined || result[1].gt(utils.ZERO)) {\n      throw new VerifyError('Invalid claim value');\n    }\n\n    if (!claimAmount.eq(result[1].neg())) {\n      throw new VerifyError('Invalid claim value');\n    }\n  }\n}\n"]}
|