UNPKG

@c4tplatform/caminojs

Version:
175 lines 20.8 kB
"use strict"; /** * @packageDocumentation * @module Common-AssetAmount */ var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.StandardAssetAmountDestination = exports.AssetAmount = void 0; const buffer_1 = require("buffer/"); const bn_js_1 = __importDefault(require("bn.js")); const errors_1 = require("../utils/errors"); /** * Class for managing asset amounts in the UTXOSet fee calcuation */ class AssetAmount { constructor(assetID, amount, burn) { // assetID that is amount is managing. this.assetID = buffer_1.Buffer.alloc(32); // amount of this asset that should be sent. this.amount = new bn_js_1.default(0); // burn is the amount of this asset that should be burned. this.burn = new bn_js_1.default(0); // spent is the total amount of this asset that has been consumed. this.spent = new bn_js_1.default(0); // stakeableLockSpent is the amount of this asset that has been consumed that // was locked. this.stakeableLockSpent = new bn_js_1.default(0); // change is the excess amount of this asset that was consumed over the amount // requested to be consumed(amount + burn). this.change = new bn_js_1.default(0); // stakeableLockChange is a flag to mark if the input that generated the // change was locked. this.stakeableLockChange = false; // finished is a convenience flag to track "spent >= amount + burn" this.finished = false; this.getAssetID = () => { return this.assetID; }; this.getAssetIDString = () => { return this.assetID.toString("hex"); }; this.getAmount = () => { return this.amount; }; this.getSpent = () => { return this.spent; }; this.getBurn = () => { return this.burn; }; this.getChange = () => { return this.change; }; this.getStakeableLockSpent = () => { return this.stakeableLockSpent; }; this.getStakeableLockChange = () => { return this.stakeableLockChange; }; this.isFinished = () => { return this.finished; }; // spendAmount should only be called if this asset is still awaiting more // funds to consume. this.spendAmount = (amt, stakeableLocked = false) => { if (this.finished) { /* istanbul ignore next */ throw new errors_1.InsufficientFundsError("Error - AssetAmount.spendAmount: attempted to spend " + "excess funds"); } this.spent = this.spent.add(amt); if (stakeableLocked) { this.stakeableLockSpent = this.stakeableLockSpent.add(amt); } const total = this.amount.add(this.burn); if (this.spent.gte(total)) { this.change = this.spent.sub(total); if (stakeableLocked) { this.stakeableLockChange = true; } this.finished = true; } return this.finished; }; this.assetID = assetID; this.amount = typeof amount === "undefined" ? new bn_js_1.default(0) : amount; this.burn = typeof burn === "undefined" ? new bn_js_1.default(0) : burn; this.spent = new bn_js_1.default(0); this.stakeableLockSpent = new bn_js_1.default(0); this.stakeableLockChange = false; } } exports.AssetAmount = AssetAmount; class StandardAssetAmountDestination { constructor(destinations, destinationsThreshold, senders, changeAddresses, changeAddressesThreshold) { this.amounts = []; this.destinations = []; this.destinationsThreshold = 0; this.senders = []; this.changeAddresses = []; this.changeAddressesThreshold = 0; this.amountkey = {}; this.inputs = []; this.outputs = []; this.change = []; // TODO: should this function allow for repeated calls with the same // assetID? this.addAssetAmount = (assetID, amount, burn) => { let aa = new AssetAmount(assetID, amount, burn); this.amounts.push(aa); this.amountkey[aa.getAssetIDString()] = aa; }; this.addInput = (input) => { this.inputs.push(input); }; this.addOutput = (output) => { this.outputs.push(output); }; this.addChange = (output) => { this.change.push(output); }; this.getAmounts = () => { return this.amounts; }; this.getDestinations = () => { return this.destinations; }; this.getDestinationsThreshold = () => { return this.destinationsThreshold; }; this.getSenders = () => { return this.senders; }; this.getChangeAddresses = () => { return this.changeAddresses; }; this.getChangeAddressesThreshold = () => { return this.changeAddressesThreshold; }; this.getAssetAmount = (assetHexStr) => { return this.amountkey[`${assetHexStr}`]; }; this.assetExists = (assetHexStr) => { return assetHexStr in this.amountkey; }; this.getInputs = () => { return this.inputs; }; this.getOutputs = () => { return this.outputs; }; this.getChangeOutputs = () => { return this.change; }; this.getAllOutputs = () => { return this.outputs.concat(this.change); }; this.canComplete = () => { for (let i = 0; i < this.amounts.length; i++) { if (!this.amounts[`${i}`].isFinished()) { return false; } } return true; }; this.destinations = destinations; this.destinationsThreshold = destinationsThreshold; this.changeAddresses = changeAddresses; this.changeAddressesThreshold = changeAddressesThreshold; this.senders = senders; } } exports.StandardAssetAmountDestination = StandardAssetAmountDestination; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assetamount.js","sourceRoot":"","sources":["../../src/common/assetamount.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @packageDocumentation\n * @module Common-AssetAmount\n */\n\nimport { Buffer } from \"buffer/\"\nimport BN from \"bn.js\"\nimport { StandardTransferableOutput } from \"./output\"\nimport { StandardTransferableInput } from \"./input\"\nimport { InsufficientFundsError } from \"../utils/errors\"\n\n/**\n * Class for managing asset amounts in the UTXOSet fee calcuation\n */\nexport class AssetAmount {\n  // assetID that is amount is managing.\n  protected assetID: Buffer = Buffer.alloc(32)\n  // amount of this asset that should be sent.\n  protected amount: BN = new BN(0)\n  // burn is the amount of this asset that should be burned.\n  protected burn: BN = new BN(0)\n\n  // spent is the total amount of this asset that has been consumed.\n  protected spent: BN = new BN(0)\n  // stakeableLockSpent is the amount of this asset that has been consumed that\n  // was locked.\n  protected stakeableLockSpent: BN = new BN(0)\n\n  // change is the excess amount of this asset that was consumed over the amount\n  // requested to be consumed(amount + burn).\n  protected change: BN = new BN(0)\n  // stakeableLockChange is a flag to mark if the input that generated the\n  // change was locked.\n  protected stakeableLockChange: boolean = false\n\n  // finished is a convenience flag to track \"spent >= amount + burn\"\n  protected finished: boolean = false\n\n  getAssetID = (): Buffer => {\n    return this.assetID\n  }\n\n  getAssetIDString = (): string => {\n    return this.assetID.toString(\"hex\")\n  }\n\n  getAmount = (): BN => {\n    return this.amount\n  }\n\n  getSpent = (): BN => {\n    return this.spent\n  }\n\n  getBurn = (): BN => {\n    return this.burn\n  }\n\n  getChange = (): BN => {\n    return this.change\n  }\n\n  getStakeableLockSpent = (): BN => {\n    return this.stakeableLockSpent\n  }\n\n  getStakeableLockChange = (): boolean => {\n    return this.stakeableLockChange\n  }\n\n  isFinished = (): boolean => {\n    return this.finished\n  }\n\n  // spendAmount should only be called if this asset is still awaiting more\n  // funds to consume.\n  spendAmount = (amt: BN, stakeableLocked: boolean = false): boolean => {\n    if (this.finished) {\n      /* istanbul ignore next */\n      throw new InsufficientFundsError(\n        \"Error - AssetAmount.spendAmount: attempted to spend \" + \"excess funds\"\n      )\n    }\n    this.spent = this.spent.add(amt)\n    if (stakeableLocked) {\n      this.stakeableLockSpent = this.stakeableLockSpent.add(amt)\n    }\n\n    const total: BN = this.amount.add(this.burn)\n    if (this.spent.gte(total)) {\n      this.change = this.spent.sub(total)\n      if (stakeableLocked) {\n        this.stakeableLockChange = true\n      }\n      this.finished = true\n    }\n    return this.finished\n  }\n\n  constructor(assetID: Buffer, amount: BN, burn: BN) {\n    this.assetID = assetID\n    this.amount = typeof amount === \"undefined\" ? new BN(0) : amount\n    this.burn = typeof burn === \"undefined\" ? new BN(0) : burn\n    this.spent = new BN(0)\n    this.stakeableLockSpent = new BN(0)\n    this.stakeableLockChange = false\n  }\n}\n\nexport abstract class StandardAssetAmountDestination<\n  TO extends StandardTransferableOutput,\n  TI extends StandardTransferableInput\n> {\n  protected amounts: AssetAmount[] = []\n  protected destinations: Buffer[] = []\n  protected destinationsThreshold: number = 0\n  protected senders: Buffer[] = []\n  protected changeAddresses: Buffer[] = []\n  protected changeAddressesThreshold: number = 0\n  protected amountkey: object = {}\n  protected inputs: TI[] = []\n  protected outputs: TO[] = []\n  protected change: TO[] = []\n\n  // TODO: should this function allow for repeated calls with the same\n  //       assetID?\n  addAssetAmount = (assetID: Buffer, amount: BN, burn: BN) => {\n    let aa: AssetAmount = new AssetAmount(assetID, amount, burn)\n    this.amounts.push(aa)\n    this.amountkey[aa.getAssetIDString()] = aa\n  }\n\n  addInput = (input: TI) => {\n    this.inputs.push(input)\n  }\n\n  addOutput = (output: TO) => {\n    this.outputs.push(output)\n  }\n\n  addChange = (output: TO) => {\n    this.change.push(output)\n  }\n\n  getAmounts = (): AssetAmount[] => {\n    return this.amounts\n  }\n\n  getDestinations = (): Buffer[] => {\n    return this.destinations\n  }\n\n  getDestinationsThreshold = (): number => {\n    return this.destinationsThreshold\n  }\n\n  getSenders = (): Buffer[] => {\n    return this.senders\n  }\n\n  getChangeAddresses = (): Buffer[] => {\n    return this.changeAddresses\n  }\n\n  getChangeAddressesThreshold = (): number => {\n    return this.changeAddressesThreshold\n  }\n\n  getAssetAmount = (assetHexStr: string): AssetAmount => {\n    return this.amountkey[`${assetHexStr}`]\n  }\n\n  assetExists = (assetHexStr: string): boolean => {\n    return assetHexStr in this.amountkey\n  }\n\n  getInputs = (): TI[] => {\n    return this.inputs\n  }\n\n  getOutputs = (): TO[] => {\n    return this.outputs\n  }\n\n  getChangeOutputs = (): TO[] => {\n    return this.change\n  }\n\n  getAllOutputs = (): TO[] => {\n    return this.outputs.concat(this.change)\n  }\n\n  canComplete = (): boolean => {\n    for (let i: number = 0; i < this.amounts.length; i++) {\n      if (!this.amounts[`${i}`].isFinished()) {\n        return false\n      }\n    }\n    return true\n  }\n\n  constructor(\n    destinations: Buffer[],\n    destinationsThreshold: number,\n    senders: Buffer[],\n    changeAddresses: Buffer[],\n    changeAddressesThreshold: number\n  ) {\n    this.destinations = destinations\n    this.destinationsThreshold = destinationsThreshold\n    this.changeAddresses = changeAddresses\n    this.changeAddressesThreshold = changeAddressesThreshold\n    this.senders = senders\n  }\n}\n"]}