{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
soulbound
2 years ago in #hive-engine by rishi556.engine (46)
$0.00
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,soulBound:!!l,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,soulBound:!!l,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,lockNfts:f,properties:p,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,soulBound:!!l,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"update","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,soulBound:!!l,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"deploy","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,soulBound:!!l,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
{"id":"ssc-testnet-hive","json":{"contractName":"contract","contractAction":"deploy","contractPayload":{"name":"nft","params":"","code":"const CONTRACT_NAME="nft",UTILITY_TOKEN_SYMBOL="BEE",MAX_NUM_AUTHORIZED_ISSUERS=10,MAX_NUM_LOCKED_TOKEN_TYPES=10,MAX_SYMBOL_LENGTH=10,MAX_DATA_PROPERTY_LENGTH=100,MAX_NUM_NFTS_ISSUABLE=10,MAX_NUM_NFTS_EDITABLE=50,MAX_NUM_NFTS_OPERABLE=50,MAX_NUM_CONTAINER_NFTS_OPERABLE=1,RESERVED_SYMBOLS={CELL:"beggars",QUST:"simplegame",TESTERA:"aggroed",SQRL:"stuffbyspencer",CRAFT:"immanuel94",MUSIC:"atomcollector",CGULL:"cgull",NFT:"cadawg",RARE:"beggars",LIC:"lictoken",MEMBER:"membertoken",COFFEE:"c0ff33a",ART:"byo",ROCK:"beggars",CRITTER:"cryptomancer",CITY:"gerber",MONSTERS:"simplegame",SETS:"lootkit.games",ANIME:"animetoken",PHOTOFT:"wwwiebe",BEER:"detlev",SPIR:"spinvest",IFG:"lion200",GUILDS:"simplegame",FCARD:"lion200",PXL:"pixelnft",COW:"stuffbyspencer",LOOOT:"stuffbyspencer",API:"steemcityapi",SPORTSMOM:"sportstester",SWT:"satren",STAR:"atomcollector"};actions.createSSC=async()=>{let e=await api.db.tableExists("nfts");if(!1===e){await api.db.createTable("nfts",["symbol"]),await api.db.createTable("params"),await api.db.createTable("pendingUndelegations",["symbol","completeTimestamp"]);let t={};t.nftCreationFee="100",t.nftIssuanceFee={BEE:"0.001",PAL:"0.001"},t.dataPropertyCreationFee="100",t.enableDelegationFee="1000",await api.db.insert("params",t)}},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{nftCreationFee:t,nftIssuanceFee:s,dataPropertyCreationFee:a,enableDelegationFee:n}=e,i=await api.db.findOne("params",{});t&&"string"==typeof t&&!api.BigNumber(t).isNaN()&&api.BigNumber(t).gte(0)&&(i.nftCreationFee=t),s&&"object"==typeof s&&(i.nftIssuanceFee=s),a&&"string"==typeof a&&!api.BigNumber(a).isNaN()&&api.BigNumber(a).gte(0)&&(i.dataPropertyCreationFee=a),n&&"string"==typeof n&&!api.BigNumber(n).isNaN()&&api.BigNumber(n).gte(0)&&(i.enableDelegationFee=n),await api.db.update("params",i)};const isTokenTransferVerified=(e,t,s,a,n,i)=>void 0===e.errors&&!!e.events&&void 0!==e.events.find(e=>"tokens"===e.contract&&e.event===i&&e.data.from===t&&e.data.to===s&&e.data.quantity===n&&e.data.symbol===a),calculateBalance=(e,t,s,a)=>a?api.BigNumber(e).plus(t).toFixed(s):api.BigNumber(e).minus(t).toFixed(s),countDecimals=e=>api.BigNumber(e).dp(),containsDuplicates=e=>new Set(e).size!==e.length,isValidHiveAccountLength=e=>e.length>=3&&e.length<=16,isValidContractLength=e=>e.length>=3&&e.length<=50,isValidAccountsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidHiveAccountLength(e)||(t=!1)}),t},isValidContractsArray=e=>{let t=!0;return e.forEach(e=>{"string"==typeof e&&isValidContractLength(e)||(t=!1)}),t},isValidDataProperties=(e,t,s,a)=>{let n=Object.keys(a).length,i=Object.keys(s.properties).length;if(!api.assert(n<=i,"cannot set more data properties than NFT has"))return!1;for(let[r,o]of Object.entries(a)){let l=!1;if(api.assert(r&&"string"==typeof r&&api.validator.isAlphanumeric(r)&&r.length>0&&r.length<=25,"invalid data property name: letters & numbers only, max length of 25")&&api.assert(r in s.properties,"data property must exist")){let d=s.properties[r];api.assert(null!=o&&(typeof o===d.type||"number"===d.type&&"string"==typeof o&&!api.BigNumber(o).isNaN()),`data property type mismatch: expected ${d.type} but got ${typeof o} for property ${r}`)&&api.assert("string"!=typeof o||o.length<=100,"string property max length is 100 characters")&&api.assert("contract"===t&&d.authorizedEditingContracts.includes(e)||"user"===t&&d.authorizedEditingAccounts.includes(e),"not allowed to set data properties")&&(l=!0,"number"===d.type&&"string"==typeof o&&(a[r]=api.BigNumber(o).toNumber()))}if(!l)return!1}return!0},isValidDataPropertiesArray=(e,t,s,a)=>{try{for(let n=0;n<a.length;n+=1){let i=!1,{id:r,properties:o}=a[n];if(api.assert(r&&"string"==typeof r&&!api.BigNumber(r).isNaN()&&api.BigNumber(r).gt(0)&&o&&"object"==typeof o,"invalid data properties")&&isValidDataProperties(e,t,s,o)&&(i=!0),!i)return!1}}catch(l){return!1}return!0},isValidNftIdArray=e=>{try{let t=0;for(let s=0;s<e.length;s+=1){let a=!1,{symbol:n,ids:i}=e[s];if(api.assert(n&&"string"==typeof n&&api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10&&i&&"object"==typeof i&&Array.isArray(i),"invalid nft list")&&(t+=i.length,api.assert(t<=50,"cannot operate on more than 50 NFT instances at once"))){for(let r=0;r<i.length;r+=1){let o=i[r];if(!api.assert(o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0),"invalid nft list"))return!1}a=!0}if(!a)return!1}}catch(l){return!1}return!0},isValidTokenBasket=async(e,t,s,a,n)=>{try{let i=Object.keys(e).length;if(i>10)return!1;for(let[r,o]of Object.entries(e)){let l=!1;if("string"==typeof r&&api.validator.isAlpha(r)&&api.validator.isUppercase(r)&&r.length>0&&r.length<=10){let d=await api.db.findOneInTable("tokens","tokens",{symbol:r});if(d&&o&&"string"==typeof o&&!api.BigNumber(o).isNaN()&&api.BigNumber(o).gt(0)&&countDecimals(o)<=d.precision){let u=r===a?calculateBalance(o,n,d.precision,!0):o,c=await api.db.findOneInTable("tokens",t,{account:s,symbol:r});c&&api.BigNumber(c.balance).gte(u)&&(l=!0)}}if(!l)return!1}}catch(f){return!1}return!0},transferAndVerifyNfts=async(e,t,s,a,n,i,r)=>{let o={success:[],fail:[]},l="user"===t?"u":"c",d="user"===a?"u":"c";await actions.transfer({fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r});let u=api.logs(),c={},f={};if(u.events)for(let p=0;p<u.events.length;p+=1){let g=u.events[p];if(g.contract&&g.event&&g.data&&"nft"===g.contract&&"transfer"===g.event&&g.data.from===e&&g.data.fromType===l&&g.data.to===s&&g.data.toType===d){let y=g.data.symbol+"-"+g.data.id;c[y]=1}}for(let m=0;m<n.length;m+=1){let{symbol:$,ids:h}=n[m],b=[],v=[];for(let w=0;w<h.length;w+=1){let N=$+"-"+h[w];N in f||(N in c?b.push(h[w].toString()):v.push(h[w].toString()),f[N]=1)}b.length>0&&o.success.push({symbol:$,ids:b}),v.length>0&&o.fail.push({symbol:$,ids:v})}return o};actions.updateUrl=async e=>{let{url:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(t.length<=255,"invalid url: max length of 255")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.parse(a.metadata);api.assert(n&&n.url,"an error occured when trying to update the url")&&(n.url=t,a.metadata=JSON.stringify(n),await api.db.update("nfts",a))}catch(i){}}},actions.updateMetadata=async e=>{let{metadata:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"object"==typeof t,"invalid params")){let a=await api.db.findOne("nfts",{symbol:s});if(a&&api.assert(a.issuer===api.sender,"must be the issuer"))try{let n=JSON.stringify(t);api.assert(n.length<=1e3,"invalid metadata: max length of 1000")&&(a.metadata=n,await api.db.update("nfts",a))}catch(i){}}},actions.updateName=async e=>{let{name:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.name=t,await api.db.update("nfts",a))}},actions.updateOrgName=async e=>{let{orgName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.orgName=t,await api.db.update("nfts",a))}},actions.updateProductName=async e=>{let{productName:t,symbol:s}=e;if(api.assert(s&&"string"==typeof s&&t&&"string"==typeof t,"invalid params")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid product name: letters, numbers, whitespaces only, max length of 50")){let a=await api.db.findOne("nfts",{symbol:s});a&&api.assert(a.issuer===api.sender,"must be the issuer")&&(a.productName=t,await api.db.update("nfts",a))}},actions.addAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim().toLowerCase(),s=!1;for(let a=0;a<i.authorizedIssuingAccounts.length;a+=1)if(t===i.authorizedIssuingAccounts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same account twice")&&api.assert(i.authorizedIssuingAccounts.length+r.length<=10,"cannot have more than 10 authorized issuing accounts")){let o=i.authorizedIssuingAccounts.concat(r);i.authorizedIssuingAccounts=o,await api.db.update("nfts",i)}}}}},actions.addAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot have more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i){let r=[];if(t.forEach(e=>{let t=e.trim(),s=!1;for(let a=0;a<i.authorizedIssuingContracts.length;a+=1)if(t===i.authorizedIssuingContracts[a]){s=!0;break}s||r.push(t)}),api.assert(i.issuer===api.sender,"must be the issuer")&&api.assert(!containsDuplicates(r),"cannot add the same contract twice")&&api.assert(i.authorizedIssuingContracts.length+r.length<=10,"cannot have more than 10 authorized issuing contracts")){let o=i.authorizedIssuingContracts.concat(r);i.authorizedIssuingContracts=o,await api.db.update("nfts",i)}}}}},actions.removeAuthorizedIssuingAccounts=async e=>{let{accounts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing accounts")){let n=isValidAccountsArray(t);if(api.assert(n,"invalid account list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingAccounts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim().toLowerCase();if(e===a)return!1}return!0});i.authorizedIssuingAccounts=r,await api.db.update("nfts",i)}}}},actions.removeAuthorizedIssuingContracts=async e=>{let{contracts:t,symbol:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(s&&"string"==typeof s&&t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot remove more than 10 authorized issuing contracts")){let n=isValidContractsArray(t);if(api.assert(n,"invalid contract list")){let i=await api.db.findOne("nfts",{symbol:s});if(i&&api.assert(i.issuer===api.sender,"must be the issuer")){let r=i.authorizedIssuingContracts.filter(e=>{for(let s=0;s<t.length;s+=1){let a=t[s].trim();if(e===a)return!1}return!0});i.authorizedIssuingContracts=r,await api.db.update("nfts",i)}}}},actions.transferOwnership=async e=>{let{symbol:t,to:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n&&api.assert(n.issuer===api.sender,"must be the issuer")){let i=s.trim().toLowerCase();api.assert(isValidHiveAccountLength(i),"invalid to")&&(n.issuer=i,await api.db.update("nfts",n))}}},actions.enableDelegation=async e=>{let{symbol:t,undelegationCooldown:s,isSignedWithActiveKey:a}=e,n=await api.db.findOne("params",{}),{enableDelegationFee:i}=n,r=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),o=!!api.BigNumber(i).lte(0)||r&&api.BigNumber(r.balance).gte(i);if(api.assert(o,"you must have enough tokens to cover fees")&&api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t,"invalid symbol")&&api.assert(s&&Number.isInteger(s)&&s>0&&s<=18250,"undelegationCooldown must be an integer between 1 and 18250")){let l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")&&api.assert(l.issuer===api.sender,"must be the issuer")&&api.assert(void 0===l.delegationEnabled||!1===l.delegationEnabled,"delegation already enabled")){if(api.BigNumber(i).gt(0)){let d=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:i,isSignedWithActiveKey:a});if(!isTokenTransferVerified(d,api.sender,"null","BEE",i,"transfer"))return!1}return l.delegationEnabled=!0,l.undelegationCooldown=s,await api.db.update("nfts",l),!0}}return!1},actions.updatePropertyDefinition=async e=>{let{symbol:t,name:s,newName:a,type:n,isReadOnly:i,isSignedWithActiveKey:r}=e;if(api.assert(!0===r,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||"string"==typeof a&&api.validator.isAlphanumeric(a)&&a.length>0&&a.length<=25,"invalid new name: letters & numbers only, max length of 25")&&api.assert(void 0===n||"string"==typeof n&&("number"===n||"string"===n||"boolean"===n),"invalid type: must be number, string, or boolean")&&api.assert(void 0===i||"boolean"==typeof i,"invalid isReadOnly: must be true or false")){let o=await api.db.findOne("nfts",{symbol:t});if(o&&api.assert(0===o.supply,"cannot change data property definition; tokens already issued")&&api.assert(s in o.properties,"property must exist")&&api.assert(o.issuer===api.sender,"must be the issuer")){if(void 0!==a&&(void 0!==o.groupBy&&o.groupBy.length>0&&!api.assert(!o.groupBy.includes(s),"cannot change data property name; property is part of groupBy")||!api.assert(a!==s,"new name must be different from old name")||!api.assert(!(a in o.properties),"there is already a data property with the given new name")))return!1;let l=!1,d=o.properties[s].type,u=o.properties[s].isReadOnly;return void 0!==n&&n!==d&&(o.properties[s].type=n,l=!0),void 0!==i&&i!==u&&(o.properties[s].isReadOnly=i,l=!0),void 0!==a&&a!==s&&(o.properties[a]=o.properties[s],delete o.properties[s],l=!0),l&&(await api.db.update("nfts",o),api.emit("updatePropertyDefinition",{symbol:t,originalName:s,originalType:d,originalIsReadOnly:u,newName:a,newType:n,newIsReadOnly:i})),!0}}return!1},actions.addProperty=async e=>{let{symbol:t,name:s,type:a,isReadOnly:n,authorizedEditingAccounts:i,authorizedEditingContracts:r,isSignedWithActiveKey:o}=e,l=await api.db.findOne("params",{}),{dataPropertyCreationFee:d}=l;if(api.assert(!0===o,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===n||"boolean"==typeof n)&&(void 0===i||i&&"object"==typeof i&&Array.isArray(i))&&(void 0===r||r&&"object"==typeof r&&Array.isArray(r))&&a&&"string"==typeof a,"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert("number"===a||"string"===a||"boolean"===a,"invalid type: must be number, string, or boolean")){let u=await api.db.findOne("nfts",{symbol:t});if(u&&api.assert(!(s in u.properties),"cannot add the same property twice")&&api.assert(u.issuer===api.sender,"must be the issuer")){let c=Object.keys(u.properties).length;if(c>=3){let f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(d).lte(0)||f&&api.BigNumber(f.balance).gte(d);if(!api.assert(p,"you must have enough tokens to cover the creation fees"))return!1;if(api.BigNumber(d).gt(0)){let g=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:d,isSignedWithActiveKey:o});if(!isTokenTransferVerified(g,api.sender,"null","BEE",d,"transfer"))return!1}}let y=void 0!==n&&n,m=void 0===i?[api.sender]:[],$={type:a,isReadOnly:y,authorizedEditingAccounts:m,authorizedEditingContracts:[]};return u.properties[s]=$,await api.db.update("nfts",u),(i||r)&&await actions.setPropertyPermissions({symbol:t,name:s,accounts:i,contracts:r,isSignedWithActiveKey:o}),!0}}return!1},actions.setPropertyPermissions=async e=>{let{symbol:t,name:s,accounts:a,contracts:n,isSignedWithActiveKey:i}=e;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"string"==typeof s&&(void 0===a||a&&"object"==typeof a&&Array.isArray(a))&&(void 0===n||n&&"object"==typeof n&&Array.isArray(n)),"invalid params")&&api.assert(api.validator.isAlphanumeric(s)&&s.length>0&&s.length<=25,"invalid name: letters & numbers only, max length of 25")&&api.assert(void 0===a||a.length<=10,"cannot have more than 10 authorized accounts")&&api.assert(void 0===n||n.length<=10,"cannot have more than 10 authorized contracts")&&api.assert(void 0===a||isValidAccountsArray(a),"invalid account list")&&api.assert(void 0===n||isValidContractsArray(n),"invalid contract list")){let r=await api.db.findOne("nfts",{symbol:t});if(r&&api.assert(s in r.properties,"property must exist")&&api.assert(r.issuer===api.sender,"must be the issuer")){let o=[],l=[];if(a&&(o=a.map(e=>e.trim().toLowerCase())),n&&(l=n.map(e=>e.trim())),api.assert(void 0===a||!containsDuplicates(o),"cannot add the same account twice")&&api.assert(void 0===n||!containsDuplicates(l),"cannot add the same contract twice")){let d=!1;a&&(r.properties[s].authorizedEditingAccounts=o,d=!0),n&&(r.properties[s].authorizedEditingContracts=l,d=!0),d&&await api.db.update("nfts",r)}}}},actions.setGroupBy=async e=>{let{symbol:t,properties:s,isSignedWithActiveKey:a}=e;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")){let n=await api.db.findOne("nfts",{symbol:t});if(n){let i=Object.keys(n.properties).length;if(api.assert(n.issuer===api.sender,"must be the issuer")&&api.assert(void 0===n.groupBy||0===n.groupBy.length,"list is already set")&&api.assert(s.length<=i,"cannot set more data properties than NFT has")&&api.assert(!containsDuplicates(s),"list cannot contain duplicates")){for(let r=0;r<s.length;r+=1){let o=s[r];if(!api.assert(o&&"string"==typeof o&&o in n.properties,"data property must exist"))return!1}return n.groupBy=s,await api.db.update("nfts",n),!0}}}return!1},actions.setProperties=async e=>{let{symbol:t,fromType:s,nfts:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===s?"user":s;if(api.assert(a&&"object"==typeof a&&Array.isArray(a)&&r&&"string"==typeof r&&i.includes(r)&&t&&"string"==typeof t&&(n||void 0===n&&"user"===r),"invalid params")&&api.assert(a.length<=50,"cannot set properties on more than 50 NFT instances at once")){let o="user"===r?api.sender:n.name,l=await api.db.findOne("nfts",{symbol:t});if(api.assert(null!==l,"symbol does not exist")){if(!isValidDataPropertiesArray(o,r,l,a))return!1;let d=t+"instances";for(let u=0;u<a.length;u+=1){let{id:c,properties:f}=a[u];if(0===Object.keys(f).length)continue;let p=await api.db.findOne(d,{_id:api.BigNumber(c).toNumber()});if(api.assert(null!==p,"nft instance does not exist")){let g=!1;for(let[y,m]of Object.entries(f)){let $=l.properties[y],h=null;$.isReadOnly?api.assert(!(y in p.properties),"cannot edit read-only properties")&&(p.properties[y]=m,g=!0):(h=p.properties[y],p.properties[y]=m,g=!0),g&&h!==m&&await api.executeSmartContract("mining","handleNftSetProperty",{symbol:t,nft:p,propertyName:y,oldValue:h})}g&&await api.db.update(d,p)}}return!0}}return!1},actions.burn=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=0,d=0,u=!1;for(let c=0;c<s.length;c+=1){let{symbol:f,ids:p}=s[c],g=await api.db.findOne("nfts",{symbol:f});if(g){let y=f+"instances";for(let m=0;m<p.length;m+=1){let $=p[m],h=await api.db.findOne(y,{_id:api.BigNumber($).toNumber()});if(h){let b=!0;if(h.lockedNfts&&h.lockedNfts.length>0?(0===d&&(u=!0),((l+=1)>1||!u)&&(b=!1)):u&&(b=!1),d+=1,h.account===o&&("u"===h.ownedBy&&"user"===r||"c"===h.ownedBy&&"contract"===r)&&void 0===h.delegatedTo&&b){let v={},w=!0;for(let[N,_]of Object.entries(h.lockedTokens)){let A=await api.transferTokens(o,N,_,r);isTokenTransferVerified(A,"nft",o,N,_,"transferFromContract")||(v[N]=_,w=!1)}api.assert(w,`unable to release locked tokens in: ${f}, id ${$}`);let T=h.lockedNfts&&h.lockedNfts.length>0?h.lockedNfts:[];if(w&&h.lockedNfts&&h.lockedNfts.length>0){let k=await transferAndVerifyNfts("nft","contract",o,r,h.lockedNfts,a,{name:"nft"});h.lockedNfts=k.fail,h.lockedNfts.length>0&&(w=!1),api.assert(w,`unable to release locked NFT instances in: ${f}, id ${$}`)}let B=h.ownedBy,E=h.lockedTokens;h.lockedTokens=v,w&&(h.previousAccount=h.account,h.previousOwnedBy=h.ownedBy,h.account="null",h.ownedBy="u",g.circulatingSupply-=1),await api.db.update(y,h),w&&api.emit("burn",{account:o,ownedBy:B,unlockedTokens:E,unlockedNfts:T,symbol:f,id:$})}}}await api.db.update("nfts",g)}}}},actions.transfer=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot transfer to self")&&api.assert(!("user"===l&&"null"===u),"cannot transfer to null; use burn action instead"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo&&!v.soulBound){let w=v.ownedBy,N="user"===l?"u":"c";v.previousAccount=v.account,v.previousOwnedBy=v.ownedBy,v.account=u,v.ownedBy=N,await api.db.update($,v),api.emit("transfer",{from:f,fromType:w,to:u,toType:N,symbol:g,id:b})}}}}}},actions.delegate=async e=>{let{fromType:t,to:s,toType:a,nfts:n,isSignedWithActiveKey:i,callingContractInfo:r}=e,o=["user","contract"],l=void 0===a?"user":a,d=void 0===t?"user":t;if(api.assert(!0===i,"you must use a custom_json signed with your active key")&&api.assert(d&&"string"==typeof d&&o.includes(d)&&s&&"string"==typeof s&&l&&"string"==typeof l&&o.includes(l)&&(r||void 0===r&&"user"===d)&&n&&"object"==typeof n&&Array.isArray(n),"invalid params")&&isValidNftIdArray(n)){let u="user"===l?s.trim().toLowerCase():s.trim(),c="user"===l?isValidHiveAccountLength(u):isValidContractLength(u),f="user"===d?api.sender:r.name;if(api.assert(c,"invalid to")&&api.assert(!(l===d&&u===f),"cannot delegate to self")&&api.assert(!("user"===l&&"null"===u),"cannot delegate to null"))for(let p=0;p<n.length;p+=1){let{symbol:g,ids:y}=n[p],m=await api.db.findOne("nfts",{symbol:g});if(m&&api.assert(!0===m.delegationEnabled,`delegation not enabled for ${g}`)){let $=g+"instances";for(let h=0;h<y.length;h+=1){let b=y[h],v=await api.db.findOne($,{_id:api.BigNumber(b).toNumber()});if(v&&v.account===f&&("u"===v.ownedBy&&"user"===d||"c"===v.ownedBy&&"contract"===d)&&void 0===v.delegatedTo){let w="user"===l?"u":"c",N={account:u,ownedBy:w};v.delegatedTo=N,await api.db.update($,v),api.emit("delegate",{from:f,fromType:v.ownedBy,to:u,toType:w,symbol:g,id:b}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:g,nft:v,add:!0})}}}}}},actions.undelegate=async e=>{let{fromType:t,nfts:s,isSignedWithActiveKey:a,callingContractInfo:n}=e,i=["user","contract"],r=void 0===t?"user":t;if(api.assert(!0===a,"you must use a custom_json signed with your active key")&&api.assert(r&&"string"==typeof r&&i.includes(r)&&(n||void 0===n&&"user"===r)&&s&&"object"==typeof s&&Array.isArray(s),"invalid params")&&isValidNftIdArray(s)){let o="user"===r?api.sender:n.name,l=new Date(`${api.hiveBlockTimestamp}.000Z`);for(let d=0;d<s.length;d+=1){let{symbol:u,ids:c}=s[d],f=await api.db.findOne("nfts",{symbol:u});if(f&&api.assert(!0===f.delegationEnabled,`delegation not enabled for ${u}`)){let p=864e5*f.undelegationCooldown,g=l.getTime()+p,y=u+"instances",m={symbol:u,ids:[],completeTimestamp:g};for(let $=0;$<c.length;$+=1){let h=c[$],b=await api.db.findOne(y,{_id:api.BigNumber(h).toNumber()});b&&b.account===o&&("u"===b.ownedBy&&"user"===r||"c"===b.ownedBy&&"contract"===r)&&b.delegatedTo&&void 0===b.delegatedTo.undelegateAt&&(b.delegatedTo.undelegateAt=g,m.ids.push(b._id),await api.db.update(y,b),api.emit("undelegateStart",{from:b.delegatedTo.account,fromType:b.delegatedTo.ownedBy,symbol:u,id:h}),await api.executeSmartContract("mining","handleNftDelegationChange",{symbol:u,nft:b,add:!1}))}m.ids.length>0&&await api.db.insert("pendingUndelegations",m)}}}};const processUndelegation=async e=>{let{symbol:t,ids:s}=e,a=t+"instances",n=await api.db.find(a,{_id:{$in:s}},50,0,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1)delete n[i].delegatedTo,await api.db.update(a,n[i],{delegatedTo:""});await api.db.remove("pendingUndelegations",e),api.emit("undelegateDone",{symbol:t,ids:s})};actions.checkPendingUndelegations=async()=>{if(api.assert("null"===api.sender,"not authorized")){let e=new Date(`${api.hiveBlockTimestamp}.000Z`),t=e.getTime(),s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}}),a=s.length;for(;a>0;){for(let n=0;n<a;n+=1){let i=s[n];await processUndelegation(i)}a=(s=await api.db.find("pendingUndelegations",{completeTimestamp:{$lte:t}})).length}}},actions.create=async e=>{let{name:t,orgName:s,productName:a,symbol:n,url:i,maxSupply:r,authorizedIssuingAccounts:o,authorizedIssuingContracts:l,isSignedWithActiveKey:d}=e,u=await api.db.findOne("params",{}),{nftCreationFee:c}=u,f=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:"BEE"}),p=!!api.BigNumber(c).lte(0)||f&&api.BigNumber(f.balance).gte(c);if(api.assert(p,"you must have enough tokens to cover the creation fees")&&api.assert(!0===d,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&n&&"string"==typeof n&&(void 0===i||i&&"string"==typeof i)&&(void 0===s||s&&"string"==typeof s)&&(void 0===a||a&&"string"==typeof a)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"object"==typeof l&&Array.isArray(l))&&(void 0===r||r&&"string"==typeof r&&!api.BigNumber(r).isNaN()),"invalid params")&&api.assert(api.validator.isAlpha(n)&&api.validator.isUppercase(n)&&n.length>0&&n.length<=10,"invalid symbol: uppercase letters only, max length of 10")&&api.assert(void 0===RESERVED_SYMBOLS[n]||api.sender===RESERVED_SYMBOLS[n],"cannot use this symbol")&&api.assert(api.validator.isAlphanumeric(api.validator.blacklist(t," "))&&t.length>0&&t.length<=50,"invalid name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===s||api.validator.isAlphanumeric(api.validator.blacklist(s," "))&&s.length>0&&s.length<=50,"invalid org name: letters, numbers, whitespaces only, max length of 50")&&api.assert(void 0===a||api.validator.isAlphanumeric(api.validator.blacklist(a," "))&&a.length>0&&a.length<=50,"invalid product 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(void 0===r||api.BigNumber(r).gt(0),"maxSupply must be positive")&&api.assert(void 0===r||api.BigNumber(r).lte(Number.MAX_SAFE_INTEGER),`maxSupply must be lower than ${Number.MAX_SAFE_INTEGER}`)){let g=await api.db.findOne("nfts",{symbol:n});if(api.assert(null===g,"symbol already exists")){if(api.BigNumber(c).gt(0)){let y=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:"BEE",quantity:c,isSignedWithActiveKey:d});if(!isTokenTransferVerified(y,api.sender,"null","BEE",c,"transfer"))return!1}let m=void 0===r?0:api.BigNumber(r).integerValue(api.BigNumber.ROUND_DOWN).toNumber(),$=void 0===s?"":s,h=void 0===a?"":a,b=void 0===i?"":i,v={url:b};v=JSON.stringify(v);let w=void 0===o?[api.sender]:[],N={issuer:api.sender,symbol:n,name:t,orgName:$,productName:h,metadata:v,maxSupply:m,supply:0,circulatingSupply:0,delegationEnabled:!1,undelegationCooldown:0,authorizedIssuingAccounts:w,authorizedIssuingContracts:[],properties:{},groupBy:[]},_=n+"instances",A=await api.db.tableExists(_);return!1===A&&await api.db.createTable(_,["account","ownedBy"]),await api.db.insert("nfts",N),void 0!==o&&await actions.addAuthorizedIssuingAccounts({accounts:o,symbol:n,isSignedWithActiveKey:d}),void 0!==l&&await actions.addAuthorizedIssuingContracts({contracts:l,symbol:n,isSignedWithActiveKey:d}),!0}}return!1},actions.issue=async e=>{let{symbol:t,fromType:s,to:a,toType:n,feeSymbol:i,lockTokens:r,lockNfts:o,soulBound:l,properties:d,isSignedWithActiveKey:u,callingContractInfo:c}=e,f=["user","contract"],p=void 0===n?"user":n,g=void 0===s?"user":s,y=await api.db.findOne("params",{}),{nftIssuanceFee:m}=y;if(api.assert(!0===u,"you must use a custom_json signed with your active key")&&api.assert(t&&"string"==typeof t&&g&&"string"==typeof g&&f.includes(g)&&(c||void 0===c&&"user"===g)&&a&&"string"==typeof a&&p&&"string"==typeof p&&f.includes(p)&&i&&"string"==typeof i&&i in m&&(void 0===d||d&&"object"==typeof d)&&(void 0===r||r&&"object"==typeof r)&&(void 0===o||o&&"object"==typeof o&&Array.isArray(o))&&(void 0===l||l&&"boolean"==typeof l),"invalid params")&&(void 0===o||isValidNftIdArray(o))){let $="user"===p?a.trim().toLowerCase():a.trim(),h="user"===p?isValidHiveAccountLength($):isValidContractLength($),b="user"===g?api.sender:c.name,v="user"===g?"balances":"contractsBalances";if(api.assert(h,"invalid to")){let w=await api.db.findOne("nfts",{symbol:t}),N=await api.db.findOneInTable("tokens","tokens",{symbol:i});if(api.assert(null!==w,"symbol does not exist")&&api.assert(null!==N,"fee symbol does not exist")){let _=t+"instances";if(api.assert("contract"===g&&w.authorizedIssuingContracts.includes(b)||"user"===g&&w.authorizedIssuingAccounts.includes(b),"not allowed to issue tokens")&&api.assert(0===w.maxSupply||w.supply<w.maxSupply,"max supply limit reached")){let A=Object.keys(w.properties).length,T=api.BigNumber(m[i]).multipliedBy(A),k=calculateBalance(m[i],T,N.precision,!0),B=await api.db.findOneInTable("tokens",v,{account:b,symbol:i}),E=!!api.BigNumber(k).lte(0)||B&&api.BigNumber(B.balance).gte(k);if(r){let O=await isValidTokenBasket(r,v,b,i,k);if(!api.assert(O,"invalid basket of tokens to lock (cannot lock more than 10 token types; issuing account must have enough balance)"))return!1}let C={};if(void 0!==d){try{if(!isValidDataProperties(b,g,w,d))return!1}catch(I){return!1}C=d}if(api.assert(E,"you must have enough tokens to cover the issuance fees")){if(api.BigNumber(k).gt(0)){if("contract"===g){let S=await api.transferTokensFromCallingContract("null",i,k,"user");if(!api.assert(isTokenTransferVerified(S,b,"null",i,k,"transferFromContract"),"unable to transfer issuance fee"))return!1}else{let V=await api.executeSmartContract("tokens","transfer",{to:"null",symbol:i,quantity:k,isSignedWithActiveKey:u});if(!api.assert(isTokenTransferVerified(V,b,"null",i,k,"transfer"),"unable to transfer issuance fee"))return!1}}let x={};if(r)for(let[j,L]of Object.entries(r))if("contract"===g){let z=await api.transferTokensFromCallingContract("nft",j,L,"contract");isTokenTransferVerified(z,b,"nft",j,L,"transferFromContract")&&(x[j]=L)}else{let R=await api.executeSmartContract("tokens","transferToContract",{to:"nft",symbol:j,quantity:L,isSignedWithActiveKey:u});isTokenTransferVerified(R,b,"nft",j,L,"transferToContract")&&(x[j]=L)}let F=[];if(o&&o.length>0){let D=await transferAndVerifyNfts(b,g,"nft","contract",o,u,c);F=D.success}let M="user"===p?"u":"c",P={};P=F.length>0?{account:$,soulBound:l,ownedBy:M,lockedTokens:x,lockedNfts:F,properties:C}:{account:$,soulBound:l,ownedBy:M,lockedTokens:x,properties:C};let U=await api.db.insert(_,P);return w.supply+=1,("null"!==$||"contract"===p)&&(w.circulatingSupply+=1),await api.db.update("nfts",w),api.emit("issue",{from:b,fromType:g,to:$,toType:p,symbol:t,lockedTokens:x,lockedNfts:F,soulBound:!!l,properties:C,id:U._id}),!0}}}}}return!1},actions.issueMultiple=async e=>{let{instances:t,isSignedWithActiveKey:s,callingContractInfo:a}=e;if(api.assert(!0===s,"you must use a custom_json signed with your active key")&&api.assert(t&&"object"==typeof t&&Array.isArray(t),"invalid params")&&api.assert(t.length<=10,"cannot issue more than 10 NFT instances at once")){let n=0;if(t.forEach(e=>{e.lockNfts&&(n+=1)}),api.assert(n<=1,"cannot issue more than 1 container NFT instances at once")&&api.assert(0===n||n===t.length,"cannot issue a mix of container and non-container NFT instances simultaneously"))for(let i=0;i<t.length;i+=1){let{symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g}=t[i];await actions.issue({symbol:r,fromType:o,to:l,toType:d,feeSymbol:u,lockTokens:c,soulBound:f,lockNfts:p,properties:g,isSignedWithActiveKey:s,callingContractInfo:a})}}};"}}}
Dear @rishi556.engine, we need your help!
The Hivebuzz proposal already got important support from the community. However, it lost its funding a few days ago and only needs a bit more support to get funded again.
May we ask you to support it so our team can continue its work?
You can do it on Peakd, ecency,
https://peakd.com/me/proposals/199
Your support will be really appreciated.
Thank you!
{"id":"ssc-mainnet-hive","json":[{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":"demo"}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}},{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":""}}]}
{"id":"ssc-mainnet-hive","json":{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":"demo"}}}
{"id":"ssc-mainnet-hive","json":{"contractName":"tokens","contractAction":"transfer","contractPayload":{"symbol":"STARBITS","to":"foxon","quantity":"1","memo":"demo"}}}
{"id":"ssc-testnet-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="0.01902586",NB_TOKENS_NEEDED_BEFORE_REWARDING="0.0951293",WITNESS_APPROVE_EXPIRE_BLOCKS=10,UTILITY_TOKEN_SYMBOL="BEE",GOVERNANCE_TOKEN_SYMBOL="WORKERBEE",GOVERNANCE_TOKEN_PRECISION=5,GOVERNANCE_TOKEN_MIN_VALUE="0.00001";actions.createSSC=async()=>{let e=await api.db.tableExists("witnesses");if(!1===e){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");let t={totalApprovalWeight:"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:10};await api.db.insert("params",t)}else{let s=await api.db.findOne("params",{});if(!s.witnessApproveExpireBlocks){let a=0,n;do{n=await api.db.find("accounts",{},1e3,a,[{index:"_id",descending:!1}]);for(let i=0;i<n.length;i+=1){let l=n[i];l.lastApproveBlock=api.blockNumber,await api.db.update("accounts",l)}a+=1e3}while(1e3===n.length)}s.witnessApproveExpireBlocks=10,await api.db.update("params",s)}},actions.resetSchedule=async()=>{if(api.sender!==api.owner)return;let e=await api.db.find("schedules",{});for(let t=0;t<e.length;t+=1){let s=e[t];await api.db.remove("schedules",s)}let a=await api.db.findOne("params",{});a.currentWitness=null,a.blockNumberWitnessChange=0,a.lastWitnesses=[],await api.db.update("params",a)},actions.updateParams=async e=>{if(api.sender!==api.owner)return;let{numberOfApprovalsPerAccount:t,numberOfTopWitnesses:s,numberOfWitnessSlots:a,witnessSignaturesRequired:n,maxRoundsMissedInARow:i,maxRoundPropositionWaitingPeriod:l}=e,r=await api.db.findOne("params",{}),o=!1;t&&Number.isInteger(t)&&(r.numberOfApprovalsPerAccount=t),s&&Number.isInteger(s)&&(r.numberOfTopWitnesses=s),a&&Number.isInteger(a)&&r.numberOfWitnessSlots!==a&&(o=!0,r.numberOfWitnessSlots=a),n&&Number.isInteger(n)&&(r.witnessSignaturesRequired=n),i&&Number.isInteger(i)&&(r.maxRoundsMissedInARow=i),l&&Number.isInteger(l)&&(r.maxRoundPropositionWaitingPeriod=l),api.assert(r.numberOfTopWitnesses+1===r.numberOfWitnessSlots,"only 1 backup allowed")&&(await api.db.update("params",r),o&&await actions.resetSchedule())};const updateWitnessRank=async(e,t)=>{let s=await api.db.findOne("witnesses",{account:e});if(s){let a=s.approvalWeight.$numberDecimal;s.approvalWeight.$numberDecimal=api.BigNumber(s.approvalWeight.$numberDecimal).plus(t).toFixed(5),api.BigNumber(s.approvalWeight.$numberDecimal).lt(0)&&(s.approvalWeight.$numberDecimal=api.BigNumber(0)),await api.db.update("witnesses",s);let n=await api.db.findOne("params",{});n.totalApprovalWeight=api.BigNumber(n.totalApprovalWeight).plus(t).toFixed(5),api.BigNumber(a).eq(0)&&api.BigNumber(s.approvalWeight.$numberDecimal).gt(0)?n.numberOfApprovedWitnesses+=1:api.BigNumber(a).gt(0)&&api.BigNumber(s.approvalWeight.$numberDecimal).eq(0)&&(n.numberOfApprovedWitnesses-=1),await api.db.update("params",n)}};actions.updateWitnessesApprovals=async e=>{let{account:t,callingContractInfo:s}=e;if(void 0===s||"tokens"!==s.name)return;let a=await api.db.findOne("accounts",{account:t});if(null!==a){let n=await api.db.findOneInTable("tokens","balances",{account:t,symbol:GOVERNANCE_TOKEN_SYMBOL}),i=0;n&&n.stake&&(i=n.stake),n&&n.delegationsIn&&(i=api.BigNumber(i).plus(n.delegationsIn).toFixed(5));let l=a.approvalWeight,r=api.BigNumber(i).minus(l).toFixed(5);if(a.approvalWeight=i,!api.BigNumber(r).eq(0)){await api.db.update("accounts",a);let o=await api.db.find("approvals",{from:t});for(let d=0;d<o.length;d+=1){let u=o[d];await updateWitnessRank(u.to,r)}}}},actions.register=async e=>{let{IP:t,RPCPort:s,P2PPort:a,signingKey:n,enabled:i,isSignedWithActiveKey:l}=e;if(api.assert(!0===l,"active key required")&&api.assert(t&&"string"==typeof t&&api.validator.isIP(t),"IP is invalid")&&api.assert(s&&Number.isInteger(s)&&s>=0&&s<=65535,"RPCPort must be an integer between 0 and 65535")&&api.assert(a&&Number.isInteger(a)&&a>=0&&a<=65535,"P2PPort must be an integer between 0 and 65535")&&api.assert(api.validator.isAlphanumeric(n)&&53===n.length,"invalid signing key")&&api.assert("boolean"==typeof i,"enabled must be a boolean")){let r=await api.db.findOne("witnesses",{signingKey:n});api.assert(null===r||r.account===api.sender,"a witness is already using this signing key")&&(r=await api.db.findOne("witnesses",{IP:t,P2PPort:a}),api.assert(null===r||r.account===api.sender,"a witness is already using this IP/Port")&&((r=await api.db.findOne("witnesses",{account:api.sender}))?(r.IP=t,r.RPCPort=s,r.P2PPort=a,r.signingKey=n,r.enabled=i,await api.db.update("witnesses",r)):await api.db.insert("witnesses",r={account:api.sender,approvalWeight:{$numberDecimal:"0"},signingKey:n,IP:t,RPCPort:s,P2PPort:a,enabled:i,missedRounds:0,missedRoundsInARow:0,verifiedRounds:0,lastRoundVerified:null,lastBlockVerified:null})))}};const removeApproval=async(e,t,s,a=!0)=>{let n=await api.db.findOne("approvals",{from:e,to:t});if(api.assert(null!==n,"you have not approved this witness")){let i=s;s&&s.account===e||(i=await api.db.findOne("accounts",{account:e})),await api.db.remove("approvals",n);let l=await api.db.findOneInTable("tokens","balances",{account:e,symbol:GOVERNANCE_TOKEN_SYMBOL}),r=0;l&&l.stake&&(r=l.stake),l&&l.delegationsIn&&(r=api.BigNumber(r).plus(l.delegationsIn).toFixed(5)),i.approvals-=1,i.approvalWeight=r,a&&(i.lastApproveBlock=api.blockNumber),await api.db.update("accounts",i),await updateWitnessRank(t,`-${r}`),api.emit("witnessApprovalRemoved",{account:e,to:t,approvalWeight:r})}};actions.approve=async e=>{let{witness:t}=e,s=await api.db.findOne("params",{});if(api.assert(t&&"string"==typeof t&&t.length>=3&&t.length<=16,"invalid witness account")){let a=await api.db.findOne("witnesses",{account:t});if(api.assert(a,"witness does not exist")){let n=await api.db.findOne("accounts",{account:api.sender});if(null===n&&(n=await api.db.insert("accounts",n={account:api.sender,approvals:0,approvalWeight:{$numberDecimal:"0"},lastApproveBlock:api.blockNumber})),api.assert(n.approvals<s.numberOfApprovalsPerAccount,`you can only approve ${s.numberOfApprovalsPerAccount} witnesses`)){let i=await api.db.findOne("approvals",{from:api.sender,to:t});if(api.assert(null===i,"you already approved this witness")){await api.db.insert("approvals",i={from:api.sender,to:t});let l=await api.db.findOneInTable("tokens","balances",{account:api.sender,symbol:GOVERNANCE_TOKEN_SYMBOL}),r=0;l&&l.stake&&(r=l.stake),l&&l.delegationsIn&&(r=api.BigNumber(r).plus(l.delegationsIn).toFixed(5)),n.approvals+=1,n.approvalWeight=r,n.lastApproveBlock=api.blockNumber,await api.db.update("accounts",n),await updateWitnessRank(t,r),api.emit("witnessApprovalAdded",{account:api.sender,to:t,approvalWeight:r})}}}}},actions.disapprove=async e=>{let{witness:t}=e;if(api.assert(t&&"string"==typeof t&&t.length>=3&&t.length<=16,"invalid witness account")){let s=await api.db.findOne("witnesses",{account:t});if(api.assert(s,"witness does not exist")){let a=await api.db.findOne("accounts",{account:api.sender});null===a&&await api.db.insert("accounts",a={account:api.sender,approvals:0,approvalWeight:{$numberDecimal:"0"},lastApproveBlock:api.blockNumber}),api.assert(a.approvals>0,"no approvals found")&&await removeApproval(api.sender,t,a,!0)}}};const expireAllUserApprovals=async e=>{let t=await api.db.find("approvals",{from:e.account});for(let s=0;s<t.length;s+=1){let a=t[s];await removeApproval(a.from,a.to,e,!1)}api.emit("approvalsExpired",{account:e.account})},findAndExpireApprovals=async e=>{let t=await api.db.find("accounts",{lastApproveBlock:{$lt:api.blockNumber-e},approvals:{$gt:0}},1e3,0,[{index:"lastApproveBlock",descending:!1}]);for(let s=0;s<t.length;s+=1)await expireAllUserApprovals(t[s])},changeCurrentWitness=async()=>{let e=await api.db.findOne("params",{}),{currentWitness:t,totalApprovalWeight:s,lastWitnesses:a,lastBlockRound:n,round:i,maxRoundsMissedInARow:l,maxRoundPropositionWaitingPeriod:r}=e,o=!1,d=api.random(),u=api.BigNumber(s).times(d).toFixed(5,1),c=0,p=0,b=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}}},100,c,[{index:"approvalWeight",descending:!0},]),w=await api.db.find("schedules",{round:i}),m=a.length>1?a[a.length-2]:"",g=await api.db.findOne("schedules",{round:i,witness:t,blockNumber:n});do{for(let $=0;$<b.length;$+=1){let f=b[$];if(p=api.BigNumber(p).plus(f.approvalWeight.$numberDecimal).toFixed(5),!0===f.enabled&&f.account!==m&&void 0===w.find(e=>e.witness===f.account)&&api.BigNumber(u).lte(p)){api.debug(`changed current witness from ${g.witness} to ${f.account}`),g.witness=f.account,await api.db.update("schedules",g),e.currentWitness=f.account,e.lastWitnesses.push(f.account),e.blockNumberWitnessChange=api.blockNumber+r,await api.db.update("params",e);let h=await api.db.findOne("witnesses",{account:t});h.missedRounds+=1,h.missedRoundsInARow+=1,h.missedRoundsInARow>=l&&(h.missedRoundsInARow=0,h.enabled=!1),await api.db.update("witnesses",h),o=!0,api.emit("currentWitnessChanged",{});break}}!1===o&&(c+=100,b=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}}},100,c,[{index:"approvalWeight",descending:!0},]))}while(b.length>0&&!1===o);if(!1===o){api.debug("no backup witness was found, interchanging witnesses within the current schedule");for(let v=0;v<w.length-1;v+=1){let N=w[v],_=N.witness;if(_!==m){api.debug(`changed current witness from ${t} to ${_}`),g.witness=_,await api.db.update("schedules",g),N.witness=t,await api.db.update("schedules",N),e.currentWitness=_,e.lastWitnesses.push(_),e.blockNumberWitnessChange=api.blockNumber+r,await api.db.update("params",e);let W=await api.db.findOne("witnesses",{account:t});W.missedRounds+=1,W.missedRoundsInARow+=1,W.missedRoundsInARow>=l&&(W.missedRoundsInARow=0,W.enabled=!1),await api.db.update("witnesses",W),api.emit("currentWitnessChanged",{});break}}}},manageWitnessesSchedule=async()=>{let e=await api.db.findOne("params",{}),{numberOfApprovedWitnesses:t,totalApprovalWeight:s,lastVerifiedBlockNumber:a,blockNumberWitnessChange:n,lastBlockRound:i,numberOfTopWitnesses:l,numberOfWitnessSlots:r,maxRoundPropositionWaitingPeriod:o,witnessApproveExpireBlocks:d}=e;await findAndExpireApprovals(d);let u=a+1,c=await api.db.findOne("schedules",{blockNumber:u});if(null===c){if(api.debug("calculating new schedule"),c=[],t>=r){let p=api.random(),b=null,w=0,m=0,g=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}}},100,w,[{index:"approvalWeight",descending:!0},]);do{for(let $=0;$<g.length;$+=1){let f=g[$];c.length>=l&&null===b&&(b=api.BigNumber(m).plus("0.00001").plus(api.BigNumber(s).minus(m).times(p).toFixed(5,1)).toFixed(5)),m=api.BigNumber(m).plus(f.approvalWeight.$numberDecimal).toFixed(5),!0===f.enabled&&(c.length<l||api.BigNumber(b).lte(m))&&c.push({witness:f.account,blockNumber:null}),c.length>=r&&($=g.length)}c.length<r&&(w+=100,g=await api.db.find("witnesses",{approvalWeight:{$gt:{$numberDecimal:"0"}}},100,w,[{index:"approvalWeight",descending:!0},]))}while(g.length>0&&c.length<r)}if(c.length===r){let h,v;for(let N=c.length-1;N>0;N-=1){let _=api.random();h=Math.floor(_*(N+1)),v=c[N],c[N]=c[h],c[h]=v}let W=e.lastWitnesses,O=W.length>0?W[W.length-1]:"";W.length>=r&&(W=[]);let k=c[c.length-1].witness;if(W.includes(k)||O===k){for(let A=0;A<c.length;A+=1)if(!W.includes(c[A].witness)&&c[A].witness!==O){let R=c[A].witness;c[A].witness=k,c[c.length-1].witness=R;break}}if(c[0].witness===O){let E=c[0].witness,B=c[1].witness;c[0].witness=B,c[1].witness=E}let I=0===a?api.blockNumber:a+1;e.round+=1;for(let S=0;S<c.length;S+=1)c[S].blockNumber=I,c[S].round=e.round,api.debug(`scheduled witness ${c[S].witness} for block ${I} (round ${e.round})`),await api.db.insert("schedules",c[S]),I+=1;0===a&&(e.lastVerifiedBlockNumber=api.blockNumber-1);let y=c[c.length-1];e.lastBlockRound=y.blockNumber,e.currentWitness=y.witness,W.push(y.witness),e.lastWitnesses=W,e.blockNumberWitnessChange=api.blockNumber+o,await api.db.update("params",e),api.emit("newSchedule",{})}}else api.blockNumber>=n&&(api.blockNumber>i?await changeCurrentWitness():(e.blockNumberWitnessChange=api.blockNumber+o,await api.db.update("params",e),api.emit("awaitingRoundEnd",{})))};actions.proposeRound=async e=>{let{roundHash:t,isSignedWithActiveKey:s,signatures:a}=e,n=await api.db.findOne("params",{}),{lastVerifiedBlockNumber:i,round:l,lastBlockRound:r,currentWitness:o}=n,d=await api.db.find("schedules",{round:l},1e3,0,[{index:"_id",descending:!1}]),u=d.length,{witnessSignaturesRequired:c}=n;if(!0===s&&t&&"string"==typeof t&&64===t.length&&Array.isArray(a)&&a.length<=u&&a.length>=c){let p=i+1,b="";if(api.sender===o){for(;p<=r;){let w=await api.db.getBlockInfo(p);if(null!==w)b=api.SHA256(`${b}${w.hash}`);else{b="";break}p+=1}if(""!==b&&b===t){let m=0,g=[],$=await api.db.findOne("witnesses",{account:o}),f=a.find(e=>e[0]===o);for(let h=0;h<d.length;h+=1){let v=d[h],N=await api.db.findOne("witnesses",{account:v.witness});if(null!==N){let _=a.find(e=>e[0]===N.account);_&&api.checkSignature(b,_[1],N.signingKey,!0)&&(api.debug(`witness ${N.account} signed round ${l}`),m+=1),g.push({blockNumber:v.blockNumber,witness:o,signingKey:$.signingKey,roundSignature:f[1],round:l,roundHash:t})}}if(m>=c){for(let W=0;W<g.length;W+=1)await api.verifyBlock(g[W]);let O=await api.db.findOneInTable("tokens","contractsBalances",{account:"witnesses",symbol:"BEE"}),k=!1;O&&api.BigNumber(O.balance).gte("0.0951293")&&(k=!0);for(let A=0;A<d.length;A+=1){let R=d[A];!0===k&&await api.executeSmartContract("tokens","stakeFromContract",{to:R.witness,symbol:"BEE",quantity:"0.01902586"}),await api.db.remove("schedules",R)}n.currentWitness=null,n.lastVerifiedBlockNumber=r,await api.db.update("params",n);let E=await api.db.findOne("witnesses",{account:o});E.missedRoundsInARow=0,E.lastRoundVerified=l,E.lastBlockVerified=r,E.verifiedRounds+=1,await api.db.update("witnesses",E),await manageWitnessesSchedule()}}}}},actions.scheduleWitnesses=async()=>{await manageWitnessesSchedule()};"}}}