tx · 6rKDqx4EXLoSJdB9fAbCQ4JWSpwHc9pMceCj5oc1jvTs 3N2D6tJAXzcTDgUiTWmcdCHd6ctYwNENuBn: -0.01000000 Waves 2022.11.01 14:05 [2297967] smart account 3N2D6tJAXzcTDgUiTWmcdCHd6ctYwNENuBn > SELF 0.00000000 Waves
{ "type": 13, "id": "6rKDqx4EXLoSJdB9fAbCQ4JWSpwHc9pMceCj5oc1jvTs", "fee": 1000000, "feeAssetId": null, "timestamp": 1667300704134, "version": 2, "chainId": 84, "sender": "3N2D6tJAXzcTDgUiTWmcdCHd6ctYwNENuBn", "senderPublicKey": "Cq4RbcVcQJpCWyHiACDqUf2PbU8PaX949Dun9wRSeDzE", "proofs": [ "4ZQxoqXa68ExXFZDkt1UgETnPJTkNjsWahuDhjzjNrMtictdtcBJ2CTVFojVcJjjMzoKZPh7FK495XVZepNxTu6z" ], "script": "base64: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", "height": 2297967, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 6VTDJ68Wkk9JEBK88fpao6zg9eEXqXSDi7APguEGhtit Next: 4oCboyW3iKd9zTC8rQ1LM8YrYjvKXiUMDdTPVih4hYb6 Full:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | - | func tryGetInteger (key) = match getInteger(this, key) { | |
5 | - | case b: Int => | |
4 | + | let ownerAddress = base58'' | |
5 | + | ||
6 | + | let nftAddress = base58'' | |
7 | + | ||
8 | + | let artefactsAddress = base58'' | |
9 | + | ||
10 | + | let feeAggregator = base58'' | |
11 | + | ||
12 | + | let nftAssetId = base58'' | |
13 | + | ||
14 | + | let kGlobalCounter = "GLOBAL_COUNTER" | |
15 | + | ||
16 | + | let feePercent = 5 | |
17 | + | ||
18 | + | let artefactsFeePercent = 20 | |
19 | + | ||
20 | + | func tryGetInteger (key) = { | |
21 | + | let val = match getInteger(this, key) { | |
22 | + | case b: Int => | |
23 | + | b | |
24 | + | case _ => | |
25 | + | 0 | |
26 | + | } | |
27 | + | val | |
28 | + | } | |
29 | + | ||
30 | + | ||
31 | + | func tryGetString (key) = { | |
32 | + | let val = match getString(this, key) { | |
33 | + | case b: String => | |
34 | + | b | |
35 | + | case _ => | |
36 | + | "" | |
37 | + | } | |
38 | + | val | |
39 | + | } | |
40 | + | ||
41 | + | ||
42 | + | func tryGetBoolean (key) = match getBoolean(this, key) { | |
43 | + | case b: Boolean => | |
6 | 44 | b | |
7 | 45 | case _ => | |
8 | - | | |
46 | + | false | |
9 | 47 | } | |
10 | 48 | ||
11 | 49 | ||
12 | - | func tryGetString (key) = match getString(this, key) { | |
13 | - | case b: String => | |
14 | - | b | |
15 | - | case _ => | |
16 | - | "" | |
17 | - | } | |
50 | + | func getGlobalCounter () = tryGetInteger(kGlobalCounter) | |
18 | 51 | ||
19 | 52 | ||
20 | - | func checkNFT (asset) = if (if (if ((asset.issuer.bytes == this.bytes)) | |
21 | - | then (asset.decimals == 0) | |
22 | - | else false) | |
23 | - | then (asset.reissuable == false) | |
24 | - | else false) | |
25 | - | then (asset.quantity == 1) | |
26 | - | else false | |
53 | + | func getAssetGenotype (assetId) = if ((value(assetInfo(assetId)).issuer == Address(artefactsAddress))) | |
54 | + | then "ART" | |
55 | + | else { | |
56 | + | let asset1 = match getString(Address(ownerAddress), toBase58String(assetId)) { | |
57 | + | case b: String => | |
58 | + | b | |
59 | + | case _ => | |
60 | + | "" | |
61 | + | } | |
62 | + | if ((asset1 != "")) | |
63 | + | then asset1 | |
64 | + | else { | |
65 | + | let asset2 = match getString(Address(nftAddress), toBase58String(assetId)) { | |
66 | + | case c: String => | |
67 | + | c | |
68 | + | case _ => | |
69 | + | "" | |
70 | + | } | |
71 | + | if ((asset2 != "")) | |
72 | + | then asset2 | |
73 | + | else "" | |
74 | + | } | |
75 | + | } | |
27 | 76 | ||
28 | 77 | ||
29 | 78 | @Callable(i) | |
30 | - | func createNFT (nftName,nftDesc,image,collectionName) = if (if ((toString(i.caller) != "")) | |
31 | - | then (toString(i.caller) != "") | |
32 | - | else false) | |
33 | - | then throw("issuance of new NFTs is limited") | |
34 | - | else if ((indexOf(image, "https://") != 0)) | |
35 | - | then throw("https link to image required") | |
36 | - | else { | |
37 | - | let asset = Issue(nftName, nftDesc, 1, 0, false, unit, 0) | |
38 | - | let assetId = calculateAssetId(asset) | |
39 | - | let assetIdStr = toBase58String(assetId) | |
40 | - | let collectionDomain = ((toString(i.caller) + "_") + (if ((size(collectionName) > 0)) | |
41 | - | then collectionName | |
42 | - | else "basic")) | |
43 | - | let nftNum = (tryGetInteger((("collection_" + collectionDomain) + "_size")) + 1) | |
44 | - | $Tuple2([StringEntry((("nft_" + assetIdStr) + "_data"), (((("{'collection': '" + collectionDomain) + "', 'num': ") + toString(nftNum)) + "}")), StringEntry((("nft_" + assetIdStr) + "_image"), image), StringEntry((("nft_" + assetIdStr) + "_issuer"), toString(i.caller)), IntegerEntry((("collection_" + collectionDomain) + "_size"), nftNum), StringEntry(((("collection_" + collectionDomain) + "_assetId_") + assetIdStr), assetIdStr), asset, ScriptTransfer(i.caller, 1, assetId)], assetIdStr) | |
45 | - | } | |
46 | - | ||
47 | - | ||
48 | - | ||
49 | - | @Callable(i) | |
50 | - | func burnNFT () = { | |
51 | - | let assetId = value(value(i.payments[0]).assetId) | |
52 | - | let asset = value(assetInfo(assetId)) | |
53 | - | if (if ((value(i.payments[0]).amount == 1)) | |
54 | - | then checkNFT(asset) | |
55 | - | else false) | |
56 | - | then { | |
57 | - | let assetIdStr = toBase58String(assetId) | |
58 | - | let issuer = tryGetString((("nft_" + assetIdStr) + "_issuer")) | |
59 | - | let data = tryGetString((("nft_" + assetIdStr) + "_data")) | |
60 | - | let partsData = split(data, ", ") | |
61 | - | let partsCollection = split(partsData[0], ": ") | |
62 | - | let collectionDomain = dropRight(drop(partsCollection[1], 1), 1) | |
63 | - | let nftNum = tryGetInteger((("collection_" + collectionDomain) + "_size")) | |
64 | - | if ((nftNum > 0)) | |
65 | - | then { | |
66 | - | let updateCollection = if ((nftNum == 1)) | |
67 | - | then DeleteEntry((("collection_" + collectionDomain) + "_size")) | |
68 | - | else IntegerEntry((("collection_" + collectionDomain) + "_size"), (nftNum - 1)) | |
69 | - | [updateCollection, Burn(assetId, 1), DeleteEntry((("nft_" + assetIdStr) + "_data")), DeleteEntry((("nft_" + assetIdStr) + "_image")), DeleteEntry((("nft_" + assetIdStr) + "_issuer")), DeleteEntry(((("collection_" + collectionDomain) + "_assetId_") + assetIdStr))] | |
70 | - | } | |
71 | - | else throw("Ivalid collection name") | |
72 | - | } | |
73 | - | else throw("Ivalid NFT attached") | |
79 | + | func nftBuy (nftId) = { | |
80 | + | let isnftart = tryGetBoolean((("nft_" + nftId) + "_nftart")) | |
81 | + | let paymentAssetId = if ((isnftart == true)) | |
82 | + | then nftAssetId | |
83 | + | else unit | |
84 | + | let paymentAssetName = if ((isnftart == true)) | |
85 | + | then "nftart" | |
86 | + | else "WAVES" | |
87 | + | let pmt = value(i.payments[0]) | |
88 | + | let amount = pmt.amount | |
89 | + | let assetId = tryGetString((("nft_" + nftId) + "_assetId")) | |
90 | + | let bidId = toBase58String(i.transactionId) | |
91 | + | let nftStatus = tryGetString((("nft_" + nftId) + "_status")) | |
92 | + | let finalFee = if ((tryGetBoolean((("nft_" + nftId) + "_isArtefact")) == true)) | |
93 | + | then artefactsFeePercent | |
94 | + | else feePercent | |
95 | + | if ((pmt.assetId != paymentAssetId)) | |
96 | + | then throw((("nft does only support " + paymentAssetName) + " payments.")) | |
97 | + | else if ((nftStatus != "open")) | |
98 | + | then throw("nft is closed. You cannot place a bid.") | |
99 | + | else { | |
100 | + | let instantPrice = tryGetInteger((("nft_" + nftId) + "_instantPrice")) | |
101 | + | if ((instantPrice > amount)) | |
102 | + | then throw(("For instant purchase you need to pay at least " + toString(instantPrice))) | |
103 | + | else { | |
104 | + | let nftOwner = tryGetString((("nft_" + nftId) + "_owner")) | |
105 | + | let recordPrice = tryGetInteger((assetId + "_recordPrice")) | |
106 | + | let newRecordPrice = max([recordPrice, amount]) | |
107 | + | [StringEntry((("nft_" + nftId) + "_status"), "finished"), IntegerEntry((("nft_" + nftId) + "_finalPrice"), amount), IntegerEntry((assetId + "_recordPrice"), newRecordPrice), StringEntry((((("assetId_" + assetId) + "_bid_") + bidId) + "_data"), (((((("{\"nftId\": \"" + nftId) + "\", \"author\": \"") + toString(i.originCaller)) + "\", \"amount\": \"") + toString(amount)) + "\", \"status\": \"finished\"}")), DeleteEntry((((("address_" + nftOwner) + "_nft_") + nftId) + "_lockedNFT")), ScriptTransfer(i.originCaller, 1, fromBase58String(getStringValue(this, (("nft_" + nftId) + "_assetId")))), ScriptTransfer(addressFromStringValue(nftOwner), fraction(amount, (100 - finalFee), 100), paymentAssetId), ScriptTransfer(Address(feeAggregator), fraction(amount, finalFee, 100), paymentAssetId)] | |
108 | + | } | |
109 | + | } | |
74 | 110 | } | |
75 | 111 | ||
76 | - | ||
77 | - | @Verifier(tx) | |
78 | - | func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
79 | 112 |
github/deemru/w8io/169f3d6 34.06 ms ◑![]()