{"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})}}};"}}}
You are viewing a single comment's thread from: