tx · 7pptHg1fqj5afMtT5LxL6hAQyWaUiMpzCpJcbYW5GLrd 3NBwZbvD45CscLGF78qTaDRBZ7mDBHt3Yib: -0.01000000 Waves 2022.08.10 08:51 [2177949] smart account 3NBwZbvD45CscLGF78qTaDRBZ7mDBHt3Yib > SELF 0.00000000 Waves
{ "type": 13, "id": "7pptHg1fqj5afMtT5LxL6hAQyWaUiMpzCpJcbYW5GLrd", "fee": 1000000, "feeAssetId": null, "timestamp": 1660110689600, "version": 2, "chainId": 84, "sender": "3NBwZbvD45CscLGF78qTaDRBZ7mDBHt3Yib", "senderPublicKey": "3q3N8Kv49DbzNHyTB4A87UYz2DdquXCgP3GNCcD9Zj66", "proofs": [ "3UVU5vmAgZvGrpuRSgs81SNUHtY7DnPj8LDsiinH8d82razfperkFE4c1T7Cfs39obkFCQt6zMVixmkyWcmh2g1g" ], "script": "base64:AAIFAAAAAAAAAC0IAhIGCgQICAgIEgQKAggBEgQKAggIEgQKAggIEgQKAggIEgQKAgQIEgMKAQgAAAAxAAAAAAtDT05GX0lOSVRfSwIAAAALY29uZmlnX2luaXQAAAAAC0NPTkZfQU1JTl9LAgAAAAxjb25maWdfYWRtaW4AAAAADUNPTkZfT1JBQ0xFX0sCAAAADWNvbmZpZ19vcmFjbGUAAAAAE0NPTkZfQVJUV09SS19EQVBQX0sCAAAAE2NvbmZpZ19hcnR3b3JrX2RhcHAAAAAAEUNPTkZfVVNFUlNfREFQUF9LAgAAABFjb25maWdfdXNlcnNfZGFwcAAAAAAQQ09ORl9GRUVfQVNTRVRfSwIAAAAQY29uZmlnX2ZlZV9hc3NldAAAAAARQ09ORl9GRUVfQU1PVU5UX0sCAAAAEWNvbmZpZ19mZWVfYW1vdW50AAAAAA5kYXBwUnVubmluZ0tleQIAAAAUY29uZl9kYXBwX2lzX3J1bm5pbmcAAAAAEW1haW50ZW5hbmNlTVNHS2V5AgAAABRjb25mX21haW50ZW5hbmNlX21zZwAAAAALZGFwcFJ1bm5pbmcJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQbAAAAAgUAAAAEdGhpcwUAAAAOZGFwcFJ1bm5pbmdLZXkGAAAAAA5tYWludGVuYW5jZU1TRwkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABB0AAAACBQAAAAR0aGlzBQAAABFtYWludGVuYW5jZU1TR0tleQIAAAAAAAAAAA1zdGF0dXNfTE9DS0VEAgAAAAZMT0NLRUQAAAAAD3N0YXR1c19VTkxPQ0tFRAIAAAAIVU5MT0NLRUQAAAAADnN0YXR1c19QRU5ESU5HAgAAAAdQRU5ESU5HAAAAAA1zdGF0dXNfQ0xPU0VEAgAAAAZDTE9TRUQAAAAAEWNyZWF0b3JSZWdpc3RlcmVkAgAAAApSRUdJU1RFUkVEAAAAAA9jcmVhdG9yVmVyaWZpZWQCAAAACFZFUklGSUVEAAAAAAphcnRGbGFnZ2VkAgAAAAdGTEFHR0VEAAAAAAphcnRJbGxlZ2FsAgAAAAdJTExFR0FMAAAAAAphcnRQZW5kaW5nAgAAAA9DSEFOR0VfUkVRVUlSRUQAAAAACE5FVFdPUktTCQAETAAAAAICAAAAA0VUSAkABEwAAAACAgAAAAdQT0xZR09OBQAAAANuaWwBAAAADGtleU5GVHN0YXR1cwAAAAEAAAAFbmZ0SWQJAAEsAAAAAgkAASwAAAACAgAAAARuZnRfBQAAAAVuZnRJZAIAAAAHX3N0YXR1cwEAAAAOa2V5VHJhbnNmZXJORlQAAAACAAAABW5mdElkAAAABHR4aWQJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAACXRyYW5zZmVyXwUAAAAFbmZ0SWQCAAAAAV8FAAAABHR4aWQAAAAAD2tleUZlZU5vbk1pbnRlZAIAAAAPY29zdF9ub25fbWludGVkAAAAAAxrZXlGZWVNaW50ZWQCAAAAC2Nvc3RfbWludGVkAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEHQAAAAIFAAAABHRoaXMFAAAAA2tleQIAAAAAAQAAAA9nZXRJbnRlZ2VyQnlLZXkAAAABAAAAA2tleQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABBoAAAACBQAAAAR0aGlzBQAAAANrZXkAAAAAAAAAAAABAAAAD2dldEJvb2xlYW5CeUtleQAAAAEAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEGwAAAAIFAAAABHRoaXMFAAAAA2tleQcAAAAABWFkbWluCQEAAAAFdmFsdWUAAAABCQAEJgAAAAEJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEFAAAAC0NPTkZfQU1JTl9LAAAAAAxhcnR3b3Jrc0RhcHAJAQAAAAV2YWx1ZQAAAAEJAAQmAAAAAQkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAATQ09ORl9BUlRXT1JLX0RBUFBfSwAAAAAJdXNlcnNEYXBwCQEAAAAFdmFsdWUAAAABCQAEJgAAAAEJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEFAAAAEUNPTkZfVVNFUlNfREFQUF9LAAAAAApvcmFjbGVBZGRyCQEAAAAFdmFsdWUAAAABCQAEJgAAAAEJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEFAAAADUNPTkZfT1JBQ0xFX0sAAAAAEGZlZUFjY2VwdGVkQXNzZXQJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEFAAAAEENPTkZfRkVFX0FTU0VUX0sAAAAAGGZlZUFjY2VwdGVkQW1vdW50Rm9yU2lnbgkBAAAAD2dldEludGVnZXJCeUtleQAAAAEFAAAAEUNPTkZfRkVFX0FNT1VOVF9LAQAAABdnZXRTdHJpbmdGcm9tVXNlcnNCeUtleQAAAAEAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEHQAAAAIFAAAACXVzZXJzRGFwcAUAAAADa2V5AgAAAAABAAAAGmdldFN0cmluZ0Zyb21BcnR3b3Jrc0J5S2V5AAAAAQAAAANrZXkJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQdAAAAAgUAAAAMYXJ0d29ya3NEYXBwBQAAAANrZXkCAAAAAAEAAAAZZ2V0SW50ZWdlckJ5S2V5RnJvbU9yYWNsZQAAAAEAAAADa2V5CQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQaAAAAAgUAAAAKb3JhY2xlQWRkcgUAAAADa2V5AgAAAB9JbnRlZ2VyIHVuZGVmaW5lIG9yIDAgaW4gb3JhY2xlAQAAAAxhc3NldElzVmFsaWQAAAABAAAADGFzc2V0RGV0YWlscwMDAwkAAAAAAAACCAUAAAAMYXNzZXREZXRhaWxzAAAACHF1YW50aXR5AAAAAAAAAAABCQAAAAAAAAIIBQAAAAxhc3NldERldGFpbHMAAAAIZGVjaW1hbHMAAAAAAAAAAAAHCQAAAAAAAAIIBQAAAAxhc3NldERldGFpbHMAAAAKcmVpc3N1YWJsZQcHCQAAAAAAAAIIBQAAAAxhc3NldERldGFpbHMAAAAGaXNzdWVyBQAAAAxhcnR3b3Jrc0RhcHAHAQAAAAxpc1NpZ25BcnRORlQAAAABAAAAAmlkBAAAAAxhc3NldERldGFpbHMJAQAAAAV2YWx1ZQAAAAEJAAPsAAAAAQUAAAACaWQJAQAAAAxhc3NldElzVmFsaWQAAAABBQAAAAxhc3NldERldGFpbHMBAAAACWlzUGF5bWVudAAAAAEAAAABaQMJAAAAAAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAkAAAIAAAABAgAAABNObyBwYXltZW50IGF0dGFjaGVkBAAAAAtuZnRfcGF5bWVudAkBAAAABXZhbHVlAAAAAQkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAQAAAAKbmZ0X2Ftb3VudAkBAAAABXZhbHVlAAAAAQgFAAAAC25mdF9wYXltZW50AAAABmFtb3VudAQAAAAPZ2F0ZXdheV9wYXltZW50CQEAAAAFdmFsdWUAAAABCQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAABBAAAAA5nYXRld2F5X2Ftb3VudAkBAAAABXZhbHVlAAAAAQgFAAAAD2dhdGV3YXlfcGF5bWVudAAAAAZhbW91bnQJAAUWAAAABAUAAAALbmZ0X3BheW1lbnQFAAAACm5mdF9hbW91bnQFAAAAD2dhdGV3YXlfcGF5bWVudAUAAAAOZ2F0ZXdheV9hbW91bnQAAAAABmlzSW5pdAkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkBAAAAD2dldEJvb2xlYW5CeUtleQAAAAEFAAAAC0NPTkZfSU5JVF9LBwEAAAAJb3duZXJPbmx5AAAAAQAAAAFpBAAAAAZjYWxsZXIJAAQlAAAAAQgFAAAAAWkAAAAGY2FsbGVyBAAAAAJpZAkAAlgAAAABCAUAAAABaQAAAA10cmFuc2FjdGlvbklkBAAAAAphbGxvd2VkQXJyAwUAAAAGaXNJbml0CQAETAAAAAIJAAQlAAAAAQUAAAAFYWRtaW4JAARMAAAAAgkABCUAAAABBQAAAAR0aGlzBQAAAANuaWwJAARMAAAAAgkABCUAAAABBQAAAAR0aGlzBQAAAANuaWwDCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgUAAAAKYWxsb3dlZEFycgUAAAAGY2FsbGVyBgcBAAAADGlkQ2FsbGVyRGF0ZQAAAAEAAAABaQMJAQAAAAEhAAAAAQUAAAALZGFwcFJ1bm5pbmcJAAACAAAAAQUAAAAObWFpbnRlbmFuY2VNU0cJAAUVAAAAAwkAAlgAAAABCAUAAAABaQAAAA10cmFuc2FjdGlvbklkCQACWAAAAAEICAUAAAABaQAAAAZjYWxsZXIAAAAFYnl0ZXMIBQAAAAlsYXN0QmxvY2sAAAAJdGltZXN0YW1wAQAAABNnZXRDcmVhdG9yRnJvbU5GVGlkAAAAAQAAAAVuZnRpZAQAAAAHbmZ0RGF0YQkBAAAAGmdldFN0cmluZ0Zyb21BcnR3b3Jrc0J5S2V5AAAAAQkAASwAAAACAgAAAARuZnRfBQAAAAVuZnRpZAQAAAAHY3JlYXRvcgkAAZEAAAACCQAEtQAAAAIFAAAAB25mdERhdGECAAAAAV8AAAAAAAAAAAYEAAAABWFydElkCQABkQAAAAIJAAS1AAAAAgUAAAAHbmZ0RGF0YQIAAAABXwAAAAAAAAAABQkABRQAAAACBQAAAAdjcmVhdG9yBQAAAAVhcnRJZAEAAAATZ2V0Q3JlYXRvckVSQzIwQWRkcgAAAAEAAAAEYWRkcgQAAAAJZXJjMjBBZGRyCQEAAAAXZ2V0U3RyaW5nRnJvbVVzZXJzQnlLZXkAAAABCQABLAAAAAICAAAAC3VzZXJfRVJDMjBfBQAAAARhZGRyBQAAAAllcmMyMEFkZHIBAAAAEGdldENyZWF0b3JTdGF0dXMAAAABAAAABGFkZHIEAAAABnN0YXR1cwkBAAAAF2dldFN0cmluZ0Zyb21Vc2Vyc0J5S2V5AAAAAQkAASwAAAACAgAAAAx1c2VyX3N0YXR1c18FAAAABGFkZHIFAAAABnN0YXR1cwEAAAAMZ2V0QXJ0U3RhdHVzAAAAAQAAAAVhcnRJZAQAAAAGc3RhdHVzCQEAAAAaZ2V0U3RyaW5nRnJvbUFydHdvcmtzQnlLZXkAAAABCQABLAAAAAICAAAADHVzZXJfc3RhdHVzXwUAAAAFYXJ0SWQFAAAABnN0YXR1cwEAAAAOaXNWYWxpZENyZWF0b3IAAAABAAAABnN0YXR1cwMJAAAAAAAAAgUAAAAGc3RhdHVzBQAAABFjcmVhdG9yUmVnaXN0ZXJlZAYJAAAAAAAAAgUAAAAGc3RhdHVzBQAAAA9jcmVhdG9yVmVyaWZpZWQBAAAACmlzVmFsaWRBcnQAAAABAAAABWFydElkAwMJAQAAAAIhPQAAAAIFAAAABWFydElkBQAAAAphcnRGbGFnZ2VkBgkBAAAAAiE9AAAAAgUAAAAFYXJ0SWQFAAAACmFydElsbGVnYWwGCQEAAAACIT0AAAACBQAAAAVhcnRJZAUAAAAKYXJ0UGVuZGluZwAAAAcAAAABaQEAAAAEaW5pdAAAAAQAAAAJYWRtaW5BZGRyAAAACm9yYWNsZUFkZHIAAAAMYXJ0d29ya3NEYXBwAAAACXVzZXJzRGFwcAMJAQAA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AAIAAAABCQABLAAAAAICAAAAE0NyZWF0b3Igc3RhdHVzIGlzIDoJAQAAABBnZXRDcmVhdG9yU3RhdHVzAAAAAQUAAAAHY3JlYXRvcgMJAQAAAAEhAAAAAQkBAAAACmlzVmFsaWRBcnQAAAABCQEAAAAMZ2V0QXJ0U3RhdHVzAAAAAQUAAAAFYXJ0SWQJAAACAAAAAQkAASwAAAACAgAAABNBcnR3b3JrIHN0YXR1cyBpcyA6CQEAAAAMZ2V0QXJ0U3RhdHVzAAAAAQUAAAAFYXJ0SWQEAAAACWVyYzIwQWRkcgkBAAAAE2dldENyZWF0b3JFUkMyMEFkZHIAAAABBQAAAAdjcmVhdG9yAwkAAAAAAAACCQABMQAAAAEFAAAACWVyYzIwQWRkcgAAAAAAAAAAAAkAAAIAAAABAgAAACxUaGlzIGFydGlzdCBkaWRuJ3QgYWxsb3cgY3Jvc3NjaGFpbiB0cmFuc2ZlcgQAAAAJbmZ0U3RhdHVzCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAMa2V5TkZUc3RhdHVzAAAAAQUAAAAFbmZ0SWQDAwkAAAAAAAACBQAAAAluZnRTdGF0dXMFAAAADXN0YXR1c19MT0NLRUQGCQAAAAAAAAIFAAAACW5mdFN0YXR1cwUAAAAOc3RhdHVzX1BFTkRJTkcJAAACAAAAAQIAAAASTkZUIGFscmVhZHkgbG9ja2VkBAAAAA9nYXNGZWVOb25NaW50ZWQJAQAAABlnZXRJbnRlZ2VyQnlLZXlGcm9tT3JhY2xlAAAAAQUAAAAPa2V5RmVlTm9uTWludGVkBAAAAAxnYXNGZWVNaW50ZWQJAQAAABlnZXRJbnRlZ2VyQnlLZXlGcm9tT3JhY2xlAAAAAQUAAAAMa2V5RmVlTWludGVkBAAAAAhpc01pbnRlZAMJAAAAAAAAAgUAAAAJbmZ0U3RhdHVzBQAAAA9zdGF0dXNfVU5MT0NLRUQGBwMDCQEAAAABIQAAAAEJAQAAAAlpc0RlZmluZWQAAAABCAUAAAAPZ2F0ZXdheV9wYXltZW50AAAAB2Fzc2V0SWQGCQEAAAACIT0AAAACCQACWAAAAAEJAQAAAAV2YWx1ZQAAAAEIBQAAAA9nYXRld2F5X3BheW1lbnQAAAAHYXNzZXRJZAUAAAAQZmVlQWNjZXB0ZWRB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", "height": 2177949, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 3aGqYGUhmHpimanQ8Ujifh6M5ZbNtPSxe7PPzSZJsfm2 Next: 216QeYiktwm5XZ7GBUrG1YeV5eWG8sHq7ut6uxYjSvme Diff:
Old | New | Differences | |
---|---|---|---|
242 | 242 | else if (if (!(isMinted)) | |
243 | 243 | then ((gateway_amount + feeAcceptedAmountForSign) > gasFeeNonMinted) | |
244 | 244 | else false) | |
245 | - | then throw("Wrong payment amount to mint and unlock") | |
245 | + | then throw(("Wrong payment amount to mint and unlock " + toString((gateway_amount + feeAcceptedAmountForSign)))) | |
246 | 246 | else [StringEntry(keyNFTstatus(nftId), status_PENDING), StringEntry(((keyTransferNFT(nftId, id) + "_") + status_PENDING), ((((((((((((recipient + "_") + network) + "_") + caller) + "_") + id) + "_") + value(assetInfo(value(nft_payment.assetId))).name) + "_") + toString(date)) + "_") + toString(height)))] | |
247 | 247 | } | |
248 | 248 | } |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let CONF_INIT_K = "config_init" | |
5 | 5 | ||
6 | 6 | let CONF_AMIN_K = "config_admin" | |
7 | 7 | ||
8 | 8 | let CONF_ORACLE_K = "config_oracle" | |
9 | 9 | ||
10 | 10 | let CONF_ARTWORK_DAPP_K = "config_artwork_dapp" | |
11 | 11 | ||
12 | 12 | let CONF_USERS_DAPP_K = "config_users_dapp" | |
13 | 13 | ||
14 | 14 | let CONF_FEE_ASSET_K = "config_fee_asset" | |
15 | 15 | ||
16 | 16 | let CONF_FEE_AMOUNT_K = "config_fee_amount" | |
17 | 17 | ||
18 | 18 | let dappRunningKey = "conf_dapp_is_running" | |
19 | 19 | ||
20 | 20 | let maintenanceMSGKey = "conf_maintenance_msg" | |
21 | 21 | ||
22 | 22 | let dappRunning = valueOrElse(getBoolean(this, dappRunningKey), true) | |
23 | 23 | ||
24 | 24 | let maintenanceMSG = valueOrElse(getString(this, maintenanceMSGKey), "") | |
25 | 25 | ||
26 | 26 | let status_LOCKED = "LOCKED" | |
27 | 27 | ||
28 | 28 | let status_UNLOCKED = "UNLOCKED" | |
29 | 29 | ||
30 | 30 | let status_PENDING = "PENDING" | |
31 | 31 | ||
32 | 32 | let status_CLOSED = "CLOSED" | |
33 | 33 | ||
34 | 34 | let creatorRegistered = "REGISTERED" | |
35 | 35 | ||
36 | 36 | let creatorVerified = "VERIFIED" | |
37 | 37 | ||
38 | 38 | let artFlagged = "FLAGGED" | |
39 | 39 | ||
40 | 40 | let artIllegal = "ILLEGAL" | |
41 | 41 | ||
42 | 42 | let artPending = "CHANGE_REQUIRED" | |
43 | 43 | ||
44 | 44 | let NETWORKS = ["ETH", "POLYGON"] | |
45 | 45 | ||
46 | 46 | func keyNFTstatus (nftId) = (("nft_" + nftId) + "_status") | |
47 | 47 | ||
48 | 48 | ||
49 | 49 | func keyTransferNFT (nftId,txid) = ((("transfer_" + nftId) + "_") + txid) | |
50 | 50 | ||
51 | 51 | ||
52 | 52 | let keyFeeNonMinted = "cost_non_minted" | |
53 | 53 | ||
54 | 54 | let keyFeeMinted = "cost_minted" | |
55 | 55 | ||
56 | 56 | func getStringByKey (key) = valueOrElse(getString(this, key), "") | |
57 | 57 | ||
58 | 58 | ||
59 | 59 | func getIntegerByKey (key) = valueOrElse(getInteger(this, key), 0) | |
60 | 60 | ||
61 | 61 | ||
62 | 62 | func getBooleanByKey (key) = valueOrElse(getBoolean(this, key), false) | |
63 | 63 | ||
64 | 64 | ||
65 | 65 | let admin = value(addressFromString(getStringByKey(CONF_AMIN_K))) | |
66 | 66 | ||
67 | 67 | let artworksDapp = value(addressFromString(getStringByKey(CONF_ARTWORK_DAPP_K))) | |
68 | 68 | ||
69 | 69 | let usersDapp = value(addressFromString(getStringByKey(CONF_USERS_DAPP_K))) | |
70 | 70 | ||
71 | 71 | let oracleAddr = value(addressFromString(getStringByKey(CONF_ORACLE_K))) | |
72 | 72 | ||
73 | 73 | let feeAcceptedAsset = getStringByKey(CONF_FEE_ASSET_K) | |
74 | 74 | ||
75 | 75 | let feeAcceptedAmountForSign = getIntegerByKey(CONF_FEE_AMOUNT_K) | |
76 | 76 | ||
77 | 77 | func getStringFromUsersByKey (key) = valueOrElse(getString(usersDapp, key), "") | |
78 | 78 | ||
79 | 79 | ||
80 | 80 | func getStringFromArtworksByKey (key) = valueOrElse(getString(artworksDapp, key), "") | |
81 | 81 | ||
82 | 82 | ||
83 | 83 | func getIntegerByKeyFromOracle (key) = valueOrErrorMessage(getInteger(oracleAddr, key), "Integer undefine or 0 in oracle") | |
84 | 84 | ||
85 | 85 | ||
86 | 86 | func assetIsValid (assetDetails) = if (if (if ((assetDetails.quantity == 1)) | |
87 | 87 | then (assetDetails.decimals == 0) | |
88 | 88 | else false) | |
89 | 89 | then (assetDetails.reissuable == false) | |
90 | 90 | else false) | |
91 | 91 | then (assetDetails.issuer == artworksDapp) | |
92 | 92 | else false | |
93 | 93 | ||
94 | 94 | ||
95 | 95 | func isSignArtNFT (id) = { | |
96 | 96 | let assetDetails = value(assetInfo(id)) | |
97 | 97 | assetIsValid(assetDetails) | |
98 | 98 | } | |
99 | 99 | ||
100 | 100 | ||
101 | 101 | func isPayment (i) = if ((size(i.payments) == 0)) | |
102 | 102 | then throw("No payment attached") | |
103 | 103 | else { | |
104 | 104 | let nft_payment = value(i.payments[0]) | |
105 | 105 | let nft_amount = value(nft_payment.amount) | |
106 | 106 | let gateway_payment = value(i.payments[1]) | |
107 | 107 | let gateway_amount = value(gateway_payment.amount) | |
108 | 108 | $Tuple4(nft_payment, nft_amount, gateway_payment, gateway_amount) | |
109 | 109 | } | |
110 | 110 | ||
111 | 111 | ||
112 | 112 | let isInit = valueOrElse(getBooleanByKey(CONF_INIT_K), false) | |
113 | 113 | ||
114 | 114 | func ownerOnly (i) = { | |
115 | 115 | let caller = toString(i.caller) | |
116 | 116 | let id = toBase58String(i.transactionId) | |
117 | 117 | let allowedArr = if (isInit) | |
118 | 118 | then [toString(admin), toString(this)] | |
119 | 119 | else [toString(this)] | |
120 | 120 | if (containsElement(allowedArr, caller)) | |
121 | 121 | then true | |
122 | 122 | else false | |
123 | 123 | } | |
124 | 124 | ||
125 | 125 | ||
126 | 126 | func idCallerDate (i) = if (!(dappRunning)) | |
127 | 127 | then throw(maintenanceMSG) | |
128 | 128 | else $Tuple3(toBase58String(i.transactionId), toBase58String(i.caller.bytes), lastBlock.timestamp) | |
129 | 129 | ||
130 | 130 | ||
131 | 131 | func getCreatorFromNFTid (nftid) = { | |
132 | 132 | let nftData = getStringFromArtworksByKey(("nft_" + nftid)) | |
133 | 133 | let creator = split(nftData, "_")[6] | |
134 | 134 | let artId = split(nftData, "_")[5] | |
135 | 135 | $Tuple2(creator, artId) | |
136 | 136 | } | |
137 | 137 | ||
138 | 138 | ||
139 | 139 | func getCreatorERC20Addr (addr) = { | |
140 | 140 | let erc20Addr = getStringFromUsersByKey(("user_ERC20_" + addr)) | |
141 | 141 | erc20Addr | |
142 | 142 | } | |
143 | 143 | ||
144 | 144 | ||
145 | 145 | func getCreatorStatus (addr) = { | |
146 | 146 | let status = getStringFromUsersByKey(("user_status_" + addr)) | |
147 | 147 | status | |
148 | 148 | } | |
149 | 149 | ||
150 | 150 | ||
151 | 151 | func getArtStatus (artId) = { | |
152 | 152 | let status = getStringFromArtworksByKey(("user_status_" + artId)) | |
153 | 153 | status | |
154 | 154 | } | |
155 | 155 | ||
156 | 156 | ||
157 | 157 | func isValidCreator (status) = if ((status == creatorRegistered)) | |
158 | 158 | then true | |
159 | 159 | else (status == creatorVerified) | |
160 | 160 | ||
161 | 161 | ||
162 | 162 | func isValidArt (artId) = if (if ((artId != artFlagged)) | |
163 | 163 | then true | |
164 | 164 | else (artId != artIllegal)) | |
165 | 165 | then true | |
166 | 166 | else (artId != artPending) | |
167 | 167 | ||
168 | 168 | ||
169 | 169 | @Callable(i) | |
170 | 170 | func init (adminAddr,oracleAddr,artworksDapp,usersDapp) = if (ownerOnly(i)) | |
171 | 171 | then [StringEntry(CONF_AMIN_K, adminAddr), StringEntry(CONF_ORACLE_K, oracleAddr), StringEntry(CONF_ARTWORK_DAPP_K, artworksDapp), StringEntry(CONF_USERS_DAPP_K, usersDapp), BooleanEntry(CONF_INIT_K, true)] | |
172 | 172 | else throw("Not allowed") | |
173 | 173 | ||
174 | 174 | ||
175 | 175 | ||
176 | 176 | @Callable(i) | |
177 | 177 | func setFee (feeAsset,feeAmount) = { | |
178 | 178 | let assetDetails = assetInfo(fromBase58String(feeAsset)) | |
179 | 179 | let decimal = value(assetDetails).decimals | |
180 | 180 | let base10FeeAmount = pow(feeAmount, 0, 10, 1, decimal, HALFUP) | |
181 | 181 | if (ownerOnly(i)) | |
182 | 182 | then [StringEntry(CONF_FEE_ASSET_K, feeAsset), IntegerEntry(CONF_FEE_AMOUNT_K, base10FeeAmount)] | |
183 | 183 | else throw("Not allowed") | |
184 | 184 | } | |
185 | 185 | ||
186 | 186 | ||
187 | 187 | ||
188 | 188 | @Callable(i) | |
189 | 189 | func lock (recipient,network) = if (!(isInit)) | |
190 | 190 | then throw("dApp not ready") | |
191 | 191 | else if (!(containsElement(NETWORKS, network))) | |
192 | 192 | then throw("Network not supported") | |
193 | 193 | else { | |
194 | 194 | let $t051385177 = idCallerDate(i) | |
195 | 195 | let id = $t051385177._1 | |
196 | 196 | let caller = $t051385177._2 | |
197 | 197 | let date = $t051385177._3 | |
198 | 198 | let $t051805257 = isPayment(i) | |
199 | 199 | let nft_payment = $t051805257._1 | |
200 | 200 | let nft_amount = $t051805257._2 | |
201 | 201 | let gateway_payment = $t051805257._3 | |
202 | 202 | let gateway_amount = $t051805257._4 | |
203 | 203 | if ((nft_amount != 1)) | |
204 | 204 | then throw("No NFT attached") | |
205 | 205 | else { | |
206 | 206 | let nftId = if (if (isDefined(nft_payment.assetId)) | |
207 | 207 | then isSignArtNFT(value(nft_payment.assetId)) | |
208 | 208 | else false) | |
209 | 209 | then toBase58String(value(nft_payment.assetId)) | |
210 | 210 | else throw("Only SIGN Art NFT accepted.") | |
211 | 211 | let $t055665647 = getCreatorFromNFTid(nftId) | |
212 | 212 | let creator = $t055665647._1 | |
213 | 213 | let artId = $t055665647._2 | |
214 | 214 | if (!(isValidCreator(getCreatorStatus(creator)))) | |
215 | 215 | then throw(("Creator status is :" + getCreatorStatus(creator))) | |
216 | 216 | else if (!(isValidArt(getArtStatus(artId)))) | |
217 | 217 | then throw(("Artwork status is :" + getArtStatus(artId))) | |
218 | 218 | else { | |
219 | 219 | let erc20Addr = getCreatorERC20Addr(creator) | |
220 | 220 | if ((size(erc20Addr) == 0)) | |
221 | 221 | then throw("This artist didn't allow crosschain transfer") | |
222 | 222 | else { | |
223 | 223 | let nftStatus = getStringByKey(keyNFTstatus(nftId)) | |
224 | 224 | if (if ((nftStatus == status_LOCKED)) | |
225 | 225 | then true | |
226 | 226 | else (nftStatus == status_PENDING)) | |
227 | 227 | then throw("NFT already locked") | |
228 | 228 | else { | |
229 | 229 | let gasFeeNonMinted = getIntegerByKeyFromOracle(keyFeeNonMinted) | |
230 | 230 | let gasFeeMinted = getIntegerByKeyFromOracle(keyFeeMinted) | |
231 | 231 | let isMinted = if ((nftStatus == status_UNLOCKED)) | |
232 | 232 | then true | |
233 | 233 | else false | |
234 | 234 | if (if (!(isDefined(gateway_payment.assetId))) | |
235 | 235 | then true | |
236 | 236 | else (toBase58String(value(gateway_payment.assetId)) != feeAcceptedAsset)) | |
237 | 237 | then throw("Wrong payment asset") | |
238 | 238 | else if (if (isMinted) | |
239 | 239 | then ((gateway_amount + feeAcceptedAmountForSign) > gasFeeMinted) | |
240 | 240 | else false) | |
241 | 241 | then throw("Wrong payment amount to unlock") | |
242 | 242 | else if (if (!(isMinted)) | |
243 | 243 | then ((gateway_amount + feeAcceptedAmountForSign) > gasFeeNonMinted) | |
244 | 244 | else false) | |
245 | - | then throw("Wrong payment amount to mint and unlock") | |
245 | + | then throw(("Wrong payment amount to mint and unlock " + toString((gateway_amount + feeAcceptedAmountForSign)))) | |
246 | 246 | else [StringEntry(keyNFTstatus(nftId), status_PENDING), StringEntry(((keyTransferNFT(nftId, id) + "_") + status_PENDING), ((((((((((((recipient + "_") + network) + "_") + caller) + "_") + id) + "_") + value(assetInfo(value(nft_payment.assetId))).name) + "_") + toString(date)) + "_") + toString(height)))] | |
247 | 247 | } | |
248 | 248 | } | |
249 | 249 | } | |
250 | 250 | } | |
251 | 251 | } | |
252 | 252 | ||
253 | 253 | ||
254 | 254 | ||
255 | 255 | @Callable(i) | |
256 | 256 | func confirmLock (nftId,transferId) = if (!(isInit)) | |
257 | 257 | then throw("dApp not ready") | |
258 | 258 | else if (ownerOnly(i)) | |
259 | 259 | then { | |
260 | 260 | let nftStatus = getStringByKey(keyNFTstatus(nftId)) | |
261 | 261 | if ((nftStatus != status_PENDING)) | |
262 | 262 | then throw("NFT not pending") | |
263 | 263 | else { | |
264 | 264 | let val = getStringByKey(((keyTransferNFT(nftId, transferId) + "_") + status_PENDING)) | |
265 | 265 | let newKey = ((keyTransferNFT(nftId, transferId) + "_") + status_CLOSED) | |
266 | 266 | [DeleteEntry(((keyTransferNFT(nftId, transferId) + "_") + status_PENDING)), StringEntry(newKey, val), StringEntry(keyNFTstatus(nftId), status_LOCKED)] | |
267 | 267 | } | |
268 | 268 | } | |
269 | 269 | else throw("Not allowed") | |
270 | 270 | ||
271 | 271 | ||
272 | 272 | ||
273 | 273 | @Callable(i) | |
274 | 274 | func unlock (recipient,nftId) = if (!(isInit)) | |
275 | 275 | then throw("dApp not ready") | |
276 | 276 | else if (ownerOnly(i)) | |
277 | 277 | then { | |
278 | 278 | let nftStatus = getStringByKey(keyNFTstatus(nftId)) | |
279 | 279 | if ((nftStatus != status_LOCKED)) | |
280 | 280 | then throw("NFT not locked, finalize first if pending") | |
281 | 281 | else [StringEntry(keyNFTstatus(nftId), status_UNLOCKED), ScriptTransfer(Address(fromBase58String(recipient)), 1, fromBase58String(nftId))] | |
282 | 282 | } | |
283 | 283 | else throw("Not allowed") | |
284 | 284 | ||
285 | 285 | ||
286 | 286 | ||
287 | 287 | @Callable(i) | |
288 | 288 | func appRunning (isLive,message) = if (!(isInit)) | |
289 | 289 | then throw("dApp not ready") | |
290 | 290 | else if (ownerOnly(i)) | |
291 | 291 | then [BooleanEntry(dappRunningKey, isLive), StringEntry(maintenanceMSGKey, message)] | |
292 | 292 | else throw("You are not allowed to change this") | |
293 | 293 | ||
294 | 294 | ||
295 | 295 | ||
296 | 296 | @Callable(i) | |
297 | 297 | func deleteEntry (entry) = if (ownerOnly(i)) | |
298 | 298 | then [DeleteEntry(entry)] | |
299 | 299 | else throw("no") | |
300 | 300 | ||
301 | 301 |
github/deemru/w8io/169f3d6 96.04 ms ◑