tx · EPK3kUQ7uHZsDDjJnoUz22BcTRewynzF1HRSYzreG89i 3P2sk1KncSxRaZs8b4CWGPw2jkvvav74u4D: -0.01000000 Waves 2022.12.12 02:54 [3421833] smart account 3P2sk1KncSxRaZs8b4CWGPw2jkvvav74u4D > SELF 0.00000000 Waves
{ "type": 13, "id": "EPK3kUQ7uHZsDDjJnoUz22BcTRewynzF1HRSYzreG89i", "fee": 1000000, "feeAssetId": null, "timestamp": 1670802914007, "version": 2, "chainId": 87, "sender": "3P2sk1KncSxRaZs8b4CWGPw2jkvvav74u4D", "senderPublicKey": "FHUpuxJwKFS6b8g1bFZnhamMkf5sT5zzAqgLfv1XzFcM", "proofs": [ "G3BHiyKWEp4kwZ4LYmT1Ytg9fsNfxada7fUAmDpPsJv18hPDswkPKxE84hFUUwinHifztr5K3Spyt8zYAZEoD3f" ], "script": "base64:BgIPCAISBQoDAQEBEgQKAhERHgAFTVVMVDYAwIQ9AAVNVUxUOACAwtcvAAt1c2RuQXNzZXRJZAEgtiYpwwT1zlORpA5LdSQvZIxRsfrfr1QpvUjSHSqyqtEAC2JhY2tFbmRBZGRyCQERQGV4dHJOYXRpdmUoMTA2MikBAiMzUEJLZXU3OHhkTnlhS3c1azJVYTZucHFlOE1SUWVnb3ExNwAPc3Rha2luZ0NvbnRyYWN0CQERQGV4dHJOYXRpdmUoMTA2MikBAiMzUExuY1h0UzFVODNENmNRYkZEM0g4ckJIUExnenhTRktaMQAPREVGQVVMVExPQ0FUSU9OAg9BZnJpY2FfRl9BZnJpY2EABk5VTVJFUwAGABNGQUNUT1JZTUFYV0FSRUhPVVNFAIDIr6AlABBSRVNPVVJDRVBSSUNFTUlOANXWCQAIcmVzVHlwZXMJAMwIAgIDT2lsCQDMCAICA09yZQkAzAgCAgRXb29kCQDMCAICBFNhbmQJAMwIAgIEQ2xheQkAzAgCAgdPcmdhbmljBQNuaWwACmNvbnRpbmVudHMJAMwIAgIIQW1lcmljYXMJAMwIAgIGRXVyb3BlCQDMCAICBEFzaWEJAMwIAgIGQWZyaWNhCQDMCAICB09jZWFuaWEFA25pbAEea2V5RmFjdG9yeVdhcmVob3VzZUJ5SWRBbmRUeXBlAglmYWN0b3J5SWQHcmVzVHlwZQkArAICCQCsAgIJAKwCAgIbZmFjdG9yeVdoQnlDb250aW5lbnRBbmRSZXNfBQlmYWN0b3J5SWQCAV8JAKQDAQUHcmVzVHlwZQERa2V5QXNzZXRJZFRvT3duZXIBB2Fzc2V0SWQJAKwCAgIJbmZ0T3duZXJfBQdhc3NldElkARZrZXlTdGFrZWRUaW1lQnlBc3NldElkAQdhc3NldElkCQCsAgICC3N0YWtlZFRpbWVfBQdhc3NldElkARRrZXlTdGFrZWREdWNrQnlPd25lcgEJb3duZXJBZGRyCQCsAgICEnN0YWtlZER1Y2tCeU93bmVyXwUJb3duZXJBZGRyARFrZXlCYWNrcGFja0J5RHVjawELZHVja0Fzc2V0SWQJAKwCAgIJYmFja1BhY2tfBQtkdWNrQXNzZXRJZAEPa2V5RHVja0xvY2F0aW9uAQtkdWNrQXNzZXRJZAkArAICAg1kdWNrTG9jYXRpb25fBQtkdWNrQXNzZXRJZAAHaWR4VHlwZQAAAAtpZHhRdWFudGl0eQABAAhpZHhQcmljZQACAA9sb2NJZHhDb250aW5lbnQAAAAKbG9jSWR4VHlwZQABAAhsb2NJZHhJZAACAApicElkeExldmVsAAAACGJwSWR4UmVzAAEACGJwSWR4TWF0AAIACWJwSWR4UHJvZAADAQhhc1N0cmluZwEBdgQHJG1hdGNoMAUBdgMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAXMFByRtYXRjaDAFAXMJAAIBAhhmYWlsIHRvIGNhc3QgaW50byBTdHJpbmcBBnN1YlJlcwMHcmVzTGlzdAdyZXNUeXBlBmFtb3VudAoBBnN1YmJlcgIDYWNjAWkJAM0IAgUDYWNjAwkAAAIFAWkFB3Jlc1R5cGUJAKQDAQkAZQIJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQdyZXNMaXN0BQFpBQZhbW91bnQJAJEDAgUHcmVzTGlzdAUBaQQBcgoAAiRsCQDMCAIAAAkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFBQNuaWwKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBBnN1YmJlcgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQITTGlzdCBzaXplIGV4Y2VlZHMgNgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgkAuQkCBQFyAgFfAQtidXlJbnRlcm5hbAQFbG9jSWQHcmVzVHlwZQZhbW91bnQIbWluUHJpY2UEBXdoS2V5CQEea2V5RmFjdG9yeVdhcmVob3VzZUJ5SWRBbmRUeXBlAgUFbG9jSWQFB3Jlc1R5cGUEAncwCQELdmFsdWVPckVsc2UCCQCfCAEFBXdoS2V5AAAEAnIwAwkAZgIFAncwBRNGQUNUT1JZTUFYV0FSRUhPVVNFAAADCQBmAgkAZAIFAncwBQZhbW91bnQFE0ZBQ1RPUllNQVhXQVJFSE9VU0UJAGUCBRNGQUNUT1JZTUFYV0FSRUhPVVNFBQJ3MAUGYW1vdW50BAx1c2RuUmVjZWl2ZWQJAGQCCQBrAwUCcjAJAGUCCQBoAgACBRBSRVNPVVJDRVBSSUNFTUlOCQBrAwkAZAIFAncwCQBpAgUCcjAAAgUQUkVTT1VSQ0VQUklDRU1JTgUTRkFDVE9SWU1BWFdBUkVIT1VTRQUFTVVMVDgJAGsDCQBlAgUGYW1vdW50BQJyMAUQUkVTT1VSQ0VQUklDRU1JTgUFTVVMVDgEBW1pbjk5CQBlAgUIbWluUHJpY2UJAGkCBQhtaW5QcmljZQBkAwkAZgIJAGgCBQVtaW45OQUGYW1vdW50CQBoAgUMdXNkblJlY2VpdmVkBQVNVUxUOAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgIPQWN0dWFsIHByaWNlID0gCQCkAwEFDHVzZG5SZWNlaXZlZAIDIC8gCQCkAwEFBmFtb3VudAIOIDwgbWluUHJpY2UgPSAJAKQDAQUIbWluUHJpY2UCAywgKAUFbG9jSWQCAiwgCQCRAwIFCHJlc1R5cGVzBQdyZXNUeXBlAgEpCQCUCgIJAQxJbnRlZ2VyRW50cnkCBQV3aEtleQkAZAIFAncwBQZhbW91bnQFDHVzZG5SZWNlaXZlZAIBaQEMc2VsbFJlc291cmNlAwdyZXNUeXBlBmFtb3VudAhtaW5QcmljZQMDCQBmAgAABQdyZXNUeXBlBgkAZwIFB3Jlc1R5cGUFBk5VTVJFUwkAAgEJAKwCAgISVW5rbm93biByZXNvdXJjZTogCQCkAwEFB3Jlc1R5cGUDCQBnAgAABQZhbW91bnQJAAIBCQCsAgICG0Ftb3VudCBzaG91bGQgYmUgcG9zaXRpdmUhIAkApAMBBQZhbW91bnQEC2R1Y2tBc3NldElkCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ0IAgUPc3Rha2luZ0NvbnRyYWN0CQEUa2V5U3Rha2VkRHVja0J5T3duZXIBCQClCAEIBQFpBmNhbGxlcgIcWW91IGRvbid0IGhhdmUgYSBkdWNrIHN0YWtlZAMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAAkAAgECKnNlbGxSZXNvdXJjZXMgZG9lc24ndCByZXF1aXJlIGFueSBwYXltZW50cwQLY3VyTG9jYXRpb24JALUJAgkBC3ZhbHVlT3JFbHNlAgkAnQgCBQ9zdGFraW5nQ29udHJhY3QJAQ9rZXlEdWNrTG9jYXRpb24BBQtkdWNrQXNzZXRJZAUPREVGQVVMVExPQ0FUSU9OAgFfAwkBAiE9AgkAkQMCBQtjdXJMb2NhdGlvbgUKbG9jSWR4VHlwZQIBRgkAAgEJAKwCAgItRHVjayBsb2NhdGlvbiB0eXBlIHNob3VsZCBiZSBGYWN0b3J5LCBidXQgaXMgCQCRAwIFC2N1ckxvY2F0aW9uBQpsb2NJZHhUeXBlBAtjdXJyZW50UGFjawkAtQkCCQELdmFsdWVPckVsc2UCCQCdCAIFD3N0YWtpbmdDb250cmFjdAkBEWtleUJhY2twYWNrQnlEdWNrAQULZHVja0Fzc2V0SWQCDzA6MF8wXzBfMF8wXzA6OgIBOgQHcmVzTGlzdAkAtQkCCQCRAwIFC2N1cnJlbnRQYWNrBQhicElkeFJlcwIBXwQKY3VycmVudFJlcwkBDXBhcnNlSW50VmFsdWUBCQCRAwIFB3Jlc0xpc3QFB3Jlc1R5cGUDCQBmAgUGYW1vdW50BQpjdXJyZW50UmVzCQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgICCVlvdSBoYXZlIAkApAMBBQpjdXJyZW50UmVzAgQgb2YgCQCRAwIFCHJlc1R5cGVzBQdyZXNUeXBlAiAgaW4gYmFja3BhY2ssIGJ1dCB0cmllZCB0byBzZWxsIAkApAMBBQZhbW91bnQEAWIJAQtidXlJbnRlcm5hbAQJAJEDAgULY3VyTG9jYXRpb24FCGxvY0lkeElkBQdyZXNUeXBlBQZhbW91bnQFCG1pblByaWNlBAVicFJlcwkBBnN1YlJlcwMFB3Jlc0xpc3QFB3Jlc1R5cGUFBmFtb3VudAQHbmV3UGFjawkAuQkCCQDMCAIJAJEDAgULY3VycmVudFBhY2sFCmJwSWR4TGV2ZWwJAMwIAgUFYnBSZXMJAMwIAgkAkQMCBQtjdXJyZW50UGFjawUIYnBJZHhNYXQJAMwIAgkAkQMCBQtjdXJyZW50UGFjawUJYnBJZHhQcm9kBQNuaWwCAToEBnJlc3VsdAkBCGFzU3RyaW5nAQkA/AcEBQ9zdGFraW5nQ29udHJhY3QCDnVwZGF0ZUJhY2twYWNrCQDMCAIFC2R1Y2tBc3NldElkCQDMCAIFB25ld1BhY2sFA25pbAUDbmlsCQCUCgIJAMwIAggFAWICXzEJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCAUBYgJfMgULdXNkbkFzc2V0SWQFA25pbAUGcmVzdWx0AWkBDXNlbGxSZXNvdXJjZXMCCHJlc1R5cGVzCW1pblByaWNlcwQLZHVja0Fzc2V0SWQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQ9zdGFraW5nQ29udHJhY3QJARRrZXlTdGFrZWREdWNrQnlPd25lcgEJAKUIAQgFAWkGY2FsbGVyAhxZb3UgZG9uJ3QgaGF2ZSBhIGR1Y2sgc3Rha2VkAwkBAiE9AgkAkAMBCAUBaQhwYXltZW50cwAACQACAQIqc2VsbFJlc291cmNlcyBkb2Vzbid0IHJlcXVpcmUgYW55IHBheW1lbnRzBAtjdXJMb2NhdGlvbgkAtQkCCQELdmFsdWVPckVsc2UCCQCdCAIFD3N0YWtpbmdDb250cmFjdAkBD2tleUR1Y2tMb2NhdGlvbgEFC2R1Y2tBc3NldElkBQ9ERUZBVUxUTE9DQVRJT04CAV8DCQECIT0CCQCRAwIFC2N1ckxvY2F0aW9uBQpsb2NJZHhUeXBlAgFGCQACAQkArAICAi1EdWNrIGxvY2F0aW9uIHR5cGUgc2hvdWxkIGJlIEZhY3RvcnksIGJ1dCBpcyAJAJEDAgULY3VyTG9jYXRpb24FCmxvY0lkeFR5cGUEBWxvY0lkCQCRAwIFC2N1ckxvY2F0aW9uBQhsb2NJZHhJZAQLY3VycmVudFBhY2sJALUJAgkBC3ZhbHVlT3JFbHNlAgkAnQgCBQ9zdGFraW5nQ29udHJhY3QJARFrZXlCYWNrcGFja0J5RHVjawEFC2R1Y2tBc3NldElkAg8wOjBfMF8wXzBfMF8wOjoCAToEB3Jlc0xpc3QJALUJAgkAkQMCBQtjdXJyZW50UGFjawUIYnBJZHhSZXMCAV8KAQVhZGRlcgIDYWNjA2lkeAQBagkAzwgCBQhyZXNUeXBlcwUDaWR4AwkBCWlzRGVmaW5lZAEFAWoEAWIJAQtidXlJbnRlcm5hbAQFBWxvY0lkBQNpZHgJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQdyZXNMaXN0BQNpZHgJAJEDAgUJbWluUHJpY2VzCQEFdmFsdWUBBQFqCQCVCgMJAM0IAggFA2FjYwJfMQgFAWICXzEJAM0IAggFA2FjYwJfMgIBMAkAZAIIBQNhY2MCXzMIBQFiAl8yCQCVCgMIBQNhY2MCXzEJAM0IAggFA2FjYwJfMgkAkQMCBQdyZXNMaXN0BQNpZHgIBQNhY2MCXzMEBm1lcmdlZAoAAiRsCQDMCAIAAAkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFBQNuaWwKAAIkcwkAkAMBBQIkbAoABSRhY2MwCQCVCgMFA25pbAUDbmlsAAAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBBWFkZGVyAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhNMaXN0IHNpemUgZXhjZWVkcyA2CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGBAduZXdQYWNrCQC5CQIJAMwIAgkAkQMCBQtjdXJyZW50UGFjawUKYnBJZHhMZXZlbAkAzAgCCQC5CQIIBQZtZXJnZWQCXzICAV8JAMwIAgkAkQMCBQtjdXJyZW50UGFjawUIYnBJZHhNYXQJAMwIAgkAkQMCBQtjdXJyZW50UGFjawUJYnBJZHhQcm9kBQNuaWwCAToEBnJlc3VsdAkBCGFzU3RyaW5nAQkA/AcEBQ9zdGFraW5nQ29udHJhY3QCDnVwZGF0ZUJhY2twYWNrCQDMCAIFC2R1Y2tBc3NldElkCQDMCAIFB25ld1BhY2sFA25pbAUDbmlsCQCUCgIJAM0IAggFBm1lcmdlZAJfMQkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCAUGbWVyZ2VkAl8zBQt1c2RuQXNzZXRJZAUGcmVzdWx0AE5DprM=", "height": 3421833, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: A3wnHzspbDYUaYLcUQh3bBrBf2bCv3Haexj2cTNbAMEm Next: 7y9ha72u9j5vjqZo1zH7nVircjib6c5Mum6K6WAj8XKv Diff:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let MULT6 = 1000000 | |
5 | + | ||
4 | 6 | let MULT8 = 100000000 | |
5 | 7 | ||
6 | 8 | let usdnAssetId = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p' | |
7 | 9 | ||
8 | 10 | let backEndAddr = addressFromStringValue("3PBKeu78xdNyaKw5k2Ua6npqe8MRQegoq17") | |
9 | 11 | ||
10 | - | let allowedResources = ["Oil", "Ore", "Wood", "Sand", "Clay", "Organic"] | |
12 | + | let stakingContract = addressFromStringValue("3PLncXtS1U83D6cQbFD3H8rBHPLgzxSFKZ1") | |
13 | + | ||
14 | + | let DEFAULTLOCATION = "Africa_F_Africa" | |
15 | + | ||
16 | + | let NUMRES = 6 | |
17 | + | ||
18 | + | let FACTORYMAXWAREHOUSE = 10000000000 | |
19 | + | ||
20 | + | let RESOURCEPRICEMIN = 158549 | |
21 | + | ||
22 | + | let resTypes = ["Oil", "Ore", "Wood", "Sand", "Clay", "Organic"] | |
11 | 23 | ||
12 | 24 | let continents = ["Americas", "Europe", "Asia", "Africa", "Oceania"] | |
13 | 25 | ||
14 | - | func keyStoreResByContinentAndType (continent,resType) = ((("storeResourceByContinentAndType_" + continent) + "_") + resType) | |
26 | + | func keyFactoryWarehouseByIdAndType (factoryId,resType) = ((("factoryWhByContinentAndRes_" + factoryId) + "_") + toString(resType)) | |
27 | + | ||
28 | + | ||
29 | + | func keyAssetIdToOwner (assetId) = ("nftOwner_" + assetId) | |
30 | + | ||
31 | + | ||
32 | + | func keyStakedTimeByAssetId (assetId) = ("stakedTime_" + assetId) | |
33 | + | ||
34 | + | ||
35 | + | func keyStakedDuckByOwner (ownerAddr) = ("stakedDuckByOwner_" + ownerAddr) | |
36 | + | ||
37 | + | ||
38 | + | func keyBackpackByDuck (duckAssetId) = ("backPack_" + duckAssetId) | |
39 | + | ||
40 | + | ||
41 | + | func keyDuckLocation (duckAssetId) = ("duckLocation_" + duckAssetId) | |
15 | 42 | ||
16 | 43 | ||
17 | 44 | let idxType = 0 | |
20 | 47 | ||
21 | 48 | let idxPrice = 2 | |
22 | 49 | ||
50 | + | let locIdxContinent = 0 | |
51 | + | ||
52 | + | let locIdxType = 1 | |
53 | + | ||
54 | + | let locIdxId = 2 | |
55 | + | ||
56 | + | let bpIdxLevel = 0 | |
57 | + | ||
58 | + | let bpIdxRes = 1 | |
59 | + | ||
60 | + | let bpIdxMat = 2 | |
61 | + | ||
62 | + | let bpIdxProd = 3 | |
63 | + | ||
64 | + | func asString (v) = match v { | |
65 | + | case s: String => | |
66 | + | s | |
67 | + | case _ => | |
68 | + | throw("fail to cast into String") | |
69 | + | } | |
70 | + | ||
71 | + | ||
72 | + | func subRes (resList,resType,amount) = { | |
73 | + | func subber (acc,i) = (acc :+ (if ((i == resType)) | |
74 | + | then toString((parseIntValue(resList[i]) - amount)) | |
75 | + | else resList[i])) | |
76 | + | ||
77 | + | let r = { | |
78 | + | let $l = [0, 1, 2, 3, 4, 5] | |
79 | + | let $s = size($l) | |
80 | + | let $acc0 = nil | |
81 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
82 | + | then $a | |
83 | + | else subber($a, $l[$i]) | |
84 | + | ||
85 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
86 | + | then $a | |
87 | + | else throw("List size exceeds 6") | |
88 | + | ||
89 | + | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
90 | + | } | |
91 | + | makeString(r, "_") | |
92 | + | } | |
93 | + | ||
94 | + | ||
95 | + | func buyInternal (locId,resType,amount,minPrice) = { | |
96 | + | let whKey = keyFactoryWarehouseByIdAndType(locId, resType) | |
97 | + | let w0 = valueOrElse(getInteger(whKey), 0) | |
98 | + | let r0 = if ((w0 > FACTORYMAXWAREHOUSE)) | |
99 | + | then 0 | |
100 | + | else if (((w0 + amount) > FACTORYMAXWAREHOUSE)) | |
101 | + | then (FACTORYMAXWAREHOUSE - w0) | |
102 | + | else amount | |
103 | + | let usdnReceived = (fraction(r0, ((2 * RESOURCEPRICEMIN) - fraction((w0 + (r0 / 2)), RESOURCEPRICEMIN, FACTORYMAXWAREHOUSE)), MULT8) + fraction((amount - r0), RESOURCEPRICEMIN, MULT8)) | |
104 | + | let min99 = (minPrice - (minPrice / 100)) | |
105 | + | if (((min99 * amount) > (usdnReceived * MULT8))) | |
106 | + | then throw((((((((((("Actual price = " + toString(usdnReceived)) + " / ") + toString(amount)) + " < minPrice = ") + toString(minPrice)) + ", (") + locId) + ", ") + resTypes[resType]) + ")")) | |
107 | + | else $Tuple2(IntegerEntry(whKey, (w0 + amount)), usdnReceived) | |
108 | + | } | |
109 | + | ||
110 | + | ||
23 | 111 | @Callable(i) | |
24 | - | func sellResource (userAddressStr,continent,resource) = if ((i.caller != backEndAddr)) | |
25 | - | then throw("permission denied") | |
26 | - | else if ((size(i.payments) != 0)) | |
27 | - | then throw("sellResources doesn't require any payments") | |
28 | - | else if (!(containsElement(continents, continent))) | |
29 | - | then throw(("Unknown continent: " + continent)) | |
30 | - | else { | |
31 | - | let userAddress = addressFromStringValue(userAddressStr) | |
32 | - | let parts = split(resource, "_") | |
33 | - | let resType = parts[idxType] | |
34 | - | if (!(containsElement(allowedResources, resType))) | |
35 | - | then throw(("Unknown resource: " + resType)) | |
36 | - | else { | |
37 | - | let resKey = keyStoreResByContinentAndType(continent, resType) | |
38 | - | let q = parseIntValue(parts[idxQuantity]) | |
39 | - | [IntegerEntry(resKey, (valueOrElse(getInteger(resKey), 0) + q)), ScriptTransfer(userAddress, fraction(q, parseIntValue(parts[idxPrice]), MULT8), usdnAssetId)] | |
40 | - | } | |
41 | - | } | |
112 | + | func sellResource (resType,amount,minPrice) = if (if ((0 > resType)) | |
113 | + | then true | |
114 | + | else (resType >= NUMRES)) | |
115 | + | then throw(("Unknown resource: " + toString(resType))) | |
116 | + | else if ((0 >= amount)) | |
117 | + | then throw(("Amount should be positive! " + toString(amount))) | |
118 | + | else { | |
119 | + | let duckAssetId = valueOrErrorMessage(getString(stakingContract, keyStakedDuckByOwner(toString(i.caller))), "You don't have a duck staked") | |
120 | + | if ((size(i.payments) != 0)) | |
121 | + | then throw("sellResources doesn't require any payments") | |
122 | + | else { | |
123 | + | let curLocation = split(valueOrElse(getString(stakingContract, keyDuckLocation(duckAssetId)), DEFAULTLOCATION), "_") | |
124 | + | if ((curLocation[locIdxType] != "F")) | |
125 | + | then throw(("Duck location type should be Factory, but is " + curLocation[locIdxType])) | |
126 | + | else { | |
127 | + | let currentPack = split(valueOrElse(getString(stakingContract, keyBackpackByDuck(duckAssetId)), "0:0_0_0_0_0_0::"), ":") | |
128 | + | let resList = split(currentPack[bpIdxRes], "_") | |
129 | + | let currentRes = parseIntValue(resList[resType]) | |
130 | + | if ((amount > currentRes)) | |
131 | + | then throw(((((("You have " + toString(currentRes)) + " of ") + resTypes[resType]) + " in backpack, but tried to sell ") + toString(amount))) | |
132 | + | else { | |
133 | + | let b = buyInternal(curLocation[locIdxId], resType, amount, minPrice) | |
134 | + | let bpRes = subRes(resList, resType, amount) | |
135 | + | let newPack = makeString([currentPack[bpIdxLevel], bpRes, currentPack[bpIdxMat], currentPack[bpIdxProd]], ":") | |
136 | + | let result = asString(invoke(stakingContract, "updateBackpack", [duckAssetId, newPack], nil)) | |
137 | + | $Tuple2([b._1, ScriptTransfer(i.caller, b._2, usdnAssetId)], result) | |
138 | + | } | |
139 | + | } | |
140 | + | } | |
141 | + | } | |
42 | 142 | ||
43 | 143 | ||
44 | 144 | ||
45 | 145 | @Callable(i) | |
46 | - | func sellResources ( | |
47 | - | | |
48 | - | | |
146 | + | func sellResources (resTypes,minPrices) = { | |
147 | + | let duckAssetId = valueOrErrorMessage(getString(stakingContract, keyStakedDuckByOwner(toString(i.caller))), "You don't have a duck staked") | |
148 | + | if ((size(i.payments) != 0)) | |
49 | 149 | then throw("sellResources doesn't require any payments") | |
50 | - | else if (!(containsElement(continents, continent))) | |
51 | - | then throw(("Unknown continent: " + continent)) | |
52 | - | else { | |
53 | - | let userAddress = addressFromStringValue(userAddressStr) | |
54 | - | func adder (acc,elem) = { | |
55 | - | let parts = split(elem, "_") | |
56 | - | let resType = parts[idxType] | |
57 | - | if (!(containsElement(allowedResources, resType))) | |
58 | - | then throw(("Unknown resource: " + resType)) | |
59 | - | else { | |
60 | - | let resKey = keyStoreResByContinentAndType(continent, resType) | |
61 | - | let q = parseIntValue(parts[idxQuantity]) | |
62 | - | $Tuple2((acc._1 :+ IntegerEntry(resKey, (valueOrElse(getInteger(resKey), 0) + q))), (acc._2 + fraction(q, parseIntValue(parts[idxPrice]), MULT8))) | |
63 | - | } | |
150 | + | else { | |
151 | + | let curLocation = split(valueOrElse(getString(stakingContract, keyDuckLocation(duckAssetId)), DEFAULTLOCATION), "_") | |
152 | + | if ((curLocation[locIdxType] != "F")) | |
153 | + | then throw(("Duck location type should be Factory, but is " + curLocation[locIdxType])) | |
154 | + | else { | |
155 | + | let locId = curLocation[locIdxId] | |
156 | + | let currentPack = split(valueOrElse(getString(stakingContract, keyBackpackByDuck(duckAssetId)), "0:0_0_0_0_0_0::"), ":") | |
157 | + | let resList = split(currentPack[bpIdxRes], "_") | |
158 | + | func adder (acc,idx) = { | |
159 | + | let j = indexOf(resTypes, idx) | |
160 | + | if (isDefined(j)) | |
161 | + | then { | |
162 | + | let b = buyInternal(locId, idx, parseIntValue(resList[idx]), minPrices[value(j)]) | |
163 | + | $Tuple3((acc._1 :+ b._1), (acc._2 :+ "0"), (acc._3 + b._2)) | |
164 | + | } | |
165 | + | else $Tuple3(acc._1, (acc._2 :+ resList[idx]), acc._3) | |
166 | + | } | |
167 | + | ||
168 | + | let merged = { | |
169 | + | let $l = [0, 1, 2, 3, 4, 5] | |
170 | + | let $s = size($l) | |
171 | + | let $acc0 = $Tuple3(nil, nil, 0) | |
172 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
173 | + | then $a | |
174 | + | else adder($a, $l[$i]) | |
175 | + | ||
176 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
177 | + | then $a | |
178 | + | else throw("List size exceeds 6") | |
179 | + | ||
180 | + | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
181 | + | } | |
182 | + | let newPack = makeString([currentPack[bpIdxLevel], makeString(merged._2, "_"), currentPack[bpIdxMat], currentPack[bpIdxProd]], ":") | |
183 | + | let result = asString(invoke(stakingContract, "updateBackpack", [duckAssetId, newPack], nil)) | |
184 | + | $Tuple2((merged._1 :+ ScriptTransfer(i.caller, merged._3, usdnAssetId)), result) | |
64 | 185 | } | |
65 | - | ||
66 | - | let result = { | |
67 | - | let $l = resources | |
68 | - | let $s = size($l) | |
69 | - | let $acc0 = $Tuple2(nil, 0) | |
70 | - | func $f0_1 ($a,$i) = if (($i >= $s)) | |
71 | - | then $a | |
72 | - | else adder($a, $l[$i]) | |
73 | - | ||
74 | - | func $f0_2 ($a,$i) = if (($i >= $s)) | |
75 | - | then $a | |
76 | - | else throw("List size exceeds 10") | |
77 | - | ||
78 | - | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10) | |
79 | - | } | |
80 | - | (result._1 :+ ScriptTransfer(userAddress, result._2, usdnAssetId)) | |
81 | - | } | |
186 | + | } | |
187 | + | } | |
82 | 188 | ||
83 | 189 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let MULT6 = 1000000 | |
5 | + | ||
4 | 6 | let MULT8 = 100000000 | |
5 | 7 | ||
6 | 8 | let usdnAssetId = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p' | |
7 | 9 | ||
8 | 10 | let backEndAddr = addressFromStringValue("3PBKeu78xdNyaKw5k2Ua6npqe8MRQegoq17") | |
9 | 11 | ||
10 | - | let allowedResources = ["Oil", "Ore", "Wood", "Sand", "Clay", "Organic"] | |
12 | + | let stakingContract = addressFromStringValue("3PLncXtS1U83D6cQbFD3H8rBHPLgzxSFKZ1") | |
13 | + | ||
14 | + | let DEFAULTLOCATION = "Africa_F_Africa" | |
15 | + | ||
16 | + | let NUMRES = 6 | |
17 | + | ||
18 | + | let FACTORYMAXWAREHOUSE = 10000000000 | |
19 | + | ||
20 | + | let RESOURCEPRICEMIN = 158549 | |
21 | + | ||
22 | + | let resTypes = ["Oil", "Ore", "Wood", "Sand", "Clay", "Organic"] | |
11 | 23 | ||
12 | 24 | let continents = ["Americas", "Europe", "Asia", "Africa", "Oceania"] | |
13 | 25 | ||
14 | - | func keyStoreResByContinentAndType (continent,resType) = ((("storeResourceByContinentAndType_" + continent) + "_") + resType) | |
26 | + | func keyFactoryWarehouseByIdAndType (factoryId,resType) = ((("factoryWhByContinentAndRes_" + factoryId) + "_") + toString(resType)) | |
27 | + | ||
28 | + | ||
29 | + | func keyAssetIdToOwner (assetId) = ("nftOwner_" + assetId) | |
30 | + | ||
31 | + | ||
32 | + | func keyStakedTimeByAssetId (assetId) = ("stakedTime_" + assetId) | |
33 | + | ||
34 | + | ||
35 | + | func keyStakedDuckByOwner (ownerAddr) = ("stakedDuckByOwner_" + ownerAddr) | |
36 | + | ||
37 | + | ||
38 | + | func keyBackpackByDuck (duckAssetId) = ("backPack_" + duckAssetId) | |
39 | + | ||
40 | + | ||
41 | + | func keyDuckLocation (duckAssetId) = ("duckLocation_" + duckAssetId) | |
15 | 42 | ||
16 | 43 | ||
17 | 44 | let idxType = 0 | |
18 | 45 | ||
19 | 46 | let idxQuantity = 1 | |
20 | 47 | ||
21 | 48 | let idxPrice = 2 | |
22 | 49 | ||
50 | + | let locIdxContinent = 0 | |
51 | + | ||
52 | + | let locIdxType = 1 | |
53 | + | ||
54 | + | let locIdxId = 2 | |
55 | + | ||
56 | + | let bpIdxLevel = 0 | |
57 | + | ||
58 | + | let bpIdxRes = 1 | |
59 | + | ||
60 | + | let bpIdxMat = 2 | |
61 | + | ||
62 | + | let bpIdxProd = 3 | |
63 | + | ||
64 | + | func asString (v) = match v { | |
65 | + | case s: String => | |
66 | + | s | |
67 | + | case _ => | |
68 | + | throw("fail to cast into String") | |
69 | + | } | |
70 | + | ||
71 | + | ||
72 | + | func subRes (resList,resType,amount) = { | |
73 | + | func subber (acc,i) = (acc :+ (if ((i == resType)) | |
74 | + | then toString((parseIntValue(resList[i]) - amount)) | |
75 | + | else resList[i])) | |
76 | + | ||
77 | + | let r = { | |
78 | + | let $l = [0, 1, 2, 3, 4, 5] | |
79 | + | let $s = size($l) | |
80 | + | let $acc0 = nil | |
81 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
82 | + | then $a | |
83 | + | else subber($a, $l[$i]) | |
84 | + | ||
85 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
86 | + | then $a | |
87 | + | else throw("List size exceeds 6") | |
88 | + | ||
89 | + | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
90 | + | } | |
91 | + | makeString(r, "_") | |
92 | + | } | |
93 | + | ||
94 | + | ||
95 | + | func buyInternal (locId,resType,amount,minPrice) = { | |
96 | + | let whKey = keyFactoryWarehouseByIdAndType(locId, resType) | |
97 | + | let w0 = valueOrElse(getInteger(whKey), 0) | |
98 | + | let r0 = if ((w0 > FACTORYMAXWAREHOUSE)) | |
99 | + | then 0 | |
100 | + | else if (((w0 + amount) > FACTORYMAXWAREHOUSE)) | |
101 | + | then (FACTORYMAXWAREHOUSE - w0) | |
102 | + | else amount | |
103 | + | let usdnReceived = (fraction(r0, ((2 * RESOURCEPRICEMIN) - fraction((w0 + (r0 / 2)), RESOURCEPRICEMIN, FACTORYMAXWAREHOUSE)), MULT8) + fraction((amount - r0), RESOURCEPRICEMIN, MULT8)) | |
104 | + | let min99 = (minPrice - (minPrice / 100)) | |
105 | + | if (((min99 * amount) > (usdnReceived * MULT8))) | |
106 | + | then throw((((((((((("Actual price = " + toString(usdnReceived)) + " / ") + toString(amount)) + " < minPrice = ") + toString(minPrice)) + ", (") + locId) + ", ") + resTypes[resType]) + ")")) | |
107 | + | else $Tuple2(IntegerEntry(whKey, (w0 + amount)), usdnReceived) | |
108 | + | } | |
109 | + | ||
110 | + | ||
23 | 111 | @Callable(i) | |
24 | - | func sellResource (userAddressStr,continent,resource) = if ((i.caller != backEndAddr)) | |
25 | - | then throw("permission denied") | |
26 | - | else if ((size(i.payments) != 0)) | |
27 | - | then throw("sellResources doesn't require any payments") | |
28 | - | else if (!(containsElement(continents, continent))) | |
29 | - | then throw(("Unknown continent: " + continent)) | |
30 | - | else { | |
31 | - | let userAddress = addressFromStringValue(userAddressStr) | |
32 | - | let parts = split(resource, "_") | |
33 | - | let resType = parts[idxType] | |
34 | - | if (!(containsElement(allowedResources, resType))) | |
35 | - | then throw(("Unknown resource: " + resType)) | |
36 | - | else { | |
37 | - | let resKey = keyStoreResByContinentAndType(continent, resType) | |
38 | - | let q = parseIntValue(parts[idxQuantity]) | |
39 | - | [IntegerEntry(resKey, (valueOrElse(getInteger(resKey), 0) + q)), ScriptTransfer(userAddress, fraction(q, parseIntValue(parts[idxPrice]), MULT8), usdnAssetId)] | |
40 | - | } | |
41 | - | } | |
112 | + | func sellResource (resType,amount,minPrice) = if (if ((0 > resType)) | |
113 | + | then true | |
114 | + | else (resType >= NUMRES)) | |
115 | + | then throw(("Unknown resource: " + toString(resType))) | |
116 | + | else if ((0 >= amount)) | |
117 | + | then throw(("Amount should be positive! " + toString(amount))) | |
118 | + | else { | |
119 | + | let duckAssetId = valueOrErrorMessage(getString(stakingContract, keyStakedDuckByOwner(toString(i.caller))), "You don't have a duck staked") | |
120 | + | if ((size(i.payments) != 0)) | |
121 | + | then throw("sellResources doesn't require any payments") | |
122 | + | else { | |
123 | + | let curLocation = split(valueOrElse(getString(stakingContract, keyDuckLocation(duckAssetId)), DEFAULTLOCATION), "_") | |
124 | + | if ((curLocation[locIdxType] != "F")) | |
125 | + | then throw(("Duck location type should be Factory, but is " + curLocation[locIdxType])) | |
126 | + | else { | |
127 | + | let currentPack = split(valueOrElse(getString(stakingContract, keyBackpackByDuck(duckAssetId)), "0:0_0_0_0_0_0::"), ":") | |
128 | + | let resList = split(currentPack[bpIdxRes], "_") | |
129 | + | let currentRes = parseIntValue(resList[resType]) | |
130 | + | if ((amount > currentRes)) | |
131 | + | then throw(((((("You have " + toString(currentRes)) + " of ") + resTypes[resType]) + " in backpack, but tried to sell ") + toString(amount))) | |
132 | + | else { | |
133 | + | let b = buyInternal(curLocation[locIdxId], resType, amount, minPrice) | |
134 | + | let bpRes = subRes(resList, resType, amount) | |
135 | + | let newPack = makeString([currentPack[bpIdxLevel], bpRes, currentPack[bpIdxMat], currentPack[bpIdxProd]], ":") | |
136 | + | let result = asString(invoke(stakingContract, "updateBackpack", [duckAssetId, newPack], nil)) | |
137 | + | $Tuple2([b._1, ScriptTransfer(i.caller, b._2, usdnAssetId)], result) | |
138 | + | } | |
139 | + | } | |
140 | + | } | |
141 | + | } | |
42 | 142 | ||
43 | 143 | ||
44 | 144 | ||
45 | 145 | @Callable(i) | |
46 | - | func sellResources ( | |
47 | - | | |
48 | - | | |
146 | + | func sellResources (resTypes,minPrices) = { | |
147 | + | let duckAssetId = valueOrErrorMessage(getString(stakingContract, keyStakedDuckByOwner(toString(i.caller))), "You don't have a duck staked") | |
148 | + | if ((size(i.payments) != 0)) | |
49 | 149 | then throw("sellResources doesn't require any payments") | |
50 | - | else if (!(containsElement(continents, continent))) | |
51 | - | then throw(("Unknown continent: " + continent)) | |
52 | - | else { | |
53 | - | let userAddress = addressFromStringValue(userAddressStr) | |
54 | - | func adder (acc,elem) = { | |
55 | - | let parts = split(elem, "_") | |
56 | - | let resType = parts[idxType] | |
57 | - | if (!(containsElement(allowedResources, resType))) | |
58 | - | then throw(("Unknown resource: " + resType)) | |
59 | - | else { | |
60 | - | let resKey = keyStoreResByContinentAndType(continent, resType) | |
61 | - | let q = parseIntValue(parts[idxQuantity]) | |
62 | - | $Tuple2((acc._1 :+ IntegerEntry(resKey, (valueOrElse(getInteger(resKey), 0) + q))), (acc._2 + fraction(q, parseIntValue(parts[idxPrice]), MULT8))) | |
63 | - | } | |
150 | + | else { | |
151 | + | let curLocation = split(valueOrElse(getString(stakingContract, keyDuckLocation(duckAssetId)), DEFAULTLOCATION), "_") | |
152 | + | if ((curLocation[locIdxType] != "F")) | |
153 | + | then throw(("Duck location type should be Factory, but is " + curLocation[locIdxType])) | |
154 | + | else { | |
155 | + | let locId = curLocation[locIdxId] | |
156 | + | let currentPack = split(valueOrElse(getString(stakingContract, keyBackpackByDuck(duckAssetId)), "0:0_0_0_0_0_0::"), ":") | |
157 | + | let resList = split(currentPack[bpIdxRes], "_") | |
158 | + | func adder (acc,idx) = { | |
159 | + | let j = indexOf(resTypes, idx) | |
160 | + | if (isDefined(j)) | |
161 | + | then { | |
162 | + | let b = buyInternal(locId, idx, parseIntValue(resList[idx]), minPrices[value(j)]) | |
163 | + | $Tuple3((acc._1 :+ b._1), (acc._2 :+ "0"), (acc._3 + b._2)) | |
164 | + | } | |
165 | + | else $Tuple3(acc._1, (acc._2 :+ resList[idx]), acc._3) | |
166 | + | } | |
167 | + | ||
168 | + | let merged = { | |
169 | + | let $l = [0, 1, 2, 3, 4, 5] | |
170 | + | let $s = size($l) | |
171 | + | let $acc0 = $Tuple3(nil, nil, 0) | |
172 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
173 | + | then $a | |
174 | + | else adder($a, $l[$i]) | |
175 | + | ||
176 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
177 | + | then $a | |
178 | + | else throw("List size exceeds 6") | |
179 | + | ||
180 | + | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
181 | + | } | |
182 | + | let newPack = makeString([currentPack[bpIdxLevel], makeString(merged._2, "_"), currentPack[bpIdxMat], currentPack[bpIdxProd]], ":") | |
183 | + | let result = asString(invoke(stakingContract, "updateBackpack", [duckAssetId, newPack], nil)) | |
184 | + | $Tuple2((merged._1 :+ ScriptTransfer(i.caller, merged._3, usdnAssetId)), result) | |
64 | 185 | } | |
65 | - | ||
66 | - | let result = { | |
67 | - | let $l = resources | |
68 | - | let $s = size($l) | |
69 | - | let $acc0 = $Tuple2(nil, 0) | |
70 | - | func $f0_1 ($a,$i) = if (($i >= $s)) | |
71 | - | then $a | |
72 | - | else adder($a, $l[$i]) | |
73 | - | ||
74 | - | func $f0_2 ($a,$i) = if (($i >= $s)) | |
75 | - | then $a | |
76 | - | else throw("List size exceeds 10") | |
77 | - | ||
78 | - | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10) | |
79 | - | } | |
80 | - | (result._1 :+ ScriptTransfer(userAddress, result._2, usdnAssetId)) | |
81 | - | } | |
186 | + | } | |
187 | + | } | |
82 | 188 | ||
83 | 189 |
github/deemru/w8io/3ef1775 48.61 ms ◑