Update Tokens Contract

in #he-dev4 years ago

{"id":"ssc-testnet-reaz","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"tokens","params":"","code":"const ACCOUNT_BLACKLIST={gateiodeposit:1,deepcrypto8:1,bittrex:1,poloniex:1,"huobi-pro":1,"binance-hot":1,bitvavo:1,blocktrades:1,probitsteem:1,probithive:1,ionomy:1,mxchive:1,coinbasebase:1,orinoco:1,"user.dunamu":1},HE_ACCOUNTS={"hive-engine":1,"swap-eth":1,"btc-swap":1,"graphene-swap":1,"honey-swap":1},RESERVED_SYMBOLS={ENG:"null",STEEMP:"steem-peg",BTCP:"btcpeg",LTCP:"ltcp",DOGEP:"dogep",BCHP:"bchp",SMTT:"steemmonsters",EM:"steem-eng",EMFOUR:"steem-eng",HIVEP:"steem-tokens"},VERIFIED_ISSUERS=["mining","tokenfunds"],calculateBalance=(a,e,t,i)=>(i?api.BigNumber(a).plus(e):api.BigNumber(a).minus(e)).toFixed(t),countDecimals=a=>api.BigNumber(a).dp(),findAndProcessAll=async(a,e,t)=>{let i=0;var n;let s=!1;for(;!s;)if(n=await api.db.find(a,e,1e3,i)){for(let a=0;a<n.length;a+=1)await t(n[a]);n.length<1e3?s=!0:i+=1e3}};actions.createSSC=async()=>{if(!1===await api.db.tableExists("tokens")){await api.db.createTable("tokens",["symbol"]),await api.db.createTable("balances",["account"]),await api.db.createTable("contractsBalances",["account"]),await api.db.createTable("params"),await api.db.createTable("pendingUnstakes",["account","unstakeCompleteTimestamp"]),await api.db.createTable("delegations",["from","to"]),await api.db.createTable("pendingUndelegations",["account","completeTimestamp"]);const a={tokenCreationFee:"0",enableDelegationFee:"0",enableStakingFee:"0"};await api.db.insert("params",a)}else{const e=await api.db.findOne("params",{});if(!e.blacklist){e.blacklist=ACCOUNT_BLACKLIST,e.heAccounts=HE_ACCOUNTS;const t={};let a=!1;e.fixMultiTxUnstakeBalance&&(delete e.fixMultiTxUnstakeBalance,t.fixMultiTxUnstakeBalance="",a=!0),e.cancelBadUnstakes&&(delete e.cancelBadUnstakes,t.cancelBadUnstakes="",a=!0),a?await api.db.update("params",e,t):await api.db.update("params",e)}}};const balanceTemplate={account:null,symbol:null,balance:"0",stake:"0",pendingUnstake:"0",delegationsIn:"0",delegationsOut:"0",pendingUndelegations:"0"},addStake=async(a,e,t)=>{let i=await api.db.findOne("balances",{account:a,symbol:e.symbol});null===i&&(i=balanceTemplate,i.account=a,i.symbol=e.symbol,i=await api.db.insert("balances",i)),void 0===i.stake&&(i.stake="0",i.pendingUnstake="0");a=i.stake;return i.stake=calculateBalance(i.stake,t,e.precision,!0),!!api.assert(api.BigNumber(i.stake).gt(a),"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)},subBalance=async(a,e,t,i)=>{const n=await api.db.findOne(i,{account:a,symbol:e.symbol});if(api.assert(null!==n,"balance does not exist")&&api.assert(api.BigNumber(n.balance).gte(t),"overdrawn balance")){a=n.balance;if(n.balance=calculateBalance(n.balance,t,e.precision,!1),api.assert(api.BigNumber(n.balance).lt(a),"cannot subtract"))return await api.db.update(i,n),!0}return!1},addBalance=async(a,e,t,i)=>{let n=await api.db.findOne(i,{account:a,symbol:e.symbol});if(null===n)return n=balanceTemplate,n.account=a,n.symbol=e.symbol,n.balance=t,await api.db.insert(i,n),!0;a=n.balance;return n.balance=calculateBalance(n.balance,t,e.precision,!0),!!api.assert(api.BigNumber(n.balance).gt(a),"cannot add")&&(await api.db.update(i,n),!0)};actions.updateParams=async a=>{if(api.sender===api.owner){var{tokenCreationFee:e,enableDelegationFee:t,enableStakingFee:i,blacklist:n,heAccounts:a}=a;const s=await api.db.findOne("params",{});e&&"string"==typeof e&&!api.BigNumber(e).isNaN()&&api.BigNumber(e).gte(0)&&(s.tokenCreationFee=e),t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(s.enableDelegationFee=t),i&&"string"==typeof i&&!api.BigNumber(i).isNaN()&&api.BigNumber(i).gte(0)&&(s.enableStakingFee=i),n&&"object"==typeof n&&(s.blacklist=n),a&&"object"==typeof a&&(s.heAccounts=a),await api.db.update("params",s)}},actions.updateUrl=async a=>{var{url:e,symbol:a}=a;if(api.assert(a&&"string"==typeof a&&e&&"string"==typeof e,"invalid params")&&api.assert(e.length<=255,"invalid url: max length of 255")){const t=await api.db.findOne("tokens",{symbol:a});if(t&&api.assert(t.issuer===api.sender,"must be the issuer"))try{const i=JSON.parse(t.metadata);api.assert(i&&i.url,"an error occured when trying to update the url")&&(i.url=e,t.metadata=JSON.stringify(i),await api.db.update("tokens",t))}catch(a){}}},actions.updateMetadata=async a=>{var{metadata:e,symbol:a}=a;if(api.assert(a&&"string"==typeof a&&e&&"object"==typeof e,"invalid params")){const i=await api.db.findOne("tokens",{symbol:a});if(i&&api.assert(i.issuer===api.sender,"must be the issuer"))try{var t=JSON.stringify(e);api.assert(t.length<=1e3,"invalid metadata: max length of 1000")&&(i.metadata=t,await api.db.update("tokens",i))}catch(a){}}},actions.updatePrecision=async a=>{var{symbol:e,precision:t,isSignedWithActiveKey:a}=a;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e)&&api.assert(0<t&&t<=8&&Number.isInteger(t),"invalid precision")){const i=await api.db.findOne("tokens",{symbol:e});i&&api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(t>i.precision,"precision can only be increased")&&(i.precision=t,await api.db.update("tokens",i))}},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 n=await api.db.findOne("tokens",{symbol:e});n&&api.assert(n.issuer===api.sender,"must be the issuer")&&(a=t.trim(),api.assert(api.isValidAccountName(a),"invalid to")&&(n.issuer=a,await api.db.update("tokens",n)))}},actions.create=async a=>{const{name:e,symbol:t,url:i,precision:n,maxSupply:s,isSignedWithActiveKey:o}=a;var{tokenCreationFee:l,heAccounts:r}=await api.db.findOne("params",{}),a=await api.db.findOne("balances",{account:api.sender,symbol:"BEE"}),a=!(!api.BigNumber(l).lte(0)&&1!==r[api.sender])||a&&api.BigNumber(a.balance).gte(l);api.assert(a,"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)&&(n&&"number"==typeof n||0===n)&&s&&"string"==typeof s&&!api.BigNumber(s).isNaN(),"invalid params")&&api.assert(0<t.length&&t.length<=10&&api.validator.isAlpha(api.validator.blacklist(t,"."))&&api.validator.isUppercase(t)&&(-1===t.indexOf(".")||0<t.indexOf(".")&&t.indexOf(".")<t.length-1&&t.indexOf(".")===t.lastIndexOf(".")),'invalid symbol: uppercase letters only and one "." allowed, max length of 10')&&api.assert(void 0===RESERVED_SYMBOLS[t]||api.sender===RESERVED_SYMBOLS[t],"cannot use this symbol")&&api.assert(1===r[api.sender]||-1===t.indexOf("SWAP"),"invalid symbol: not allowed to use SWAP")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(e," "))&&0<e.length&&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(0<=n&&n<=8&&Number.isInteger(n),"invalid precision")&&api.assert(api.BigNumber(s).gt(0),"maxSupply must be positive")&&api.assert(api.BigNumber(s).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)&&(a=await api.db.findOne("tokens",{symbol:t}),api.assert(null===a,"symbol already exists")&&(a={url:void 0===i?"":i},a=JSON.stringify(a),a={issuer:api.sender,symbol:t,name:e,metadata:a,precision:n,maxSupply:api.BigNumber(s).toFixed(n),supply:"0",circulatingSupply:"0",stakingEnabled:!1,unstakingCooldown:1,delegationEnabled:!1,undelegationCooldown:0},await api.db.insert("tokens",a),api.BigNumber(l).gt(0)&&void 0===r[api.sender]&&await actions.transfer({to:"null",symbol:"BEE",quantity:l,isSignedWithActiveKey:o})))},actions.issue=async e=>{const{to:a,symbol:t,quantity:i,isSignedWithActiveKey:n,callingContractInfo:s}=e;var o="null"===api.sender&&-1!==VERIFIED_ISSUERS.indexOf(s.name);if(o||api.assert(!0===n,"you must use a custom_json signed with your active key")&&api.assert(a&&"string"==typeof a&&t&&"string"==typeof t&&i&&"string"==typeof i&&!api.BigNumber(i).isNaN(),"invalid params")){e=a.trim();const l=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(o||l.issuer===api.sender,"not allowed to issue tokens")&&api.assert(countDecimals(i)<=l.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must issue positive quantity")&&api.assert(api.BigNumber(l.maxSupply).minus(l.supply).gte(i),"quantity exceeds available supply")&&api.assert(api.isValidAccountName(e),"invalid to")){let a=await addBalance(l.issuer,l,i,"balances");!0===a&&e!==l.issuer&&await subBalance(l.issuer,l,i,"balances")&&(a=await addBalance(e,l,i,"balances"),!1===a&&await addBalance(l.issuer,l,i,"balances")),!0===a&&(l.supply=calculateBalance(l.supply,i,l.precision,!0),"null"!==e&&(l.circulatingSupply=calculateBalance(l.circulatingSupply,i,l.precision,!0)),await api.db.update("tokens",l),api.emit("transferFromContract",{from:"tokens",to:e,symbol:t,quantity:i}))}}},actions.issueToContract=async a=>{const{to:e,symbol:t,quantity:i,isSignedWithActiveKey:n,callingContractInfo:s}=a;var o="null"===api.sender&&-1!==VERIFIED_ISSUERS.indexOf(s.name);if(o||api.assert(!0===n,"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")){a=e.trim();const l=await api.db.findOne("tokens",{symbol:t});api.assert(null!==l,"symbol does not exist")&&api.assert(o||l.issuer===api.sender,"not allowed to issue tokens")&&api.assert(countDecimals(i)<=l.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must issue positive quantity")&&api.assert(api.BigNumber(l.maxSupply).minus(l.supply).gte(i),"quantity exceeds available supply")&&api.assert(3<=a.length&&a.length<=50,"invalid to")&&!0===await addBalance(a,l,i,"contractsBalances")&&(l.supply=calculateBalance(l.supply,i,l.precision,!0),"null"!==a&&(l.circulatingSupply=calculateBalance(l.circulatingSupply,i,l.precision,!0)),await api.db.update("tokens",l),api.emit("issueToContract",{from:"tokens",to:a,symbol:t,quantity:i}))}},actions.transfer=async a=>{const{to:e,symbol:t,quantity:i,isSignedWithActiveKey:n}=a;if(api.assert(!0===n,"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")){var s=e.trim();if(api.assert(s!==api.sender,"cannot transfer to self")){var{blacklist:a}=await api.db.findOne("params",{});if(api.assert(api.isValidAccountName(s),"invalid to")&&api.assert(void 0===a[s],`not allowed to send to ${s}`)){const o=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==o,"symbol does not exist")&&api.assert(countDecimals(i)<=o.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(i).gt(0),"must transfer positive quantity"))if(await subBalance(api.sender,o,i,"balances"))return!1===await addBalance(s,o,i,"balances")?(await addBalance(api.sender,o,i,"balances"),!1):("null"===s&&(o.circulatingSupply=calculateBalance(o.circulatingSupply,i,o.precision,!1),await api.db.update("tokens",o)),api.emit("transfer",{from:api.sender,to:s,symbol:t,quantity:i}),!0)}}}return!1},actions.transferToContract=async a=>{const{from:e,to:t,symbol:i,quantity:n,isSignedWithActiveKey:s}=a;var o=void 0===e||"null"!==api.sender?api.sender:e;if(api.assert(!0===s||"null"===api.sender,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&i&&"string"==typeof i&&n&&"string"==typeof n&&!api.BigNumber(n).isNaN(),"invalid params")){a=t.trim().toLowerCase();if(api.assert(a!==o,"cannot transfer to self")&&api.assert(3<=a.length&&a.length<=50,"invalid to")){const l=await api.db.findOne("tokens",{symbol:i});api.assert(null!==l,"symbol does not exist")&&api.assert(countDecimals(n)<=l.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(n).gt(0),"must transfer positive quantity")&&await subBalance(o,l,n,"balances")&&(!1===await addBalance(a,l,n,"contractsBalances")?await addBalance(o,l,n,"balances"):("null"===a&&(l.circulatingSupply=calculateBalance(l.circulatingSupply,n,l.precision,!1),await api.db.update("tokens",l)),api.emit("transferToContract",{from:o,to:a,symbol:i,quantity:n})))}}},actions.transferFromContract=async a=>{if(api.assert("null"===api.sender,"not authorized")){const{from:i,to:n,symbol:s,quantity:o,type:l}=a,r=["user","contract"];if(api.assert(n&&"string"==typeof n&&i&&"string"==typeof i&&s&&"string"==typeof s&&l&&r.includes(l)&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN(),"invalid params")){var e=n.trim(),t="user"===l?"balances":"contractsBalances";if(api.assert("user"===l||"contract"===l&&e!==i,"cannot transfer to self")){a="user"===l?api.isValidAccountName(e):3<=e.length&&e.length<=50;if(api.assert(!0===a,"invalid to")){const p=await api.db.findOne("tokens",{symbol:s});api.assert(null!==p,"symbol does not exist")&&api.assert(countDecimals(o)<=p.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(o).gt(0),"must transfer positive quantity")&&await subBalance(i,p,o,"contractsBalances")&&(!1===await addBalance(e,p,o,t)?await addBalance(i,p,o,"contractsBalances"):("null"===e&&(p.circulatingSupply=calculateBalance(p.circulatingSupply,o,p.precision,!1),await api.db.update("tokens",p)),api.emit("transferFromContract",{from:i,to:e,symbol:s,quantity:o})))}}}}};const processUnstake=async a=>{var{account:e,symbol:t,quantity:i,quantityLeft:n,numberTransactionsLeft:s}=a;const o=a,l=await api.db.findOne("balances",{account:e,symbol:t}),r=await api.db.findOne("tokens",{symbol:t});let p=0,c=0;api.assert(null!==l,"balance does not exist")&&(1===s?(p=n,await api.db.remove("pendingUnstakes",a)):(p=api.BigNumber(i).dividedBy(r.numberTransactions).toFixed(r.precision,api.BigNumber.ROUND_DOWN),o.quantityLeft=api.BigNumber(o.quantityLeft).minus(p).toFixed(r.precision),--o.numberTransactionsLeft,c=1===o.numberTransactionsLeft?o.quantityLeft:p,o.nextTransactionTimestamp=api.BigNumber(o.nextTransactionTimestamp).plus(o.millisecPerPeriod).toNumber(),await api.db.update("pendingUnstakes",o)),api.BigNumber(p).gt(0)&&(a=l.balance,i=l.pendingUnstake,l.balance=calculateBalance(l.balance,p,r.precision,!0),l.pendingUnstake=calculateBalance(l.pendingUnstake,p,r.precision,!1),api.assert(api.BigNumber(l.pendingUnstake).lt(i)&&api.BigNumber(l.balance).gt(a),"cannot subtract")&&(api.BigNumber(c).gt(0)&&(l.stake=calculateBalance(l.stake,c,r.precision,!1),r.totalStaked=calculateBalance(r.totalStaked,c,r.precision,!1),"WORKERBEE"===t&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:e}),await api.executeSmartContract("mining","handleStakeChange",{account:e,symbol:t,quantity:api.BigNumber(c).negated()}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:e,token:r})),await api.db.update("balances",l),await api.db.update("tokens",r),api.emit("unstake",{account:e,symbol:t,quantity:p}))))};actions.checkPendingUnstakes=async()=>{if(api.assert("null"===api.sender,"not authorized")){const n=new Date(`${api.hiveBlockTimestamp}.000Z`);var a=n.getTime();let e=await api.db.find("pendingUnstakes",{nextTransactionTimestamp:{$lte:a}}),t=e.length;for(;0<t;){for(let a=0;a<t;a+=1){var i=e[a];await processUnstake(i)}e=await api.db.find("pendingUnstakes",{nextTransactionTimestamp:{$lte:a}}),t=e.length}}},actions.enableStaking=async a=>{var{symbol:e,unstakingCooldown:t,numberTransactions:i,isSignedWithActiveKey:n}=a,{enableStakingFee:s}=await api.db.findOne("params",{}),a=await api.db.findOne("balances",{account:api.sender,symbol:"BEE"}),a=a&&api.BigNumber(a.balance).gte(s),a=void 0===s||api.BigNumber(s).lte(0)||a;if(api.assert(a,"you must have enough tokens to cover  fees")&&api.assert(!0===n,"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)&&0<t&&t<=18250,"unstakingCooldown must be an integer between 1 and 18250")&&api.assert(i&&Number.isInteger(i)&&0<i&&i<=18250,"numberTransactions must be an integer between 1 and 18250")){const o=await api.db.findOne("tokens",{symbol:e});api.assert(null!==o,"symbol does not exist")&&api.assert(o.issuer===api.sender,"must be the issuer")&&api.assert(void 0===o.stakingEnabled||!1===o.stakingEnabled,"staking already enabled")&&(o.stakingEnabled=!0,o.totalStaked="0",o.unstakingCooldown=t,o.numberTransactions=i,await api.db.update("tokens",o),api.BigNumber(s).gt(0)&&await actions.transfer({to:"null",symbol:"BEE",quantity:s,isSignedWithActiveKey:n}))}},actions.stake=async a=>{const{symbol:e,quantity:t,to:i,isSignedWithActiveKey:n}=a;var s;api.assert(!0===n,"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")&&(s=await api.db.findOne("tokens",{symbol:e}),a=i.trim(),api.assert(api.isValidAccountName(a),"invalid to")&&api.assert(null!==s,"symbol does not exist")&&api.assert(countDecimals(t)<=s.precision,"symbol precision mismatch")&&api.assert(!0===s.stakingEnabled,"staking not enabled")&&api.assert(api.BigNumber(t).gt(0),"must stake positive quantity")&&await subBalance(api.sender,s,t,"balances")&&(!1===await addStake(a,s,t)?await addBalance(api.sender,s,t,"balances"):(api.emit("stake",{account:a,symbol:e,quantity:t}),"WORKERBEE"===e&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:a}),await api.executeSmartContract("mining","handleStakeChange",{account:a,symbol:e,quantity:t}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:a,token:s}))))},actions.stakeFromContract=async a=>{const{symbol:e,quantity:t,to:i,callingContractInfo:n}=a;var s;n&&api.assert(e&&"string"==typeof e&&i&&"string"==typeof i&&t&&"string"==typeof t&&!api.BigNumber(t).isNaN(),"invalid params")&&(s=await api.db.findOne("tokens",{symbol:e}),a=i.trim(),api.assert(api.isValidAccountName(a),"invalid to")&&api.assert(null!==s,"symbol does not exist")&&api.assert(countDecimals(t)<=s.precision,"symbol precision mismatch")&&api.assert(!0===s.stakingEnabled,"staking not enabled")&&api.assert(api.BigNumber(t).gt(0),"must stake positive quantity")&&await subBalance(n.name,s,t,"contractsBalances")&&(!1===await addStake(a,s,t)?await addBalance(n.name,s,t,"balances"):(api.emit("stakeFromContract",{account:a,symbol:e,quantity:t}),"WORKERBEE"===e&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:a}),await api.executeSmartContract("mining","handleStakeChange",{account:a,symbol:e,quantity:t}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:a,token:s}))))};const validateAvailableStake=async(a,t,e)=>{let i=api.BigNumber(a.stake);return await findAndProcessAll("pendingUnstakes",{symbol:a.symbol,account:a.account},async a=>{var e;1<a.numberTransactionsLeft&&(e=api.BigNumber(a.quantity).dividedBy(t.numberTransactions).toFixed(t.precision,api.BigNumber.ROUND_DOWN),i=i.minus(a.quantityLeft).plus(e))}),api.assert(i.gte(e),"overdrawn stake")},startUnstake=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")||!await validateAvailableStake(i,e,t))return!1;var n=i.stake,s=i.pendingUnstake,o=1<e.numberTransactions?api.BigNumber(t).dividedBy(e.numberTransactions).toFixed(e.precision,api.BigNumber.ROUND_DOWN):t;i.stake=calculateBalance(i.stake,o,e.precision,!1),i.pendingUnstake=calculateBalance(i.pendingUnstake,t,e.precision,!0),api.assert(api.BigNumber(i.stake).lt(n)&&api.BigNumber(i.pendingUnstake).gt(s),"cannot subtract")&&(await api.db.update("balances",i),e.totalStaked=calculateBalance(e.totalStaked,o,e.precision,!1),await api.db.update("tokens",e),"WORKERBEE"===e.symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:a}),await api.executeSmartContract("mining","handleStakeChange",{account:a,symbol:e.symbol,quantity:api.BigNumber(o).negated()}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:a,token:e}));const l=new Date(`${api.hiveBlockTimestamp}.000Z`);s=24*e.unstakingCooldown*3600*1e3,o=api.BigNumber(s).dividedBy(e.numberTransactions).integerValue(api.BigNumber.ROUND_DOWN),s=api.BigNumber(l.getTime()).plus(o).toNumber(),o={account:a,symbol:e.symbol,quantity:t,quantityLeft:t,nextTransactionTimestamp:s,numberTransactionsLeft:e.numberTransactions,millisecPerPeriod:o,txID:api.transactionId};return await api.db.insert("pendingUnstakes",o),!0};actions.unstake=async a=>{var{symbol:e,quantity:t,isSignedWithActiveKey:a}=a;api.assert(!0===a,"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")&&(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 startUnstake(api.sender,a,t)&&api.emit("unstakeStart",{account:api.sender,symbol:e,quantity:t}))};const processCancelUnstake=async a=>{var{account:e,symbol:t,quantity:i,quantityLeft:n,numberTransactionsLeft:s}=a;const o=await api.db.findOne("balances",{account:e,symbol:t}),l=await api.db.findOne("tokens",{symbol:t});if(api.assert(null!==o,"balance does not exist")&&api.assert(api.BigNumber(o.pendingUnstake).gte(n),"overdrawn pendingUnstake")){var r=o.stake,a=o.pendingUnstake,i=1<s?api.BigNumber(i).dividedBy(l.numberTransactions).toFixed(l.precision,api.BigNumber.ROUND_DOWN):n;if(o.stake=calculateBalance(o.stake,i,l.precision,!0),o.pendingUnstake=calculateBalance(o.pendingUnstake,n,l.precision,!1),api.assert(api.BigNumber(o.pendingUnstake).lt(a)&&api.BigNumber(o.stake).gt(r),"cannot subtract"))return await api.db.update("balances",o),l.totalStaked=calculateBalance(l.totalStaked,i,l.precision,!0),await api.db.update("tokens",l),api.emit("unstakeCancel",{account:e,symbol:t,quantity:n}),"WORKERBEE"===t&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:e}),await api.executeSmartContract("mining","handleStakeChange",{account:e,symbol:t,quantity:i}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:e,token:l}),!0}return!1};actions.cancelUnstake=async a=>{var{txID:e,isSignedWithActiveKey:a}=a;api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(e&&"string"==typeof e,"invalid params")&&(e=await api.db.findOne("pendingUnstakes",{account:api.sender,txID:e}),api.assert(e,"unstake does not exist")&&await processCancelUnstake(e)&&await api.db.remove("pendingUnstakes",e))},actions.enableDelegation=async a=>{var{symbol:e,undelegationCooldown:t,isSignedWithActiveKey:i}=a,{enableDelegationFee:n}=await api.db.findOne("params",{}),a=await api.db.findOne("balances",{account:api.sender,symbol:"BEE"}),a=a&&api.BigNumber(a.balance).gte(n),a=void 0===n||api.BigNumber(n).lte(0)||a;if(api.assert(a,"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)&&0<t&&t<=18250,"undelegationCooldown must be an integer between 1 and 18250")){const s=await api.db.findOne("tokens",{symbol:e});api.assert(null!==s,"symbol does not exist")&&api.assert(s.issuer===api.sender,"must be the issuer")&&api.assert(!0===s.stakingEnabled,"staking not enabled")&&api.assert(void 0===s.delegationEnabled||!1===s.delegationEnabled,"delegation already enabled")&&(s.delegationEnabled=!0,s.undelegationCooldown=t,await api.db.update("tokens",s),api.BigNumber(n).gt(0)&&await actions.transfer({to:"null",symbol:"BEE",quantity:n,isSignedWithActiveKey:i}))}},actions.delegate=async t=>{const{symbol:i,quantity:n,to:a,isSignedWithActiveKey:e}=t;if(api.assert(!0===e,"you must use a custom_json signed with your active key")&&api.assert(i&&"string"==typeof i&&a&&"string"==typeof a&&n&&"string"==typeof n&&!api.BigNumber(n).isNaN(),"invalid params")){var s=a.trim();if(api.assert(api.isValidAccountName(s),"invalid to")){var o=await api.db.findOne("tokens",{symbol:i});if(api.assert(null!==o,"symbol does not exist")&&api.assert(countDecimals(n)<=o.precision,"symbol precision mismatch")&&api.assert(!0===o.delegationEnabled,"delegation not enabled")&&api.assert(s!==api.sender,"cannot delegate to yourself")&&api.assert(api.BigNumber(n).gt(0),"must delegate positive quantity")){const l=await api.db.findOne("balances",{account:api.sender,symbol:i});if(api.assert(null!==l,"balanceFrom does not exist")&&await validateAvailableStake(l,o,n)){void 0===l.stake?(l.stake="0",l.pendingUnstake="0",l.delegationsIn="0",l.delegationsOut="0",l.pendingUndelegations="0"):void 0===l.delegationsIn&&(l.delegationsIn="0",l.delegationsOut="0",l.pendingUndelegations="0",l.delegatedStake&&(delete l.delegatedStake,delete l.receivedStake));let a=await api.db.findOne("balances",{account:s,symbol:i});null===a?(a=balanceTemplate,a.account=s,a.symbol=i,a=await api.db.insert("balances",a)):void 0===a.stake?(a.stake="0",a.pendingUnstake="0",a.delegationsIn="0",a.delegationsOut="0",a.pendingUndelegations="0"):void 0===a.delegationsIn&&(a.delegationsIn="0",a.delegationsOut="0",a.pendingUndelegations="0",a.delegatedStake&&(delete a.delegatedStake,delete a.receivedStake));let e=await api.db.findOne("delegations",{to:s,from:api.sender,symbol:i});const r=new Date(`${api.hiveBlockTimestamp}.000Z`);t=r.getTime();null==e?(l.stake=calculateBalance(l.stake,n,o.precision,!1),l.delegationsOut=calculateBalance(l.delegationsOut,n,o.precision,!0),await api.db.update("balances",l),a.delegationsIn=calculateBalance(a.delegationsIn,n,o.precision,!0),await api.db.update("balances",a),e={},e.from=api.sender,e.to=s,e.symbol=i,e.quantity=n,e.created=t,e.updated=t,await api.db.insert("delegations",e)):(l.stake=calculateBalance(l.stake,n,o.precision,!1),l.delegationsOut=calculateBalance(l.delegationsOut,n,o.precision,!0),await api.db.update("balances",l),a.delegationsIn=calculateBalance(a.delegationsIn,n,o.precision,!0),await api.db.update("balances",a),e.quantity=calculateBalance(e.quantity,n,o.precision,!0),e.updated=t,await api.db.update("delegations",e)),api.emit("delegate",{to:s,symbol:i,quantity:n}),"WORKERBEE"===i&&(await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:api.sender}),await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:s})),await api.executeSmartContract("mining","handleStakeChange",{account:s,symbol:i,quantity:n,delegated:!0}),await api.executeSmartContract("mining","handleStakeChange",{account:api.sender,symbol:i,quantity:api.BigNumber(n).negated()}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:api.sender,token:o}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:s,token:o})}}}}},actions.undelegate=async a=>{const{symbol:e,quantity:t,from:i,isSignedWithActiveKey:n}=a;if(api.assert(!0===n,"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")){var s=i.trim();if(api.assert(3<=s.length&&s.length<=16,"invalid from")){var o=await api.db.findOne("tokens",{symbol:e});if(api.assert(null!==o,"symbol does not exist")&&api.assert(countDecimals(t)<=o.precision,"symbol precision mismatch")&&api.assert(!0===o.delegationEnabled,"delegation not enabled")&&api.assert(s!==api.sender,"cannot undelegate from yourself")&&api.assert(api.BigNumber(t).gt(0),"must undelegate positive quantity")){const l=await api.db.findOne("balances",{account:api.sender,symbol:e});if(api.assert(null!==l,"balanceTo does not exist")&&api.assert(api.BigNumber(l.delegationsOut).gte(t),"overdrawn delegation")){const r=await api.db.findOne("balances",{account:s,symbol:e});if(api.assert(null!==r,"balanceFrom does not exist")){const p=await api.db.findOne("delegations",{to:s,from:api.sender,symbol:e});if(api.assert(null!==p,"delegation does not exist")&&api.assert(api.BigNumber(p.quantity).gte(t),"overdrawn delegation")){l.pendingUndelegations=calculateBalance(r.pendingUndelegations,t,o.precision,!0),l.delegationsOut=calculateBalance(l.delegationsOut,t,o.precision,!1),await api.db.update("balances",l),r.delegationsIn=calculateBalance(r.delegationsIn,t,o.precision,!1),await api.db.update("balances",r),p.quantity=calculateBalance(p.quantity,t,o.precision,!1),api.BigNumber(p.quantity).gt(0)?await api.db.update("delegations",p):await api.db.remove("delegations",p);const c=new Date(`${api.hiveBlockTimestamp}.000Z`);a=24*o.undelegationCooldown*3600*1e3,a=c.getTime()+a,a={account:api.sender,symbol:o.symbol,quantity:t,completeTimestamp:a,txID:api.transactionId};await api.db.insert("pendingUndelegations",a),api.emit("undelegateStart",{from:s,symbol:e,quantity:t}),"WORKERBEE"===e&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:s}),await api.executeSmartContract("mining","handleStakeChange",{account:s,symbol:e,quantity:api.BigNumber(t).negated(),delegated:!0}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:s,token:o})}}}}}}};const processUndelegation=async a=>{var{account:e,symbol:t,quantity:i}=a;const n=await api.db.findOne("balances",{account:e,symbol:t});var s,o,l=await api.db.findOne("tokens",{symbol:t});api.assert(null!==n,"balance does not exist")&&(s=n.stake,o=n.pendingUndelegations,n.stake=calculateBalance(n.stake,i,l.precision,!0),n.pendingUndelegations=calculateBalance(n.pendingUndelegations,i,l.precision,!1),api.assert(api.BigNumber(n.pendingUndelegations).lt(o)&&api.BigNumber(n.stake).gt(s),"cannot subtract")&&(await api.db.update("balances",n),await api.db.remove("pendingUndelegations",a),api.emit("undelegateDone",{account:e,symbol:t,quantity:i}),"WORKERBEE"===t&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:e}),await api.executeSmartContract("mining","handleStakeChange",{account:e,symbol:t,quantity:i}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:e,token:l})))};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){const n=new Date(`${api.hiveBlockTimestamp}.000Z`);var a=n.getTime();let e=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:a}}),t=e.length;for(;0<t;){for(let a=0;a<t;a+=1){var i=e[a];await processUndelegation(i)}e=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:a}}),t=e.length}}};"}}}