test-triam-base-contract
Version:
Low level triam smart cotnract support library
461 lines (424 loc) • 14.9 kB
JavaScript
import {default as xdr} from "./generated/stellar-xdr_generated";
import {Keypair} from "./keypair";
import {hash} from "./hashing";
import {StrKey} from "./strkey";
import {Hyper} from "js-xdr";
import {Asset} from "./asset";
import {ContractInput} from "./contractinput";
import BigNumber from 'bignumber.js';
import {best_r} from "./util/continued_fraction";
import trimEnd from 'lodash/trimEnd';
import isUndefined from 'lodash/isUndefined';
import isString from 'lodash/isString';
import isNumber from 'lodash/isNumber';
import isFinite from 'lodash/isFinite';
import {ContrOperation} from './contr_operation';
import * as ops from './operations/index';
const ONE = 10000000;
const MAX_INT64 = '9223372036854775807';
/**
* When set using `{@link Operation.setOptions}` option, requires the issuing account to
* give other accounts permission before they can hold the issuing account’s credit.
* @constant
* @see [Account flags](https://www.stellar.org/developers/guides/concepts/accounts.html#flags)
*/
export const AuthRequiredFlag = 1 << 0;
/**
* When set using `{@link Operation.setOptions}` option, allows the issuing account to
* revoke its credit held by other accounts.
* @constant
* @see [Account flags](https://www.stellar.org/developers/guides/concepts/accounts.html#flags)
*/
export const AuthRevocableFlag = 1 << 1;
/**
* When set using `{@link Operation.setOptions}` option, then none of the authorization flags
* can be set and the account can never be deleted.
* @constant
* @see [Account flags](https://www.stellar.org/developers/guides/concepts/accounts.html#flags)
*/
export const AuthImmutableFlag = 1 << 2;
/**
* `Operation` class represents [operations](https://www.stellar.org/developers/learn/concepts/operations.html) in Stellar network.
* Use one of static methods to create operations:
* * `{@link Operation.createAccount}`
* * `{@link Operation.payment}`
* * `{@link Operation.pathPayment}`
* * `{@link Operation.manageOffer}`
* * `{@link Operation.createPassiveOffer}`
* * `{@link Operation.setOptions}`
* * `{@link Operation.changeTrust}`
* * `{@link Operation.allowTrust}`
* * `{@link Operation.accountMerge}`
* * `{@link Operation.inflation}`
* * `{@link Operation.manageData}`
* * `{@link Operation.bumpSequence}`
* * * * * * * * * *
* * `{@link Operation.createAsset}`
* * `{@link Operation.changeAsset}`
* * `{@link Operation.limitAsset}`
* * thuannd start
* * `{@link Operation.createContract}`
* * `{@link Operation.callContract}`
* * `{@link Operation.sendAsset}`
* * `{@link Operation.checkContract}`
* * `{@link Operation.testContract}`
* * thuannd end
*
* @class Operation
*/
export class Operation {
static setSourceAccount(opAttributes, opts) {
if (opts.source) {
if (!StrKey.isValidEd25519PublicKey(opts.source)) {
throw new Error("Source address is invalid");
}
opAttributes.sourceAccount = Keypair.fromPublicKey(opts.source).xdrAccountId();
}
}
/**
* Converts the XDR Operation object to the opts object used to create the XDR
* operation.
* @param {xdr.Operation} operation - An XDR Operation.
* @return {Operation}
*/
static fromXDRObject(operation) {
function accountIdtoAddress(accountId) {
return StrKey.encodeEd25519PublicKey(accountId.ed25519());
}
// thuannd start
function contractIdtoAddress(contractId) {
//return StrKey.encodeContractKey(contractId.contract());
return StrKey.encodeContractKey(contractId.contract());
}
// thuannd end
let result = {};
if (operation.sourceAccount()) {
result.source = accountIdtoAddress(operation.sourceAccount());
}
let attrs = operation.body().value();
switch (operation.body().switch().name) {
case "createAccount":
result.type = "createAccount";
result.destination = accountIdtoAddress(attrs.destination());
result.startingBalance = this._fromXDRAmount(attrs.startingBalance());
break;
case "payment":
result.type = "payment";
result.destination = accountIdtoAddress(attrs.destination());
result.asset = Asset.fromOperation(attrs.asset());
result.amount = this._fromXDRAmount(attrs.amount());
break;
case "pathPayment":
result.type = "pathPayment";
result.sendAsset = Asset.fromOperation(attrs.sendAsset());
result.sendMax = this._fromXDRAmount(attrs.sendMax());
result.destination = accountIdtoAddress(attrs.destination());
result.destAsset = Asset.fromOperation(attrs.destAsset());
result.destAmount = this._fromXDRAmount(attrs.destAmount());
let path = attrs.path();
result.path = [];
for (let i in path) {
result.path.push(Asset.fromOperation(path[i]));
}
break;
case "changeTrust":
result.type = "changeTrust";
result.line = Asset.fromOperation(attrs.line());
result.limit = this._fromXDRAmount(attrs.limit());
break;
case "allowTrust":
result.type = "allowTrust";
result.trustor = accountIdtoAddress(attrs.trustor());
result.assetCode = attrs.asset().value().toString();
result.assetCode = trimEnd(result.assetCode, "\0");
result.authorize = attrs.authorize();
break;
case "setOption":
result.type = "setOptions";
if (attrs.inflationDest()) {
result.inflationDest = accountIdtoAddress(attrs.inflationDest());
}
result.clearFlags = attrs.clearFlags();
result.setFlags = attrs.setFlags();
result.masterWeight = attrs.masterWeight();
result.lowThreshold = attrs.lowThreshold();
result.medThreshold = attrs.medThreshold();
result.highThreshold = attrs.highThreshold();
// home_domain is checked by iscntrl in stellar-core
result.homeDomain = attrs.homeDomain() ? attrs.homeDomain().toString('ascii') : null;
if (attrs.signer()) {
let signer = {};
let arm = attrs.signer().key().arm();
if (arm == "ed25519") {
signer.ed25519PublicKey = accountIdtoAddress(attrs.signer().key());
} else if (arm == "preAuthTx") {
signer.preAuthTx = attrs.signer().key().preAuthTx();
} else if (arm == "hashX") {
signer.sha256Hash = attrs.signer().key().hashX();
}
signer.weight = attrs.signer().weight();
result.signer = signer;
}
break;
case "manageOffer":
result.type = "manageOffer";
result.selling = Asset.fromOperation(attrs.selling());
result.buying = Asset.fromOperation(attrs.buying());
result.amount = this._fromXDRAmount(attrs.amount());
result.price = this._fromXDRPrice(attrs.price());
result.offerId = attrs.offerId().toString();
break;
case "createPassiveOffer":
result.type = "createPassiveOffer";
result.selling = Asset.fromOperation(attrs.selling());
result.buying = Asset.fromOperation(attrs.buying());
result.amount = this._fromXDRAmount(attrs.amount());
result.price = this._fromXDRPrice(attrs.price());
break;
case "accountMerge":
result.type = "accountMerge";
result.destination = accountIdtoAddress(attrs);
break;
case "manageDatum":
result.type = "manageData";
// manage_data.name is checked by iscntrl in stellar-core
result.name = attrs.dataName().toString('ascii');
result.value = attrs.dataValue();
break;
case "inflation":
result.type = "inflation";
break;
case "bumpSequence":
result.type = "bumpSequence";
result.bumpTo = attrs.bumpTo().toString();
break;
// thuannd start: fee task
case "createAsset":
result.type = "createAsset";
result.asset = Asset.fromOperation(attrs.asset());
result.beneficiary = attrs.beneficiary().toString('ascii');
result.fee = attrs.fee();
result.ratio = attrs.ratio();
result.minfee = this._fromXDRAmount(attrs.minfee());
break;
case "changeAsset":
result.type = "changeAsset";
result.asset = Asset.fromOperation(attrs.asset());
result.beneficiary = attrs.beneficiary().toString('ascii');
break;
case "limitAsset":
result.type = "limitAsset";
result.asset = Asset.fromOperation(attrs.asset());
result.islimited = attrs.islimited();
break;
// thuannd end: fee task
// thuannd start: contract task
case "createContract":
result.type = "createContract";
result.contractId = attrs.contractId().toString('ascii');
result.startingBalance = this._fromXDRAmount(attrs.startingBalance());
result.contractAddr = attrs.contractAddr();
result.fileHash = attrs.fileHash().toString('ascii');
result.state = attrs.state();
result.errFlag = attrs.errFlag();
break;
case "callContract":
result.type = "callContract";
result.contractId = contractIdtoAddress(attrs.contractId());
// thuannd notes data
result.data = ContractInput.fromOperation(attrs.data());
// if (attrs.data()) {
// let data = {};
// data.funcName = attrs.data().funcName().toString();
// data.contractParams = attrs.data().contractParams();
// result.data = data;
// }
// ContrOperations
result.contrOps = ContrOperation.fromXDRObject(attrs.contrOps());
result.errFlag = attrs.errFlag();
break;
case "sendAsset":
result.type = "sendAsset";
result.contractId = contractIdtoAddress(attrs.contractId());
result.asset = Asset.fromOperation(attrs.asset());
result.amount = this._fromXDRAmount(attrs.amount());
result.state = attrs.state();
result.errFlag = attrs.errFlag();
break;
case "checkContract":
result.type = "checkContract";
result.isContract = attrs.isContract();
break;
case "testContract":
result.type = "testContract";
result.startingBalance = attrs.startingBalance();
break;
// thuannd end
default:
throw new Error("Unknown operation");
}
return result;
}
static isValidAmount(value, allowZero = false) {
if (!isString(value)) {
return false;
}
let amount;
try {
amount = new BigNumber(value);
} catch (e) {
return false;
}
switch (true) {
// == 0
case !allowZero && amount.isZero():
// < 0
case amount.isNegative():
// > Max value
case amount.times(ONE).greaterThan(new BigNumber(MAX_INT64).toString()):
// Decimal places (max 7)
case amount.decimalPlaces() > 7:
// NaN or Infinity
case (amount.isNaN() || !amount.isFinite()):
return false;
default:
return true;
}
}
// thuannd start
static isValidIsContract(value, allowZero = false) {
if (!isString(value)) {
return false;
}
let isContract;
try {
isContract = new BigNumber(value);
} catch (e) {
return false;
}
switch (true) {
// < 0
case isContract.isNegative():
// NaN or Infinity
case (isContract.isNaN() || !isContract.isFinite()):
return false;
default:
return true;
}
}
static constructIsContractRequirementsError(arg) {
return `${arg} argument must be of type String, represent a positive number after the decimal`;
}
// thuannd end
static constructAmountRequirementsError(arg) {
return `${arg} argument must be of type String, represent a positive number and have at most 7 digits after the decimal`;
}
/**
* Returns value converted to uint32 value or undefined.
* If `value` is not `Number`, `String` or `Undefined` then throws an error.
* Used in {@link Operation.setOptions}.
* @private
* @param {string} name Name of the property (used in error message only)
* @param {*} value Value to check
* @param {function(value, name)} isValidFunction Function to check other constraints (the argument will be a `Number`)
* @returns {undefined|Number}
* @private
*/
static _checkUnsignedIntValue(name, value, isValidFunction = null) {
if (isUndefined(value)) {
return undefined;
}
if (isString(value)) {
value = parseFloat(value);
}
switch (true) {
case !isNumber(value) || !isFinite(value) || value % 1 !== 0:
throw new Error(`${name} value is invalid`);
case value < 0:
throw new Error(`${name} value must be unsigned`);
case !isValidFunction || (isValidFunction && isValidFunction(value, name)):
return value;
default:
throw new Error(`${name} value is invalid`);
}
}
/**
* @private
*/
static _toXDRAmount(value) {
let amount = new BigNumber(value).mul(ONE);
return Hyper.fromString(amount.toString());
}
/**
* @private
*/
static _fromXDRAmount(value) {
return new BigNumber(value).div(ONE).toString();
}
// thuannd start
/**
* @private
*/
static _toXDRIsContract(value) {
let isContract = new BigNumber(value);
return Hyper.fromString(isContract.toString());
}
/**
* @private
*/
static _fromXDRIsContract(value) {
return new BigNumber(value).toString();
}
// thuannd end
/**
* @private
*/
static _fromXDRPrice(price) {
let n = new BigNumber(price.n());
return n.div(new BigNumber(price.d())).toString();
}
/**
* @private
*/
static _toXDRPrice(price) {
let xdrObject;
if (price.n && price.d) {
xdrObject = new xdr.Price(price);
} else {
price = new BigNumber(price);
let approx = best_r(price);
xdrObject = new xdr.Price({
n: parseInt(approx[0]),
d: parseInt(approx[1])
});
}
if (xdrObject.n() < 0 || xdrObject.d() < 0) {
throw new Error('price must be positive');
}
return xdrObject;
}
}
// Attach all imported operations as static methods on the Operation class
Operation.accountMerge = ops.accountMerge;
Operation.allowTrust = ops.allowTrust;
Operation.bumpSequence = ops.bumpSequence;
Operation.changeTrust = ops.changeTrust;
Operation.createAccount = ops.createAccount;
Operation.createPassiveOffer = ops.createPassiveOffer;
Operation.inflation = ops.inflation;
Operation.manageData = ops.manageData;
Operation.manageOffer = ops.manageOffer;
Operation.pathPayment = ops.pathPayment;
Operation.payment = ops.payment;
Operation.setOptions = ops.setOptions;
//////////////////////
Operation.createAsset = ops.createAsset;
Operation.changeAsset = ops.changeAsset;
Operation.limitAsset = ops.limitAsset;
// thuannd start
Operation.createContract = ops.createContract;
Operation.callContract = ops.callContract;
Operation.sendAsset = ops.sendAsset;
Operation.checkContract = ops.checkContract;
Operation.testContract = ops.testContract;
// thuannd end