tx · 68how4fSFhdCGt2zDTNMzBE6h5HmFj1t3ZFVPeJ1Ffoc 3N2Z6EjzrzaCpD8ypDYLWDKe2PPVU8y9dJs: -0.01400000 Waves 2019.08.27 22:32 [650025] smart account 3N2Z6EjzrzaCpD8ypDYLWDKe2PPVU8y9dJs > SELF 0.00000000 Waves
{ "type": 13, "id": "68how4fSFhdCGt2zDTNMzBE6h5HmFj1t3ZFVPeJ1Ffoc", "fee": 1400000, "feeAssetId": null, "timestamp": 1566934365825, "version": 1, "sender": "3N2Z6EjzrzaCpD8ypDYLWDKe2PPVU8y9dJs", "senderPublicKey": "HSTmdhPrJF81ozDBaqVn8QVtV76oD7gzDfcHPQRdxdwW", "proofs": [ "uh1SYDy3Dv96pm7V2j62xzCujtR3cyvuGAWjdmRzmjch7C16LgrZGaH2M16xBe9RTKQUvP9LQn3yJmXa2Hi3MRn" ], "script": "base64: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", "chainId": 84, "height": 650025, "spentComplexity": 0 } View: original | compacted Prev: FmzX7wcfv8r7Kd3fUHx6tjvDdhABKdpu8ZiRWTihxPPD Next: 8sMTk9AjyQURHowaEPnMurZ9YqGDhQykwAQAyX4fChVz Diff:
Old | New | Differences | |
---|---|---|---|
34 | 34 | } | |
35 | 35 | ||
36 | 36 | ||
37 | - | func | |
37 | + | func getBalance (key) = { | |
38 | 38 | let playerBalanceNum = match getInteger(this, key) { | |
39 | 39 | case playerBalanceNum: Int => | |
40 | 40 | playerBalanceNum | |
56 | 56 | } | |
57 | 57 | ||
58 | 58 | ||
59 | + | func getPriceByItemID (itemId) = if ((itemId == "1")) | |
60 | + | then 1 | |
61 | + | else if ((itemId == "2")) | |
62 | + | then 4 | |
63 | + | else if ((itemId == "3")) | |
64 | + | then 6 | |
65 | + | else throw("Неверный ID item") | |
66 | + | ||
67 | + | ||
59 | 68 | @Callable(i) | |
60 | 69 | func buy (itemId) = { | |
61 | 70 | let buyTXId = toBase58String(i.transactionId) | |
62 | 71 | let playerAddress = toString(i.caller) | |
63 | 72 | let key2Items = (playerAddress + "_items") | |
73 | + | let balanceMinus = getPriceByItemID(itemId) | |
74 | + | let NewBalance = (getBalance((playerAddress + "_balance")) - balanceMinus) | |
75 | + | let newBalanceData = DataEntry((playerAddress + "_balance"), NewBalance) | |
64 | 76 | let oldItemsList = ExtractItemsList(key2Items) | |
65 | 77 | if ((oldItemsList == "")) | |
66 | - | then WriteSet([DataEntry(key2Items, (itemId + ":1")), DataEntry(BUYITEM, buyTXId)]) | |
78 | + | then WriteSet([DataEntry(key2Items, (itemId + ":1")), DataEntry(BUYITEM, buyTXId), newBalanceData]) | |
67 | 79 | else if ((oldItemsList != "")) | |
68 | - | then WriteSet([DataEntry(key2Items, (((oldItemsList + " ") + itemId) + ":1")), DataEntry(BUYITEM, buyTXId)]) | |
80 | + | then WriteSet([DataEntry(key2Items, (((oldItemsList + " ") + itemId) + ":1")), DataEntry(BUYITEM, buyTXId), newBalanceData]) | |
69 | 81 | else throw("Что-то пошло совсем не так c Items") | |
70 | 82 | } | |
71 | 83 | ||
111 | 123 | then { | |
112 | 124 | let donateNum = ((donateAmt - COMMISSION) / WAVELET) | |
113 | 125 | let key2Balance = (playerAddress + "_balance") | |
114 | - | let newBalance = ( | |
126 | + | let newBalance = (getBalance(key2Balance) + (donateNum * TokenPrice)) | |
115 | 127 | let donateData = [DataEntry(DONATECOUNTERKEY, newGameNum), DataEntry(key2Balance, newBalance), DataEntry(donateId, donateNum)] | |
116 | 128 | WriteSet(donateData) | |
117 | 129 | } |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 3 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let WAVELET = ((100 * 1000) * 1000) | |
5 | 5 | ||
6 | 6 | let COMMISSION = ((5 * WAVELET) / 1000) | |
7 | 7 | ||
8 | 8 | let TokenPrice = 10 | |
9 | 9 | ||
10 | 10 | let DONATECOUNTERKEY = "$DONATE_NUM" | |
11 | 11 | ||
12 | 12 | let BUYITEM = "$BUY_ITEM" | |
13 | 13 | ||
14 | 14 | let Bag1 = (1 * WAVELET) | |
15 | 15 | ||
16 | 16 | let Bag5 = (5 * WAVELET) | |
17 | 17 | ||
18 | 18 | let Bag10 = (10 * WAVELET) | |
19 | 19 | ||
20 | 20 | let Bag20 = (20 * WAVELET) | |
21 | 21 | ||
22 | 22 | let Bag50 = (50 * WAVELET) | |
23 | 23 | ||
24 | 24 | let Bag100 = (100 * WAVELET) | |
25 | 25 | ||
26 | 26 | func IncrementDonateNum () = { | |
27 | 27 | let gameNum = match getInteger(this, DONATECOUNTERKEY) { | |
28 | 28 | case num: Int => | |
29 | 29 | num | |
30 | 30 | case _ => | |
31 | 31 | 0 | |
32 | 32 | } | |
33 | 33 | (gameNum + 1) | |
34 | 34 | } | |
35 | 35 | ||
36 | 36 | ||
37 | - | func | |
37 | + | func getBalance (key) = { | |
38 | 38 | let playerBalanceNum = match getInteger(this, key) { | |
39 | 39 | case playerBalanceNum: Int => | |
40 | 40 | playerBalanceNum | |
41 | 41 | case _ => | |
42 | 42 | 0 | |
43 | 43 | } | |
44 | 44 | playerBalanceNum | |
45 | 45 | } | |
46 | 46 | ||
47 | 47 | ||
48 | 48 | func ExtractItemsList (key) = { | |
49 | 49 | let rawDataStr = match getString(this, key) { | |
50 | 50 | case str: String => | |
51 | 51 | str | |
52 | 52 | case _ => | |
53 | 53 | throw(("Couldn't find list of player items by key:" + key)) | |
54 | 54 | } | |
55 | 55 | rawDataStr | |
56 | 56 | } | |
57 | 57 | ||
58 | 58 | ||
59 | + | func getPriceByItemID (itemId) = if ((itemId == "1")) | |
60 | + | then 1 | |
61 | + | else if ((itemId == "2")) | |
62 | + | then 4 | |
63 | + | else if ((itemId == "3")) | |
64 | + | then 6 | |
65 | + | else throw("Неверный ID item") | |
66 | + | ||
67 | + | ||
59 | 68 | @Callable(i) | |
60 | 69 | func buy (itemId) = { | |
61 | 70 | let buyTXId = toBase58String(i.transactionId) | |
62 | 71 | let playerAddress = toString(i.caller) | |
63 | 72 | let key2Items = (playerAddress + "_items") | |
73 | + | let balanceMinus = getPriceByItemID(itemId) | |
74 | + | let NewBalance = (getBalance((playerAddress + "_balance")) - balanceMinus) | |
75 | + | let newBalanceData = DataEntry((playerAddress + "_balance"), NewBalance) | |
64 | 76 | let oldItemsList = ExtractItemsList(key2Items) | |
65 | 77 | if ((oldItemsList == "")) | |
66 | - | then WriteSet([DataEntry(key2Items, (itemId + ":1")), DataEntry(BUYITEM, buyTXId)]) | |
78 | + | then WriteSet([DataEntry(key2Items, (itemId + ":1")), DataEntry(BUYITEM, buyTXId), newBalanceData]) | |
67 | 79 | else if ((oldItemsList != "")) | |
68 | - | then WriteSet([DataEntry(key2Items, (((oldItemsList + " ") + itemId) + ":1")), DataEntry(BUYITEM, buyTXId)]) | |
80 | + | then WriteSet([DataEntry(key2Items, (((oldItemsList + " ") + itemId) + ":1")), DataEntry(BUYITEM, buyTXId), newBalanceData]) | |
69 | 81 | else throw("Что-то пошло совсем не так c Items") | |
70 | 82 | } | |
71 | 83 | ||
72 | 84 | ||
73 | 85 | ||
74 | 86 | @Callable(i) | |
75 | 87 | func reset () = { | |
76 | 88 | let playerAddress = toString(i.caller) | |
77 | 89 | let donateData = [DataEntry(DONATECOUNTERKEY, 0), DataEntry((playerAddress + "_balance"), 0)] | |
78 | 90 | WriteSet(donateData) | |
79 | 91 | } | |
80 | 92 | ||
81 | 93 | ||
82 | 94 | ||
83 | 95 | @Callable(i) | |
84 | 96 | func donate () = { | |
85 | 97 | let newGameNum = IncrementDonateNum() | |
86 | 98 | let donateId = toBase58String(i.transactionId) | |
87 | 99 | let pmt = extract(i.payment) | |
88 | 100 | let playerAddress = toString(i.caller) | |
89 | 101 | let betNotInWaves = isDefined(pmt.assetId) | |
90 | 102 | let feeNotInWaves = isDefined(pmt.assetId) | |
91 | 103 | let donateAmt = pmt.amount | |
92 | 104 | if (betNotInWaves) | |
93 | 105 | then throw("Bet amount must be in Waves") | |
94 | 106 | else if (feeNotInWaves) | |
95 | 107 | then throw("Transaction's fee must be in Waves") | |
96 | 108 | else { | |
97 | 109 | let donateAmtValid = if (if (if (if (if (if ((donateAmt == (Bag1 + COMMISSION))) | |
98 | 110 | then true | |
99 | 111 | else (donateAmt == (Bag5 + COMMISSION))) | |
100 | 112 | then true | |
101 | 113 | else (donateAmt == (Bag10 + COMMISSION))) | |
102 | 114 | then true | |
103 | 115 | else (donateAmt == (Bag20 + COMMISSION))) | |
104 | 116 | then true | |
105 | 117 | else (donateAmt == (Bag50 + COMMISSION))) | |
106 | 118 | then true | |
107 | 119 | else (donateAmt == (Bag100 + COMMISSION))) | |
108 | 120 | then true | |
109 | 121 | else false | |
110 | 122 | if (donateAmtValid) | |
111 | 123 | then { | |
112 | 124 | let donateNum = ((donateAmt - COMMISSION) / WAVELET) | |
113 | 125 | let key2Balance = (playerAddress + "_balance") | |
114 | - | let newBalance = ( | |
126 | + | let newBalance = (getBalance(key2Balance) + (donateNum * TokenPrice)) | |
115 | 127 | let donateData = [DataEntry(DONATECOUNTERKEY, newGameNum), DataEntry(key2Balance, newBalance), DataEntry(donateId, donateNum)] | |
116 | 128 | WriteSet(donateData) | |
117 | 129 | } | |
118 | 130 | else throw("You paid the wrong amount") | |
119 | 131 | } | |
120 | 132 | } | |
121 | 133 | ||
122 | 134 | ||
123 | 135 | @Verifier(tx) | |
124 | 136 | func verify () = if (sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)) | |
125 | 137 | then match tx { | |
126 | 138 | case ttx: TransferTransaction => | |
127 | 139 | true | |
128 | 140 | case stx: SetScriptTransaction => | |
129 | 141 | true | |
130 | 142 | case _ => | |
131 | 143 | false | |
132 | 144 | } | |
133 | 145 | else false | |
134 | 146 |
github/deemru/w8io/169f3d6 79.55 ms ◑![]()