tokens Smart Contract

in #hive-engine2 months ago

{"id":"ssc-mainnet-hive","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",GLINT:"steemmonsters"},VERIFIED_ISSUERS=["comments","mining","tokenfunds","beedollar"],calculateBalance=(balance,quantity,precision,add)=>add?api.BigNumber(balance).plus(quantity).toFixed(precision):api.BigNumber(balance).minus(quantity).toFixed(precision),countDecimals=value=>api.BigNumber(value).dp(),findAndProcessAll=async(table,query,callback)=>{let offset=0,results=[],done=!1;for(;!done;)if(results=await api.db.find(table,query,1e3,offset),results){for(let i=0;i<results.length;i+=1)await callback(results[i]);results.length<1e3?done=!0:offset+=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 params={tokenCreationFee:"0",enableDelegationFee:"0",enableStakingFee:"0"};await api.db.insert("params",params)}else{const params=await api.db.findOne("params",{});if(!params.blacklist){params.blacklist=ACCOUNT_BLACKLIST,params.heAccounts=HE_ACCOUNTS;const unsets={};let useUnsets=!1;params.fixMultiTxUnstakeBalance&&(delete params.fixMultiTxUnstakeBalance,unsets.fixMultiTxUnstakeBalance="",useUnsets=!0),params.cancelBadUnstakes&&(delete params.cancelBadUnstakes,unsets.cancelBadUnstakes="",useUnsets=!0),useUnsets?await api.db.update("params",params,unsets):await api.db.update("params",params)}}};const balanceTemplate={account:null,symbol:null,balance:"0",stake:"0",pendingUnstake:"0",delegationsIn:"0",delegationsOut:"0",pendingUndelegations:"0"},addStake=async(account,token,quantity)=>{let balance=await api.db.findOne("balances",{account:account,symbol:token.symbol});null===balance&&(balance=balanceTemplate,balance.account=account,balance.symbol=token.symbol,balance=await api.db.insert("balances",balance)),void 0===balance.stake&&(balance.stake="0",balance.pendingUnstake="0");const originalStake=balance.stake;return balance.stake=calculateBalance(balance.stake,quantity,token.precision,!0),!!api.assert(api.BigNumber(balance.stake).gt(originalStake),"cannot add")&&(await api.db.update("balances",balance),void 0===token.totalStaked&&(token.totalStaked="0"),token.totalStaked=calculateBalance(token.totalStaked,quantity,token.precision,!0),await api.db.update("tokens",token),!0)},subBalance=async(account,token,quantity,table)=>{const balance=await api.db.findOne(table,{account:account,symbol:token.symbol});if(api.assert(null!==balance,"balance does not exist")&&api.assert(api.BigNumber(balance.balance).gte(quantity),"overdrawn balance")){const originalBalance=balance.balance;if(balance.balance=calculateBalance(balance.balance,quantity,token.precision,!1),api.assert(api.BigNumber(balance.balance).lt(originalBalance),"cannot subtract"))return await api.db.update(table,balance),!0}return!1},addBalance=async(account,token,quantity,table)=>{let balance=await api.db.findOne(table,{account:account,symbol:token.symbol});if(null===balance)return balance=balanceTemplate,balance.account=account,balance.symbol=token.symbol,balance.balance=quantity,await api.db.insert(table,balance),!0;const originalBalance=balance.balance;return balance.balance=calculateBalance(balance.balance,quantity,token.precision,!0),!!api.assert(api.BigNumber(balance.balance).gt(originalBalance),"cannot add")&&(await api.db.update(table,balance),!0)};actions.updateParams=async payload=>{if(api.sender!==api.owner)return;const{tokenCreationFee:tokenCreationFee,enableDelegationFee:enableDelegationFee,enableStakingFee:enableStakingFee,blacklist:blacklist,heAccounts:heAccounts}=payload,params=await api.db.findOne("params",{});tokenCreationFee&&"string"==typeof tokenCreationFee&&!api.BigNumber(tokenCreationFee).isNaN()&&api.BigNumber(tokenCreationFee).gte(0)&&(params.tokenCreationFee=tokenCreationFee),enableDelegationFee&&"string"==typeof enableDelegationFee&&!api.BigNumber(enableDelegationFee).isNaN()&&api.BigNumber(enableDelegationFee).gte(0)&&(params.enableDelegationFee=enableDelegationFee),enableStakingFee&&"string"==typeof enableStakingFee&&!api.BigNumber(enableStakingFee).isNaN()&&api.BigNumber(enableStakingFee).gte(0)&&(params.enableStakingFee=enableStakingFee),blacklist&&"object"==typeof blacklist&&(params.blacklist=blacklist),heAccounts&&"object"==typeof heAccounts&&(params.heAccounts=heAccounts),await api.db.update("params",params)},actions.updateUrl=async payload=>{const{url:url,symbol:symbol}=payload;if(api.assert(symbol&&"string"==typeof symbol&&url&&"string"==typeof url,"invalid params")&&api.assert(url.length<=255,"invalid url: max length of 255")){const token=await api.db.findOne("tokens",{symbol:symbol});if(token&&api.assert(token.issuer===api.sender,"must be the issuer"))try{const metadata=JSON.parse(token.metadata);api.assert(metadata&&metadata.url,"an error occured when trying to update the url")&&(metadata.url=url,token.metadata=JSON.stringify(metadata),await api.db.update("tokens",token))}catch(e){}}},actions.updateMetadata=async payload=>{const{metadata:metadata,symbol:symbol,callingContractInfo:callingContractInfo}=payload,fromVerifiedContract="hive-engine"===api.sender&&callingContractInfo&&-1!==VERIFIED_ISSUERS.indexOf(callingContractInfo.name);if(api.assert(symbol&&"string"==typeof symbol&&metadata&&"object"==typeof metadata,"invalid params")){const token=await api.db.findOne("tokens",{symbol:symbol});if(token&&api.assert(fromVerifiedContract||token.issuer===api.sender,"must be the issuer"))try{const finalMetadata=JSON.stringify(metadata);api.assert(finalMetadata.length<=1e3,"invalid metadata: max length of 1000")&&(token.metadata=finalMetadata,await api.db.update("tokens",token))}catch(e){}}},actions.updatePrecision=async payload=>{const{symbol:symbol,precision:precision,isSignedWithActiveKey:isSignedWithActiveKey}=payload;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol)&&api.assert(precision>0&&precision<=8&&Number.isInteger(precision),"invalid precision")){const token=await api.db.findOne("tokens",{symbol:symbol});token&&api.assert(token.issuer===api.sender,"must be the issuer")&&api.assert(precision>token.precision,"precision can only be increased")&&(token.precision=precision,await api.db.update("tokens",token))}},actions.transferOwnership=async payload=>{const{symbol:symbol,to:to,isSignedWithActiveKey:isSignedWithActiveKey}=payload;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol&&to&&"string"==typeof to,"invalid params")){const token=await api.db.findOne("tokens",{symbol:symbol});if(token&&api.assert(token.issuer===api.sender,"must be the issuer")){const finalTo=to.trim();api.assert(api.isValidAccountName(finalTo),"invalid to")&&(token.issuer=finalTo,await api.db.update("tokens",token),api.emit("transferOwnership",{from:api.sender,to:finalTo,symbol:symbol}))}}},actions.create=async payload=>{const{name:name,symbol:symbol,url:url,precision:precision,maxSupply:maxSupply,isSignedWithActiveKey:isSignedWithActiveKey,callingContractInfo:callingContractInfo}=payload,params=await api.db.findOne("params",{}),{tokenCreationFee:tokenCreationFee,heAccounts:heAccounts}=params,fromVerifiedContract="hive-engine"===api.sender&&callingContractInfo&&-1!==VERIFIED_ISSUERS.indexOf(callingContractInfo.name),utilityTokenBalance=fromVerifiedContract?null:await api.db.findOne("balances",{account:api.sender,symbol:"BEE"}),authorizedCreation=!(!fromVerifiedContract&&!api.BigNumber(tokenCreationFee).lte(0)&&1!==heAccounts[api.sender])||utilityTokenBalance&&api.BigNumber(utilityTokenBalance.balance).gte(tokenCreationFee);if(api.assert(authorizedCreation,"you must have enough tokens to cover the creation fees")&&api.assert(fromVerifiedContract||!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(name&&"string"==typeof name&&symbol&&"string"==typeof symbol&&(void 0===url||url&&"string"==typeof url)&&(precision&&"number"==typeof precision||0===precision)&&maxSupply&&"string"==typeof maxSupply&&!api.BigNumber(maxSupply).isNaN(),"invalid params")&&api.assert(symbol.length>0&&symbol.length<=10&&api.validator.isAlpha(api.validator.blacklist(symbol,"."))&&api.validator.isUppercase(symbol)&&(-1===symbol.indexOf(".")||symbol.indexOf(".")>0&&symbol.indexOf(".")<symbol.length-1&&symbol.indexOf(".")===symbol.lastIndexOf(".")),'invalid symbol: uppercase letters only and one "." allowed, max length of 10')&&api.assert(void 0===RESERVED_SYMBOLS[symbol]||api.sender===RESERVED_SYMBOLS[symbol],"cannot use this symbol")&&api.assert(1===heAccounts[api.sender]||-1===symbol.indexOf("SWAP"),"invalid symbol: not allowed to use SWAP")&&api.assert(1===heAccounts[api.sender]||-1===symbol.indexOf("ETH"),"invalid symbol: not allowed to use ETH")&&api.assert(1===heAccounts[api.sender]||-1===symbol.indexOf("BSC"),"invalid symbol: not allowed to use BSC")&&api.assert(1===heAccounts[api.sender]||-1===symbol.indexOf("."),'invalid symbol: usage of "." is restricted')&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(name," "))&&name.length>0&&name.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===url||url.length<=255,"invalid url: max length of 255")&&api.assert(precision>=0&&precision<=8&&Number.isInteger(precision),"invalid precision")&&api.assert(api.BigNumber(maxSupply).gt(0),"maxSupply must be positive")&&api.assert(api.BigNumber(maxSupply).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){const token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null===token,"symbol already exists")){let metadata={url:void 0===url?"":url};metadata=JSON.stringify(metadata);const newToken={issuer:fromVerifiedContract?"null":api.sender,symbol:symbol,name:name,metadata:metadata,precision:precision,maxSupply:api.BigNumber(maxSupply).toFixed(precision),supply:"0",circulatingSupply:"0",stakingEnabled:!1,unstakingCooldown:1,delegationEnabled:!1,undelegationCooldown:0};await api.db.insert("tokens",newToken),api.BigNumber(tokenCreationFee).gt(0)&&void 0===heAccounts[api.sender]&&!fromVerifiedContract&&await actions.transfer({to:"null",symbol:"BEE",quantity:tokenCreationFee,isSignedWithActiveKey:isSignedWithActiveKey})}}},actions.issue=async payload=>{const{to:to,symbol:symbol,isSignedWithActiveKey:isSignedWithActiveKey,callingContractInfo:callingContractInfo}=payload;let quantity=payload.quantity;const fromVerifiedContract="null"===api.sender&&-1!==VERIFIED_ISSUERS.indexOf(callingContractInfo.name)||callingContractInfo&&"beedollar"===callingContractInfo.name;if(fromVerifiedContract||api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(to&&"string"==typeof to&&symbol&&"string"==typeof symbol&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalTo=to.trim(),token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(fromVerifiedContract||token.issuer===api.sender,"not allowed to issue tokens")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(quantity).gt(0),"must issue positive quantity")&&api.assert(api.BigNumber(token.maxSupply).minus(token.supply).gte(quantity),"quantity exceeds available supply")&&api.assert(api.isValidAccountName(finalTo),"invalid to")){quantity=api.BigNumber(quantity).toFixed(token.precision);let res=await addBalance(token.issuer,token,quantity,"balances");!0===res&&finalTo!==token.issuer&&await subBalance(token.issuer,token,quantity,"balances")&&(res=await addBalance(finalTo,token,quantity,"balances"),!1===res&&await addBalance(token.issuer,token,quantity,"balances")),!0===res&&(token.supply=calculateBalance(token.supply,quantity,token.precision,!0),"null"!==finalTo&&(token.circulatingSupply=calculateBalance(token.circulatingSupply,quantity,token.precision,!0)),await api.db.update("tokens",token),api.emit("transferFromContract",{from:"tokens",to:finalTo,symbol:symbol,quantity:quantity}))}}},actions.issueToContract=async payload=>{const{to:to,symbol:symbol,isSignedWithActiveKey:isSignedWithActiveKey,callingContractInfo:callingContractInfo}=payload;let quantity=payload.quantity;const fromVerifiedContract="null"===api.sender&&-1!==VERIFIED_ISSUERS.indexOf(callingContractInfo.name);if(fromVerifiedContract||api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(to&&"string"==typeof to&&symbol&&"string"==typeof symbol&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalTo=to.trim(),token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(fromVerifiedContract||token.issuer===api.sender,"not allowed to issue tokens")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(quantity).gt(0),"must issue positive quantity")&&api.assert(api.BigNumber(token.maxSupply).minus(token.supply).gte(quantity),"quantity exceeds available supply")&&api.assert(finalTo.length>=3&&finalTo.length<=50,"invalid to")){quantity=api.BigNumber(quantity).toFixed(token.precision);!0===await addBalance(finalTo,token,quantity,"contractsBalances")&&(token.supply=calculateBalance(token.supply,quantity,token.precision,!0),"null"!==finalTo&&(token.circulatingSupply=calculateBalance(token.circulatingSupply,quantity,token.precision,!0)),await api.db.update("tokens",token),api.emit("issueToContract",{from:"tokens",to:finalTo,symbol:symbol,quantity:quantity}))}}},actions.transfer=async payload=>{const{to:to,symbol:symbol,isSignedWithActiveKey:isSignedWithActiveKey}=payload;let quantity=payload.quantity;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(to&&"string"==typeof to&&symbol&&"string"==typeof symbol&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalTo=to.trim();if(api.assert(finalTo!==api.sender,"cannot transfer to self")){const params=await api.db.findOne("params",{}),{blacklist:blacklist}=params;if(api.assert(api.isValidAccountName(finalTo),"invalid to")&&api.assert(void 0===blacklist[finalTo],`not allowed to send to ${finalTo}`)){const token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(quantity).gt(0),"must transfer positive quantity")&&await subBalance(api.sender,token,quantity,"balances")){return!1===await addBalance(finalTo,token,quantity,"balances")?(await addBalance(api.sender,token,quantity,"balances"),!1):("null"===finalTo&&(token.circulatingSupply=calculateBalance(token.circulatingSupply,quantity,token.precision,!1),await api.db.update("tokens",token)),api.emit("transfer",{from:api.sender,to:finalTo,symbol:symbol,quantity:quantity}),!0)}}}}return!1},actions.transferToContract=async payload=>{const{from:from,to:to,symbol:symbol,isSignedWithActiveKey:isSignedWithActiveKey}=payload;let quantity=payload.quantity;const finalFrom=void 0===from||"null"!==api.sender?api.sender:from;if(api.assert(!0===isSignedWithActiveKey||"null"===api.sender,"you must use a custom_json signed with your active key")&&api.assert(to&&"string"==typeof to&&symbol&&"string"==typeof symbol&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalTo=to.trim().toLowerCase();if(api.assert(finalTo!==finalFrom,"cannot transfer to self")&&api.assert(finalTo.length>=3&&finalTo.length<=50,"invalid to")){const token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(quantity).gt(0),"must transfer positive quantity")&&(quantity=api.BigNumber(quantity).toFixed(token.precision),await subBalance(finalFrom,token,quantity,"balances"))){!1===await addBalance(finalTo,token,quantity,"contractsBalances")?await addBalance(finalFrom,token,quantity,"balances"):("null"===finalTo&&(token.circulatingSupply=calculateBalance(token.circulatingSupply,quantity,token.precision,!1),await api.db.update("tokens",token)),api.emit("transferToContract",{from:finalFrom,to:finalTo,symbol:symbol,quantity:quantity}))}}}},actions.transferFromContract=async payload=>{if(api.assert("null"===api.sender,"not authorized")){const{from:from,to:to,symbol:symbol,type:type}=payload;let quantity=payload.quantity;const types=["user","contract"];if(api.assert(to&&"string"==typeof to&&from&&"string"==typeof from&&symbol&&"string"==typeof symbol&&type&&types.includes(type)&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalTo=to.trim(),table="user"===type?"balances":"contractsBalances";if(api.assert("user"===type||"contract"===type&&finalTo!==from,"cannot transfer to self")){const toValid="user"===type?api.isValidAccountName(finalTo):finalTo.length>=3&&finalTo.length<=50;if(api.assert(!0===toValid,"invalid to")){const token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(quantity).gt(0),"must transfer positive quantity")&&(quantity=api.BigNumber(quantity).toFixed(token.precision),await subBalance(from,token,quantity,"contractsBalances"))){!1===await addBalance(finalTo,token,quantity,table)?await addBalance(from,token,quantity,"contractsBalances"):("null"===finalTo&&(token.circulatingSupply=calculateBalance(token.circulatingSupply,quantity,token.precision,!1),await api.db.update("tokens",token)),api.emit("transferFromContract",{from:from,to:finalTo,symbol:symbol,quantity:quantity}))}}}}}};const processUnstake=async unstake=>{const{account:account,symbol:symbol,quantity:quantity,quantityLeft:quantityLeft,numberTransactionsLeft:numberTransactionsLeft}=unstake,newUnstake=unstake,balance=await api.db.findOne("balances",{account:account,symbol:symbol}),token=await api.db.findOne("tokens",{symbol:symbol});let tokensToRelease=0,nextTokensToRelease=0;if(api.assert(null!==balance,"balance does not exist")&&(1===numberTransactionsLeft?(tokensToRelease=quantityLeft,await api.db.remove("pendingUnstakes",unstake)):(tokensToRelease=api.BigNumber(quantity).dividedBy(token.numberTransactions).toFixed(token.precision,api.BigNumber.ROUND_DOWN),newUnstake.quantityLeft=api.BigNumber(newUnstake.quantityLeft).minus(tokensToRelease).toFixed(token.precision),newUnstake.numberTransactionsLeft-=1,nextTokensToRelease=1===newUnstake.numberTransactionsLeft?newUnstake.quantityLeft:tokensToRelease,newUnstake.nextTransactionTimestamp=api.BigNumber(newUnstake.nextTransactionTimestamp).plus(newUnstake.millisecPerPeriod).toNumber(),await api.db.update("pendingUnstakes",newUnstake)),api.BigNumber(tokensToRelease).gt(0))){const originalBalance=balance.balance,originalPendingStake=balance.pendingUnstake;balance.balance=calculateBalance(balance.balance,tokensToRelease,token.precision,!0),balance.pendingUnstake=calculateBalance(balance.pendingUnstake,tokensToRelease,token.precision,!1),api.assert(api.BigNumber(balance.pendingUnstake).lt(originalPendingStake)&&api.BigNumber(balance.balance).gt(originalBalance),"cannot subtract")&&(api.BigNumber(nextTokensToRelease).gt(0)&&(balance.stake=calculateBalance(balance.stake,nextTokensToRelease,token.precision,!1),token.totalStaked=calculateBalance(token.totalStaked,nextTokensToRelease,token.precision,!1),await api.executeSmartContract("mining","handleStakeChange",{account:account,symbol:symbol,quantity:api.BigNumber(nextTokensToRelease).negated()})),await api.db.update("balances",balance),await api.db.update("tokens",token),"WORKERBEE"===symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:account}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:account,token:token}),api.emit("unstake",{account:account,symbol:symbol,quantity:tokensToRelease}))}};actions.checkPendingUnstakes=async()=>{if(api.assert("null"===api.sender,"not authorized")){const timestamp=new Date(`${api.hiveBlockTimestamp}.000Z`).getTime();let pendingUnstakes=await api.db.find("pendingUnstakes",{nextTransactionTimestamp:{$lte:timestamp}},1e3,0,[{index:"_id",descending:!1}]),nbPendingUnstakes=pendingUnstakes.length;for(;nbPendingUnstakes>0;){for(let index=0;index<nbPendingUnstakes;index+=1){const pendingUnstake=pendingUnstakes[index];await processUnstake(pendingUnstake)}pendingUnstakes=await api.db.find("pendingUnstakes",{nextTransactionTimestamp:{$lte:timestamp}},1e3,0,[{index:"_id",descending:!1}]),nbPendingUnstakes=pendingUnstakes.length}}},actions.enableStaking=async payload=>{const{symbol:symbol,unstakingCooldown:unstakingCooldown,numberTransactions:numberTransactions,isSignedWithActiveKey:isSignedWithActiveKey}=payload,params=await api.db.findOne("params",{}),{enableStakingFee:enableStakingFee}=params,utilityTokenBalance=await api.db.findOne("balances",{account:api.sender,symbol:"BEE"}),enoughFunds=utilityTokenBalance&&api.BigNumber(utilityTokenBalance.balance).gte(enableStakingFee),authorized=void 0===enableStakingFee||api.BigNumber(enableStakingFee).lte(0)||enoughFunds;if(api.assert(authorized,"you must have enough tokens to cover  fees")&&api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol,"invalid symbol")&&api.assert(unstakingCooldown&&Number.isInteger(unstakingCooldown)&&unstakingCooldown>0&&unstakingCooldown<=18250,"unstakingCooldown must be an integer between 1 and 18250")&&api.assert(numberTransactions&&Number.isInteger(numberTransactions)&&numberTransactions>0&&numberTransactions<=18250,"numberTransactions must be an integer between 1 and 18250")){const token=await api.db.findOne("tokens",{symbol:symbol});api.assert(null!==token,"symbol does not exist")&&api.assert(token.issuer===api.sender,"must be the issuer")&&api.assert(void 0===token.stakingEnabled||!1===token.stakingEnabled,"staking already enabled")&&(token.stakingEnabled=!0,token.totalStaked="0",token.unstakingCooldown=unstakingCooldown,token.numberTransactions=numberTransactions,await api.db.update("tokens",token),api.BigNumber(enableStakingFee).gt(0)&&await actions.transfer({to:"null",symbol:"BEE",quantity:enableStakingFee,isSignedWithActiveKey:isSignedWithActiveKey}))}},actions.stake=async payload=>{const{symbol:symbol,to:to,isSignedWithActiveKey:isSignedWithActiveKey}=payload;let quantity=payload.quantity;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol&&to&&"string"==typeof to&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const token=await api.db.findOne("tokens",{symbol:symbol}),finalTo=to.trim();if(api.assert(api.isValidAccountName(finalTo),"invalid to")&&api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(!0===token.stakingEnabled,"staking not enabled")&&api.assert(api.BigNumber(quantity).gt(0),"must stake positive quantity")&&(quantity=api.BigNumber(quantity).toFixed(token.precision),await subBalance(api.sender,token,quantity,"balances"))){!1===await addStake(finalTo,token,quantity)?await addBalance(api.sender,token,quantity,"balances"):(api.emit("stake",{account:finalTo,symbol:symbol,quantity:quantity}),"WORKERBEE"===symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:finalTo}),await api.executeSmartContract("mining","handleStakeChange",{account:finalTo,symbol:symbol,quantity:quantity}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:finalTo,token:token}))}}},actions.stakeFromContract=async payload=>{const{symbol:symbol,to:to,callingContractInfo:callingContractInfo}=payload;let quantity=payload.quantity;if(callingContractInfo&&api.assert(symbol&&"string"==typeof symbol&&to&&"string"==typeof to&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const token=await api.db.findOne("tokens",{symbol:symbol}),finalTo=to.trim();if(api.assert(api.isValidAccountName(finalTo),"invalid to")&&api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(!0===token.stakingEnabled,"staking not enabled")&&api.assert(api.BigNumber(quantity).gt(0),"must stake positive quantity")&&(quantity=api.BigNumber(quantity).toFixed(token.precision),await subBalance(callingContractInfo.name,token,quantity,"contractsBalances"))){!1===await addStake(finalTo,token,quantity)?await addBalance(callingContractInfo.name,token,quantity,"balances"):(api.emit("stakeFromContract",{account:finalTo,symbol:symbol,quantity:quantity}),"WORKERBEE"===symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:finalTo}),await api.executeSmartContract("mining","handleStakeChange",{account:finalTo,symbol:symbol,quantity:quantity}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:finalTo,token:token}))}}};const validateAvailableStake=async(balance,token,quantity)=>{let availableStakeBalance=api.BigNumber(balance.stake);return await findAndProcessAll("pendingUnstakes",{symbol:balance.symbol,account:balance.account},(async pendingUnstake=>{if(pendingUnstake.numberTransactionsLeft>1){const tokensToRelease=api.BigNumber(pendingUnstake.quantity).dividedBy(token.numberTransactions).toFixed(token.precision,api.BigNumber.ROUND_DOWN);availableStakeBalance=availableStakeBalance.minus(pendingUnstake.quantityLeft).plus(tokensToRelease)}})),api.assert(availableStakeBalance.gte(quantity),"overdrawn stake")},startUnstake=async(account,token,quantity)=>{const balance=await api.db.findOne("balances",{account:account,symbol:token.symbol});if(!api.assert(null!==balance,"balance does not exist")||!await validateAvailableStake(balance,token,quantity))return!1;{const originalStake=balance.stake,originalPendingStake=balance.pendingUnstake,nextTokensToRelease=token.numberTransactions>1?api.BigNumber(quantity).dividedBy(token.numberTransactions).toFixed(token.precision,api.BigNumber.ROUND_DOWN):quantity;balance.stake=calculateBalance(balance.stake,nextTokensToRelease,token.precision,!1),balance.pendingUnstake=calculateBalance(balance.pendingUnstake,quantity,token.precision,!0),api.assert(api.BigNumber(balance.stake).lt(originalStake)&&api.BigNumber(balance.pendingUnstake).gt(originalPendingStake),"cannot subtract")&&(await api.db.update("balances",balance),token.totalStaked=calculateBalance(token.totalStaked,nextTokensToRelease,token.precision,!1),await api.db.update("tokens",token),"WORKERBEE"===token.symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:account}),await api.executeSmartContract("mining","handleStakeChange",{account:account,symbol:token.symbol,quantity:api.BigNumber(nextTokensToRelease).negated()}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:account,token:token}))}const blockDate=new Date(`${api.hiveBlockTimestamp}.000Z`),cooldownPeriodMillisec=24*token.unstakingCooldown*3600*1e3,millisecPerPeriod=api.BigNumber(cooldownPeriodMillisec).dividedBy(token.numberTransactions).integerValue(api.BigNumber.ROUND_DOWN),nextTransactionTimestamp=api.BigNumber(blockDate.getTime()).plus(millisecPerPeriod).toNumber(),unstake={account:account,symbol:token.symbol,quantity:quantity,quantityLeft:quantity,nextTransactionTimestamp:nextTransactionTimestamp,numberTransactionsLeft:token.numberTransactions,millisecPerPeriod:millisecPerPeriod,txID:api.transactionId};return await api.db.insert("pendingUnstakes",unstake),!0};actions.unstake=async payload=>{const{symbol:symbol,isSignedWithActiveKey:isSignedWithActiveKey}=payload;let quantity=payload.quantity;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const token=await api.db.findOne("tokens",{symbol:symbol});api.assert(null!==token,"symbol does not exist")&&api.assert(!0===token.stakingEnabled,"staking not enabled")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(api.BigNumber(quantity).gt(0),"must unstake positive quantity")&&(quantity=api.BigNumber(quantity).toFixed(token.precision),await startUnstake(api.sender,token,quantity)&&api.emit("unstakeStart",{account:api.sender,symbol:symbol,quantity:quantity}))}};const processCancelUnstake=async unstake=>{const{account:account,symbol:symbol,quantity:quantity,quantityLeft:quantityLeft,numberTransactionsLeft:numberTransactionsLeft}=unstake,balance=await api.db.findOne("balances",{account:account,symbol:symbol}),token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==balance,"balance does not exist")&&api.assert(api.BigNumber(balance.pendingUnstake).gte(quantityLeft),"overdrawn pendingUnstake")){const originalStake=balance.stake,originalPendingStake=balance.pendingUnstake,tokensToRelease=numberTransactionsLeft>1?api.BigNumber(quantity).dividedBy(token.numberTransactions).toFixed(token.precision,api.BigNumber.ROUND_DOWN):quantityLeft;if(balance.stake=calculateBalance(balance.stake,tokensToRelease,token.precision,!0),balance.pendingUnstake=calculateBalance(balance.pendingUnstake,quantityLeft,token.precision,!1),api.assert(api.BigNumber(balance.pendingUnstake).lt(originalPendingStake)&&api.BigNumber(balance.stake).gt(originalStake),"cannot subtract"))return await api.db.update("balances",balance),token.totalStaked=calculateBalance(token.totalStaked,tokensToRelease,token.precision,!0),await api.db.update("tokens",token),api.emit("unstakeCancel",{account:account,symbol:symbol,quantity:quantityLeft}),"WORKERBEE"===symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:account}),await api.executeSmartContract("mining","handleStakeChange",{account:account,symbol:symbol,quantity:tokensToRelease}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:account,token:token}),!0}return!1};actions.cancelUnstake=async payload=>{const{txID:txID,isSignedWithActiveKey:isSignedWithActiveKey}=payload;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(txID&&"string"==typeof txID,"invalid params")){const unstake=await api.db.findOne("pendingUnstakes",{account:api.sender,txID:txID});api.assert(unstake,"unstake does not exist")&&await processCancelUnstake(unstake)&&await api.db.remove("pendingUnstakes",unstake)}},actions.enableDelegation=async payload=>{const{symbol:symbol,undelegationCooldown:undelegationCooldown,isSignedWithActiveKey:isSignedWithActiveKey}=payload,params=await api.db.findOne("params",{}),{enableDelegationFee:enableDelegationFee}=params,utilityTokenBalance=await api.db.findOne("balances",{account:api.sender,symbol:"BEE"}),enoughFunds=utilityTokenBalance&&api.BigNumber(utilityTokenBalance.balance).gte(enableDelegationFee),authorized=void 0===enableDelegationFee||api.BigNumber(enableDelegationFee).lte(0)||enoughFunds;if(api.assert(authorized,"you must have enough tokens to cover  fees")&&api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol,"invalid symbol")&&api.assert(undelegationCooldown&&Number.isInteger(undelegationCooldown)&&undelegationCooldown>0&&undelegationCooldown<=18250,"undelegationCooldown must be an integer between 1 and 18250")){const token=await api.db.findOne("tokens",{symbol:symbol});api.assert(null!==token,"symbol does not exist")&&api.assert(token.issuer===api.sender,"must be the issuer")&&api.assert(!0===token.stakingEnabled,"staking not enabled")&&api.assert(void 0===token.delegationEnabled||!1===token.delegationEnabled,"delegation already enabled")&&(token.delegationEnabled=!0,token.undelegationCooldown=undelegationCooldown,await api.db.update("tokens",token),api.BigNumber(enableDelegationFee).gt(0)&&await actions.transfer({to:"null",symbol:"BEE",quantity:enableDelegationFee,isSignedWithActiveKey:isSignedWithActiveKey}))}},actions.delegate=async payload=>{const{symbol:symbol,to:to,isSignedWithActiveKey:isSignedWithActiveKey}=payload;let quantity=payload.quantity;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol&&to&&"string"==typeof to&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalTo=to.trim();if(api.assert(api.isValidAccountName(finalTo),"invalid to")){const token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(!0===token.delegationEnabled,"delegation not enabled")&&api.assert(finalTo!==api.sender,"cannot delegate to yourself")&&api.assert(api.BigNumber(quantity).gt(0),"must delegate positive quantity")){quantity=api.BigNumber(quantity).toFixed(token.precision);const balanceFrom=await api.db.findOne("balances",{account:api.sender,symbol:symbol});if(api.assert(null!==balanceFrom,"balanceFrom does not exist")&&await validateAvailableStake(balanceFrom,token,quantity)){void 0===balanceFrom.stake?(balanceFrom.stake="0",balanceFrom.pendingUnstake="0",balanceFrom.delegationsIn="0",balanceFrom.delegationsOut="0",balanceFrom.pendingUndelegations="0"):void 0===balanceFrom.delegationsIn&&(balanceFrom.delegationsIn="0",balanceFrom.delegationsOut="0",balanceFrom.pendingUndelegations="0",balanceFrom.delegatedStake&&(delete balanceFrom.delegatedStake,delete balanceFrom.receivedStake));let balanceTo=await api.db.findOne("balances",{account:finalTo,symbol:symbol});null===balanceTo?(balanceTo=balanceTemplate,balanceTo.account=finalTo,balanceTo.symbol=symbol,balanceTo=await api.db.insert("balances",balanceTo)):void 0===balanceTo.stake?(balanceTo.stake="0",balanceTo.pendingUnstake="0",balanceTo.delegationsIn="0",balanceTo.delegationsOut="0",balanceTo.pendingUndelegations="0"):void 0===balanceTo.delegationsIn&&(balanceTo.delegationsIn="0",balanceTo.delegationsOut="0",balanceTo.pendingUndelegations="0",balanceTo.delegatedStake&&(delete balanceTo.delegatedStake,delete balanceTo.receivedStake));let delegation=await api.db.findOne("delegations",{to:finalTo,from:api.sender,symbol:symbol});const timestamp=new Date(`${api.hiveBlockTimestamp}.000Z`).getTime();null==delegation?(balanceFrom.stake=calculateBalance(balanceFrom.stake,quantity,token.precision,!1),balanceFrom.delegationsOut=calculateBalance(balanceFrom.delegationsOut,quantity,token.precision,!0),await api.db.update("balances",balanceFrom),balanceTo.delegationsIn=calculateBalance(balanceTo.delegationsIn,quantity,token.precision,!0),await api.db.update("balances",balanceTo),delegation={},delegation.from=api.sender,delegation.to=finalTo,delegation.symbol=symbol,delegation.quantity=quantity,delegation.created=timestamp,delegation.updated=timestamp,await api.db.insert("delegations",delegation),api.emit("delegate",{to:finalTo,symbol:symbol,quantity:quantity}),"WORKERBEE"===symbol&&(await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:api.sender}),await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:finalTo})),await api.executeSmartContract("mining","handleStakeChange",{account:finalTo,symbol:symbol,quantity:quantity,delegated:!0}),await api.executeSmartContract("mining","handleStakeChange",{account:api.sender,symbol:symbol,quantity:api.BigNumber(quantity).negated()}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:api.sender,token:token}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:finalTo,token:token})):(balanceFrom.stake=calculateBalance(balanceFrom.stake,quantity,token.precision,!1),balanceFrom.delegationsOut=calculateBalance(balanceFrom.delegationsOut,quantity,token.precision,!0),await api.db.update("balances",balanceFrom),balanceTo.delegationsIn=calculateBalance(balanceTo.delegationsIn,quantity,token.precision,!0),await api.db.update("balances",balanceTo),delegation.quantity=calculateBalance(delegation.quantity,quantity,token.precision,!0),delegation.updated=timestamp,await api.db.update("delegations",delegation),api.emit("delegate",{to:finalTo,symbol:symbol,quantity:quantity}),"WORKERBEE"===symbol&&(await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:api.sender}),await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:finalTo})),await api.executeSmartContract("mining","handleStakeChange",{account:finalTo,symbol:symbol,quantity:quantity,delegated:!0}),await api.executeSmartContract("mining","handleStakeChange",{account:api.sender,symbol:symbol,quantity:api.BigNumber(quantity).negated()}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:api.sender,token:token}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:finalTo,token:token}))}}}}},actions.undelegate=async payload=>{const{symbol:symbol,from:from,isSignedWithActiveKey:isSignedWithActiveKey}=payload;let quantity=payload.quantity;if(api.assert(!0===isSignedWithActiveKey,"you must use a custom_json signed with your active key")&&api.assert(symbol&&"string"==typeof symbol&&from&&"string"==typeof from&&quantity&&"string"==typeof quantity&&!api.BigNumber(quantity).isNaN(),"invalid params")){const finalFrom=from.trim();if(api.assert(finalFrom.length>=3&&finalFrom.length<=16,"invalid from")){const token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==token,"symbol does not exist")&&api.assert(countDecimals(quantity)<=token.precision,"symbol precision mismatch")&&api.assert(!0===token.delegationEnabled,"delegation not enabled")&&api.assert(finalFrom!==api.sender,"cannot undelegate from yourself")&&api.assert(api.BigNumber(quantity).gt(0),"must undelegate positive quantity")){quantity=api.BigNumber(quantity).toFixed(token.precision);const balanceTo=await api.db.findOne("balances",{account:api.sender,symbol:symbol});if(api.assert(null!==balanceTo,"balanceTo does not exist")&&api.assert(api.BigNumber(balanceTo.delegationsOut).gte(quantity),"overdrawn delegation")){const balanceFrom=await api.db.findOne("balances",{account:finalFrom,symbol:symbol});if(api.assert(null!==balanceFrom,"balanceFrom does not exist")){const delegation=await api.db.findOne("delegations",{to:finalFrom,from:api.sender,symbol:symbol});if(api.assert(null!==delegation,"delegation does not exist")&&api.assert(api.BigNumber(delegation.quantity).gte(quantity),"overdrawn delegation")){balanceTo.pendingUndelegations=calculateBalance(balanceTo.pendingUndelegations,quantity,token.precision,!0),balanceTo.delegationsOut=calculateBalance(balanceTo.delegationsOut,quantity,token.precision,!1),await api.db.update("balances",balanceTo),balanceFrom.delegationsIn=calculateBalance(balanceFrom.delegationsIn,quantity,token.precision,!1),await api.db.update("balances",balanceFrom),delegation.quantity=calculateBalance(delegation.quantity,quantity,token.precision,!1),api.BigNumber(delegation.quantity).gt(0)?await api.db.update("delegations",delegation):await api.db.remove("delegations",delegation);const blockDate=new Date(`${api.hiveBlockTimestamp}.000Z`),cooldownPeriodMillisec=24*token.undelegationCooldown*3600*1e3,completeTimestamp=blockDate.getTime()+cooldownPeriodMillisec,undelegation={account:api.sender,symbol:token.symbol,quantity:quantity,completeTimestamp:completeTimestamp,txID:api.transactionId};await api.db.insert("pendingUndelegations",undelegation),api.emit("undelegateStart",{from:finalFrom,symbol:symbol,quantity:quantity}),"WORKERBEE"===symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:finalFrom}),await api.executeSmartContract("mining","handleStakeChange",{account:finalFrom,symbol:symbol,quantity:api.BigNumber(quantity).negated(),delegated:!0}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:finalFrom,token:token})}}}}}}};const processUndelegation=async undelegation=>{const{account:account,symbol:symbol,quantity:quantity}=undelegation,balance=await api.db.findOne("balances",{account:account,symbol:symbol}),token=await api.db.findOne("tokens",{symbol:symbol});if(api.assert(null!==balance,"balance does not exist")){const originalStake=balance.stake,originalPendingUndelegations=balance.pendingUndelegations;balance.stake=calculateBalance(balance.stake,quantity,token.precision,!0),balance.pendingUndelegations=calculateBalance(balance.pendingUndelegations,quantity,token.precision,!1),api.assert(api.BigNumber(balance.pendingUndelegations).lt(originalPendingUndelegations)&&api.BigNumber(balance.stake).gt(originalStake),"cannot subtract")&&(await api.db.update("balances",balance),await api.db.remove("pendingUndelegations",undelegation),api.emit("undelegateDone",{account:account,symbol:symbol,quantity:quantity}),"WORKERBEE"===symbol&&await api.executeSmartContract("witnesses","updateWitnessesApprovals",{account:account}),await api.executeSmartContract("mining","handleStakeChange",{account:account,symbol:symbol,quantity:quantity}),await api.executeSmartContract("tokenfunds","updateProposalApprovals",{account:account,token:token}))}};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){const timestamp=new Date(`${api.hiveBlockTimestamp}.000Z`).getTime();let pendingUndelegations=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:timestamp}},1e3,0,[{index:"_id",descending:!1}]),nbPendingUndelegations=pendingUndelegations.length;for(;nbPendingUndelegations>0;){for(let index=0;index<nbPendingUndelegations;index+=1){const pendingUndelegation=pendingUndelegations[index];await processUndelegation(pendingUndelegation)}pendingUndelegations=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:timestamp}},1e3,0,[{index:"_id",descending:!1}]),nbPendingUndelegations=pendingUndelegations.length}}};"}}}

Sort:  

{"id":"ssc-mainnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"witnesses","params":"","code":"const NB_APPROVALS_ALLOWED=30,NB_TOP_WITNESSES=20,NB_BACKUP_WITNESSES=1,NB_WITNESSES=21,NB_WITNESSES_SIGNATURES_REQUIRED=14,MAX_ROUNDS_MISSED_IN_A_ROW=3,MAX_ROUND_PROPOSITION_WAITING_PERIOD=40,NB_TOKENS_TO_REWARD_PER_BLOCK="0.01902586",NB_TOKENS_NEEDED_BEFORE_REWARDING="0.39954306",WITNESS_APPROVE_EXPIRE_BLOCKS=5184e3,WITNESS_MAX_ACCOUNT_EXPIRE_PER_BLOCK=10,UTILITY_TOKEN_SYMBOL="BEE",UTILITY_TOKEN_PRECISION=8,GOVERNANCE_TOKEN_SYMBOL="WORKERBEE",GOVERNANCE_TOKEN_PRECISION=5,GOVERNANCE_TOKEN_MIN_VALUE="0.00001",recalcTotalEnabledApprovalWeight=async()=>{let wits,totalEnabledApprovalWeight="0",offset=0;do{wits=await api.db.find("witnesses",{},1e3,offset,[{index:"_id",descending:!1}]);for(let i=0;i<wits.length;i+=1){const wit=wits[i];wit.enabled&&(totalEnabledApprovalWeight=api.BigNumber(totalEnabledApprovalWeight).plus(wit.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION))}offset+=1e3}while(1e3===wits.length);return totalEnabledApprovalWeight};actions.createSSC=async()=>{if(!1===await api.db.tableExists("witnesses")){await api.db.createTable("witnesses",["approvalWeight"]),await api.db.createTable("approvals",["from","to"]),await api.db.createTable("accounts",["account"]),await api.db.createTable("schedules"),await api.db.createTable("params");const params={totalApprovalWeight:"0",totalEnabledApprovalWeight:"0",numberOfApprovedWitnesses:0,lastVerifiedBlockNumber:0,round:0,lastBlockRound:0,currentWitness:null,blockNumberWitnessChange:0,lastWitnesses:[],numberOfApprovalsPerAccount:30,numberOfTopWitnesses:20,numberOfWitnessSlots:21,witnessSignaturesRequired:14,maxRoundsMissedInARow:3,maxRoundPropositionWaitingPeriod:40,witnessApproveExpireBlocks:5184e3};await api.db.insert("params",params)}else{const params=await api.db.findOne("params",{});params.totalEnabledApprovalWeight&&"NaN"!==params.totalEnabledApprovalWeight||(params.totalEnabledApprovalWeight=await recalcTotalEnabledApprovalWeight(),await api.db.update("params",params))}},actions.resetSchedule=async()=>{if(api.sender!==api.owner)return;const schedules=await api.db.find("schedules",{});for(let index=0;index<schedules.length;index+=1){const schedule=schedules[index];await api.db.remove("schedules",schedule)}const params=await api.db.findOne("params",{});params.currentWitness=null,params.blockNumberWitnessChange=0,params.lastWitnesses=[],params.totalEnabledApprovalWeight=await recalcTotalEnabledApprovalWeight(),await api.db.update("params",params)},actions.recalculateApprovals=async payload=>{if(api.sender!==api.owner)return;const witnessRec=await api.db.findOne("witnesses",{account:payload.witness});if(!witnessRec)return;let newApprovalWeight=api.BigNumber(0);const approvals=await api.db.find("approvals",{to:payload.witness});for(let i=0;i<approvals.length;i+=1){const approval=approvals[i],account=await api.db.findOne("accounts",{account:approval.from});account&&(newApprovalWeight=api.BigNumber(newApprovalWeight).plus(account.approvalWeight).toFixed(GOVERNANCE_TOKEN_PRECISION))}const oldApprovalWeight=witnessRec.approvalWeight.$numberDecimal,deltaApprovalWeight=api.BigNumber(newApprovalWeight).minus(oldApprovalWeight).toFixed(GOVERNANCE_TOKEN_PRECISION);await updateWitnessRank(payload.witness,deltaApprovalWeight)},actions.updateParams=async payload=>{if(api.sender!==api.owner)return;const{numberOfApprovalsPerAccount:numberOfApprovalsPerAccount,numberOfTopWitnesses:numberOfTopWitnesses,numberOfWitnessSlots:numberOfWitnessSlots,witnessSignaturesRequired:witnessSignaturesRequired,maxRoundsMissedInARow:maxRoundsMissedInARow,maxRoundPropositionWaitingPeriod:maxRoundPropositionWaitingPeriod,witnessApproveExpireBlocks:witnessApproveExpireBlocks}=payload,params=await api.db.findOne("params",{});let shouldResetSchedule=!1;numberOfApprovalsPerAccount&&Number.isInteger(numberOfApprovalsPerAccount)&&(params.numberOfApprovalsPerAccount=numberOfApprovalsPerAccount),numberOfTopWitnesses&&Number.isInteger(numberOfTopWitnesses)&&(params.numberOfTopWitnesses=numberOfTopWitnesses),numberOfWitnessSlots&&Number.isInteger(numberOfWitnessSlots)&&params.numberOfWitnessSlots!==numberOfWitnessSlots&&(shouldResetSchedule=!0,params.numberOfWitnessSlots=numberOfWitnessSlots),witnessSignaturesRequired&&Number.isInteger(witnessSignaturesRequired)&&(params.witnessSignaturesRequired=witnessSignaturesRequired),maxRoundsMissedInARow&&Number.isInteger(maxRoundsMissedInARow)&&(params.maxRoundsMissedInARow=maxRoundsMissedInARow),maxRoundPropositionWaitingPeriod&&Number.isInteger(maxRoundPropositionWaitingPeriod)&&(params.maxRoundPropositionWaitingPeriod=maxRoundPropositionWaitingPeriod),api.assert(params.numberOfTopWitnesses+1===params.numberOfWitnessSlots,"only 1 backup allowed")&&(witnessApproveExpireBlocks&&Number.isInteger(witnessApproveExpireBlocks)&&api.assert(witnessApproveExpireBlocks>params.numberOfWitnessSlots,"witnessApproveExpireBlocks should be greater than numberOfWitnessSlots")&&(params.witnessApproveExpireBlocks=witnessApproveExpireBlocks),await api.db.update("params",params),shouldResetSchedule&&await actions.resetSchedule())};const updateWitnessRank=async(witness,approvalWeight)=>{const witnessRec=await api.db.findOne("witnesses",{account:witness});if(witnessRec){const oldApprovalWeight=witnessRec.approvalWeight.$numberDecimal;witnessRec.approvalWeight.$numberDecimal=api.BigNumber(witnessRec.approvalWeight.$numberDecimal).plus(approvalWeight).toFixed(GOVERNANCE_TOKEN_PRECISION),api.BigNumber(witnessRec.approvalWeight.$numberDecimal).lt(0)&&(witnessRec.approvalWeight.$numberDecimal=api.BigNumber(0)),await api.db.update("witnesses",witnessRec);const params=await api.db.findOne("params",{});params.totalApprovalWeight=api.BigNumber(params.totalApprovalWeight).plus(approvalWeight).toFixed(GOVERNANCE_TOKEN_PRECISION),witnessRec.enabled&&(params.totalEnabledApprovalWeight=api.BigNumber(params.totalEnabledApprovalWeight).plus(approvalWeight).toFixed(GOVERNANCE_TOKEN_PRECISION)),api.BigNumber(oldApprovalWeight).eq(0)&&api.BigNumber(witnessRec.approvalWeight.$numberDecimal).gt(0)?params.numberOfApprovedWitnesses+=1:api.BigNumber(oldApprovalWeight).gt(0)&&api.BigNumber(witnessRec.approvalWeight.$numberDecimal).eq(0)&&(params.numberOfApprovedWitnesses-=1),await api.db.update("params",params)}};actions.updateWitnessesApprovals=async payload=>{const{account:account,callingContractInfo:callingContractInfo}=payload;if(api.sender!==api.owner){if(void 0===callingContractInfo)return;if("tokens"!==callingContractInfo.name)return}const acct=await api.db.findOne("accounts",{account:account});if(null!==acct){const balance=await api.db.findOneInTable("tokens","balances",{account:account,symbol:GOVERNANCE_TOKEN_SYMBOL});let approvalWeight=0;balance&&balance.stake&&(approvalWeight=balance.stake),balance&&balance.delegationsIn&&(approvalWeight=api.BigNumber(approvalWeight).plus(balance.delegationsIn).toFixed(GOVERNANCE_TOKEN_PRECISION));const oldApprovalWeight=acct.approvalWeight,deltaApprovalWeight=api.BigNumber(approvalWeight).minus(oldApprovalWeight).toFixed(GOVERNANCE_TOKEN_PRECISION);if(acct.approvalWeight=approvalWeight,!api.BigNumber(deltaApprovalWeight).eq(0)){await api.db.update("accounts",acct);const approvals=await api.db.find("approvals",{from:account});for(let index=0;index<approvals.length;index+=1){const approval=approvals[index];await updateWitnessRank(approval.to,deltaApprovalWeight)}}}},actions.register=async payload=>{const{domain:domain,IP:IP,RPCPort:RPCPort,P2PPort:P2PPort,signingKey:signingKey,enabled:enabled,isSignedWithActiveKey:isSignedWithActiveKey}=payload;if(api.assert(!0===isSignedWithActiveKey,"active key required")&&api.assert(domain||IP,"neither domain nor ip provided")&&api.assert(!(domain&&IP),"both domain and ip provided")&&(domain&&api.assert(domain&&"string"==typeof domain&&api.validator.isFQDN(domain),"domain is invalid")||IP&&api.assert(IP&&"string"==typeof IP&&api.validator.isIP(IP),"IP is invalid"))&&api.assert(RPCPort&&Number.isInteger(RPCPort)&&RPCPort>=0&&RPCPort<=65535,"RPCPort must be an integer between 0 and 65535")&&api.assert(P2PPort&&Number.isInteger(P2PPort)&&P2PPort>=0&&P2PPort<=65535,"P2PPort must be an integer between 0 and 65535")&&api.assert(api.validator.isAlphanumeric(signingKey)&&53===signingKey.length,"invalid signing key")&&api.assert("boolean"==typeof enabled,"enabled must be a boolean")){let witness=await api.db.findOne("witnesses",{signingKey:signingKey});if(api.assert(null===witness||witness.account===api.sender,"a witness is already using this signing key")&&(witness=IP?await api.db.findOne("witnesses",{IP:IP,P2PPort:P2PPort}):await api.db.findOne("witnesses",{domain:domain,P2PPort:P2PPort}),api.assert(null===witness||witness.account===api.sender,`a witness is already using this ${IP?"IP":"domain"}/Port`)))if(witness=await api.db.findOne("witnesses",{account:api.sender}),witness){const enabledChanged=witness.enabled!==enabled;let useUnsets=!1;const unsets={};IP?(witness.IP=IP,witness.domain&&(delete witness.domain,unsets.domain="",useUnsets=!0)):(witness.domain=domain,witness.IP&&(delete witness.IP,unsets.IP="",useUnsets=!0)),witness.RPCPort=RPCPort,witness.P2PPort=P2PPort,witness.signingKey=signingKey,witness.enabled=enabled,useUnsets?await api.db.update("witnesses",witness,unsets):await api.db.update("witnesses",witness);const params=await api.db.findOne("params",{});enabledChanged&&witness.enabled?params.totalEnabledApprovalWeight=api.BigNumber(params.totalEnabledApprovalWeight).plus(witness.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION):enabledChanged&&!witness.enabled&&(params.totalEnabledApprovalWeight=api.BigNumber(params.totalEnabledApprovalWeight).minus(witness.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION)),await api.db.update("params",params)}else witness={account:api.sender,approvalWeight:{$numberDecimal:"0"},signingKey:signingKey,RPCPort:RPCPort,P2PPort:P2PPort,enabled:enabled,missedRounds:0,missedRoundsInARow:0,verifiedRounds:0,lastRoundVerified:null,lastBlockVerified:null},IP?witness.IP=IP:witness.domain=domain,await api.db.insert("witnesses",witness)}};const removeApproval=async(approval,acct,blnce,manual=!0)=>{if(api.assert(null!==approval,"you have not approved this witness")){const{from:from,to:to}=approval;let account=acct;acct&&acct.account===from||(account=await api.db.findOne("accounts",{account:from})),await api.db.remove("approvals",approval);let balance=blnce;balance||(balance=await api.db.findOneInTable("tokens","balances",{account:from,symbol:GOVERNANCE_TOKEN_SYMBOL}));let approvalWeight=0;balance&&balance.stake&&(approvalWeight=balance.stake),balance&&balance.delegationsIn&&(approvalWeight=api.BigNumber(approvalWeight).plus(balance.delegationsIn).toFixed(GOVERNANCE_TOKEN_PRECISION)),manual&&(account.approvals-=1,account.approvalWeight=approvalWeight,account.lastApproveBlock=api.blockNumber,await api.db.update("accounts",account)),await updateWitnessRank(to,`-${approvalWeight}`),api.emit("witnessApprovalRemoved",{account:from,to:to,approvalWeight:approvalWeight})}};actions.approve=async payload=>{const{witness:witness}=payload,params=await api.db.findOne("params",{});if(api.assert(witness&&"string"==typeof witness&&witness.length>=3&&witness.length<=16,"invalid witness account")){const witnessRec=await api.db.findOne("witnesses",{account:witness});if(api.assert(witnessRec,"witness does not exist")){let acct=await api.db.findOne("accounts",{account:api.sender});if(null===acct&&(acct={account:api.sender,approvals:0,approvalWeight:{$numberDecimal:"0"},lastApproveBlock:api.blockNumber},acct=await api.db.insert("accounts",acct)),api.assert(acct.approvals<params.numberOfApprovalsPerAccount,`you can only approve ${params.numberOfApprovalsPerAccount} witnesses`)){let approval=await api.db.findOne("approvals",{from:api.sender,to:witness});if(api.assert(null===approval,"you already approved this witness")){approval={from:api.sender,to:witness},await api.db.insert("approvals",approval);const balance=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:GOVERNANCE_TOKEN_SYMBOL});let approvalWeight=0;balance&&balance.stake&&(approvalWeight=balance.stake),balance&&balance.delegationsIn&&(approvalWeight=api.BigNumber(approvalWeight).plus(balance.delegationsIn).toFixed(GOVERNANCE_TOKEN_PRECISION)),acct.approvals+=1,acct.approvalWeight=approvalWeight,acct.lastApproveBlock=api.blockNumber,await api.db.update("accounts",acct),await updateWitnessRank(witness,approvalWeight),api.emit("witnessApprovalAdded",{account:api.sender,to:witness,approvalWeight:approvalWeight})}}}}},actions.disapprove=async payload=>{const{witness:witness}=payload;if(api.assert(witness&&"string"==typeof witness&&witness.length>=3&&witness.length<=16,"invalid witness account")){const witnessRec=await api.db.findOne("witnesses",{account:witness});if(api.assert(witnessRec,"witness does not exist")){let acct=await api.db.findOne("accounts",{account:api.sender});if(null===acct&&(acct={account:api.sender,approvals:0,approvalWeight:{$numberDecimal:"0"},lastApproveBlock:api.blockNumber},await api.db.insert("accounts",acct)),api.assert(acct.approvals>0,"no approvals found")){const approval=await api.db.findOne("approvals",{from:acct.account,to:witness}),balance=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:GOVERNANCE_TOKEN_SYMBOL});await removeApproval(approval,acct,balance,!0)}}}};const expireAllUserApprovals=async acct=>{const approvals=await api.db.find("approvals",{from:acct.account}),balance=await api.db.findOneInTable("tokens","balances",{account:acct.account,symbol:GOVERNANCE_TOKEN_SYMBOL});for(let i=0;i<approvals.length;i+=1){const approval=approvals[i];await removeApproval(approval,acct,balance,!1)}let approvalWeight=0;balance&&balance.stake&&(approvalWeight=balance.stake),balance&&balance.delegationsIn&&(approvalWeight=api.BigNumber(approvalWeight).plus(balance.delegationsIn).toFixed(GOVERNANCE_TOKEN_PRECISION));const account=acct;account.approvals=0,account.approvalWeight=approvalWeight,await api.db.update("accounts",account),api.emit("witnessApprovalsExpired",{account:acct.account})},findAndExpireApprovals=async witnessApproveExpireBlocks=>{const accounts=await api.db.find("accounts",{lastApproveBlock:{$lt:api.blockNumber-witnessApproveExpireBlocks},approvals:{$gt:0}},10,0,[{index:"lastApproveBlock",descending:!1}]);for(let i=0;i<accounts.length;i+=1)await expireAllUserApprovals(accounts[i])},changeCurrentWitness=async()=>{const params=await api.db.findOne("params",{}),{currentWitness:currentWitness,totalEnabledApprovalWeight:totalEnabledApprovalWeight,lastWitnesses:lastWitnesses,lastBlockRound:lastBlockRound,round:round,maxRoundsMissedInARow:maxRoundsMissedInARow,maxRoundPropositionWaitingPeriod:maxRoundPropositionWaitingPeriod,lastVerifiedBlockNumber:lastVerifiedBlockNumber}=params;let witnessFound=!1;const random=api.random(),randomWeight=api.BigNumber(totalEnabledApprovalWeight).times(random).toFixed(GOVERNANCE_TOKEN_PRECISION,1);let offset=0,accWeight=0,witnesses=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}},enabled:!0},100,offset,[{index:"approvalWeight",descending:!0}]);const schedules=await api.db.find("schedules",{round:round}),currentWitnessSchedule=schedules.find((s=>s.witness===currentWitness)),previousRoundWitness=lastWitnesses.length>1?lastWitnesses[lastWitnesses.length-2]:"";do{for(let index=0;index<witnesses.length;index+=1){const witness=witnesses[index];if(accWeight=api.BigNumber(accWeight).plus(witness.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION),!0===witness.enabled&&witness.account!==previousRoundWitness&&void 0===schedules.find((s=>s.witness===witness.account))&&api.BigNumber(randomWeight).lte(accWeight)){let witnessToChange=currentWitness;if(currentWitnessSchedule.verifiedRound)for(let index=0;index<schedules.length;index+=1)schedules[index].verifiedRound||(witnessToChange=schedules[index].witness);api.debug(`changed witness from ${witnessToChange} to ${witness.account}`),api.emit("witnessChanged",{removed:witnessToChange,added:witness.account});const newWitnessOrder=[witness.account];for(let index=0;index<schedules.length;index+=1){const schedule=schedules[index];schedule.witness!==witnessToChange&&newWitnessOrder.push(schedule.witness),await api.db.remove("schedules",schedule)}let blockNumber=0===lastVerifiedBlockNumber?api.blockNumber:lastVerifiedBlockNumber+1;for(let i=0;i<newWitnessOrder.length;i+=1){const newSchedule={witness:newWitnessOrder[i],blockNumber:blockNumber,round:round};await api.db.insert("schedules",newSchedule),blockNumber+=1}if(params.currentWitness!==newWitnessOrder[newWitnessOrder.length-1]&&(params.currentWitness=newWitnessOrder[newWitnessOrder.length-1],params.lastWitnesses.push(newWitnessOrder[newWitnessOrder.length-1])),params.blockNumberWitnessChange=api.blockNumber+maxRoundPropositionWaitingPeriod,!currentWitnessSchedule.verifiedRound){const scheduledWitness=await api.db.findOne("witnesses",{account:currentWitness});scheduledWitness.missedRounds+=1,scheduledWitness.missedRoundsInARow+=1,api.emit("witnessMissedRound",{witness:scheduledWitness.account}),scheduledWitness.missedRoundsInARow>=maxRoundsMissedInARow&&(scheduledWitness.missedRoundsInARow=0,scheduledWitness.enabled&&(params.totalEnabledApprovalWeight=api.BigNumber(params.totalEnabledApprovalWeight).minus(scheduledWitness.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION)),scheduledWitness.enabled=!1,api.emit("witnessDisabledForMissingTooManyRoundsInARow",{witness:scheduledWitness.account})),await api.db.update("witnesses",scheduledWitness)}await api.db.update("params",params),witnessFound=!0;break}}!1===witnessFound&&(offset+=100,witnesses=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}}},100,offset,[{index:"approvalWeight",descending:!0}]))}while(witnesses.length>0&&!1===witnessFound);if(!1===witnessFound){api.debug("no backup witness was found, interchanging witnesses within the current schedule");for(let index=0;index<schedules.length-1;index+=1){const sched=schedules[index],newWitness=sched.witness;if(newWitness!==previousRoundWitness){api.debug(`changed current witness from ${currentWitness} to ${newWitness}`),schedule.witness=newWitness,await api.db.update("schedules",schedule),sched.witness=currentWitness,await api.db.update("schedules",sched),params.currentWitness=newWitness,params.lastWitnesses.push(newWitness),params.blockNumberWitnessChange=api.blockNumber+maxRoundPropositionWaitingPeriod;const scheduledWitness=await api.db.findOne("witnesses",{account:currentWitness});scheduledWitness.missedRounds+=1,scheduledWitness.missedRoundsInARow+=1,api.emit("witnessMissedRound",{witness:scheduledWitness.account}),scheduledWitness.missedRoundsInARow>=maxRoundsMissedInARow&&(scheduledWitness.missedRoundsInARow=0,scheduledWitness.enabled&&(params.totalEnabledApprovalWeight=api.BigNumber(params.totalEnabledApprovalWeight).minus(scheduledWitness.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION)),scheduledWitness.enabled=!1,api.emit("witnessDisabledForMissingTooManyRoundsInARow",{witness:scheduledWitness.account})),await api.db.update("params",params),await api.db.update("witnesses",scheduledWitness),api.emit("currentWitnessChanged",{});break}}}},manageWitnessesSchedule=async()=>{if("null"!==api.sender)return;const params=await api.db.findOne("params",{}),{numberOfApprovedWitnesses:numberOfApprovedWitnesses,totalEnabledApprovalWeight:totalEnabledApprovalWeight,lastVerifiedBlockNumber:lastVerifiedBlockNumber,blockNumberWitnessChange:blockNumberWitnessChange,lastBlockRound:lastBlockRound,numberOfTopWitnesses:numberOfTopWitnesses,numberOfWitnessSlots:numberOfWitnessSlots,maxRoundPropositionWaitingPeriod:maxRoundPropositionWaitingPeriod,witnessApproveExpireBlocks:witnessApproveExpireBlocks}=params;await findAndExpireApprovals(witnessApproveExpireBlocks);const currentBlock=lastVerifiedBlockNumber+1;let schedule=await api.db.findOne("schedules",{blockNumber:currentBlock});if(null===schedule){if(api.debug("calculating new schedule"),schedule=[],numberOfApprovedWitnesses>=numberOfWitnessSlots){const random=api.random();let randomWeight=null,offset=0,accWeight=0,witnesses=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}},enabled:!0},100,offset,[{index:"approvalWeight",descending:!0}]);do{for(let index=0;index<witnesses.length;index+=1){const witness=witnesses[index];schedule.length>=numberOfTopWitnesses&&null===randomWeight&&(randomWeight=api.BigNumber(accWeight).plus(GOVERNANCE_TOKEN_MIN_VALUE).plus(api.BigNumber(totalEnabledApprovalWeight).minus(accWeight).times(random).toFixed(GOVERNANCE_TOKEN_PRECISION,1)).toFixed(GOVERNANCE_TOKEN_PRECISION)),accWeight=api.BigNumber(accWeight).plus(witness.approvalWeight.$numberDecimal).toFixed(GOVERNANCE_TOKEN_PRECISION),!0===witness.enabled&&(schedule.length<numberOfTopWitnesses||api.BigNumber(randomWeight).lte(accWeight))&&schedule.push({witness:witness.account,blockNumber:null}),schedule.length>=numberOfWitnessSlots&&(index=witnesses.length)}schedule.length<numberOfWitnessSlots&&(offset+=100,witnesses=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}}},100,offset,[{index:"approvalWeight",descending:!0}]))}while(witnesses.length>0&&schedule.length<numberOfWitnessSlots)}if(schedule.length===numberOfWitnessSlots){let j,x;for(let i=schedule.length-1;i>0;i-=1){const random=api.random();j=Math.floor(random*(i+1)),x=schedule[i],schedule[i]=schedule[j],schedule[j]=x}let lastWitnesses=params.lastWitnesses;const previousRoundWitness=lastWitnesses.length>0?lastWitnesses[lastWitnesses.length-1]:"";lastWitnesses.length>=numberOfWitnessSlots&&(lastWitnesses=[]);const lastWitness=schedule[schedule.length-1].witness;if(lastWitnesses.includes(lastWitness)||previousRoundWitness===lastWitness)for(let i=0;i<schedule.length;i+=1)if(!lastWitnesses.includes(schedule[i].witness)&&schedule[i].witness!==previousRoundWitness){const thisWitness=schedule[i].witness;schedule[i].witness=lastWitness,schedule[schedule.length-1].witness=thisWitness;break}if(schedule[0].witness===previousRoundWitness){const firstWitness=schedule[0].witness,secondWitness=schedule[1].witness;schedule[0].witness=secondWitness,schedule[1].witness=firstWitness}let blockNumber=0===lastVerifiedBlockNumber?api.blockNumber:lastVerifiedBlockNumber+1;params.round+=1;for(let i=0;i<schedule.length;i+=1)schedule[i].blockNumber=blockNumber,schedule[i].round=params.round,api.debug(`scheduled witness ${schedule[i].witness} for block ${blockNumber} (round ${params.round})`),await api.db.insert("schedules",schedule[i]),blockNumber+=1;0===lastVerifiedBlockNumber&&(params.lastVerifiedBlockNumber=api.blockNumber-1);const lastWitnessRoundSchedule=schedule[schedule.length-1];params.lastBlockRound=lastWitnessRoundSchedule.blockNumber,params.currentWitness=lastWitnessRoundSchedule.witness,lastWitnesses.push(lastWitnessRoundSchedule.witness),params.lastWitnesses=lastWitnesses,params.blockNumberWitnessChange=api.blockNumber+maxRoundPropositionWaitingPeriod,await api.db.update("params",params),api.emit("newSchedule",{})}}else api.blockNumber>=blockNumberWitnessChange&&(api.blockNumber>lastBlockRound?await changeCurrentWitness():(params.blockNumberWitnessChange=api.blockNumber+maxRoundPropositionWaitingPeriod,await api.db.update("params",params),api.emit("awaitingRoundEnd",{})))};actions.proposeRound=async payload=>{const{roundHash:roundHash,isSignedWithActiveKey:isSignedWithActiveKey,signatures:signatures}=payload,params=await api.db.findOne("params",{}),{lastVerifiedBlockNumber:lastVerifiedBlockNumber,round:round,lastBlockRound:lastBlockRound,currentWitness:currentWitness}=params,schedules=await api.db.find("schedules",{round:round},1e3,0,[{index:"_id",descending:!1}]);if(!api.assert(schedules&&schedules.length>0,"invalid round"))return;const numberOfWitnessSlots=schedules.length,{witnessSignaturesRequired:witnessSignaturesRequired}=params;if(!api.assert(isSignedWithActiveKey,"you must use a transaction signed with your active key"))return;if(!api.assert(roundHash&&"string"==typeof roundHash&&64===roundHash.length,"invalid round hash"))return;if(!api.assert(Array.isArray(signatures)&&signatures.length<=numberOfWitnessSlots,"invalid signatures"))return;let currentBlock=lastVerifiedBlockNumber+1,calculatedRoundHash="";if(!api.assert(api.sender===currentWitness,"must be current witness"))return;for(;currentBlock<=lastBlockRound;){const block=await api.db.getBlockInfo(currentBlock);if(null===block){calculatedRoundHash="";break}calculatedRoundHash=api.SHA256(`${calculatedRoundHash}${block.hash}`),currentBlock+=1}if(!api.assert(""!==calculatedRoundHash&&calculatedRoundHash===roundHash,"round hash mismatch"))return;let signaturesChecked=0;const verifiedBlockInformation=[],currentWitnessInfo=await api.db.findOne("witnesses",{account:currentWitness}),currentWitnessSignature=signatures.find((s=>s[0]===currentWitness));for(let index=0;index<schedules.length;index+=1){const scheduledWitness=schedules[index],witness=await api.db.findOne("witnesses",{account:scheduledWitness.witness});if(null!==witness){const signature=signatures.find((s=>s[0]===witness.account));signature&&api.checkSignature(calculatedRoundHash,signature[1],witness.signingKey,!0)&&(api.debug(`witness ${witness.account} signed round ${round}`),signaturesChecked+=1,scheduledWitness.verifiedRound=!0,await api.db.update("schedules",scheduledWitness)),verifiedBlockInformation.push({blockNumber:scheduledWitness.blockNumber,witness:currentWitness,signingKey:currentWitnessInfo.signingKey,roundSignature:currentWitnessSignature[1],round:round,roundHash:roundHash})}}if(!api.assert(signaturesChecked>=witnessSignaturesRequired,"valid round hash but not enough signatures"))return;for(let index=0;index<verifiedBlockInformation.length;index+=1)await api.verifyBlock(verifiedBlockInformation[index]);for(let index=0;index<schedules.length;index+=1){const schedule=schedules[index];await api.db.remove("schedules",schedule)}const contractBalance=await api.db.findOneInTable("tokens","contractsBalances",{account:"witnesses",symbol:UTILITY_TOKEN_SYMBOL});if(contractBalance&&api.BigNumber(contractBalance.balance).gte("0.39954306")){const rewardAmount=api.BigNumber("0.01902586").multipliedBy(numberOfWitnessSlots).toFixed(UTILITY_TOKEN_PRECISION);await api.executeSmartContract("tokens","stakeFromContract",{to:currentWitness,symbol:UTILITY_TOKEN_SYMBOL,quantity:rewardAmount})}params.currentWitness=null,params.lastVerifiedBlockNumber=lastBlockRound,await api.db.update("params",params);const witness=await api.db.findOne("witnesses",{account:currentWitness});witness.missedRoundsInARow=0,witness.lastRoundVerified=round,witness.lastBlockVerified=lastBlockRound,witness.verifiedRounds+=1,await api.db.update("witnesses",witness),await manageWitnessesSchedule()},actions.scheduleWitnesses=async()=>{"null"===api.sender&&await manageWitnessesSchedule()};"}}}