Tokens Smart Contract

in #steem-engine5 years ago (edited)

{
"id": "ssc-mainnet1",
"json": {
"contractName": "contract",
"contractAction": "update",
"contractPayload": {
"name": "tokens",
"params": "",
"code": "actions.createSSC=(async()=>{let a=await api.db.tableExists("tokens");if(!1===a){await api.db.createTable("tokens",["symbol"]),await api.db.createTable("balances",["account"]),await api.db.createTable("contractsBalances",["account"]),await api.db.createTable("params");const a={tokenCreationFee:"0"};await api.db.insert("params",a)}else{const a=await api.db.findOne("params",{});a.enableDelegationFee="1000",a.enableStakingFee="1000",await api.db.update("params",a)}!1===(a=await api.db.tableExists("pendingUnstakes"))&&await api.db.createTable("pendingUnstakes",["account","unstakeCompleteTimestamp"]),!1===(a=await api.db.tableExists("delegations"))&&(await api.db.createTable("delegations",["from","to"]),await api.db.createTable("pendingUndelegations",["account","completeTimestamp"]));let e=await api.db.findOne("tokens",{symbol:"STEEMP"});e&&e.precision<8&&(e.precision=8,await api.db.update("tokens",e)),void 0!==(e=await api.db.findOne("tokens",{symbol:"ENG"})).stakingEnabled&&!1!==e.stakingEnabled||(e.stakingEnabled=!0,e.totalStaked="0",e.unstakingCooldown=40,e.numberTransactions=4,e.delegationEnabled=!0,e.undelegationCooldown=7,await api.db.update("tokens",e))});const balanceTemplate={account:null,symbol:null,balance:"0",stake:"0",pendingUnstake:"0",delegationsIn:"0",delegationsOut:"0",pendingUndelegations:"0"},calculateBalance=(a,e,t,i)=>i?api.BigNumber(a).plus(e).toFixed(t):api.BigNumber(a).minus(e).toFixed(t),countDecimals=a=>api.BigNumber(a).dp(),addStake=async(a,e,t)=>{let i=await api.db.findOne("balances",{account:a,symbol:e.symbol});null===i&&((i=balanceTemplate).account=a,i.symbol=e.symbol,i=await api.db.insert("balances",i)),void 0===i.stake&&(i.stake="0",i.pendingUnstake="0");const s=i.stake;return i.stake=calculateBalance(i.stake,t,e.precision,!0),!!api.assert(api.BigNumber(i.stake).gt(s),"cannot add")&&(await api.db.update("balances",i),void 0===e.totalStaked&&(e.totalStaked="0"),e.totalStaked=calculateBalance(e.totalStaked,t,e.precision,!0),await api.db.update("tokens",e),!0)},subStake=async(a,e,t)=>{const i=await api.db.findOne("balances",{account:a,symbol:e.symbol});if(api.assert(null!==i,"balance does not exist")&&api.assert(api.BigNumber(i.stake).gte(t),"overdrawn stake")){const a=i.stake,s=i.pendingUnstake;if(i.stake=calculateBalance(i.stake,t,e.precision,!1),i.pendingUnstake=calculateBalance(i.pendingUnstake,t,e.precision,!0),api.assert(api.BigNumber(i.stake).lt(a)&&api.BigNumber(i.pendingUnstake).gt(s),"cannot subtract"))return await api.db.update("balances",i),!0}return!1},subBalance=async(a,e,t,i)=>{const s=await api.db.findOne(i,{account:a,symbol:e.symbol});if(api.assert(null!==s,"balance does not exist")&&api.assert(api.BigNumber(s.balance).gte(t),"overdrawn balance")){const a=s.balance;if(s.balance=calculateBalance(s.balance,t,e.precision,!1),api.assert(api.BigNumber(s.balance).lt(a),"cannot subtract"))return await api.db.update(i,s),!0}return!1},addBalance=async(a,e,t,i)=>{let s=await api.db.findOne(i,{account:a,symbol:e.symbol});if(null===s)return(s=balanceTemplate).account=a,s.symbol=e.symbol,s.balance=t,await api.db.insert(i,s),!0;const n=s.balance;return s.balance=calculateBalance(s.balance,t,e.precision,!0),!!api.assert(api.BigNumber(s.balance).gt(n),"cannot add")&&(await api.db.update(i,s),!0)};actions.updateParams=(async a=>{if(api.sender!==api.owner)return;const{tokenCreationFee:e}=a,t=await api.db.findOne("params",{});t.tokenCreationFee=e,await api.db.update("params",t)}),actions.updateUrl=(async a=>{const{url:e,symbol:t}=a;if(api.assert(t&&"string"==typeof t&&e&&"string"==typeof e,"invalid params")&&api.assert(e.length<=255,"invalid url: max length of 255")){const a=await api.db.findOne("tokens",{symbol:t});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{const t=JSON.parse(a.metadata);api.assert(t&&t.url,"an error occured when trying to update the url")&&(t.url=e,a.metadata=JSON.stringify(t),await api.db.update("tokens",a))}catch(a){}}}),actions.updateMetadata=(async a=>{const{metadata:e,symbol:t}=a;if(api.assert(t&&"string"==typeof t&&e&&"object"==typeof e,"invalid params")){const a=await api.db.findOne("tokens",{symbol:t});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{const t=JSON.stringify(e);api.assert(t.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=t,await api.db.update("tokens",a))}catch(a){}}}),actions.updatePrecision=(async a=>{const{symbol:e,precision:t,isSignedWithActiveKey:i}=a;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e)&&api.assert(t>0&&t<=8&&Number.isInteger(t),"invalid precision")){const a=await api.db.findOne("tokens",{symbol:e});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&api.assert(t>a.precision,"precision can only be increased")&&(a.precision=t,await api.db.update("tokens",a))}}),actions.transferOwnership=(async a=>{const{symbol:e,to:t,isSignedWithActiveKey:i}=a;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t,"invalid params")){const a=await api.db.findOne("tokens",{symbol:e});if(a&&api.assert(a.issuer===api.sender,"must be the issuer")){const e=t.trim();api.assert(api.isValidAccountName(e),"invalid to")&&(a.issuer=e,await api.db.update("tokens",a))}}}),actions.create=(async a=>{const{name:e,symbol:t,url:i,precision:s,maxSupply:n,isSignedWithActiveKey:o}=a,l=await api.db.findOne("params",{}),{tokenCreationFee:p}=l,c=await api.db.findOne("balances",{account:api.sender,symbol:"ENG"}),r=!!api.BigNumber(p).lte(0)||c&&api.BigNumber(c.balance).gte(p);if(api.assert(r,"you must have enough tokens to cover the creation fees")&&api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t&&(void 0===i||i&&"string"==typeof i)&&(s&&"number"==typeof s||0===s)&&n&&"string"==typeof n&&!api.BigNumber(n).isNaN(),"invalid params")&&api.assert(api.validator.isAlpha(t)&&api.validator.isUppercase(t)&&t.length>0&&t.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(e," "))&&e.length>0&&e.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===i||i.length<=255,"invalid url: max length of 255")&&api.assert(s>=0&&s<=8&&Number.isInteger(s),"invalid precision")&&api.assert(api.BigNumber(n).gt(0),"maxSupply must be positive")&&api.assert(api.BigNumber(n).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){const a=await api.db.findOne("tokens",{symbol:t});if(api.assert(null===a,"symbol already exists")){let a={url:void 0===i?"":i};a=JSON.stringify(a);const l={issuer:api.sender,symbol:t,name:e,metadata:a,precision:s,maxSupply:api.BigNumber(n).toFixed(s),supply:"0",circulatingSupply:"0",stakingEnabled:!1,unstakingCooldown:1,delegationEnabled:!1,undelegationCooldown:0};await api.db.insert("tokens",l),api.BigNumber(p).gt(0)&&await actions.transfer({to:"null",symbol:"ENG",quantity:p,isSignedWithActiveKey:o})}}}),actions.issue=(async a=>{const{to:e,symbol:t,quantity:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t&&i&&"string"==typeof i&&!api.BigNumber(i).isNaN(),"invalid params")){const a=e.trim(),s=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==s,"symbol does not exist")&&api.assert(s.issuer===api.sender,"not allowed to issue tokens")&&api.assert(countDecimals(i)<=s.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must issue positive quantity")&&api.assert(api.BigNumber(s.maxSupply).minus(s.supply).gte(i),"quantity exceeds available supply")&&api.assert(api.isValidAccountName(a),"invalid to")){let e=await addBalance(s.issuer,s,i,"balances");!0===e&&a!==s.issuer&&await subBalance(s.issuer,s,i,"balances")&&!1===(e=await addBalance(a,s,i,"balances"))&&await addBalance(s.issuer,s,i,"balances"),!0===e&&(s.supply=calculateBalance(s.supply,i,s.precision,!0),"null"!==a&&(s.circulatingSupply=calculateBalance(s.circulatingSupply,i,s.precision,!0)),await api.db.update("tokens",s),api.emit("transferFromContract",{from:"tokens",to:a,symbol:t,quantity:i}))}}}),actions.issueToContract=(async a=>{const{to:e,symbol:t,quantity:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t&&i&&"string"==typeof i&&!api.BigNumber(i).isNaN(),"invalid params")){const a=e.trim(),s=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==s,"symbol does not exist")&&api.assert(s.issuer===api.sender,"not allowed to issue tokens")&&api.assert(countDecimals(i)<=s.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must issue positive quantity")&&api.assert(api.BigNumber(s.maxSupply).minus(s.supply).gte(i),"quantity exceeds available supply")&&api.assert(a.length>=3&&a.length<=50,"invalid to")){!0===await addBalance(a,s,i,"contractsBalances")&&(s.supply=calculateBalance(s.supply,i,s.precision,!0),"null"!==a&&(s.circulatingSupply=calculateBalance(s.circulatingSupply,i,s.precision,!0)),await api.db.update("tokens",s),api.emit("issueToContract",{from:"tokens",to:a,symbol:t,quantity:i}))}}}),actions.transfer=(async a=>{const{to:e,symbol:t,quantity:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t&&i&&"string"==typeof i&&!api.BigNumber(i).isNaN(),"invalid params")){const a=e.trim();if(api.assert(a!==api.sender,"cannot transfer to self")&&api.assert(api.isValidAccountName(a),"invalid to")){const e=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==e,"symbol does not exist")&&api.assert(countDecimals(i)<=e.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must transfer positive quantity")&&await subBalance(api.sender,e,i,"balances")){return!1===await addBalance(a,e,i,"balances")?(await addBalance(api.sender,e,i,"balances"),!1):("null"===a&&(e.circulatingSupply=calculateBalance(e.circulatingSupply,i,e.precision,!1),await api.db.update("tokens",e)),api.emit("transfer",{from:api.sender,to:a,symbol:t,quantity:i}),!0)}}}return!1}),actions.transferToContract=(async a=>{const{to:e,symbol:t,quantity:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t&&i&&"string"==typeof i&&!api.BigNumber(i).isNaN(),"invalid params")){const a=e.trim().toLowerCase();if(api.assert(a!==api.sender,"cannot transfer to self")&&api.assert(a.length>=3&&a.length<=50,"invalid to")){const e=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==e,"symbol does not exist")&&api.assert(countDecimals(i)<=e.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must transfer positive quantity")&&await subBalance(api.sender,e,i,"balances")){!1===await addBalance(a,e,i,"contractsBalances")?await addBalance(api.sender,e,i,"balances"):("null"===a&&(e.circulatingSupply=calculateBalance(e.circulatingSupply,i,e.precision,!1),await api.db.update("tokens",e)),api.emit("transferToContract",{from:api.sender,to:a,symbol:t,quantity:i}))}}}}),actions.transferFromContract=(async a=>{if(api.assert("null"===api.sender,"not authorized")){const{from:e,to:t,symbol:i,quantity:s,type:n,isSignedWithActiveKey:o}=a,l=["user","contract"];if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&e&&"string"==typeof e&&i&&"string"==typeof i&&n&&l.includes(n)&&s&&"string"==typeof s&&!api.BigNumber(s).isNaN(),"invalid params")){const a=t.trim(),o="user"===n?"balances":"contractsBalances";if(api.assert("user"===n||"contract"===n&&a!==e,"cannot transfer to self")){const t="user"===n?api.isValidAccountName(a):a.length>=3&&a.length<=50;if(api.assert(!0===t,"invalid to")){const t=await api.db.findOne("tokens",{symbol:i});if(api.assert(null!==t,"symbol does not exist")&&api.assert(countDecimals(s)<=t.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(s).gt(0),"must transfer positive quantity")&&await subBalance(e,t,s,"contractsBalances")){!1===await addBalance(a,t,s,o)?await addBalance(e,t,s,"contractsBalances"):("null"===a&&(t.circulatingSupply=calculateBalance(t.circulatingSupply,s,t.precision,!1),await api.db.update("tokens",t)),api.emit("transferFromContract",{from:e,to:a,symbol:i,quantity:s}))}}}}}});const processUnstake=async a=>{const{account:e,symbol:t,quantity:i,quantityLeft:s,numberTransactionsLeft:n}=a,o=a,l=await api.db.findOne("balances",{account:e,symbol:t}),p=await api.db.findOne("tokens",{symbol:t});let c=0;if(api.assert(null!==l,"balance does not exist")&&(1===n?(c=s,await api.db.remove("pendingUnstakes",a)):(c=api.BigNumber(i).dividedBy(p.numberTransactions).toFixed(p.precision,api.BigNumber.ROUND_DOWN),o.quantityLeft=api.BigNumber(o.quantityLeft).minus(c).toFixed(p.precision),o.numberTransactionsLeft-=1,o.nextTransactionTimestamp=api.BigNumber(o.nextTransactionTimestamp).plus(o.millisecPerPeriod).toNumber(),await api.db.update("pendingUnstakes",o)),api.BigNumber(c).gt(0))){const a=l.balance,i=l.pendingUnstake;l.balance=calculateBalance(l.balance,c,p.precision,!0),l.pendingUnstake=calculateBalance(l.pendingUnstake,c,p.precision,!1),api.assert(api.BigNumber(l.pendingUnstake).lt(i)&&api.BigNumber(l.balance).gt(a),"cannot subtract")&&(await api.db.update("balances",l),p.totalStaked=calculateBalance(p.totalStaked,c,p.precision,!1),await api.db.update("tokens",p),api.emit("unstake",{account:e,symbol:t,quantity:c}),"ENG"===t&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:e}))}};actions.checkPendingUnstakes=(async()=>{if(api.assert("null"===api.sender,"not authorized")){const a=new Date(`${api.steemBlockTimestamp}.000Z`).getTime();let e=await api.db.find("pendingUnstakes",{nextTransactionTimestamp:{$lte:a}}),t=e.length;for(;t>0;){for(let a=0;a<t;a+=1){const t=e[a];await processUnstake(t)}t=(e=await api.db.find("pendingUnstakes",{nextTransactionTimestamp:{$lte:a}})).length}}}),actions.enableStaking=(async a=>{const{symbol:e,unstakingCooldown:t,numberTransactions:i,isSignedWithActiveKey:s}=a,n=await api.db.findOne("params",{}),{enableStakingFee:o}=n,l=await api.db.findOne("balances",{account:api.sender,symbol:"ENG"}),p=l&&api.BigNumber(l.balance).gte(o),c=void 0===o||api.BigNumber(o).lte(0)||p;if(api.assert(c,"you must have enough tokens to cover  fees")&&api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e,"invalid symbol")&&api.assert(t&&Number.isInteger(t)&&t>0&&t<=18250,"unstakingCooldown must be an integer between 1 and 18250")&&api.assert(i&&Number.isInteger(i)&&i>0&&i<=18250,"numberTransactions must be an integer between 1 and 18250")){const a=await api.db.findOne("tokens",{symbol:e});api.assert(null!==a,"symbol does not exist")&&api.assert(a.issuer===api.sender,"must be the issuer")&&api.assert(void 0===a.stakingEnabled||!1===a.stakingEnabled,"staking already enabled")&&(a.stakingEnabled=!0,a.totalStaked="0",a.unstakingCooldown=t,a.numberTransactions=i,await api.db.update("tokens",a),api.BigNumber(o).gt(0)&&await actions.transfer({to:"null",symbol:"ENG",quantity:o,isSignedWithActiveKey:s}))}}),actions.stake=(async a=>{const{symbol:e,quantity:t,to:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&i&&"string"==typeof i&&t&&"string"==typeof t&&!api.BigNumber(t).isNaN(),"invalid params")){const a=await api.db.findOne("tokens",{symbol:e}),s=i.trim();if(api.assert(api.isValidAccountName(s),"invalid to")&&api.assert(null!==a,"symbol does not exist")&&api.assert(countDecimals(t)<=a.precision,"symbol precision mismatch")&&api.assert(!0===a.stakingEnabled,"staking not enabled")&&api.assert(api.BigNumber(t).gt(0),"must stake positive quantity")&&await subBalance(api.sender,a,t,"balances")){!1===await addStake(s,a,t)?await addBalance(api.sender,a,t,"balances"):(api.emit("stake",{account:s,symbol:e,quantity:t}),"ENG"===e&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:api.sender}))}}}),actions.stakeFromContract=(async a=>{const{symbol:e,quantity:t,to:i,callingContractInfo:s}=a;if(s&&api.assert(e&&"string"==typeof e&&i&&"string"==typeof i&&t&&"string"==typeof t&&!api.BigNumber(t).isNaN(),"invalid params")){const a=await api.db.findOne("tokens",{symbol:e}),n=i.trim();if(api.assert(api.isValidAccountName(n),"invalid to")&&api.assert(null!==a,"symbol does not exist")&&api.assert(countDecimals(t)<=a.precision,"symbol precision mismatch")&&api.assert(!0===a.stakingEnabled,"staking not enabled")&&api.assert(api.BigNumber(t).gt(0),"must stake positive quantity")&&await subBalance(s.name,a,t,"contractsBalances")){!1===await addStake(n,a,t)?await addBalance(s.name,a,t,"balances"):(api.emit("stakeFromContract",{account:n,symbol:e,quantity:t}),"ENG"===e&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:n}))}}});const startUnstake=async(a,e,t)=>{const i=new Date(`${api.steemBlockTimestamp}.000Z`),s=24*e.unstakingCooldown*3600*1e3,n=api.BigNumber(s).dividedBy(e.numberTransactions).integerValue(api.BigNumber.ROUND_DOWN),o=api.BigNumber(i.getTime()).plus(n).toNumber(),l={account:a,symbol:e.symbol,quantity:t,quantityLeft:t,nextTransactionTimestamp:o,numberTransactionsLeft:e.numberTransactions,millisecPerPeriod:n,txID:api.transactionId};await api.db.insert("pendingUnstakes",l)};actions.unstake=(async a=>{const{symbol:e,quantity:t,isSignedWithActiveKey:i}=a;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&t&&"string"==typeof t&&!api.BigNumber(t).isNaN(),"invalid params")){const a=await api.db.findOne("tokens",{symbol:e});api.assert(null!==a,"symbol does not exist")&&api.assert(!0===a.stakingEnabled,"staking not enabled")&&api.assert(countDecimals(t)<=a.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(t).gt(0),"must unstake positive quantity")&&await subStake(api.sender,a,t)&&(await startUnstake(api.sender,a,t),api.emit("unstakeStart",{account:api.sender,symbol:e,quantity:t}))}});const processCancelUnstake=async a=>{const{account:e,symbol:t,quantityLeft:i}=a,s=await api.db.findOne("balances",{account:e,symbol:t}),n=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==s,"balance does not exist")&&api.assert(api.BigNumber(s.pendingUnstake).gte(i),"overdrawn pendingUnstake")){const a=s.stake,o=s.pendingUnstake;if(s.stake=calculateBalance(s.stake,i,n.precision,!0),s.pendingUnstake=calculateBalance(s.pendingUnstake,i,n.precision,!1),api.assert(api.BigNumber(s.pendingUnstake).lt(o)&&api.BigNumber(s.stake).gt(a),"cannot subtract"))return await api.db.update("balances",s),api.emit("unstake",{account:e,symbol:t,quantity:i}),!0}return!1};actions.cancelUnstake=(async a=>{const{txID:e,isSignedWithActiveKey:t}=a;if(api.assert(!0===t,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e,"invalid params")){const a=await api.db.findOne("pendingUnstakes",{account:api.sender,txID:e});api.assert(a,"unstake does not exist")&&await processCancelUnstake(a)&&await api.db.remove("pendingUnstakes",a)}}),actions.enableDelegation=(async a=>{const{symbol:e,undelegationCooldown:t,isSignedWithActiveKey:i}=a,s=await api.db.findOne("params",{}),{enableDelegationFee:n}=s,o=await api.db.findOne("balances",{account:api.sender,symbol:"ENG"}),l=o&&api.BigNumber(o.balance).gte(n),p=void 0===n||api.BigNumber(n).lte(0)||l;if(api.assert(p,"you must have enough tokens to cover  fees")&&api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e,"invalid symbol")&&api.assert(t&&Number.isInteger(t)&&t>0&&t<=18250,"undelegationCooldown must be an integer between 1 and 18250")){const a=await api.db.findOne("tokens",{symbol:e});api.assert(null!==a,"symbol does not exist")&&api.assert(a.issuer===api.sender,"must be the issuer")&&api.assert(!0===a.stakingEnabled,"staking not enabled")&&api.assert(void 0===a.delegationEnabled||!1===a.delegationEnabled,"delegation already enabled")&&(a.delegationEnabled=!0,a.undelegationCooldown=t,await api.db.update("tokens",a),api.BigNumber(n).gt(0)&&await actions.transfer({to:"null",symbol:"ENG",quantity:n,isSignedWithActiveKey:i}))}}),actions.delegate=(async a=>{const{symbol:e,quantity:t,to:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&i&&"string"==typeof i&&t&&"string"==typeof t&&!api.BigNumber(t).isNaN(),"invalid params")){const a=i.trim();if(api.assert(api.isValidAccountName(a),"invalid to")){const i=await api.db.findOne("tokens",{symbol:e});if(api.assert(null!==i,"symbol does not exist")&&api.assert(countDecimals(t)<=i.precision,"symbol precision mismatch")&&api.assert(!0===i.delegationEnabled,"delegation not enabled")&&api.assert(a!==api.sender,"cannot delegate to yourself")&&api.assert(api.BigNumber(t).gt(0),"must delegate positive quantity")){const s=await api.db.findOne("balances",{account:api.sender,symbol:e});if(api.assert(null!==s,"balanceFrom does not exist")&&api.assert(api.BigNumber(s.stake).gte(t),"overdrawn stake")){void 0===s.stake?(s.stake="0",s.pendingUnstake="0",s.delegationsIn="0",s.delegationsOut="0",s.pendingUndelegations="0"):void 0===s.delegationsIn&&(s.delegationsIn="0",s.delegationsOut="0",s.pendingUndelegations="0",s.delegatedStake&&(delete s.delegatedStake,delete s.receivedStake));let n=await api.db.findOne("balances",{account:a,symbol:e});null===n?((n=balanceTemplate).account=a,n.symbol=e,n=await api.db.insert("balances",n)):void 0===n.stake?(n.stake="0",n.pendingUnstake="0",n.delegationsIn="0",n.delegationsOut="0",n.pendingUndelegations="0"):void 0===n.delegationsIn&&(n.delegationsIn="0",n.delegationsOut="0",n.pendingUndelegations="0",n.delegatedStake&&(delete n.delegatedStake,delete n.receivedStake));let o=await api.db.findOne("delegations",{to:a,from:api.sender,symbol:e});const l=new Date(`${api.steemBlockTimestamp}.000Z`).getTime();null==o?(s.stake=calculateBalance(s.stake,t,i.precision,!1),s.delegationsOut=calculateBalance(s.delegationsOut,t,i.precision,!0),await api.db.update("balances",s),n.delegationsIn=calculateBalance(n.delegationsIn,t,i.precision,!0),await api.db.update("balances",n),(o={}).from=api.sender,o.to=a,o.symbol=e,o.quantity=t,o.created=l,o.updated=l,await api.db.insert("delegations",o),api.emit("delegate",{to:a,symbol:e,quantity:t}),"ENG"===e&&(await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:api.sender}),await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:a}))):(s.stake=calculateBalance(s.stake,t,i.precision,!1),s.delegationsOut=calculateBalance(s.delegationsOut,t,i.precision,!0),await api.db.update("balances",s),n.delegationsIn=calculateBalance(n.delegationsIn,t,i.precision,!0),await api.db.update("balances",n),o.quantity=calculateBalance(o.quantity,t,i.precision,!0),o.updated=l,await api.db.update("delegations",o),api.emit("delegate",{to:a,symbol:e,quantity:t}),"ENG"===e&&(await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:api.sender}),await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:a})))}}}}}),actions.undelegate=(async a=>{const{symbol:e,quantity:t,from:i,isSignedWithActiveKey:s}=a;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e&&i&&"string"==typeof i&&t&&"string"==typeof t&&!api.BigNumber(t).isNaN(),"invalid params")){const a=i.trim();if(api.assert(a.length>=3&&a.length<=16,"invalid from")){const i=await api.db.findOne("tokens",{symbol:e});if(api.assert(null!==i,"symbol does not exist")&&api.assert(countDecimals(t)<=i.precision,"symbol precision mismatch")&&api.assert(!0===i.delegationEnabled,"delegation not enabled")&&api.assert(a!==api.sender,"cannot undelegate from yourself")&&api.assert(api.BigNumber(t).gt(0),"must undelegate positive quantity")){const s=await api.db.findOne("balances",{account:api.sender,symbol:e});if(api.assert(null!==s,"balanceTo does not exist")&&api.assert(api.BigNumber(s.delegationsOut).gte(t),"overdrawn delegation")){const n=await api.db.findOne("balances",{account:a,symbol:e});if(api.assert(null!==n,"balanceFrom does not exist")){const o=await api.db.findOne("delegations",{to:a,from:api.sender,symbol:e});if(api.assert(null!==o,"delegation does not exist")&&api.assert(api.BigNumber(o.quantity).gte(t),"overdrawn delegation")){s.pendingUndelegations=calculateBalance(n.pendingUndelegations,t,i.precision,!0),s.delegationsOut=calculateBalance(s.delegationsOut,t,i.precision,!1),await api.db.update("balances",s),n.delegationsIn=calculateBalance(n.delegationsIn,t,i.precision,!1),await api.db.update("balances",n),o.quantity=calculateBalance(o.quantity,t,i.precision,!1),api.BigNumber(o.quantity).gt(0)?await api.db.update("delegations",o):await api.db.remove("delegations",o);const l=new Date(`${api.steemBlockTimestamp}.000Z`),p=24*i.undelegationCooldown*3600*1e3,c=l.getTime()+p,r={account:api.sender,symbol:i.symbol,quantity:t,completeTimestamp:c,txID:api.transactionId};await api.db.insert("pendingUndelegations",r),api.emit("undelegateStart",{from:a,symbol:e,quantity:t}),"ENG"===e&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:a})}}}}}}});const processUndelegation=async a=>{const{account:e,symbol:t,quantity:i}=a,s=await api.db.findOne("balances",{account:e,symbol:t}),n=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==s,"balance does not exist")){const o=s.stake,l=s.pendingUndelegations;s.stake=calculateBalance(s.stake,i,n.precision,!0),s.pendingUndelegations=calculateBalance(s.pendingUndelegations,i,n.precision,!1),api.assert(api.BigNumber(s.pendingUndelegations).lt(l)&&api.BigNumber(s.stake).gt(o),"cannot subtract")&&(await api.db.update("balances",s),await api.db.remove("pendingUndelegations",a),api.emit("undelegateDone",{account:e,symbol:t,quantity:i}),"ENG"===t&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:e}))}};actions.checkPendingUndelegations=(async()=>{if(api.assert("null"===api.sender,"not authorized")){const a=new Date(`${api.steemBlockTimestamp}.000Z`).getTime();let e=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:a}}),t=e.length;for(;t>0;){for(let a=0;a<t;a+=1){const t=e[a];await processUndelegation(t)}t=(e=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:a}})).length}}});"
}
}
}

Coin Marketplace

STEEM 0.28
TRX 0.12
JST 0.033
BTC 69626.36
ETH 3667.86
USDT 1.00
SBD 3.82