2022.07.01 16:28 [3186245] smart account 3P4QfR6fewW85epUg68fjgeFt3XBVpgLxmd > SELF 0.00000000 Waves
{ "type": 13, "id": "62CQmhZrrqd4x3yZbTLaqaQMVPhwQ7WscY6YTa8Q3vA7", "fee": 1600000, "feeAssetId": null, "timestamp": 1656682105489, "version": 2, "chainId": 87, "sender": "3P4QfR6fewW85epUg68fjgeFt3XBVpgLxmd", "senderPublicKey": "26Yopt2jwHSycwVY98xW7w7XkQN8xoijxgsLeEmcEX34", "proofs": [ "4SmtPtfL13ZbvjJ6qZCgRfpSGh3R8wExpWh9tkPsEeucHKLR7NiSrhfdNhVzJ4dSXPN6FnRQQn7zqwX3WbiVabUa" ], "script": "base64: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ZWQhAQAAAApnZXRORlROYW1lAAAAAAkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEIgAAAAECAAAADnN0YXRpY19uZnROYW1lAgAAACMzR05OOiBUaGVyZSBpcyBubyBuZnQgbmFtZSBkZWZpbmVkIQEAAAATZ2V0SW5jdWJhdG9yQWRkcmVzcwAAAAAJAQAAAAdBZGRyZXNzAAAAAQkAAlkAAAABCQEAAAAUdHJ5R2V0U3RyaW5nRXh0ZXJuYWwAAAACCQEAAAAJZ2V0T3JhY2xlAAAAAAIAAAAXc3RhdGljX2luY3ViYXRvckFkZHJlc3MBAAAAEWdldEJyZWVkZXJBZGRyZXNzAAAAAAkBAAAAB0FkZHJlc3MAAAABCQACWQAAAAEJAQAAABR0cnlHZXRTdHJpbmdFeHRlcm5hbAAAAAIJAQAAAAlnZXRPcmFjbGUAAAAAAgAAABVzdGF0aWNfYnJlZWRlckFkZHJlc3MBAAAAEWdldEZhcm1pbmdBZGRyZXNzAAAAAAkBAAAAB0FkZHJlc3MAAAABCQACWQAAAAEJAQAAABR0cnlHZXRTdHJpbmdFeHRlcm5hbAAAAAIJAQAAAAlnZXRPcmFjbGUAAAAAAgAAABVzdGF0aWNfZmFybWluZ0FkZHJlc3MBAAAADWdldEVnZ0Fzc2V0SWQAAAAACQACWQAAAAEJAQAAABR0cnlHZXRTdHJpbmdFeHRlcm5hbAAAAAIJAQAAAAlnZXRPcmFjbGUAAAAAAgAAABFzdGF0aWNfZWdnQXNzZXRJZAEAAAAUZ2V0QWNjZXNzSXRlbUFzc2V0SWQAAAAACQACWQAAAAEJAQAAAAx0cnlHZXRTdHJpbmcAAAABAgAAABhzdGF0aWNfYWNjZXNzSXRlbUFzc2V0SWQBAAAAEmdldEFjY2Vzc0l0ZW1QcmljZQAAAAAJAQAAAA10cnlHZXRJbnRlZ2VyAAAAAQIAAAAWc3RhdGljX2FjY2Vzc0l0ZW1QcmljZQEAAAASZ2V0TWluTG9ja0R1cmF0aW9uAAAAAAkBAAAADXRyeUdldEludGVnZXIAAAABAgAAABZzdGF0aWNfbWluTG9ja0R1cmF0aW9uAQAAABNnZXRCb3N0ZXJCdXlBc3NldElkAAAAAAkAAlkAAAABCQEAAAAMdHJ5R2V0U3RyaW5nAAAAAQIAAAAYc3RhdGljX2Jvb3N0ZXJCdXlBc3NldElkAQAAABZnZXRCb3N0ZXJQcmljZUZvckxldmVsAAAAAQAAAAVsZXZlbAkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEHwAAAAEJAAEsAAAAAgIAAAAZc3RhdGljX2Jvb3N0ZXJQcmljZUxldmVsXwkAAaQAAAABBQAAAAVsZXZlbAIAAAA5M0dCUEZMOiBUaGVyZSBpcyBubyBwcmljZSB2YWx1ZSBmb3IgdGhpcyBsZXZlbCBvZiBib29zdGVyAQAAABVnZXRCb3N0ZXJOYW1lRm9yTGV2ZWwAAAABAAAABWxldmVsCQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQiAAAAAQkAASwAAAACAgAAABNzdGF0aWNfYm9vc3Rlck5hbWVfCQABpAAAAAEFAAAABWxldmVsAgAAACAzR0JORkw6IFRoZXJlIGlzIG5vIHN1Y2ggYm9vc3RlcgEAAAAIYXNTdHJpbmcAAAABAAAABXZhbHVlBAAAAAckbWF0Y2gwBQAAAAV2YWx1ZQMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAGU3RyaW5nBAAAAAZzdHJpbmcFAAAAByRtYXRjaDAFAAAABnN0cmluZwkAAAIAAAABAgAAABx3cm9uZyB0eXBlLCBleHBlY3RlZDogU3RyaW5nAQAAABBsb2NrRHVja0ludGVybmFsAAAAAgAAAAphZGRyZXNzU3RyAAAAA3BtdAQAAAAHYXNzZXRJZAkBAAAABXZhbHVlAAAAAQgFAAAAA3BtdAAAAAdhc3NldElkBAAAAAthc3NldElzc3VlcggJAQAAAAV2YWx1ZQAAAAEJAAPsAAAAAQUAAAAHYXNzZXRJZAAAAAZpc3N1ZXIEAAAADHREdWNrRGV0YWlscwkBAAAADmdldER1Y2tEZXRhaWxzAAAAAQUAAAAHYXNzZXRJZAQAAAAKYXNzZXRDb2xvcggFAAAADHREdWNrRGV0YWlscwAAAAJfMQQAAAAJaXNKYWNrcG90CAUAAAAMdER1Y2tEZXRhaWxzAAAAAl8yBAAAAAprU3BvdHNCdXN5CQEAAAAMa2V5U3BvdHNCdXN5AAAAAQUAAAAKYWRkcmVzc1N0cgQAAAAJc3BvdHNCdXN5CQEAAAAN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AWkBAAAACnVubG9ja0R1Y2sAAAABAAAACmFzc2V0SWRTdHIEAAAACmFkZHJlc3NTdHIJAAQlAAAAAQgFAAAAAWkAAAAGY2FsbGVyBAAAAAdhc3NldElkCQACWQAAAAEFAAAACmFzc2V0SWRTdHIEAAAADHREdWNrRGV0YWlscwkBAAAADmdldER1Y2tEZXRhaWxzAAAAAQUAAAAHYXNzZXRJZAQAAAAKYXNzZXRDb2xvcggFAAAADHREdWNrRGV0YWlscwAAAAJfMQQAAAAJaXNKYWNrcG90CAUAAAAMdER1Y2tEZXRhaWxzAAAAAl8yBAAAAAprU3BvdHNCdXN5CQEAAAAMa2V5U3BvdHNCdXN5AAAAAQUAAAAKYWRkcmVzc1N0cgQAAAAKa0R1Y2tPd25lcgkBAAAADGtleUR1Y2tPd25lcgAAAAEFAAAAB2Fzc2V0SWQEAAAAEnJlbWFpbmluZ0ZvclVubG9jawkAAGUAAAACCQEAAAANdHJ5R2V0SW50ZWdlcgAAAAEJAQAAABFrZXlEdWNrVW5sb2NrVGltZQAAAAEFAAAAB2Fzc2V0SWQIBQAAAAlsYXN0QmxvY2sAAAAJdGltZXN0YW1wAwkBAAAAAiE9AAAAAgkBAAAADHRyeUdldFN0cmluZwAAAAEFAAAACmtEdWNrT3duZXIFAAAACmFkZHJlc3NTdHIJAAACAAAAAQIAAAAaM1VMOiBUaGUgZHVjayBpcyBub3QgeW91cnMDCQAAZgAAAAIFAAAAEnJlbWFpbmluZ0ZvclVubG9jawAAAAAAAAAAAAkAAAIAAAABCQABLAAAAAIJAAEsAAAAAgIAAAAgM1VMOiBZb3UgbmVlZCB0byB3YWl0IHRvIHVubG9jayAJAAGkAAAAAQkAAGkAAAACBQAAABJyZW1haW5pbmdGb3JVbmxvY2sAAAAAAAAAA+gCAAAACCBzZWNvbmRzBAAAAA91bnN0YWtlRnVuY05hbWUDCQAAAAAAAAIFAAAACWlzSmFja3BvdAYCAAAADnVuc3Rha2VKYWNrcG90AgAAAAp1bnN0YWtlTkZUBAAAAARpbnYxCQAD/AAAAAQJAQAAABFnZXRGYXJtaW5nQWRkcmVzcwAAAAAFAAAAD3Vuc3Rha2VGdW5jTmFtZQkABEwAAAACBQAAAAphc3NldElkU3RyBQAAAANuaWwFAAAAA25pbAMJAAAAAAAAAgUAAAAEaW52MQUAAAAEaW52MQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIAAAAAAAAAAAEFAAAAB2Fzc2V0SWQJAARMAAAAAgkBAAAAC0RlbGV0ZUVudHJ5AAAAAQkBAAAAE2tleUxvY2tlZER1Y2tTdGF0dXMAAAACBQAAAAphZGRyZXNzU3RyBQAAAAphc3NldElkU3RyCQAETAAAAAIJAQAAAAtEZWxldGVFbnRyeQAAAAEFAAAACmtEdWNrT3duZXIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAACmtTcG90c0J1c3kJAABlAAAAAgkBAAAADXRyeUdldEludGVnZXIAAAABBQAAAAprU3BvdHNCdXN5AAAAAAAAAAABBQAAAANuaWwJAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAAAAAF77P2U=", "height": 3186245, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 8WA1MiJXMynFBqDU1w8xa4LJEhmJjXX8q1C51rv35tfu Next: 8FjysT9SAbLbZ9CaT6kCk6rFjpmbBFhgz1Dg2Ru1vJ1d Diff:
Old | New | Differences | |
---|---|---|---|
6 | 6 | b | |
7 | 7 | case _ => | |
8 | 8 | 0 | |
9 | - | } | |
10 | - | ||
11 | - | ||
12 | - | func tryGetString (key) = match getString(this, key) { | |
13 | - | case a: String => | |
14 | - | a | |
15 | - | case _ => | |
16 | - | "" | |
17 | 9 | } | |
18 | 10 | ||
19 | 11 | ||
46 | 38 | } | |
47 | 39 | ||
48 | 40 | ||
49 | - | func getIncubatorAddress () = fromBase58String(tryGetString("static_incubatorAddress")) | |
41 | + | func tryGetStringExternal (address,key) = match getString(address, key) { | |
42 | + | case a: String => | |
43 | + | a | |
44 | + | case _ => | |
45 | + | "" | |
46 | + | } | |
50 | 47 | ||
51 | 48 | ||
52 | - | func | |
49 | + | func tryGetString (key) = tryGetStringExternal(this, key) | |
53 | 50 | ||
54 | 51 | ||
55 | - | func | |
52 | + | func getOracle () = Address(fromBase58String(tryGetString("static_oracleAddress"))) | |
56 | 53 | ||
57 | 54 | ||
58 | - | func getEggAssetId () = fromBase58String(tryGetString("static_eggAssetId")) | |
55 | + | func getGameName () = valueOrErrorMessage(getString("static_shortGameName"), "3GGN: There is no game name defined!") | |
56 | + | ||
57 | + | ||
58 | + | func getLongGameName () = valueOrErrorMessage(getString("static_longGameName"), "3GLGN: There is no long game name defined!") | |
59 | + | ||
60 | + | ||
61 | + | func getNFTName () = valueOrErrorMessage(getString("static_nftName"), "3GNN: There is no nft name defined!") | |
62 | + | ||
63 | + | ||
64 | + | func getIncubatorAddress () = Address(fromBase58String(tryGetStringExternal(getOracle(), "static_incubatorAddress"))) | |
65 | + | ||
66 | + | ||
67 | + | func getBreederAddress () = Address(fromBase58String(tryGetStringExternal(getOracle(), "static_breederAddress"))) | |
68 | + | ||
69 | + | ||
70 | + | func getFarmingAddress () = Address(fromBase58String(tryGetStringExternal(getOracle(), "static_farmingAddress"))) | |
71 | + | ||
72 | + | ||
73 | + | func getEggAssetId () = fromBase58String(tryGetStringExternal(getOracle(), "static_eggAssetId")) | |
59 | 74 | ||
60 | 75 | ||
61 | 76 | func getAccessItemAssetId () = fromBase58String(tryGetString("static_accessItemAssetId")) | |
70 | 85 | func getBosterBuyAssetId () = fromBase58String(tryGetString("static_boosterBuyAssetId")) | |
71 | 86 | ||
72 | 87 | ||
73 | - | func getBosterPriceForLevel (level) = valueOrErrorMessage(getInteger(("static_boosterPriceLevel_" + toString(level))), "There is no price value for this level of booster") | |
88 | + | func getBosterPriceForLevel (level) = valueOrErrorMessage(getInteger(("static_boosterPriceLevel_" + toString(level))), "3GBPFL: There is no price value for this level of booster") | |
74 | 89 | ||
75 | 90 | ||
76 | - | func getBosterNameForLevel (level) = valueOrErrorMessage(getString(("static_boosterName_" + toString(level))), "There is no suck booster") | |
91 | + | func getBosterNameForLevel (level) = valueOrErrorMessage(getString(("static_boosterName_" + toString(level))), "3GBNFL: There is no such booster") | |
92 | + | ||
93 | + | ||
94 | + | func asString (value) = match value { | |
95 | + | case string: String => | |
96 | + | string | |
97 | + | case _ => | |
98 | + | throw("wrong type, expected: String") | |
99 | + | } | |
100 | + | ||
101 | + | ||
102 | + | func lockDuckInternal (addressStr,pmt) = { | |
103 | + | let assetId = value(pmt.assetId) | |
104 | + | let assetIssuer = value(assetInfo(assetId)).issuer | |
105 | + | let tDuckDetails = getDuckDetails(assetId) | |
106 | + | let assetColor = tDuckDetails._1 | |
107 | + | let isJackpot = tDuckDetails._2 | |
108 | + | let kSpotsBusy = keySpotsBusy(addressStr) | |
109 | + | let spotsBusy = tryGetInteger(kSpotsBusy) | |
110 | + | let kSpotsBought = keySpotsBought(addressStr) | |
111 | + | let addressSpotsAvailable = (tryGetInteger(kSpotsBought) - spotsBusy) | |
112 | + | if ((pmt.amount != 1)) | |
113 | + | then throw("NFT is not attached") | |
114 | + | else if (if ((assetIssuer != getIncubatorAddress())) | |
115 | + | then (assetIssuer != getBreederAddress()) | |
116 | + | else false) | |
117 | + | then throw("ivalid NFT attached") | |
118 | + | else if ((0 >= addressSpotsAvailable)) | |
119 | + | then throw("No spots available") | |
120 | + | else { | |
121 | + | let kPerchesAvailable = ((("address_" + toString(this)) + "_perchesAvailable_") + assetColor) | |
122 | + | let perchesAvailable = match getInteger(getFarmingAddress(), kPerchesAvailable) { | |
123 | + | case b: Int => | |
124 | + | b | |
125 | + | case _ => | |
126 | + | 0 | |
127 | + | } | |
128 | + | if ((perchesAvailable == perchesAvailable)) | |
129 | + | then { | |
130 | + | let eggAssetId = getEggAssetId() | |
131 | + | let inv1 = if ((perchesAvailable > 0)) | |
132 | + | then unit | |
133 | + | else invoke(getFarmingAddress(), "buyPerch", [assetColor, ""], [AttachedPayment(eggAssetId, 100000000)]) | |
134 | + | if ((inv1 == inv1)) | |
135 | + | then { | |
136 | + | let inv2 = if ((isJackpot == false)) | |
137 | + | then invoke(getFarmingAddress(), "stakeNFT", nil, [AttachedPayment(assetId, 1)]) | |
138 | + | else invoke(getFarmingAddress(), "stakeJackpot", [assetColor], [AttachedPayment(assetId, 1)]) | |
139 | + | if ((inv2 == inv2)) | |
140 | + | then [IntegerEntry(kSpotsBusy, (spotsBusy + 1)), StringEntry(keyDuckOwner(assetId), addressStr), BooleanEntry(keyLockedDuckStatus(addressStr, toBase58String(assetId)), true), IntegerEntry(keyDuckUnlockTime(assetId), (lastBlock.timestamp + getMinLockDuration()))] | |
141 | + | else throw("Strict value is not equal to itself.") | |
142 | + | } | |
143 | + | else throw("Strict value is not equal to itself.") | |
144 | + | } | |
145 | + | else throw("Strict value is not equal to itself.") | |
146 | + | } | |
147 | + | } | |
148 | + | ||
149 | + | ||
150 | + | @Callable(i) | |
151 | + | func buyAccessItemInternal (addressStr) = if ((i.caller != this)) | |
152 | + | then throw("3BAII: internal use only") | |
153 | + | else { | |
154 | + | let issueAccessItem = Issue(("ACCESS-" + getGameName()), ((("[Access] " + getLongGameName()) + " access NFT for ") + getNFTName()), 1, 0, false) | |
155 | + | let accessRaceAssetId = toBase58String(calculateAssetId(issueAccessItem)) | |
156 | + | let kSpotsBought = keySpotsBought(addressStr) | |
157 | + | $Tuple2([issueAccessItem, StringEntry((("accessItem_" + accessRaceAssetId) + "_owner"), addressStr), StringEntry((("address_" + addressStr) + "_owning"), accessRaceAssetId), IntegerEntry(kSpotsBought, (tryGetInteger(kSpotsBought) + 1))], accessRaceAssetId) | |
158 | + | } | |
159 | + | ||
160 | + | ||
161 | + | ||
162 | + | @Callable(i) | |
163 | + | func configureOracle (oracle,shortname,longname,nftName) = if ((i.caller != this)) | |
164 | + | then throw("3CO: admin only") | |
165 | + | else [StringEntry("static_oracleAddress", oracle), StringEntry("static_shortGameName", shortname), StringEntry("static_longGameName", longname), StringEntry("static_nftName", nftName)] | |
166 | + | ||
77 | 167 | ||
78 | 168 | ||
79 | 169 | @Callable(i) | |
96 | 186 | ||
97 | 187 | @Callable(i) | |
98 | 188 | func deleteSetting (key) = if ((i.caller != this)) | |
99 | - | then throw("Only administrator can call this method") | |
189 | + | then throw("3DS: Only administrator can call this method") | |
100 | 190 | else [DeleteEntry(key)] | |
101 | - | ||
102 | - | ||
103 | - | ||
104 | - | @Callable(i) | |
105 | - | func issueFreeAccessItem (recipientAddress) = if ((i.caller != this)) | |
106 | - | then throw("Can't be called by this user") | |
107 | - | else { | |
108 | - | let issueAccessItem = Issue("ACCESS-RACE", "[Access] Metarace access NFT for Duckracer", 1, 0, false) | |
109 | - | let accessRaceAssetId = toBase58String(calculateAssetId(issueAccessItem)) | |
110 | - | let kSpotsBought = keySpotsBought(recipientAddress) | |
111 | - | [StringEntry((("accessItem_" + accessRaceAssetId) + "_owner"), recipientAddress), issueAccessItem, StringEntry((("address_" + recipientAddress) + "_owning"), accessRaceAssetId), IntegerEntry(kSpotsBought, (tryGetInteger(kSpotsBought) + 1))] | |
112 | - | } | |
113 | 191 | ||
114 | 192 | ||
115 | 193 | ||
122 | 200 | let price = getBosterPriceForLevel(newLevel) | |
123 | 201 | let expectedAssetId = getBosterBuyAssetId() | |
124 | 202 | if ((pmt.amount != price)) | |
125 | - | then throw(("Bad price, it should be: " + toString(price))) | |
203 | + | then throw(("3BB: Bad price, it should be: " + toString(price))) | |
126 | 204 | else if ((pmt.assetId != expectedAssetId)) | |
127 | - | then throw(("Bad payment attached, it | |
205 | + | then throw(("3BB: Bad payment attached, it should be: " + toBase58String(expectedAssetId))) | |
128 | 206 | else { | |
129 | 207 | let name = getBosterNameForLevel(newLevel) | |
130 | - | let boosterItem = Issue((" | |
208 | + | let boosterItem = Issue(((getGameName() + "-") + name), ((("[Booster] " + getLongGameName()) + " booster for the game, level = ") + toString(newLevel)), 1, 0, false) | |
131 | 209 | let boosterItemAssetId = calculateAssetId(boosterItem) | |
132 | 210 | [boosterItem, IntegerEntry(kBoughtLevel, newLevel), ScriptTransfer(i.caller, 1, boosterItemAssetId)] | |
133 | 211 | } | |
136 | 214 | ||
137 | 215 | ||
138 | 216 | @Callable(i) | |
217 | + | func buyAccessItemAndLockDuck () = { | |
218 | + | let eggPayment = value(i.payments[0]) | |
219 | + | let duckPayment = value(i.payments[1]) | |
220 | + | let addressStr = toString(i.caller) | |
221 | + | if ((eggPayment.assetId != getAccessItemAssetId())) | |
222 | + | then throw("3BAI: Wrong asset attached") | |
223 | + | else if ((eggPayment.amount != getAccessItemPrice())) | |
224 | + | then throw("3BAI: Wrong amount of assets attached") | |
225 | + | else { | |
226 | + | let accessItemData = asString(invoke(this, "buyAccessItemInternal", [addressStr], nil)) | |
227 | + | if ((accessItemData == accessItemData)) | |
228 | + | then { | |
229 | + | let lockData = lockDuckInternal(addressStr, duckPayment) | |
230 | + | $Tuple2(lockData, accessItemData) | |
231 | + | } | |
232 | + | else throw("Strict value is not equal to itself.") | |
233 | + | } | |
234 | + | } | |
235 | + | ||
236 | + | ||
237 | + | ||
238 | + | @Callable(i) | |
139 | 239 | func buyAccessItem () = { | |
140 | - | let pmt = value(i.payments[0]) | |
141 | 240 | let addressStr = toString(i.caller) | |
142 | - | if ((pmt.assetId != getAccessItemAssetId())) | |
143 | - | then throw("Wrong asset attached") | |
144 | - | else if ((pmt.amount != getAccessItemPrice())) | |
145 | - | then throw("Wrong amount of assets attached") | |
241 | + | let eggPayment = value(i.payments[0]) | |
242 | + | if ((eggPayment.assetId != getAccessItemAssetId())) | |
243 | + | then throw("3BAI: Wrong asset attached") | |
244 | + | else if ((eggPayment.amount != getAccessItemPrice())) | |
245 | + | then throw("3BAI: Wrong amount of assets attached") | |
146 | 246 | else { | |
147 | - | let issueAccessItem = Issue("ACCESS-RACE", "[Access] Metarace access NFT for Duckracer", 1, 0, false) | |
148 | - | let accessRaceAssetId = toBase58String(calculateAssetId(issueAccessItem)) | |
149 | - | let kSpotsBought = keySpotsBought(addressStr) | |
150 | - | [issueAccessItem, StringEntry((("accessItem_" + accessRaceAssetId) + "_owner"), addressStr), StringEntry((("address_" + addressStr) + "_owning"), accessRaceAssetId), IntegerEntry(kSpotsBought, (tryGetInteger(kSpotsBought) + 1))] | |
247 | + | let accessItemData = asString(invoke(this, "buyAccessItemInternal", [addressStr], nil)) | |
248 | + | if ((accessItemData == accessItemData)) | |
249 | + | then $Tuple2(nil, accessItemData) | |
250 | + | else throw("Strict value is not equal to itself.") | |
251 | + | } | |
252 | + | } | |
253 | + | ||
254 | + | ||
255 | + | ||
256 | + | @Callable(i) | |
257 | + | func buyAccessItemOther (addressOther) = { | |
258 | + | let eggPayment = value(i.payments[0]) | |
259 | + | if ((eggPayment.assetId != getAccessItemAssetId())) | |
260 | + | then throw("3BAI: Wrong asset attached") | |
261 | + | else if ((eggPayment.amount != getAccessItemPrice())) | |
262 | + | then throw("3BAI: Wrong amount of assets attached") | |
263 | + | else { | |
264 | + | let accessItemData = asString(invoke(this, "buyAccessItemInternal", [addressOther], nil)) | |
265 | + | if ((accessItemData == accessItemData)) | |
266 | + | then $Tuple2(nil, accessItemData) | |
267 | + | else throw("Strict value is not equal to itself.") | |
151 | 268 | } | |
152 | 269 | } | |
153 | 270 | ||
156 | 273 | @Callable(i) | |
157 | 274 | func lockDuck () = { | |
158 | 275 | let addressStr = toString(i.caller) | |
159 | - | let pmt = value(i.payments[0]) | |
160 | - | let assetId = value(pmt.assetId) | |
161 | - | let assetIssuer = value(assetInfo(assetId)).issuer.bytes | |
162 | - | let tDuckDetails = getDuckDetails(assetId) | |
163 | - | let assetColor = tDuckDetails._1 | |
164 | - | let isJackpot = tDuckDetails._2 | |
165 | - | let kSpotsBusy = keySpotsBusy(addressStr) | |
166 | - | let spotsBusy = tryGetInteger(kSpotsBusy) | |
167 | - | let kSpotsBought = keySpotsBought(addressStr) | |
168 | - | let addressSpotsAvailable = (tryGetInteger(kSpotsBought) - spotsBusy) | |
169 | - | if ((pmt.amount != 1)) | |
170 | - | then throw("NFT is not attached") | |
171 | - | else if (if ((assetIssuer != getIncubatorAddress())) | |
172 | - | then (assetIssuer != getBreederAddress()) | |
173 | - | else false) | |
174 | - | then throw("ivalid NFT attached") | |
175 | - | else if ((0 >= addressSpotsAvailable)) | |
176 | - | then throw("No spots available") | |
177 | - | else { | |
178 | - | let kPerchesAvailable = ((("address_" + toString(this)) + "_perchesAvailable_") + assetColor) | |
179 | - | let perchesAvailable = match getInteger(Address(getFarmingAddress()), kPerchesAvailable) { | |
180 | - | case b: Int => | |
181 | - | b | |
182 | - | case _ => | |
183 | - | 0 | |
184 | - | } | |
185 | - | if ((perchesAvailable == perchesAvailable)) | |
186 | - | then { | |
187 | - | let eggAssetId = getEggAssetId() | |
188 | - | let inv1 = if ((perchesAvailable > 0)) | |
189 | - | then unit | |
190 | - | else invoke(Address(getFarmingAddress()), "buyPerch", [assetColor, ""], [AttachedPayment(eggAssetId, 100000000)]) | |
191 | - | if ((inv1 == inv1)) | |
192 | - | then { | |
193 | - | let inv2 = if ((isJackpot == false)) | |
194 | - | then invoke(Address(getFarmingAddress()), "stakeNFT", nil, [AttachedPayment(assetId, 1)]) | |
195 | - | else invoke(Address(getFarmingAddress()), "stakeJackpot", [assetColor], [AttachedPayment(assetId, 1)]) | |
196 | - | if ((inv2 == inv2)) | |
197 | - | then [IntegerEntry(kSpotsBusy, (spotsBusy + 1)), StringEntry(keyDuckOwner(assetId), addressStr), BooleanEntry(keyLockedDuckStatus(addressStr, toBase58String(assetId)), true), IntegerEntry(keyDuckUnlockTime(assetId), (lastBlock.timestamp + getMinLockDuration()))] | |
198 | - | else throw("Strict value is not equal to itself.") | |
199 | - | } | |
200 | - | else throw("Strict value is not equal to itself.") | |
201 | - | } | |
202 | - | else throw("Strict value is not equal to itself.") | |
203 | - | } | |
276 | + | let duckPayment = value(i.payments[0]) | |
277 | + | lockDuckInternal(addressStr, duckPayment) | |
204 | 278 | } | |
205 | 279 | ||
206 | 280 | ||
216 | 290 | let kDuckOwner = keyDuckOwner(assetId) | |
217 | 291 | let remainingForUnlock = (tryGetInteger(keyDuckUnlockTime(assetId)) - lastBlock.timestamp) | |
218 | 292 | if ((tryGetString(kDuckOwner) != addressStr)) | |
219 | - | then throw("The duck is not yours") | |
293 | + | then throw("3UL: The duck is not yours") | |
220 | 294 | else if ((remainingForUnlock > 0)) | |
221 | - | then throw(((" | |
295 | + | then throw((("3UL: You need to wait to unlock " + toString((remainingForUnlock / 1000))) + " seconds")) | |
222 | 296 | else { | |
223 | 297 | let unstakeFuncName = if ((isJackpot == true)) | |
224 | 298 | then "unstakeJackpot" | |
225 | 299 | else "unstakeNFT" | |
226 | - | let inv1 = invoke( | |
300 | + | let inv1 = invoke(getFarmingAddress(), unstakeFuncName, [assetIdStr], nil) | |
227 | 301 | if ((inv1 == inv1)) | |
228 | 302 | then [ScriptTransfer(i.caller, 1, assetId), DeleteEntry(keyLockedDuckStatus(addressStr, assetIdStr)), DeleteEntry(kDuckOwner), IntegerEntry(kSpotsBusy, (tryGetInteger(kSpotsBusy) - 1))] | |
229 | 303 | else throw("Strict value is not equal to itself.") |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | func tryGetInteger (key) = match getInteger(this, key) { | |
5 | 5 | case b: Int => | |
6 | 6 | b | |
7 | 7 | case _ => | |
8 | 8 | 0 | |
9 | - | } | |
10 | - | ||
11 | - | ||
12 | - | func tryGetString (key) = match getString(this, key) { | |
13 | - | case a: String => | |
14 | - | a | |
15 | - | case _ => | |
16 | - | "" | |
17 | 9 | } | |
18 | 10 | ||
19 | 11 | ||
20 | 12 | func keyBoughtBoosterLevel (addressStr) = (("address_" + addressStr) + "_boughtBoosterLevel") | |
21 | 13 | ||
22 | 14 | ||
23 | 15 | func keyDuckUnlockTime (assetId) = (("duck_" + toBase58String(assetId)) + "_unlockTime") | |
24 | 16 | ||
25 | 17 | ||
26 | 18 | func keyDuckOwner (assetId) = (("duck_" + toBase58String(assetId)) + "_owner") | |
27 | 19 | ||
28 | 20 | ||
29 | 21 | func keySpotsBought (addressStr) = (("address_" + addressStr) + "_spotsBought") | |
30 | 22 | ||
31 | 23 | ||
32 | 24 | func keySpotsBusy (addressStr) = (("address_" + addressStr) + "_spotsBusy") | |
33 | 25 | ||
34 | 26 | ||
35 | 27 | func keyLockedDuckStatus (addressStr,assetIdStr) = (((("address_" + addressStr) + "_lockedDuck_") + assetIdStr) + "_status") | |
36 | 28 | ||
37 | 29 | ||
38 | 30 | func getDuckDetails (assetId) = { | |
39 | 31 | let assetName = value(value(assetInfo(assetId)).name) | |
40 | 32 | let assetNameParts = split(assetName, "") | |
41 | 33 | let isJackpot = (assetNameParts[(size(assetNameParts) - 2)] == "J") | |
42 | 34 | let assetColor = if (isJackpot) | |
43 | 35 | then "B" | |
44 | 36 | else assetNameParts[(size(assetNameParts) - 1)] | |
45 | 37 | $Tuple2(assetColor, isJackpot) | |
46 | 38 | } | |
47 | 39 | ||
48 | 40 | ||
49 | - | func getIncubatorAddress () = fromBase58String(tryGetString("static_incubatorAddress")) | |
41 | + | func tryGetStringExternal (address,key) = match getString(address, key) { | |
42 | + | case a: String => | |
43 | + | a | |
44 | + | case _ => | |
45 | + | "" | |
46 | + | } | |
50 | 47 | ||
51 | 48 | ||
52 | - | func | |
49 | + | func tryGetString (key) = tryGetStringExternal(this, key) | |
53 | 50 | ||
54 | 51 | ||
55 | - | func | |
52 | + | func getOracle () = Address(fromBase58String(tryGetString("static_oracleAddress"))) | |
56 | 53 | ||
57 | 54 | ||
58 | - | func getEggAssetId () = fromBase58String(tryGetString("static_eggAssetId")) | |
55 | + | func getGameName () = valueOrErrorMessage(getString("static_shortGameName"), "3GGN: There is no game name defined!") | |
56 | + | ||
57 | + | ||
58 | + | func getLongGameName () = valueOrErrorMessage(getString("static_longGameName"), "3GLGN: There is no long game name defined!") | |
59 | + | ||
60 | + | ||
61 | + | func getNFTName () = valueOrErrorMessage(getString("static_nftName"), "3GNN: There is no nft name defined!") | |
62 | + | ||
63 | + | ||
64 | + | func getIncubatorAddress () = Address(fromBase58String(tryGetStringExternal(getOracle(), "static_incubatorAddress"))) | |
65 | + | ||
66 | + | ||
67 | + | func getBreederAddress () = Address(fromBase58String(tryGetStringExternal(getOracle(), "static_breederAddress"))) | |
68 | + | ||
69 | + | ||
70 | + | func getFarmingAddress () = Address(fromBase58String(tryGetStringExternal(getOracle(), "static_farmingAddress"))) | |
71 | + | ||
72 | + | ||
73 | + | func getEggAssetId () = fromBase58String(tryGetStringExternal(getOracle(), "static_eggAssetId")) | |
59 | 74 | ||
60 | 75 | ||
61 | 76 | func getAccessItemAssetId () = fromBase58String(tryGetString("static_accessItemAssetId")) | |
62 | 77 | ||
63 | 78 | ||
64 | 79 | func getAccessItemPrice () = tryGetInteger("static_accessItemPrice") | |
65 | 80 | ||
66 | 81 | ||
67 | 82 | func getMinLockDuration () = tryGetInteger("static_minLockDuration") | |
68 | 83 | ||
69 | 84 | ||
70 | 85 | func getBosterBuyAssetId () = fromBase58String(tryGetString("static_boosterBuyAssetId")) | |
71 | 86 | ||
72 | 87 | ||
73 | - | func getBosterPriceForLevel (level) = valueOrErrorMessage(getInteger(("static_boosterPriceLevel_" + toString(level))), "There is no price value for this level of booster") | |
88 | + | func getBosterPriceForLevel (level) = valueOrErrorMessage(getInteger(("static_boosterPriceLevel_" + toString(level))), "3GBPFL: There is no price value for this level of booster") | |
74 | 89 | ||
75 | 90 | ||
76 | - | func getBosterNameForLevel (level) = valueOrErrorMessage(getString(("static_boosterName_" + toString(level))), "There is no suck booster") | |
91 | + | func getBosterNameForLevel (level) = valueOrErrorMessage(getString(("static_boosterName_" + toString(level))), "3GBNFL: There is no such booster") | |
92 | + | ||
93 | + | ||
94 | + | func asString (value) = match value { | |
95 | + | case string: String => | |
96 | + | string | |
97 | + | case _ => | |
98 | + | throw("wrong type, expected: String") | |
99 | + | } | |
100 | + | ||
101 | + | ||
102 | + | func lockDuckInternal (addressStr,pmt) = { | |
103 | + | let assetId = value(pmt.assetId) | |
104 | + | let assetIssuer = value(assetInfo(assetId)).issuer | |
105 | + | let tDuckDetails = getDuckDetails(assetId) | |
106 | + | let assetColor = tDuckDetails._1 | |
107 | + | let isJackpot = tDuckDetails._2 | |
108 | + | let kSpotsBusy = keySpotsBusy(addressStr) | |
109 | + | let spotsBusy = tryGetInteger(kSpotsBusy) | |
110 | + | let kSpotsBought = keySpotsBought(addressStr) | |
111 | + | let addressSpotsAvailable = (tryGetInteger(kSpotsBought) - spotsBusy) | |
112 | + | if ((pmt.amount != 1)) | |
113 | + | then throw("NFT is not attached") | |
114 | + | else if (if ((assetIssuer != getIncubatorAddress())) | |
115 | + | then (assetIssuer != getBreederAddress()) | |
116 | + | else false) | |
117 | + | then throw("ivalid NFT attached") | |
118 | + | else if ((0 >= addressSpotsAvailable)) | |
119 | + | then throw("No spots available") | |
120 | + | else { | |
121 | + | let kPerchesAvailable = ((("address_" + toString(this)) + "_perchesAvailable_") + assetColor) | |
122 | + | let perchesAvailable = match getInteger(getFarmingAddress(), kPerchesAvailable) { | |
123 | + | case b: Int => | |
124 | + | b | |
125 | + | case _ => | |
126 | + | 0 | |
127 | + | } | |
128 | + | if ((perchesAvailable == perchesAvailable)) | |
129 | + | then { | |
130 | + | let eggAssetId = getEggAssetId() | |
131 | + | let inv1 = if ((perchesAvailable > 0)) | |
132 | + | then unit | |
133 | + | else invoke(getFarmingAddress(), "buyPerch", [assetColor, ""], [AttachedPayment(eggAssetId, 100000000)]) | |
134 | + | if ((inv1 == inv1)) | |
135 | + | then { | |
136 | + | let inv2 = if ((isJackpot == false)) | |
137 | + | then invoke(getFarmingAddress(), "stakeNFT", nil, [AttachedPayment(assetId, 1)]) | |
138 | + | else invoke(getFarmingAddress(), "stakeJackpot", [assetColor], [AttachedPayment(assetId, 1)]) | |
139 | + | if ((inv2 == inv2)) | |
140 | + | then [IntegerEntry(kSpotsBusy, (spotsBusy + 1)), StringEntry(keyDuckOwner(assetId), addressStr), BooleanEntry(keyLockedDuckStatus(addressStr, toBase58String(assetId)), true), IntegerEntry(keyDuckUnlockTime(assetId), (lastBlock.timestamp + getMinLockDuration()))] | |
141 | + | else throw("Strict value is not equal to itself.") | |
142 | + | } | |
143 | + | else throw("Strict value is not equal to itself.") | |
144 | + | } | |
145 | + | else throw("Strict value is not equal to itself.") | |
146 | + | } | |
147 | + | } | |
148 | + | ||
149 | + | ||
150 | + | @Callable(i) | |
151 | + | func buyAccessItemInternal (addressStr) = if ((i.caller != this)) | |
152 | + | then throw("3BAII: internal use only") | |
153 | + | else { | |
154 | + | let issueAccessItem = Issue(("ACCESS-" + getGameName()), ((("[Access] " + getLongGameName()) + " access NFT for ") + getNFTName()), 1, 0, false) | |
155 | + | let accessRaceAssetId = toBase58String(calculateAssetId(issueAccessItem)) | |
156 | + | let kSpotsBought = keySpotsBought(addressStr) | |
157 | + | $Tuple2([issueAccessItem, StringEntry((("accessItem_" + accessRaceAssetId) + "_owner"), addressStr), StringEntry((("address_" + addressStr) + "_owning"), accessRaceAssetId), IntegerEntry(kSpotsBought, (tryGetInteger(kSpotsBought) + 1))], accessRaceAssetId) | |
158 | + | } | |
159 | + | ||
160 | + | ||
161 | + | ||
162 | + | @Callable(i) | |
163 | + | func configureOracle (oracle,shortname,longname,nftName) = if ((i.caller != this)) | |
164 | + | then throw("3CO: admin only") | |
165 | + | else [StringEntry("static_oracleAddress", oracle), StringEntry("static_shortGameName", shortname), StringEntry("static_longGameName", longname), StringEntry("static_nftName", nftName)] | |
166 | + | ||
77 | 167 | ||
78 | 168 | ||
79 | 169 | @Callable(i) | |
80 | 170 | func updateSetting (key,value) = if ((i.caller != this)) | |
81 | 171 | then throw("Only administrator can call this method") | |
82 | 172 | else match value { | |
83 | 173 | case int: Int => | |
84 | 174 | [IntegerEntry(key, int)] | |
85 | 175 | case s: String => | |
86 | 176 | [StringEntry(key, s)] | |
87 | 177 | case b: Boolean => | |
88 | 178 | [BooleanEntry(key, b)] | |
89 | 179 | case bv: ByteVector => | |
90 | 180 | [BinaryEntry(key, bv)] | |
91 | 181 | case _ => | |
92 | 182 | throw("Bad value type") | |
93 | 183 | } | |
94 | 184 | ||
95 | 185 | ||
96 | 186 | ||
97 | 187 | @Callable(i) | |
98 | 188 | func deleteSetting (key) = if ((i.caller != this)) | |
99 | - | then throw("Only administrator can call this method") | |
189 | + | then throw("3DS: Only administrator can call this method") | |
100 | 190 | else [DeleteEntry(key)] | |
101 | - | ||
102 | - | ||
103 | - | ||
104 | - | @Callable(i) | |
105 | - | func issueFreeAccessItem (recipientAddress) = if ((i.caller != this)) | |
106 | - | then throw("Can't be called by this user") | |
107 | - | else { | |
108 | - | let issueAccessItem = Issue("ACCESS-RACE", "[Access] Metarace access NFT for Duckracer", 1, 0, false) | |
109 | - | let accessRaceAssetId = toBase58String(calculateAssetId(issueAccessItem)) | |
110 | - | let kSpotsBought = keySpotsBought(recipientAddress) | |
111 | - | [StringEntry((("accessItem_" + accessRaceAssetId) + "_owner"), recipientAddress), issueAccessItem, StringEntry((("address_" + recipientAddress) + "_owning"), accessRaceAssetId), IntegerEntry(kSpotsBought, (tryGetInteger(kSpotsBought) + 1))] | |
112 | - | } | |
113 | 191 | ||
114 | 192 | ||
115 | 193 | ||
116 | 194 | @Callable(i) | |
117 | 195 | func buyBooster () = { | |
118 | 196 | let kBoughtLevel = keyBoughtBoosterLevel(toString(i.caller)) | |
119 | 197 | let boughtBoosterLevel = tryGetInteger(kBoughtLevel) | |
120 | 198 | let newLevel = (boughtBoosterLevel + 1) | |
121 | 199 | let pmt = value(i.payments[0]) | |
122 | 200 | let price = getBosterPriceForLevel(newLevel) | |
123 | 201 | let expectedAssetId = getBosterBuyAssetId() | |
124 | 202 | if ((pmt.amount != price)) | |
125 | - | then throw(("Bad price, it should be: " + toString(price))) | |
203 | + | then throw(("3BB: Bad price, it should be: " + toString(price))) | |
126 | 204 | else if ((pmt.assetId != expectedAssetId)) | |
127 | - | then throw(("Bad payment attached, it | |
205 | + | then throw(("3BB: Bad payment attached, it should be: " + toBase58String(expectedAssetId))) | |
128 | 206 | else { | |
129 | 207 | let name = getBosterNameForLevel(newLevel) | |
130 | - | let boosterItem = Issue((" | |
208 | + | let boosterItem = Issue(((getGameName() + "-") + name), ((("[Booster] " + getLongGameName()) + " booster for the game, level = ") + toString(newLevel)), 1, 0, false) | |
131 | 209 | let boosterItemAssetId = calculateAssetId(boosterItem) | |
132 | 210 | [boosterItem, IntegerEntry(kBoughtLevel, newLevel), ScriptTransfer(i.caller, 1, boosterItemAssetId)] | |
133 | 211 | } | |
134 | 212 | } | |
135 | 213 | ||
136 | 214 | ||
137 | 215 | ||
138 | 216 | @Callable(i) | |
217 | + | func buyAccessItemAndLockDuck () = { | |
218 | + | let eggPayment = value(i.payments[0]) | |
219 | + | let duckPayment = value(i.payments[1]) | |
220 | + | let addressStr = toString(i.caller) | |
221 | + | if ((eggPayment.assetId != getAccessItemAssetId())) | |
222 | + | then throw("3BAI: Wrong asset attached") | |
223 | + | else if ((eggPayment.amount != getAccessItemPrice())) | |
224 | + | then throw("3BAI: Wrong amount of assets attached") | |
225 | + | else { | |
226 | + | let accessItemData = asString(invoke(this, "buyAccessItemInternal", [addressStr], nil)) | |
227 | + | if ((accessItemData == accessItemData)) | |
228 | + | then { | |
229 | + | let lockData = lockDuckInternal(addressStr, duckPayment) | |
230 | + | $Tuple2(lockData, accessItemData) | |
231 | + | } | |
232 | + | else throw("Strict value is not equal to itself.") | |
233 | + | } | |
234 | + | } | |
235 | + | ||
236 | + | ||
237 | + | ||
238 | + | @Callable(i) | |
139 | 239 | func buyAccessItem () = { | |
140 | - | let pmt = value(i.payments[0]) | |
141 | 240 | let addressStr = toString(i.caller) | |
142 | - | if ((pmt.assetId != getAccessItemAssetId())) | |
143 | - | then throw("Wrong asset attached") | |
144 | - | else if ((pmt.amount != getAccessItemPrice())) | |
145 | - | then throw("Wrong amount of assets attached") | |
241 | + | let eggPayment = value(i.payments[0]) | |
242 | + | if ((eggPayment.assetId != getAccessItemAssetId())) | |
243 | + | then throw("3BAI: Wrong asset attached") | |
244 | + | else if ((eggPayment.amount != getAccessItemPrice())) | |
245 | + | then throw("3BAI: Wrong amount of assets attached") | |
146 | 246 | else { | |
147 | - | let issueAccessItem = Issue("ACCESS-RACE", "[Access] Metarace access NFT for Duckracer", 1, 0, false) | |
148 | - | let accessRaceAssetId = toBase58String(calculateAssetId(issueAccessItem)) | |
149 | - | let kSpotsBought = keySpotsBought(addressStr) | |
150 | - | [issueAccessItem, StringEntry((("accessItem_" + accessRaceAssetId) + "_owner"), addressStr), StringEntry((("address_" + addressStr) + "_owning"), accessRaceAssetId), IntegerEntry(kSpotsBought, (tryGetInteger(kSpotsBought) + 1))] | |
247 | + | let accessItemData = asString(invoke(this, "buyAccessItemInternal", [addressStr], nil)) | |
248 | + | if ((accessItemData == accessItemData)) | |
249 | + | then $Tuple2(nil, accessItemData) | |
250 | + | else throw("Strict value is not equal to itself.") | |
251 | + | } | |
252 | + | } | |
253 | + | ||
254 | + | ||
255 | + | ||
256 | + | @Callable(i) | |
257 | + | func buyAccessItemOther (addressOther) = { | |
258 | + | let eggPayment = value(i.payments[0]) | |
259 | + | if ((eggPayment.assetId != getAccessItemAssetId())) | |
260 | + | then throw("3BAI: Wrong asset attached") | |
261 | + | else if ((eggPayment.amount != getAccessItemPrice())) | |
262 | + | then throw("3BAI: Wrong amount of assets attached") | |
263 | + | else { | |
264 | + | let accessItemData = asString(invoke(this, "buyAccessItemInternal", [addressOther], nil)) | |
265 | + | if ((accessItemData == accessItemData)) | |
266 | + | then $Tuple2(nil, accessItemData) | |
267 | + | else throw("Strict value is not equal to itself.") | |
151 | 268 | } | |
152 | 269 | } | |
153 | 270 | ||
154 | 271 | ||
155 | 272 | ||
156 | 273 | @Callable(i) | |
157 | 274 | func lockDuck () = { | |
158 | 275 | let addressStr = toString(i.caller) | |
159 | - | let pmt = value(i.payments[0]) | |
160 | - | let assetId = value(pmt.assetId) | |
161 | - | let assetIssuer = value(assetInfo(assetId)).issuer.bytes | |
162 | - | let tDuckDetails = getDuckDetails(assetId) | |
163 | - | let assetColor = tDuckDetails._1 | |
164 | - | let isJackpot = tDuckDetails._2 | |
165 | - | let kSpotsBusy = keySpotsBusy(addressStr) | |
166 | - | let spotsBusy = tryGetInteger(kSpotsBusy) | |
167 | - | let kSpotsBought = keySpotsBought(addressStr) | |
168 | - | let addressSpotsAvailable = (tryGetInteger(kSpotsBought) - spotsBusy) | |
169 | - | if ((pmt.amount != 1)) | |
170 | - | then throw("NFT is not attached") | |
171 | - | else if (if ((assetIssuer != getIncubatorAddress())) | |
172 | - | then (assetIssuer != getBreederAddress()) | |
173 | - | else false) | |
174 | - | then throw("ivalid NFT attached") | |
175 | - | else if ((0 >= addressSpotsAvailable)) | |
176 | - | then throw("No spots available") | |
177 | - | else { | |
178 | - | let kPerchesAvailable = ((("address_" + toString(this)) + "_perchesAvailable_") + assetColor) | |
179 | - | let perchesAvailable = match getInteger(Address(getFarmingAddress()), kPerchesAvailable) { | |
180 | - | case b: Int => | |
181 | - | b | |
182 | - | case _ => | |
183 | - | 0 | |
184 | - | } | |
185 | - | if ((perchesAvailable == perchesAvailable)) | |
186 | - | then { | |
187 | - | let eggAssetId = getEggAssetId() | |
188 | - | let inv1 = if ((perchesAvailable > 0)) | |
189 | - | then unit | |
190 | - | else invoke(Address(getFarmingAddress()), "buyPerch", [assetColor, ""], [AttachedPayment(eggAssetId, 100000000)]) | |
191 | - | if ((inv1 == inv1)) | |
192 | - | then { | |
193 | - | let inv2 = if ((isJackpot == false)) | |
194 | - | then invoke(Address(getFarmingAddress()), "stakeNFT", nil, [AttachedPayment(assetId, 1)]) | |
195 | - | else invoke(Address(getFarmingAddress()), "stakeJackpot", [assetColor], [AttachedPayment(assetId, 1)]) | |
196 | - | if ((inv2 == inv2)) | |
197 | - | then [IntegerEntry(kSpotsBusy, (spotsBusy + 1)), StringEntry(keyDuckOwner(assetId), addressStr), BooleanEntry(keyLockedDuckStatus(addressStr, toBase58String(assetId)), true), IntegerEntry(keyDuckUnlockTime(assetId), (lastBlock.timestamp + getMinLockDuration()))] | |
198 | - | else throw("Strict value is not equal to itself.") | |
199 | - | } | |
200 | - | else throw("Strict value is not equal to itself.") | |
201 | - | } | |
202 | - | else throw("Strict value is not equal to itself.") | |
203 | - | } | |
276 | + | let duckPayment = value(i.payments[0]) | |
277 | + | lockDuckInternal(addressStr, duckPayment) | |
204 | 278 | } | |
205 | 279 | ||
206 | 280 | ||
207 | 281 | ||
208 | 282 | @Callable(i) | |
209 | 283 | func unlockDuck (assetIdStr) = { | |
210 | 284 | let addressStr = toString(i.caller) | |
211 | 285 | let assetId = fromBase58String(assetIdStr) | |
212 | 286 | let tDuckDetails = getDuckDetails(assetId) | |
213 | 287 | let assetColor = tDuckDetails._1 | |
214 | 288 | let isJackpot = tDuckDetails._2 | |
215 | 289 | let kSpotsBusy = keySpotsBusy(addressStr) | |
216 | 290 | let kDuckOwner = keyDuckOwner(assetId) | |
217 | 291 | let remainingForUnlock = (tryGetInteger(keyDuckUnlockTime(assetId)) - lastBlock.timestamp) | |
218 | 292 | if ((tryGetString(kDuckOwner) != addressStr)) | |
219 | - | then throw("The duck is not yours") | |
293 | + | then throw("3UL: The duck is not yours") | |
220 | 294 | else if ((remainingForUnlock > 0)) | |
221 | - | then throw(((" | |
295 | + | then throw((("3UL: You need to wait to unlock " + toString((remainingForUnlock / 1000))) + " seconds")) | |
222 | 296 | else { | |
223 | 297 | let unstakeFuncName = if ((isJackpot == true)) | |
224 | 298 | then "unstakeJackpot" | |
225 | 299 | else "unstakeNFT" | |
226 | - | let inv1 = invoke( | |
300 | + | let inv1 = invoke(getFarmingAddress(), unstakeFuncName, [assetIdStr], nil) | |
227 | 301 | if ((inv1 == inv1)) | |
228 | 302 | then [ScriptTransfer(i.caller, 1, assetId), DeleteEntry(keyLockedDuckStatus(addressStr, assetIdStr)), DeleteEntry(kDuckOwner), IntegerEntry(kSpotsBusy, (tryGetInteger(kSpotsBusy) - 1))] | |
229 | 303 | else throw("Strict value is not equal to itself.") | |
230 | 304 | } | |
231 | 305 | } | |
232 | 306 | ||
233 | 307 |
github/deemru/w8io/786bc32 54.45 ms ◑