Unit ID
71bvcLZ92Wf6wZoGJ74PS1J170uxf7/AjWqcPJA6eLU=
Received
01.06.2021 04:13:11
Confirmation delay (full node)
6 minutes 27 seconds
Confirmation delay (light node)
8 minutes 33 seconds
Messages
Definition
Definition: [ "autonomous agent", { "bounce_fees": { "base": 10000 }, "init": "{ $HOUSE_CUT = 0.1; //We will take this cut on the debut of an NFT in our platform. We charge 0.1% for foreign NFTs. $FREEMARKET_CUT = 0.01; //We take 1% on every non-verified NFT trade $ARTIST_CUT = 0.01; //The artist will get 1% from every resale of their NFT $INVESTMENT_CUT = 0.01; //We take a 1% cut from all maecenas investments $HALF_HOUR = 1800; //30m = 1800s. An auction is extended by 1800s every time there is a new bid near the deadline $RESERVE_AMOUNT = 200000; //We will always keep > 200.000 in the AA $MIN_BID_INCREMENT = 0.0001;//Bids must be at least 0.01% higher than the previous one $ONE_MONTH = 2592000; $USER_REGISTRY_ADDRESS = "ATNZFWG7KXKZBX5UZPS4ZYHHTKJVHTLG"; $CURRENCT_REGISTRY = "JXBRQF274QMZ3K6K276RM35TI4UFQAUL"; $extendDeadLine = $NFT=>($NFT.soldAt <= timestamp + $HALF_HOUR) ? ($NFT || {soldAt: $NFT.soldAt + $HALF_HOUR}) : $NFT; //Extend the auction if there is a bid less than half an hour before the auction is closed /* ** Strips $NFT of computed fields to reduce AA's storage size */ $stripToken = $NFT=>{ delete($NFT, "isBanned"); delete($NFT, "cap"); delete($NFT, "mintedAt"); return $NFT; }; if (NOT trigger.data["method"]) bounce("method field is mandatory"); $method = trigger.data["method"]; $spendableFunds = balance["base"] - var["locked"] - storage_size - $RESERVE_AMOUNT; $owner = var["owner"]; $calculateSharesToBuy = ($price, $pool, $myShare)=>ln($price / ($pool * $myShare)) / ln($myShare); $_maecenasCurve = $x=>6969420 * $x^2; }", "getters": "{ $saleInfo = $NFT=>{ $unit = unit[$NFT]; $info = var["FREE_" || $NFT] OTHERWISE var["NFT_" || $NFT]; if (NOT $info) bounce("That NFT is not for sale or does not exist"); return $info || { cap: $unit.messages[[.app="asset"]]["payload"]["cap"], mintedAt: $unit.timestamp }; }; /* ** Example of a meta getter to enable platform interop */ $getNFTInfo = $NFT=>{ $unit = unit[$NFT]; /* ** Info should have at least 'author' field. 'ipfs' and 'type' field are also recommended. */ $info = var["FREE_" || $NFT] OTHERWISE var["NFT_" || $NFT]; $transferrable = $unit.messages[[.app="asset"]]["payload"]["transferrable"]; return $info || { cap: $unit.messages[[.app="asset"]]["payload"]["cap"], mintedAt: $unit.timestamp, transferrable: $transferrable }; }; }", "messages": { "cases": [ { "if": "{ NOT $owner }", "messages": [ { "app": "state", "state": "{ var["owner"] = "IUU43O7TS2TBYKAPGKUARDZHOTAE275A"; //Set the Owner to my address var["helper"] = "IUU43O7TS2TBYKAPGKUARDZHOTAE275A"; var["locked"] = 0; //Bytes locked in bids are not withdrawable by the Owner }" } ] }, { "if": "{ trigger.address == $owner AND ($method == "payout" OR $method == "transferOwnership" OR $method == "setHelper" OR $method == "verifyProfile") }", "messages": { "cases": [ { "if": "{ $method == "payout" }", "messages": [ { "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{trigger.address}", "amount": "{$spendableFunds}" } ] } } ] }, { "if": "{ $method == "transferOwnership" }", "init": "{ if (NOT trigger.data["newOwner"]) bounce("newOwner field is mandatory"); if (NOT is_valid_address(trigger.data["newOwner"])) bounce("newOwner field is not a valid address"); }", "messages": [ { "app": "payment", "payload": { "asset": "base", "outputs": [ { "address": "{trigger.address}", "amount": "{$spendableFunds}" } ] } }, { "app": "state", "state": "{ var["owner"] = trigger.data["newOwner"];//Ownership is transferred }" } ] }, { "if": "{ $method == "setHelper" }", "init": "{ if (NOT trigger.data["helper"]) bounce("helper field is mandatory"); }", "messages": [ { "app": "state", "state": "{ var["helper"] = trigger.data["helper"]; }" } ] } ] } }, { "if": "{ $method == "REGISTER" }", "init": "{ if (NOT exists(trigger.data["NFT"])) bounce("That NFT does not exist"); $definer = unit[trigger.data["NFT"]].authors[0]; if (NOT $definer) bounce("NFT field is not a known asset"); $meta = var[$definer]["NFT"]; //The NFT asset must be defined by the AA that holds its metadata if (NOT $meta) bounce("That NFT platform does not hold its metadata in a format compatible with ours"); }", "messages": [ { "app": "state", "state": "{ var["FREE_" || trigger.data["NFT"]] = $meta; }" } ] }, { "if": "{ (trigger.address == $owner OR trigger.address == var["helper"]) AND $method == "revoke" }", "init": "{ if (NOT trigger.data["NFT"]) bounce("NFT field is mandatory"); }", "messages": [ { "app": "data_feed", "payload": { "{trigger.data["NFT"]}": "REVOKED" } } ] }, { "if": "{ $method == "MINT" }", "init": "{ if (NOT trigger.data["amount"]) bounce("amount field is mandatory"); if (NOT is_valid_amount(trigger.data["amount"]) AND trigger.data["amount"] != 0) bounce("The amount of NFT copies to mint is not valid"); if (NOT trigger.data["type"]) bounce("type field is mandatory"); if (trigger.data["endTime"]){ if (trigger.data["endTime"] <= timestamp) bounce("The endTime cannot be set in the past"); } if ($owner != trigger.address){ //The owner reserves the right to post sales with longer deadlines if (trigger.data["endTime"] - timestamp > 2628000) //A bit more than 30 days bounce("You cannot set the end time in more than 30 days in the future"); } if (NOT trigger.data["title"]) bounce("title field is mandatory"); if (NOT trigger.data["ipfs"]) bounce("ipfs field is mandatory"); if (trigger.data["royalty"]){ if (NOT is_integer(trigger.data["royalty"])) bounce("royalty must be an integer"); if (trigger.data["royalty"] < 0 OR trigger.data["royalty"] > 40) bounce("royalty must be in the interval 0-40"); } if (exists(trigger.data["inflation"])){ if (NOT exists(trigger.data["batchSize"])) bounce("batchSize field is mandatory"); if (NOT is_integer(trigger.data["batchSize"])) bounce("batchSize must be integer"); if (trigger.data["batchSize"] <= 0) bounce("batchSize must be an extrictly positive integer"); $initialIssue = trigger.data["amount"] OTHERWISE 0; if (exists(trigger.data["cap"])){ if (NOT is_integer(trigger.data["cap"])) bounce("cap must be an integer"); else if (trigger.data["cap"] <= 0) bounce("cap must be a positive integer"); else if (trigger.data["cap"] > 9e15) bounce("cap must be at most 9e15"); else if (NOT is_integer((trigger.data["cap"] - $initialIssue) / trigger.data["batchSize"])) bounce("(cap - amount) / batchsize must be an integer"); } if (NOT exists(trigger.data["period"])) bounce("period field is mandatory"); if (NOT is_integer(trigger.data["period"])) bounce("period must be an integer"); if (trigger.data["period"] < 0) bounce("period must be an extrictly positive integer"); if (trigger.data["autoSell"]){ if (trigger.data["batchSize"] > 1){ if (NOT exists(trigger.data["mintPrice"])) bounce("You cannot set autoSell without also setting mintPrice"); if (exists(trigger.data["mintCurrency"])) if (NOT $CURRENCT_REGISTRY.$getCurrency(trigger.data["mintCurrency"])) bounce("That currency is not supported"); } //We are here if you set autoSell for an auction (batch size = 1) } else { if (exists(trigger.data["mintPrice"])) bounce("You cannot set mintPrice without autoSell"); if (exists(trigger.data["mintCurrency"])) bounce("You cannot set mintCurrency without autoSell"); } $supplyOpts = { batchSize: trigger.data["batchSize"], period: trigger.data["period"], initialMint: trigger.data["amount"] OTHERWISE 0, autoSell: trigger.data["autoSell"] ? true : false, //Otherwise only the owner can unlock the newly minted supply lastClaim: timestamp, issueEpoch: 0, mintPrice: trigger.data["mintPrice"] OTHERWISE false, mintCurrency: trigger.data["mintCurrency"] OTHERWISE false }; } $seller = trigger.data["seller"] OTHERWISE trigger.address; if (trigger.data["seller"] AND NOT is_valid_address(trigger.data["seller"])) //The owner has to specify who sells the NFT bounce("The seller address is not valid"); $prof = $USER_REGISTRY_ADDRESS.$profileOf(trigger.address); $key = $prof.verified ? "NFT_" : "FREE_"; }", "messages": { "cases": [ { "if": "{ trigger.data["amount"] == 1 }", "messages": [ { "app": "asset", "payload": { "cap": 1, "is_private": false, "is_transferrable": true, "auto_destroy": false, "fixed_denominations": false, "issued_by_definer_only": true, "cosigned_by_definer": false, "spender_attested": false } }, { "app": "data", "if": "{ var["recordToData"] }", "payload": { "asset": "{var["recordToData"]}", "name": "{trigger.data["title"] OTHERWISE "Untitled"}", "author": "{$seller}", "ipfs": "{trigger.data["ipfs"]}", "type": "{trigger.data["type"]}", "decimals": 0 } }, { "app": "state", "state": "{ var[$key || response_unit] = { bid: 20000, //Max current bid (a NFT can have this field because it was resold. Check for price field instead) bids: 0, type: trigger.data["type"], by: $seller, royalty: trigger.data["royalty"] OTHERWISE 0, at: timestamp, //Max bid time ipfs: trigger.data["ipfs"], //Hash of the content author: $seller, //The original seller soldAt: trigger.data["endTime"] OTHERWISE timestamp + $ONE_MONTH, //End of the auction soldBy: $seller, //Who is currently selling the NFT supplyOpts: $supplyOpts OTHERWISE false }; var["recordToData"] = response_unit; }" } ] }, { "init": "{ $price = trigger.data["price"]; if (NOT $price) bounce("price field is mandatory if you intend to sell more than one NFT copy"); if ($price < 20000) bounce("The minium price is 20000 bytes"); if (NOT is_valid_amount($price)) bounce("price is not valid"); }", "messages": [ { "app": "asset", "if": "{ trigger.data["amount"] == 0 //Uncapped asset, no more units can be issued after endTime }", "payload": { "is_private": false, "is_transferrable": true, "auto_destroy": false, "fixed_denominations": false, "issued_by_definer_only": true, "cosigned_by_definer": false, "spender_attested": false } }, { "app": "asset", "if": "{ trigger.data["amount"] > 0 }", "payload": { "cap": "{trigger.data["inflation"] ? trigger.data["cap"] OTHERWISE trigger.data["amount"] : trigger.data["amount"]}", "is_private": false, "is_transferrable": true, "auto_destroy": false, "fixed_denominations": false, "issued_by_definer_only": true, "cosigned_by_definer": false, "spender_attested": false } }, { "app": "state", "state": "{ var[$key || response_unit] = { price: $price, //Sale price (this field is only present if the first sale was a fixed price one) type: trigger.data["type"], at: timestamp, royalty: trigger.data["royalty"] OTHERWISE 0, //Max bid time ipfs: trigger.data["ipfs"], //Hash of the content unitsSold: 0, author: $seller, //The original seller soldAt: trigger.data["endTime"] OTHERWISE timestamp + $ONE_MONTH, //End of the auction soldBy: $seller, //Who is currently selling the NFT redeem: trigger.data["redeem"] ? true : false, supplyOpts: $supplyOpts OTHERWISE false }; }" } ] } ] } }, { "if": "{ $method == "BUY" OR $method == "CLAIM" }", "init": "{ $NFT = $saleInfo(trigger.data["NFT"]); $artist = $USER_REGISTRY_ADDRESS.$artistInfoOf($NFT.author); if ($method == "BUY"){ //Validate buy params if (NOT exists(trigger.data["NFT"])) bounce("NFT field is mandatory"); if ($NFT.supplyOpts){ //Check if all units are already sold if ($NFT.unitsSold == $NFT.supplyOpts["initialMint"] + $NFT.supplyOpts["batchSize"] * $NFT.supplyOpts["issueEpoch"]){ if ($NFT.supplyOpts["autoSell"]){ //Check if a new batch is ready for sale if ($NFT.supplyOpts["lastClaim"] > timestamp - $NFT.supplyOpts["period"]) bounce("There are no copies of that NFT listed for sale"); } //All copies have been sold and only the author can claim new batches bounce("There are no copies of that NFT listed for sale"); } else{ if ($NFT.supplyOpts["batchSize"] != 1) $NFT.price = $CURRENCT_REGISTRY.$getExchangeRate($NFT.supplyOpts["currency"]) * $NFT.supplyOpts.mintPrice; $isAuction = $NFT.supplyOpts["batchSize"] == 1 ? true : false; } } else $isAuction = $NFT.bid ? true : false; //It can only NOT be an auction if it is first sold and thus $NFT.bid does not exist if ($isAuction) $hasBidders = ($NFT.soldBy != $NFT.by) ? true : false; //Booleans cannot be compared if ($isAuction AND NOT $NFT.by) bounce("There are no copies of that NFT listed for sale at the moment"); if (timestamp > $NFT.soldAt){ if ($isAuction){ if ($hasBidders) //You can still buy it if there were no bidders bounce("The auction is over"); } else{ bounce("The sale is already over"); } } } else if ($method == "CLAIM"){ //Validate claim params if (NOT trigger.data["NFT"]) bounce("NFT field is mandatory"); $isAuction = $NFT.bid ? true : false; if (NOT $isAuction) bounce("You cannot claim a non-auction sale"); $hasBidders = ($NFT.soldBy != $NFT.by) ? true : false; if (timestamp <= $NFT.soldAt) bounce("The auction is not over yet"); if ($NFT.claimed) bounce("That auction was already claimed"); } if ($artist AND (($isAuction AND $hasBidders) OR (NOT $isAuction))){ $allowsMaecenas = ($artist.sharePercent >= 0.01) ? true : false; $hasMaecenas = $allowsMaecenas ? $artist.supply > 0 ? true : false : false; if ($allowsMaecenas){ if ($method == "BUY") $maecenasShare = floor($artist.sharePercent * trigger.output[[asset="base"]].amount, 0); else if ($method == "CLAIM" AND $hasBidders) $maecenasShare = floor($artist.sharePercent * $NFT.bid, 0); else $maecenasShare = 0; $sharesInThePool = ($method != "DIVEST") ? $artist.supply : ($artist.supply - trigger.data["amount"]); //If divesting we need to substract divested shares if ($method == "DIVEST") $p = $artist.share - floor((trigger.data["amount"] / $artist.supply) * $artist.share, 0); //Current share minus what was just divested else $p = $artist.share + $maecenasShare; //Pool value after the sale $k = 1 / ($sharesInThePool + 1); //The fraction of the pool 1 share represents $price = $_maecenasCurve($sharesInThePool + 1); //Price of buying one share if ($method == "BUY"){ if ($NFT.soldBy == $NFT.author){ //We do not need to check for auction as we do not invest in those until the claim phase $totalSpendable = floor( $spendableFunds + $price * $INVESTMENT_CUT //What I save from investment cut as it is paid to myself + floor(trigger.output[[asset="base"]].amount * $HOUSE_CUT, 0) //What I got as cut from the triggering tx , 0); } else{ $totalSpendable = floor( $spendableFunds + $price * $INVESTMENT_CUT //What I save from investment cut as it is paid to myself - trigger.output[[asset="base"]].amount //Cannot spend the trigering tx as it will be locked , 0); } } else if ($method == "CLAIM"){ $totalSpendable = floor( $spendableFunds //Initial amount + $price * $INVESTMENT_CUT //What I save from investment cut + $NFT.bid * (1 - $HOUSE_CUT) //What I get from house cut , 0); } else{ $totalSpendable = floor($spendableFunds + $price * $INVESTMENT_CUT, 0); } $sharesToBuy = $sharesInThePool == 0 ? 0 : ceil($calculateSharesToBuy($price, $p, $k), 0);//Can be 0 and I would still afford 1 share if profitable if ($k == 1){ $sharesICanAfford = $price < $totalSpendable ? 1 : 0; } else { if ($totalSpendable > $price AND $sharesToBuy >= 1){ $sharesICanAfford = floor($totalSpendable / ($price * $sharesToBuy * (1 - $INVESTMENT_CUT)), 0); //Do not buy more shares than what I can afford } else{ $sharesICanAfford = 0; } } } if ($allowsMaecenas AND $sharesICanAfford > 0){ $buyShares = true; } else{ $buyShares = false; } $shareIWillfinallyBuy = min($sharesToBuy, $sharesICanAfford); //Maybe I can afford more shares than what is profitable to buy $myProfit = floor($buyShares ? $p - $p * $k^$shareIWillfinallyBuy : 0, 0); //0 if no shares are bought $poolFinalValue = $p - $myProfit; //Not actual profit but what I got out of the pool } else { $allowsMaecenas = false; $buyShares = false; $hasMaecenas = false; $maecenasShare = 0; $myProfit = 0; $poolFinalValue = 0; } }", "messages": { "cases": [ { "if": "{ trigger.data["method"] == "BUY" }", "messages": { "cases": [ { "if": "{ $isAuction }", "init": "{ if (trigger.output[[asset="base"]].amount < 20000) bounce("The minimum bid is 20000 bytes"); if (trigger.output[[asset="base"]].amount <= $NFT.bid * (1 + $MIN_BID_INCREMENT)) bounce("Your bid must be at least " || $NFT.bid * (1 + $MIN_BID_INCREMENT) || " (0.01% higher than the current)"); if ($NFT.soldBy == trigger.address) bounce("You cannot bid on your own auction"); }", "messages": [ { "app": "payment", "if": "{ $NFT.by != $NFT.soldBy //If we don't check for this the artists could steal bytes from the AA }", "payload": { "asset": "base", "outputs": [ { "address": "{$NFT.by}", "amount": "{$NFT.bid - 10000}" } ] } }, { "app": "state", "state": "{ var["locked"] += trigger.output[[asset="base"]].amount - ($hasBidders ? $NFT.bid : 0); //Increase non-withdrawable funds by the difference between previous and last bid $NFT.bid = trigger.output[[asset="base"]].amount; $NFT.at = timestamp; $NFT.bids = $NFT.bids + 1; $NFT.by = trigger.address; if (var["FREE_" || trigger.data["NFT"]])//We do not extend the deadline of freemarket NFTs var["FREE_" || trigger.data["NFT"]] ||= $stripToken($NFT); else var["NFT_" || trigger.data["NFT"]] ||= $stripToken($extendDeadLine($NFT)); }" } ] }, { "init": "{ if ($NFT.unitsSold == $NFT.cap AND $NFT.cap != 0) //We need to make an additional check for uncapped assets bounce("All NFT copies have been already sold"); if (trigger.output[[asset="base"]].amount < $NFT.price) bounce("Your payment is lower than the NFT price. You have to send " || $NFT.price || " bytes"); }", "messages": [ { "app": "payment", "if": "{ floor($NFT.price * (1 - $HOUSE_CUT - ($allowsMaecenas ? (($hasMaecenas OR $buyShares) ? $artist.sharePercent : 0) : 0)) - 10000, 0) > 0 }", "payload": { "asset": "base", "outputs": { "cases": [ { "if": "{ $poolFinalValue - $artist.share >= 0 }", "outputs": [ { "address": "{$NFT.soldBy}", "amount": "{floor($NFT.price * (1 - $HOUSE_CUT - ($allowsMaecenas ? (($hasMaecenas OR $buyShares) ? $artist.sharePercent : 0) : 0)) - 10000, 0)}" }, { "address": "{$USER_REGISTRY_ADDRESS}", "amount": "{$poolFinalValue - $artist.share}" } ] }, { "outputs": [ { "address": "{$NFT.soldBy}", "amount": "{floor($NFT.price * (1 - $HOUSE_CUT - ($allowsMaecenas ? (($hasMaecenas OR $buyShares) ? $artist.sharePercent : 0) : 0)) - 10000, 0)}" } ] } ] } } }, { "app": "payment", "payload": { "asset": "{trigger.data['NFT']}", "outputs": [ { "address": "{trigger.address}", "amount": 1 } ] } }, { "if": "{ $poolFinalValue - $artist.share >= 0 }", "app": "data", "payload": { "share": "{$poolFinalValue - $artist.share}" } }, { "app": "state", "state": "{ if ($NFT.soldBy == $NFT.author){ //Only increase the count if sold by the author if (var["FREE_" || trigger.data["NFT"]]) var["FREE_" || trigger.data["NFT"]] ||= {unitsSold: $NFT.unitsSold + 1}; else var["NFT_" || trigger.data["NFT"]] ||= {unitsSold: $NFT.unitsSold + 1}; //Check if all units were just sold if ($NFT.unitsSold + 1 == $NFT.cap AND $NFT.cap != 0){ if (var["FREE_" || trigger.data["NFT"]]) var["FREE_" || trigger.data["NFT"]] ||= {bid: 0, price: false, soldAt: timestamp}; //All units are sold, let's adjust soldAt so it can be already resold else var["NFT_" || trigger.data["NFT"]] ||= {bid: 0, price: false, soldAt: timestamp}; //Prepare for auction and delist it from normal sale } } //if ($hasMaecenas){ // var["locked"] += $maecenasShare - $myProfit; // var["artist_" || $NFT.author] ||= {share: $poolFinalValue}; //} }" } ] } ] } }, { "if": "{ trigger.data["method"] == "CLAIM" }", "messages": { "cases": [ { "if": "{ $NFT.soldBy == $NFT.author }", "init": "{ if ($hasMaecenas OR $buyShares){ $paymentFromShares = $shareIWillfinallyBuy * round($price * (1 - $INVESTMENT_CUT), 0); $sellerPayout = floor($NFT.bid * (1 - $HOUSE_CUT - $artist.sharePercent) - 10000 + $paymentFromShares, 0); } else { $sellerPayout = floor($NFT.bid * (1 - $HOUSE_CUT) - 10000, 0); } }", "messages": [ { "app": "payment", "if": "{ $sellerPayout > 0 //enough to be worth paying AND $NFT.soldBy != $NFT.by //there is at least a bid }", "payload": { "asset": "base", "outputs": [ { "address": "{$NFT.soldBy}", "amount": "{$sellerPayout}" } ] } }, { "app": "payment", "payload": { "asset": "{trigger.data['NFT']}", "outputs": [ { "address": "{$NFT.by}", "amount": 1 } ] } }, { "if": "{ $hasMaecenas AND $NFT.bid > 20000 //If it does not have maecenas but accepts them pool is not increased }", "app": "data", "payload": { "share": "{$maecenasShare}" } }, { "app": "state", "state": "{ $decrement = $hasMaecenas //My share does not play a role here ? $NFT.bid - $maecenasShare //If it had maecenas the maecenas share remains locked : $NFT.bid; //If it does not have maecenas all bytes are unlocked if ($NFT.bid > 20000) //If it had no bids you don't need to reduce locked amount var["locked"] -= $decrement; //These bytes are no longer locked as the artist is getting them in this same unit if (var["FREE_" || trigger.data["NFT"]]) var["FREE_" || trigger.data["NFT"]] ||= {claimed: true}; else var["NFT_" || trigger.data["NFT"]] ||= {claimed: true}; //No longer listed for sale //if ($hasMaecenas AND $NFT.bid > 20000) // var["artist_" || $NFT.author] ||= {share: $artist.share + $maecenasShare}; //Increase the artist pool value }" } ] }, { "messages": [ { "app": "payment", "if": "{ $NFT.soldBy != $NFT.by //If we don't do this check the seller could steal bytes from the AA if there were no bids }", "init": "{ $share = $NFT.bid * (1 - $HOUSE_CUT); if ($NFT.royalty > 0 AND $NFT.soldBy != $NFT.by){ $royalty = $share * $NFT.royalty / 100; $outputs = [ {//Royalty address: $NFT.author, amount: floor($royalty, 0) //Percentage of the whole bid, no fees are charged }, {//Seller payment address: $NFT.soldBy, amount: floor($share - $royalty, 0) //We take the house cut then royalty cut } ]; } else{ $outputs = [ {//Seller payment address: $NFT.soldBy, amount: floor($share, 0) } ]; } $payload = {outputs: $outputs}; }", "payload": "{$payload}" }, { "app": "payment", "payload": { "asset": "{trigger.data['NFT']}", "outputs": [ { "address": "{$NFT.by}", "amount": 1 } ] } }, { "app": "state", "state": "{ var["locked"] -= $NFT.bid; if (var["FREE_" || trigger.data["NFT"]]) var["FREE_" || trigger.data["NFT"]] ||= {claimed: true}; else var["NFT_" || trigger.data["NFT"]] ||= {claimed: true}; //No longer listed for sale }" } ] } ] } } ] } }, { "if": "{ $method == "REDEEM" }", "init": "{ if (exists(trigger.data["NFT"])){//Redeeming an inflationary NFT $NFT = $saleInfo(trigger.data["NFT"]); if ($NFT.author != trigger.address) //Check the author is the one redeeming the batch bounce("Only the NFT author can redeem the batch"); if ($NFT.supplyOpts["autoSell"]) bounce("You cannot redeem a batch that is supposed to be automatically listed for sale"); if ($NFT.lastBatchDate > timestamp - $NFT.supplyOpts["period"]) bounce("You cannot redeem a new batch batch yet"); } else{ if (NOT exists(trigger.data["signed_message"])) bounce("signed_message field is mandatory"); if (NOT exists(trigger.data["authors"])) bounce('authors field is mandatory'); if (NOT exists(trigger.data["meta"])) bounce("meta field is mandatory"); $spack = { signed_message: trigger.data["signed_message"], authors: trigger.data["authors"] } || trigger.data["meta"]; if (NOT is_valid_signed_package($spack, $spack.authors[0]["address"])) bounce("The signed message signature is invalid"); if (NOT $spack.signed_message["serial"]) bounce("The signed message does not contains a serial field"); if (NOT $spack.signed_message["NFT"]) bounce("The signer of this message did not include a NFT field"); //Check if the code has already been redeemed if (var["CODE_" || $spack.signed_message["NFT"] || "_" || sha256($spack.signed_message["serial"])]) bounce("That code was already redeemed"); $NFT = $saleInfo($spack.signed_message["NFT"]); if (NOT $NFT) bounce("That NFT does not exist"); //Prevent author from stealing their NFTs from other people if ($NFT.soldBy != $NFT.author) bounce("You cannot claim an NFT that is not being sold by their author"); if ($NFT.author != $spack.authors[0]["address"]) bounce("The signer of that message is not the author of that NFT"); //NFTs have to be created with redeemable: true to be redeemable if (NOT $NFT.redeem) bounce("That NFT is not redeemable"); //Check if sale is over and expire was set in the signed message if ($spack.signed_message["expireDate"]) if ($NFT.soldAt < timestamp OR $spack.signed_message["expireDate"] < timestamp) bounce("That NFT is no longer claimable"); //Check if the NFT is address-locked if ($spack.signed_message["claimer"]) if ($spack.signed_message["claimer"] != trigger.address) bounce("You cannot claim that NFT from that address"); $signedAmount = $spack.signed_message["amount"] OTHERWISE 1; //Uncapped asset, we can issue whatever amount was signed if (NOT $NFT.cap){ $amount = $signedAmount; } //If it is going over the cap we need to reduce the claimed amount to match the asset cap else { if ($NFT.unitsSold + $signedAmount > $NFT.cap) $amount = $NFT.cap - $NFT.unitsSold; else $amount = $signedAmount; } //All copies were issued if ($amount == 0) bounce("All copies of that NFT have been already minted"); } }", "messages": { "cases": [ { "if": "{ trigger.data["NFT"] }", "messages": [ { "app": "state", "state": "{ if (var["FREE_" || trigger.data["NFT"]]) var["FREE_" || trigger.data["NFT"]] ||= {lastBatchDate: timestamp}; else var["NFT_" || trigger.data["NFT"]] ||= {lastBatchDate: timestamp}; }" } ] }, { "messages": [ { "app": "payment", "payload": { "asset": "{$spack.signed_message["NFT"]}", "outputs": [ { "address": "{trigger.address}", "amount": "{$amount}" } ] } }, { "app": "state", "state": "{ if (var["FREE_" || $spack.signed_message["NFT"]]) var["FREE_" || $spack.signed_message["NFT"]] ||= {unitsSold: $NFT.unitsSold + $amount}; else var["NFT_" || $spack.signed_message["NFT"]] ||= {unitsSold: $NFT.unitsSold + $amount}; var["CODE_" || $spack.signed_message["NFT"] || "_" || sha256($spack.signed_message["serial"], 'base64')] = true; //Hash the serial to prevent the issuer from sending too big serials to bloat state size }" } ] } ] } }, { "if": "{ $method == "SELL" }", "init": "{ //Parameter validation if (NOT exists(trigger.data["endTime"])) bounce("endTime field is mandatory"); if (trigger.data["endTime"] < timestamp) bounce("You cannot set the end time in the past"); if (trigger.data["endTime"] > timestamp + 2628000) bounce("You cannot set the end time in more than 30 days"); if (NOT exists(trigger.data["initialPrice"])) bounce("initialPrice field is mandatory"); if (NOT is_valid_amount(trigger.data["initialPrice"])) bounce("initialPrice must be a valid amount"); if (trigger.data["initialPrice"] < 20000) bounce("The minimum initialPrice is 20000 bytes"); //NFT validation if (trigger.output[[asset!="base"]].asset == "ambigous") bounce("You cannot send more than one NFT type at a time"); //Check that the NFT is not revoked $feedData = data_feed[[oracles=this_address, feed_name=trigger.output[[asset!="base"]].asset, ifnone="OK", ifseveral="last", type="string"]]; if ($feedData == "REVOKED") bounce("We revoked the trading of that token probably due to copyright reasons"); $NFT = $saleInfo(trigger.output[[asset!="base"]].asset) OTHERWISE {}; if ($NFT.supplyOpts) bounce("You cannot resell inflationary supply NFTs here"); if (NOT $NFT) bounce("NFT not found"); if (trigger.output[[asset!="base"]].amount != 1) bounce("You cannot auction more than 1 copy of an NFT at a time"); if ($NFT.soldAt >= timestamp) bounce("A copy of that NFT is already being sold. The sale is expected to end at " || timestamp_to_string($NFT.soldAt) || " UTC but may be delayed depending on the number of bids"); if ($NFT.bid AND NOT $NFT.claimed) bounce("That NFT was auctioned but the payment has not been claimed yet. You can force the saller to claim it."); }", "messages": [ { "app": "state", "state": "{ $NFT.bid = trigger.data["initialPrice"]; $NFT.bids = 0; $NFT.by = trigger.address; $NFT.soldBy = trigger.address; $NFT.at = timestamp; $NFT.claimed = false; $NFT.soldAt = trigger.data["endTime"]; if (var["FREE_" || trigger.output[[asset!="base"]].asset]) var["FREE_" || trigger.output[[asset!="base"]].asset] = $stripToken($NFT) || {claimed: false}; else var["NFT_" || trigger.output[[asset!="base"]].asset] = $stripToken($NFT) || {claimed: false}; }" } ] }, { "if": "{ $method == "CHANGE_PRICE" }", "init": "{ if (NOT trigger.data["NFT"]) bounce("NFT field is mandatory"); $NFT = var["NFT_" || trigger.data["NFT"]]; if (NOT $NFT) bounce("NFT not found"); if (NOT trigger.data["price"]) bounce("price field is mandatory"); if (NOT is_integer(trigger.data["price"])) bounce("price must be an integer"); if (NOT is_valid_amount(trigger.data["price"])) bounce("price is invalid"); if (trigger.data["price"] < 20000) bounce("The minimum price is 20000 bytes"); if ($NFT.bid) bounce("You cannot change an auction price!"); }", "messages": [ { "app": "state", "state": "{ var["NFT_" || trigger.data["NFT"]] ||= {price: trigger.data["price"]}; }" } ] } ] } } ]
Technical information
Fees:
41,975 bytes
(452 headers, 41523 payload)
Level:2026583
Witnessed level:2026575
Main chain index:2018592
Latest included mc index:2018591
Status:stable/confirmed/final