tx · DnFcsTBvMYcqykihzMRQZb4PaU5CW2tS1E3YLQ7nG9dG

3GpJhSPmFdaB1N3nbmGGnqUeC7jSK4iMbG1:  -0.07000000 Gic

2025.06.25 22:48 [6634] smart account 3GpJhSPmFdaB1N3nbmGGnqUeC7jSK4iMbG1 > SELF 0.00000000 Gic

{ "type": 13, "id": "DnFcsTBvMYcqykihzMRQZb4PaU5CW2tS1E3YLQ7nG9dG", "fee": 7000000, "feeAssetId": null, "timestamp": 1750880964047, "version": 2, "chainId": 71, "sender": "3GpJhSPmFdaB1N3nbmGGnqUeC7jSK4iMbG1", "senderPublicKey": "3QpT3yez6M99264RVoDPZZ3oZ7JSPeYduUQDJqdhG82y", "proofs": [ "4WTdt3Ay6qVfWHXArc3ikekQ2KY9o3fckUKfeqZsNdqbHy4Fjz7vYg1s31FXbS95whBo2vZKp69SFwSYYbdwkbEq" ], "script": "base64:AAIFAAAAAAAAAGoIAhIGCgQICAEEEgMKAQgSAwoBBBIGCgQEBAQEEgMKAQESBAoCCAQSBwoFCAgBAQESBAoCCAgSBAoCCAESBQoDCAgIEgQKAggIEgUKAwgIARIFCgMICAESBAoCCAgSBAoCCAgSBQoDCAgBAAAAJgAAAAAEbWFpbgIAAAAKbWFpbl9hc3NldAAAAAAMYWRtaW5BZGRyZXNzAgAAAA1hZG1pbl9hZGRyZXNzAAAAAAJscAIAAAAJbHBfYXNzZXRfAAAAAARwb29sAgAAAAZwb29sX18AAAAABGZhcm0CAAAABmZhcm1fXwAAAAAGc3Rha2VkAgAAAAdzdGFrZWRfAAAAAA1yZXdhcmRDbGFpbWVkAgAAAA9yZXdhcmRfY2xhaW1lZF8AAAAADWRlcG9zaXRIZWlnaHQCAAAAD2RlcG9zaXRfaGVpZ2h0XwAAAAAKZmFybUhlaWdodAIAAAAMZmFybV9oZWlnaHRfAAAAAAt0b3RhbExvY2tlZAIAAAANdG90YWxfbG9ja2VkXwAAAAAHc3dhcEZlZQIAAAAIc3dhcF9mZWUAAAAADmZhcm1SZXdhcmRQb29sAgAAABFmYXJtX3Jld2FyZF9wb29sXwAAAAAHZmFybUFwcgIAAAAJZmFybV9hcHJfAAAAAA5mYXJtTG9ja0Jsb2NrcwIAAAARZmFybV9sb2NrX2Jsb2Nrc18AAAAAD2Zhcm1Ub3RhbFN0YWtlZAIAAAASZmFybV90b3RhbF9zdGFrZWRfAAAAABFibGFja2xpc3RlZFRva2VucwIAAAAMYmxhY2tsaXN0ZWRfAAAAAA50b3RhbFVzZXJzRmFybQIAAAARdG90YWxfdXNlcnNfZmFybV8AAAAAAkQ4AAAAAAAF9eEAAAAAAA9jb250cmFjdEFkZHJlc3MJAQAAAAdBZGRyZXNzAAAAAQgFAAAABHRoaXMAAAAFYnl0ZXMAAAAAAmFjAgAAAAhhY3RpdmF0ZQAAAAAGcG9vbEFjAgAAAA5hY3RpdmF0ZV9wb29scwAAAAAGc3dhcEFjAgAAAA1hY3RpdmF0ZV9zd2FwAAAAAAtsaXF1aWRpdHlBYwIAAAASYWN0aXZhdGVfbGlxdWlkaXR5AAAAAAZmYXJtQWMCAAAADmFjdGl2YXRlX2Zhcm1zAQAAAAJrcAAAAAIAAAAGYXNz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IHRva2VuAwkAAAAAAAACCQEAAAAKdmFsaWRQb29sSwAAAAEFAAAAB3Bvb2xLZXkAAAAAAAAAAAAJAAACAAAAAQIAAAATUG9vbCBkb2VzIG5vdCBleGlzdAMJAABnAAAAAgAAAAAAAAAAAAUAAAAIbHBBbW91bnQJAAACAAAAAQIAAAAaTFAgYW1vdW50IG11c3QgYmUgcG9zaXRpdmUEAAAAC3Bvb2xBbW91bnQxCQEAAAAGcG9vbEFBAAAAAgUAAAAHcG9vbEtleQUAAAAGYXNzZXQxBAAAAAtwb29sQW1vdW50MgkBAAAABnBvb2xBQQAAAAIFAAAAB3Bvb2xLZXkFAAAABmFzc2V0MgQAAAANdG90YWxMcFN1cHBseQQAAAAHJG1hdGNoMAkAA+wAAAABCQACWQAAAAEFAAAACWxwQXNzZXRJZAMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAFQXNzZXQEAAAAAWEFAAAAByRtYXRjaDAIBQAAAAFhAAAACHF1YW50aXR5AAAAAAAAAAAABAAAAAdhbW91bnQxCQAAaQAAAAIJAABoAAAAAgUAAAAIbHBBbW91bnQFAAAAC3Bvb2xBbW91bnQxBQAAAA10b3RhbExwU3VwcGx5BAAAAAdhbW91bnQyCQAAaQAAAAIJAABoAAAAAgUAAAAIbHBBbW91bnQFAAAAC3Bvb2xBbW91bnQyBQAAAA10b3RhbExwU3VwcGx5BAAAAApjdXJyZW50S2V5CQACWAAAAAEICAUAAAABaQAAAAZjYWxsZXIAAAAFYnl0ZXMEAAAAE2N1cnJlbnRBbW91bnRBc3NldDEJAQAAAA11c2VyTGlxdWlkaXR5AAAAAwUAAAAKY3VycmVudEtleQUAAAAHcG9vbEtleQUAAAAGYXNzZXQxBAAAABNjdXJyZW50QW1vdW50QXNzZXQyCQEAAAANdXNlckxpcXVpZGl0eQAAAAMFAAAACmN1cnJlbnRLZXkFAAAAB3Bvb2xLZXkFAAAABmFzc2V0MgQAAAAPbmV3QW1vdW50QXNzZXQxCQAAZQAAAAIFAAAAE2N1cnJlbnRBbW91bnRBc3NldDEFAAAAB2Ftb3VudDEEAAAAD25ld0Ftb3VudEFzc2V0MgkAAGUAAAACBQAAABNjdXJyZW50QW1vdW50QXNzZXQyBQAAAAdhbW91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", "height": 6634, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 2Q4j5dJM71RdtMbder1nGbuT4dP8HhFr4sXhARVa9Xgb Next: 8TZ8LTcVYxQZQ3MyX4qqSysCv14oYSa2XAFmJJJpRM47 Diff:
OldNewDifferences
11 {-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4+let main = "main_asset"
5+
46 let adminAddress = "admin_address"
57
6-let fee = "fee"
8+let lp = "lp_asset_"
79
8-func getAdminAddress () = match getString(adminAddress) {
9- case t: String =>
10- t
11- case _ =>
12- toBase58String(this.bytes)
13-}
10+let pool = "pool__"
11+
12+let farm = "farm__"
13+
14+let staked = "staked_"
15+
16+let rewardClaimed = "reward_claimed_"
17+
18+let depositHeight = "deposit_height_"
19+
20+let farmHeight = "farm_height_"
21+
22+let totalLocked = "total_locked_"
23+
24+let swapFee = "swap_fee"
25+
26+let farmRewardPool = "farm_reward_pool_"
27+
28+let farmApr = "farm_apr_"
29+
30+let farmLockBlocks = "farm_lock_blocks_"
31+
32+let farmTotalStaked = "farm_total_staked_"
33+
34+let blacklistedTokens = "blacklisted_"
35+
36+let totalUsersFarm = "total_users_farm_"
37+
38+let D8 = 100000000
39+
40+let contractAddress = Address(this.bytes)
41+
42+let ac = "activate"
43+
44+let poolAc = "activate_pools"
45+
46+let swapAc = "activate_swap"
47+
48+let liquidityAc = "activate_liquidity"
49+
50+let farmAc = "activate_farms"
51+
52+func kp (asset1,asset2) = (((pool + asset1) + "_") + asset2)
53+
54+
55+func fk (asset1,asset2) = (((farm + asset1) + "_") + asset2)
56+
57+
58+func sk (asset1,asset2,address) = (((staked + kp(asset1, asset2)) + "_") + address)
59+
60+
61+func rk (asset1,asset2,address) = (((rewardClaimed + kp(asset1, asset2)) + "_") + address)
62+
63+
64+func dh (asset1,asset2,address) = (((depositHeight + kp(asset1, asset2)) + "_") + address)
65+
66+
67+func fh (asset1,asset2,address) = (((farmHeight + kp(asset1, asset2)) + "_") + address)
68+
69+
70+func isValidAsset (assetId) = if ((assetId == "GIC"))
71+ then true
72+ else match assetInfo(fromBase58String(assetId)) {
73+ case a: Asset =>
74+ true
75+ case _ =>
76+ false
77+ }
78+
79+
80+func isBlacklisted (assetId) = valueOrElse(getBoolean(this, (blacklistedTokens + assetId)), false)
1481
1582
1683 func getAssetDecimals (assetId) = if ((assetId == "GIC"))
2390 }
2491
2592
26-func getAssetquantity (assetId) = if ((assetId == "GIC"))
27- then 1000000000
28- else match assetInfo(fromBase58String(assetId)) {
29- case a: Asset =>
30- a.quantity
31- case _ =>
32- 1000000000
33- }
93+func getAddressIfValid (address) = toString(valueOrErrorMessage(addressFromString(address), (("Can't parse " + address) + " as address")))
3494
3595
36-func getIsReissuable (assetId) = if ((assetId == "GIC"))
37- then true
38- else match assetInfo(fromBase58String(assetId)) {
39- case a: Asset =>
40- a.reissuable
41- case _ =>
42- true
43- }
96+func validPoolK (key) = match getInteger(this, key) {
97+ case t: Int =>
98+ t
99+ case _ =>
100+ 0
101+}
102+
103+
104+func poolAA (key,asset) = match getInteger(this, ((key + "_") + asset)) {
105+ case t: Int =>
106+ t
107+ case _ =>
108+ 0
109+}
110+
111+
112+func userLiquidity (address,key,asset) = match getInteger(this, ((((key + "_") + address) + "_") + asset)) {
113+ case t: Int =>
114+ t
115+ case _ =>
116+ 0
117+}
118+
119+
120+func getAdminAddress () = if ((valueOrElse(getStringValue(adminAddress), "") == ""))
121+ then throw("Constructor has not been initialized yet!")
122+ else getStringValue(adminAddress)
123+
124+
125+@Callable(i)
126+func constructor (MainAssetId,AdminAddress,SwapFeeBps,acceptGic) = if ((i.caller != this))
127+ then throw("Only the contract itself can invoke this function")
128+ else if (!(isValidAsset(MainAssetId)))
129+ then throw("Invalid MainAssetId")
130+ else if ((AdminAddress == ""))
131+ then throw("Invalid AdminAddress")
132+ else if (if ((0 > SwapFeeBps))
133+ then true
134+ else (SwapFeeBps > 1000))
135+ then throw("SwapFeeBps must be between 0 and 1000")
136+ else [StringEntry(main, MainAssetId), StringEntry(adminAddress, AdminAddress), IntegerEntry(swapFee, SwapFeeBps), BooleanEntry(ac, true), BooleanEntry(poolAc, false), BooleanEntry(swapAc, false), BooleanEntry(liquidityAc, false), BooleanEntry(farmAc, false), BooleanEntry("accept_gic", acceptGic)]
137+
44138
45139
46140 @Callable(i)
54148
55149
56150 @Callable(i)
57-func setNFTFee (feeBps) = {
151+func activate (v) = {
152+ let admin = getAdminAddress()
153+ if ((toBase58String(i.caller.bytes) != admin))
154+ then throw("Only the Admin itself can invoke this function")
155+ else [BooleanEntry(ac, v)]
156+ }
157+
158+
159+
160+@Callable(i)
161+func maintenance (pools,swap,stake,farms) = {
162+ let admin = getAdminAddress()
163+ if ((toBase58String(i.caller.bytes) != admin))
164+ then throw("Only the Admin itself can invoke this function")
165+ else [BooleanEntry(poolAc, pools), BooleanEntry(swapAc, swap), BooleanEntry(liquidityAc, stake), BooleanEntry(farmAc, farms)]
166+ }
167+
168+
169+
170+@Callable(i)
171+func setSwapFee (feeBps) = {
58172 let admin = getAdminAddress()
59173 if ((toBase58String(i.caller.bytes) != admin))
60174 then throw("Only the Admin itself can invoke this function")
62176 then true
63177 else (feeBps > 1000))
64178 then throw("Fee must be between 0 and 1000 basis points")
65- else [IntegerEntry(fee, feeBps)]
179+ else [IntegerEntry(swapFee, feeBps)]
66180 }
67181
68182
69183
70184 @Callable(i)
71-func createNFT (name,ipfs) = if ((100000000 > i.fee))
72- then throw("Error: NFT fee amount to be paid is less than 1 GIC")
73- else if ((ipfs == ""))
74- then throw("Error: Put a valid IPFS string.")
75- else if ((name == ""))
76- then throw("Error: Put a valid NFT name string.")
77- else {
78- let nftCreate = Issue(name, ipfs, 1, 0, false)
79- let idNft = calculateAssetId(nftCreate)
80-[nftCreate, ScriptTransfer(i.caller, 1, idNft)]
81- }
185+func blacklistToken (assetId,blacklist) = {
186+ let admin = getAdminAddress()
187+ if ((toBase58String(i.caller.bytes) != admin))
188+ then throw("Only the Admin itself can invoke this function")
189+ else if (if (!(isValidAsset(assetId)))
190+ then (assetId != "GIC")
191+ else false)
192+ then throw("Invalid assetId")
193+ else [BooleanEntry((blacklistedTokens + assetId), blacklist)]
194+ }
82195
83196
84197
85198 @Callable(i)
86-func sellNFT (ValueToSell) = if ((100 > ValueToSell))
87- then throw("Error: set a minimum value (0.00000100) to sell your NFT")
88- else {
89- let nft = i.payments[0]
90- let pmtAssetId = match nft.assetId {
91- case a: ByteVector =>
92- toBase58String(a)
93- case _ =>
94- "GIC"
95- }
96- let isRessuable = getIsReissuable(pmtAssetId)
97- let totalQuantity = getAssetquantity(pmtAssetId)
98- let totalDecimals = getAssetDecimals(pmtAssetId)
99- let OldNumberH = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
100- let numberH = (OldNumberH + 1)
101- let OldNumberHUser = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
102- let numberHUser = (OldNumberH + 1)
103- if ((pmtAssetId == "GIC"))
104- then throw("Error: Invalid NFT.")
105- else if (if (if (!(isRessuable))
106- then (totalQuantity == 1)
107- else false)
108- then (totalDecimals == 0)
109- else false)
110- then [BooleanEntry(("is_sale_" + pmtAssetId), true), IntegerEntry(("price_sale_" + pmtAssetId), ValueToSell), StringEntry(("coin_accepted_" + pmtAssetId), "GIC"), StringEntry(("saler_" + pmtAssetId), toBase58String(i.caller.bytes)), IntegerEntry(("history_" + toBase58String(i.caller.bytes)), numberHUser), StringEntry(((("history_" + toBase58String(i.caller.bytes)) + "_") + toString(numberHUser)), pmtAssetId), IntegerEntry(("history_" + pmtAssetId), numberH), StringEntry((("history_" + pmtAssetId) + toString(numberH)), "sell")]
111- else throw("Error: this is an asset, not an NFT.")
112- }
199+func createFarm (asset1,asset2,apr,lockBlocks,rewardAmount) = {
200+ let admin = getAdminAddress()
201+ if ((toBase58String(i.caller.bytes) != admin))
202+ then throw("Only the Admin itself can invoke this function")
203+ else if (!(isValidAsset(asset1)))
204+ then throw("Invalid asset1")
205+ else if (!(isValidAsset(asset2)))
206+ then throw("Invalid asset2")
207+ else if ((asset1 == asset2))
208+ then throw("Asset1 cannot be equal to Asset2")
209+ else if ((0 >= apr))
210+ then throw("APR must be positive")
211+ else if ((0 >= lockBlocks))
212+ then throw("Lock blocks must be positive")
213+ else if ((0 >= rewardAmount))
214+ then throw("Reward amount must be positive")
215+ else {
216+ let poolKey = kp(asset1, asset2)
217+ if ((validPoolK(poolKey) == 0))
218+ then throw("Pool does not exist")
219+ else {
220+ let farmKey = fk(asset1, asset2)
221+[IntegerEntry(farmKey, 1), IntegerEntry((farmApr + poolKey), apr), IntegerEntry((farmLockBlocks + poolKey), lockBlocks), IntegerEntry((farmRewardPool + poolKey), rewardAmount), IntegerEntry((farmTotalStaked + poolKey), 0), IntegerEntry((totalUsersFarm + poolKey), 0)]
222+ }
223+ }
224+ }
113225
114226
115227
116228 @Callable(i)
117-func buyNFT (nftId) = {
118- let pay = i.payments[0]
119- let pmtAssetId = match pay.assetId {
229+func fundFarm (asset1,asset2) = {
230+ let admin = getAdminAddress()
231+ if ((toBase58String(i.caller.bytes) != admin))
232+ then throw("Only the Admin itself can invoke this function")
233+ else {
234+ let pmt = i.payments[0]
235+ let pmtAssetId = match pmt.assetId {
236+ case a: ByteVector =>
237+ toBase58String(a)
238+ case _ =>
239+ "GIC"
240+ }
241+ if ((toBase58String(i.caller.bytes) != admin))
242+ then throw("Only the Admin itself can invoke this function")
243+ else if ((pmtAssetId != "GIC"))
244+ then throw("Rewards must be in GIC")
245+ else {
246+ let poolKey = kp(asset1, asset2)
247+ let farmKey = fk(asset1, asset2)
248+ if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
249+ then throw("Farm does not exist")
250+ else {
251+ let currentRewardPool = valueOrElse(getInteger(this, (farmRewardPool + poolKey)), 0)
252+ let newRewardPool = (currentRewardPool + pmt.amount)
253+[IntegerEntry((farmRewardPool + poolKey), newRewardPool)]
254+ }
255+ }
256+ }
257+ }
258+
259+
260+
261+@Callable(i)
262+func changeData (key,data) = {
263+ let admin = getAdminAddress()
264+ if ((toBase58String(i.caller.bytes) != admin))
265+ then throw("Only the Admin itself can invoke this function")
266+ else [IntegerEntry(key, data)]
267+ }
268+
269+
270+
271+@Callable(i)
272+func createPool (asset1,asset2,nameLp) = {
273+ let poolKey = kp(asset1, asset2)
274+ let admin = getAdminAddress()
275+ if ((toBase58String(i.caller.bytes) != admin))
276+ then throw("Only the Admin itself can invoke this function")
277+ else if (!(isValidAsset(asset1)))
278+ then throw("Invalid asset1")
279+ else if (!(isValidAsset(asset2)))
280+ then throw("Invalid asset2")
281+ else if ((asset1 == asset2))
282+ then throw("Asset1 cannot be equal to Asset2")
283+ else if (if (isBlacklisted(asset1))
284+ then true
285+ else isBlacklisted(asset2))
286+ then throw("One or both assets are blacklisted")
287+ else if ((validPoolK(poolKey) != 0))
288+ then throw("Pool already exists")
289+ else {
290+ let assetLpCreate = Issue(nameLp, ((("LP asset for pool " + asset1) + "_") + asset2), 1, 8, true)
291+ let idAssetLp = calculateAssetId(assetLpCreate)
292+[assetLpCreate, StringEntry((lp + poolKey), toBase58String(idAssetLp)), IntegerEntry(poolKey, 1)]
293+ }
294+ }
295+
296+
297+
298+@Callable(i)
299+func addLiquidity (asset1,asset2) = {
300+ let power = getBooleanValue(this, ac)
301+ let liquidityPower = getBooleanValue(this, liquidityAc)
302+ let acGic = getBooleanValue(this, "accept_gic")
303+ let poolKey = kp(asset1, asset2)
304+ let pmt1 = i.payments[0]
305+ let pmt2 = i.payments[1]
306+ let asset1Id = match pmt1.assetId {
120307 case a: ByteVector =>
121308 toBase58String(a)
122309 case _ =>
123310 "GIC"
124311 }
125- let feeBps = toBigInt(getIntegerValue(this, fee))
126- let isSale = valueOrElse(getBooleanValue(("is_sale_" + nftId)), false)
127- let valueSale = valueOrElse(getIntegerValue(("price_sale_" + nftId)), 0)
128- let feeToSend = ((toBigInt(pay.amount) * feeBps) / toBigInt(10000))
129- let adminAddr = getAdminAddress()
130- let OldNumberH = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
131- let numberH = (OldNumberH + 1)
132- let OldNumberHUser = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
133- let numberHUser = (OldNumberH + 1)
134- let toSendToSaler = (toBigInt(pay.amount) - feeToSend)
135- if ((pmtAssetId != "GIC"))
136- then throw("Error: Invalid Asset to buy.")
137- else if (!(isSale))
138- then throw("This NFT is no longer for sale.")
139- else if ((valueSale != pay.amount))
140- then throw("Insufficient value to make the purchase.")
141- else [ScriptTransfer(i.caller, 1, fromBase58String(nftId)), ScriptTransfer(addressFromStringValue(getStringValue(this, ("saler_" + pmtAssetId))), toInt(toSendToSaler), unit), ScriptTransfer(addressFromStringValue(adminAddr), toInt(feeToSend), unit), BooleanEntry(("is_sale_" + pmtAssetId), false), StringEntry(("saler_" + pmtAssetId), toBase58String(i.caller.bytes)), IntegerEntry(("history_" + pmtAssetId), numberH), StringEntry((("history_" + pmtAssetId) + toString(numberH)), "buy")]
312+ let asset2Id = match pmt2.assetId {
313+ case a: ByteVector =>
314+ toBase58String(a)
315+ case _ =>
316+ "GIC"
317+ }
318+ let decimals1 = getAssetDecimals(asset1Id)
319+ let decimals2 = getAssetDecimals(asset2Id)
320+ if (!(power))
321+ then throw("dApp is currently under maintenance")
322+ else if (liquidityPower)
323+ then throw("Liquidity is currently under maintenance")
324+ else if (if ((asset1Id != asset1))
325+ then true
326+ else (asset2Id != asset2))
327+ then throw("Payment assets do not match specified assets")
328+ else if ((asset1 == asset2))
329+ then throw("Asset1 cannot be equal to Asset2")
330+ else if (!(isValidAsset(asset1)))
331+ then throw("Invalid asset1")
332+ else if (!(isValidAsset(asset2)))
333+ then throw("Invalid asset2")
334+ else if (if (isBlacklisted(asset1))
335+ then true
336+ else isBlacklisted(asset2))
337+ then throw("One or both assets are blacklisted")
338+ else if (if (if ((asset1 == "GIC"))
339+ then true
340+ else (asset2 == "GIC"))
341+ then !(acGic)
342+ else false)
343+ then throw("GIC not allowed")
344+ else if ((validPoolK(poolKey) == 0))
345+ then throw("Pool does not exist. Create it first")
346+ else {
347+ let poolAmount1 = poolAA(poolKey, asset1)
348+ let poolAmount2 = poolAA(poolKey, asset2)
349+ let amount1 = pmt1.amount
350+ let amount2 = pmt2.amount
351+ let lpAmount = if (if ((poolAmount1 == 0))
352+ then true
353+ else (poolAmount2 == 0))
354+ then toBigInt(amount1)
355+ else ((toBigInt(amount1) * toBigInt(poolAmount2)) / toBigInt(poolAmount1))
356+ let currentKey = toBase58String(i.caller.bytes)
357+ let currentAmountAsset1 = userLiquidity(currentKey, poolKey, asset1)
358+ let currentAmountAsset2 = userLiquidity(currentKey, poolKey, asset2)
359+ let newAmountAsset1 = (toBigInt(currentAmountAsset1) + toBigInt(amount1))
360+ let newAmountAsset2 = (toBigInt(currentAmountAsset2) + toBigInt(amount2))
361+ let newPoolAmount1 = (toBigInt(poolAmount1) + toBigInt(amount1))
362+ let newPoolAmount2 = (toBigInt(poolAmount2) + toBigInt(amount2))
363+ let lpAssetId = getStringValue(this, (lp + poolKey))
364+ let oldAssetLpTotal = valueOrElse(getIntegerValue(poolKey), 0)
365+ let lpAsset = if ((lpAssetId == ""))
366+ then throw("Pool does not have a liquidity pair, contact admin")
367+ else fromBase58String(lpAssetId)
368+[IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset1), toInt(newAmountAsset1)), IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset2), toInt(newAmountAsset2)), IntegerEntry(((poolKey + "_") + asset1), toInt(newPoolAmount1)), IntegerEntry(((poolKey + "_") + asset2), toInt(newPoolAmount2)), IntegerEntry((((depositHeight + poolKey) + "_") + currentKey), height), IntegerEntry(poolKey, (oldAssetLpTotal + toInt(lpAmount))), Reissue(lpAsset, toInt(lpAmount), true), ScriptTransfer(i.caller, toInt(lpAmount), lpAsset)]
369+ }
370+ }
371+
372+
373+
374+@Callable(i)
375+func removeLiquidity (asset1,asset2,lpAmount) = {
376+ let power = getBooleanValue(this, ac)
377+ let liquidityPower = getBooleanValue(this, liquidityAc)
378+ let poolKey = kp(asset1, asset2)
379+ let lpAssetId = getStringValue(this, (lp + poolKey))
380+ let lpAsset = fromBase58String(lpAssetId)
381+ let pmt = i.payments[0]
382+ let pmtAssetId = match pmt.assetId {
383+ case a: ByteVector =>
384+ toBase58String(a)
385+ case _ =>
386+ "GIC"
387+ }
388+ let decimals1 = getAssetDecimals(asset1)
389+ let decimals2 = getAssetDecimals(asset2)
390+ if (!(power))
391+ then throw("dApp is currently under maintenance")
392+ else if (liquidityPower)
393+ then throw("Liquidity is currently under maintenance")
394+ else if ((lpAssetId == ""))
395+ then throw("Invalid LP asset")
396+ else if ((pmtAssetId != lpAssetId))
397+ then throw("Invalid LP token")
398+ else if ((validPoolK(poolKey) == 0))
399+ then throw("Pool does not exist")
400+ else if ((0 >= lpAmount))
401+ then throw("LP amount must be positive")
402+ else {
403+ let poolAmount1 = poolAA(poolKey, asset1)
404+ let poolAmount2 = poolAA(poolKey, asset2)
405+ let totalLpSupply = match assetInfo(fromBase58String(lpAssetId)) {
406+ case a: Asset =>
407+ a.quantity
408+ case _ =>
409+ 0
410+ }
411+ let amount1 = ((lpAmount * poolAmount1) / totalLpSupply)
412+ let amount2 = ((lpAmount * poolAmount2) / totalLpSupply)
413+ let currentKey = toBase58String(i.caller.bytes)
414+ let currentAmountAsset1 = userLiquidity(currentKey, poolKey, asset1)
415+ let currentAmountAsset2 = userLiquidity(currentKey, poolKey, asset2)
416+ let newAmountAsset1 = (currentAmountAsset1 - amount1)
417+ let newAmountAsset2 = (currentAmountAsset2 - amount2)
418+ if (if ((0 > newAmountAsset1))
419+ then true
420+ else (0 > newAmountAsset2))
421+ then throw("Insufficient liquidity balance")
422+ else {
423+ let newPoolAmount1 = (poolAmount1 - amount1)
424+ let newPoolAmount2 = (poolAmount2 - amount2)
425+ let asset1Bytes = if ((asset1 == "GIC"))
426+ then unit
427+ else fromBase58String(asset1)
428+ let asset2Bytes = if ((asset2 == "GIC"))
429+ then unit
430+ else fromBase58String(asset2)
431+ let oldAssetLpTotal = getIntegerValue(poolKey)
432+[IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset1), newAmountAsset1), IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset2), newAmountAsset2), IntegerEntry(((poolKey + "_") + asset1), newPoolAmount1), IntegerEntry(((poolKey + "_") + asset2), newPoolAmount2), IntegerEntry(poolKey, (oldAssetLpTotal - lpAmount)), Burn(lpAsset, lpAmount), ScriptTransfer(i.caller, amount1, asset1Bytes), ScriptTransfer(i.caller, amount2, asset2Bytes)]
433+ }
434+ }
435+ }
436+
437+
438+
439+@Callable(i)
440+func swap (assetIn,assetOut,maxSlippage) = {
441+ let power = getBooleanValue(this, ac)
442+ let swapActivate = getBooleanValue(this, swapAc)
443+ let acGic = getBooleanValue(this, "accept_gic")
444+ let poolKey = kp(assetIn, assetOut)
445+ let reversePoolKey = kp(assetOut, assetIn)
446+ let key = if ((validPoolK(poolKey) != 0))
447+ then poolKey
448+ else if ((validPoolK(reversePoolKey) != 0))
449+ then reversePoolKey
450+ else ""
451+ let pmt = i.payments[0]
452+ let pmtAssetId = match pmt.assetId {
453+ case a: ByteVector =>
454+ toBase58String(a)
455+ case _ =>
456+ "GIC"
457+ }
458+ let decimalsIn = getAssetDecimals(assetIn)
459+ let decimalsOut = getAssetDecimals(assetOut)
460+ let feeBps = toBigInt(getIntegerValue(this, swapFee))
461+ let adminAddr = addressFromStringValue(getStringValue(this, adminAddress))
462+ if (!(power))
463+ then throw("dApp is currently under maintenance")
464+ else if (swapActivate)
465+ then throw("Swap is currently under maintenance")
466+ else if ((pmtAssetId != assetIn))
467+ then throw("Payment asset does not match assetIn")
468+ else if (!(isValidAsset(assetIn)))
469+ then throw("Invalid assetIn")
470+ else if (!(isValidAsset(assetOut)))
471+ then throw("Invalid assetOut")
472+ else if (if (isBlacklisted(assetIn))
473+ then true
474+ else isBlacklisted(assetOut))
475+ then throw("One or both assets are blacklisted")
476+ else if ((assetIn == assetOut))
477+ then throw("assetIn cannot be equal to assetOut")
478+ else if (if (if ((assetIn == "GIC"))
479+ then true
480+ else (assetOut == "GIC"))
481+ then !(acGic)
482+ else false)
483+ then throw("GIC not allowed")
484+ else if ((key == ""))
485+ then throw("Pool does not exist")
486+ else {
487+ let isReverse = (key == reversePoolKey)
488+ let poolAsset1 = if (isReverse)
489+ then assetOut
490+ else assetIn
491+ let poolAsset2 = if (isReverse)
492+ then assetIn
493+ else assetOut
494+ let poolAmount1 = toBigInt(poolAA(key, poolAsset1))
495+ let poolAmount2 = toBigInt(poolAA(key, poolAsset2))
496+ let amountInAdjusted = toBigInt(pmt.amount)
497+ let amountOut = if (isReverse)
498+ then ((amountInAdjusted * poolAmount1) / poolAmount2)
499+ else ((amountInAdjusted * poolAmount2) / poolAmount1)
500+ let fee = ((amountOut * feeBps) / toBigInt(10000))
501+ let amountOutAfterFee = (amountOut - fee)
502+ let feeToSend = ((amountInAdjusted * feeBps) / toBigInt(10000))
503+ let minAmountOut = ((amountOutAfterFee * toBigInt((10000 - maxSlippage))) / toBigInt(10000))
504+ if ((minAmountOut > amountOutAfterFee))
505+ then throw("Slippage exceeded")
506+ else {
507+ let newPoolAmount1 = if (isReverse)
508+ then (poolAmount1 - amountOut)
509+ else (poolAmount1 + amountInAdjusted)
510+ let newPoolAmount2 = if (isReverse)
511+ then (poolAmount2 + amountInAdjusted)
512+ else (poolAmount2 - amountOut)
513+ let assettoReceive = if ((assetOut == "GIC"))
514+ then unit
515+ else fromBase58String(assetOut)
516+ let assetOutBytes = if ((assetIn == "GIC"))
517+ then unit
518+ else fromBase58String(assetIn)
519+[IntegerEntry(((key + "_") + poolAsset1), toInt(newPoolAmount1)), IntegerEntry(((key + "_") + poolAsset2), toInt(newPoolAmount2)), BooleanEntry("isReverse", isReverse), ScriptTransfer(i.caller, toInt(amountOutAfterFee), assettoReceive), ScriptTransfer(adminAddr, toInt(feeToSend), assetOutBytes)]
520+ }
521+ }
522+ }
523+
524+
525+
526+@Callable(i)
527+func farmsAddLP (asset1,asset2) = {
528+ let power = getBooleanValue(this, ac)
529+ let farmPower = getBooleanValue(this, farmAc)
530+ let poolKey = kp(asset1, asset2)
531+ let farmKey = fk(asset1, asset2)
532+ let pmt = i.payments[0]
533+ let lpAssetId = getStringValue(this, (lp + poolKey))
534+ let pmtAssetId = match pmt.assetId {
535+ case a: ByteVector =>
536+ toBase58String(a)
537+ case _ =>
538+ "GIC"
539+ }
540+ if (!(power))
541+ then throw("dApp is currently under maintenance")
542+ else if (farmPower)
543+ then throw("Farms are currently under maintenance")
544+ else if ((pmtAssetId != lpAssetId))
545+ then throw("Invalid LP token")
546+ else if ((validPoolK(poolKey) == 0))
547+ then throw("Pool does not exist")
548+ else if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
549+ then throw("Farm does not exist")
550+ else {
551+ let currentKey = toBase58String(i.caller.bytes)
552+ let stakedKey = sk(asset1, asset2, currentKey)
553+ let currentStaked = valueOrElse(getInteger(this, stakedKey), 0)
554+ let newStaked = (currentStaked + pmt.amount)
555+ let totalStakedKey = (farmTotalStaked + poolKey)
556+ let previousTotalStaked = valueOrElse(getInteger(this, totalStakedKey), 0)
557+ let newTotalStaked = (previousTotalStaked + pmt.amount)
558+ let totalUsers = valueOrElse(getInteger(this, (totalUsersFarm + poolKey)), 0)
559+ let newTotalUsers = if ((currentStaked == 0))
560+ then (totalUsers + 1)
561+ else totalUsers
562+[IntegerEntry(stakedKey, newStaked), IntegerEntry(totalStakedKey, newTotalStaked), IntegerEntry((totalUsersFarm + poolKey), newTotalUsers), IntegerEntry(fh(asset1, asset2, currentKey), height)]
563+ }
564+ }
565+
566+
567+
568+@Callable(i)
569+func farmsClaimLP (asset1,asset2) = {
570+ let power = getBooleanValue(this, ac)
571+ let farmPower = getBooleanValue(this, farmAc)
572+ let poolKey = kp(asset1, asset2)
573+ let farmKey = fk(asset1, asset2)
574+ let currentKey = toBase58String(i.caller.bytes)
575+ let stakedKey = sk(asset1, asset2, currentKey)
576+ let heightKey = fh(asset1, asset2, currentKey)
577+ let rewardKey = rk(asset1, asset2, currentKey)
578+ if (!(power))
579+ then throw("dApp is currently under maintenance")
580+ else if (farmPower)
581+ then throw("Farms are currently under maintenance")
582+ else if ((validPoolK(poolKey) == 0))
583+ then throw("Pool does not exist")
584+ else if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
585+ then throw("Farm does not exist")
586+ else {
587+ let currentStaked = valueOrElse(getInteger(this, stakedKey), 0)
588+ if ((0 >= currentStaked))
589+ then throw("No staked LP tokens")
590+ else {
591+ let rewardPool = valueOrElse(getInteger(this, (farmRewardPool + poolKey)), 0)
592+ let totalUsers = valueOrElse(getInteger(this, (totalUsersFarm + poolKey)), 1)
593+ let lockBlocks = getIntegerValue(this, (farmLockBlocks + poolKey))
594+ let apr = getIntegerValue(this, (farmApr + poolKey))
595+ let lastClaimHeight = valueOrElse(getInteger(this, heightKey), height)
596+ let blocksSinceLastClaim = (height - lastClaimHeight)
597+ if ((1 > blocksSinceLastClaim))
598+ then throw("Must wait at least 1 block since last claim")
599+ else {
600+ let rewardPerBlock = ((rewardPool * apr) / ((10000 * lockBlocks) * totalUsers))
601+ let reward = (((rewardPerBlock * blocksSinceLastClaim) * currentStaked) / D8)
602+ if ((reward > rewardPool))
603+ then throw("Insufficient reward pool")
604+ else {
605+ let newRewardPool = (rewardPool - reward)
606+[IntegerEntry(rewardKey, 0), IntegerEntry((farmRewardPool + poolKey), newRewardPool), IntegerEntry(heightKey, height), ScriptTransfer(i.caller, reward, unit)]
607+ }
608+ }
609+ }
610+ }
611+ }
612+
613+
614+
615+@Callable(i)
616+func farmsRemoveLP (asset1,asset2,amount) = {
617+ let power = getBooleanValue(this, ac)
618+ let farmPower = getBooleanValue(this, farmAc)
619+ let poolKey = kp(asset1, asset2)
620+ let farmKey = fk(asset1, asset2)
621+ let lpAssetId = getStringValue(this, (lp + poolKey))
622+ let lpAsset = if ((lpAssetId == ""))
623+ then unit
624+ else fromBase58String(lpAssetId)
625+ let currentKey = toBase58String(i.caller.bytes)
626+ let stakedKey = sk(asset1, asset2, currentKey)
627+ let heightKey = fh(asset1, asset2, currentKey)
628+ let rewardKey = rk(asset1, asset2, currentKey)
629+ if (!(power))
630+ then throw("dApp is currently under maintenance")
631+ else if (farmPower)
632+ then throw("Farms are currently under maintenance")
633+ else if ((validPoolK(poolKey) == 0))
634+ then throw("Pool does not exist")
635+ else if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
636+ then throw("Farm does not exist")
637+ else {
638+ let currentStaked = valueOrElse(getInteger(this, stakedKey), 0)
639+ if ((0 >= amount))
640+ then throw("Amount must be positive")
641+ else if ((amount > currentStaked))
642+ then throw("Insufficient staked LP tokens")
643+ else {
644+ let rewardPool = valueOrElse(getInteger(this, (farmRewardPool + poolKey)), 0)
645+ let totalUsers = valueOrElse(getInteger(this, (totalUsersFarm + poolKey)), 1)
646+ let lockBlocks = getIntegerValue(this, (farmLockBlocks + poolKey))
647+ let apr = getIntegerValue(this, (farmApr + poolKey))
648+ let lastClaimHeight = valueOrElse(getInteger(this, heightKey), height)
649+ let blocksSinceLastClaim = (height - lastClaimHeight)
650+ let rewardPerBlock = ((rewardPool * apr) / ((10000 * lockBlocks) * totalUsers))
651+ let reward = if ((blocksSinceLastClaim >= 1))
652+ then (((rewardPerBlock * blocksSinceLastClaim) * currentStaked) / D8)
653+ else 0
654+ let newRewardPool = (rewardPool - reward)
655+ let newStaked = (currentStaked - amount)
656+ let totalStakedKey = (farmTotalStaked + poolKey)
657+ let previousTotalStaked = valueOrElse(getInteger(this, totalStakedKey), 0)
658+ let newTotalStaked = (previousTotalStaked - amount)
659+ let newTotalUsers = if ((newStaked == 0))
660+ then (totalUsers - 1)
661+ else totalUsers
662+[IntegerEntry(stakedKey, newStaked), IntegerEntry(totalStakedKey, newTotalStaked), IntegerEntry((totalUsersFarm + poolKey), newTotalUsers), IntegerEntry(rewardKey, 0), IntegerEntry((farmRewardPool + poolKey), newRewardPool), IntegerEntry(heightKey, height), ScriptTransfer(i.caller, amount, lpAsset), ScriptTransfer(i.caller, reward, unit)]
663+ }
664+ }
142665 }
143666
144667
145668 @Verifier(tx)
146-func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
669+func verify () = match tx {
670+ case _ =>
671+ sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
672+}
147673
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4+let main = "main_asset"
5+
46 let adminAddress = "admin_address"
57
6-let fee = "fee"
8+let lp = "lp_asset_"
79
8-func getAdminAddress () = match getString(adminAddress) {
9- case t: String =>
10- t
11- case _ =>
12- toBase58String(this.bytes)
13-}
10+let pool = "pool__"
11+
12+let farm = "farm__"
13+
14+let staked = "staked_"
15+
16+let rewardClaimed = "reward_claimed_"
17+
18+let depositHeight = "deposit_height_"
19+
20+let farmHeight = "farm_height_"
21+
22+let totalLocked = "total_locked_"
23+
24+let swapFee = "swap_fee"
25+
26+let farmRewardPool = "farm_reward_pool_"
27+
28+let farmApr = "farm_apr_"
29+
30+let farmLockBlocks = "farm_lock_blocks_"
31+
32+let farmTotalStaked = "farm_total_staked_"
33+
34+let blacklistedTokens = "blacklisted_"
35+
36+let totalUsersFarm = "total_users_farm_"
37+
38+let D8 = 100000000
39+
40+let contractAddress = Address(this.bytes)
41+
42+let ac = "activate"
43+
44+let poolAc = "activate_pools"
45+
46+let swapAc = "activate_swap"
47+
48+let liquidityAc = "activate_liquidity"
49+
50+let farmAc = "activate_farms"
51+
52+func kp (asset1,asset2) = (((pool + asset1) + "_") + asset2)
53+
54+
55+func fk (asset1,asset2) = (((farm + asset1) + "_") + asset2)
56+
57+
58+func sk (asset1,asset2,address) = (((staked + kp(asset1, asset2)) + "_") + address)
59+
60+
61+func rk (asset1,asset2,address) = (((rewardClaimed + kp(asset1, asset2)) + "_") + address)
62+
63+
64+func dh (asset1,asset2,address) = (((depositHeight + kp(asset1, asset2)) + "_") + address)
65+
66+
67+func fh (asset1,asset2,address) = (((farmHeight + kp(asset1, asset2)) + "_") + address)
68+
69+
70+func isValidAsset (assetId) = if ((assetId == "GIC"))
71+ then true
72+ else match assetInfo(fromBase58String(assetId)) {
73+ case a: Asset =>
74+ true
75+ case _ =>
76+ false
77+ }
78+
79+
80+func isBlacklisted (assetId) = valueOrElse(getBoolean(this, (blacklistedTokens + assetId)), false)
1481
1582
1683 func getAssetDecimals (assetId) = if ((assetId == "GIC"))
1784 then 8
1885 else match assetInfo(fromBase58String(assetId)) {
1986 case a: Asset =>
2087 a.decimals
2188 case _ =>
2289 8
2390 }
2491
2592
26-func getAssetquantity (assetId) = if ((assetId == "GIC"))
27- then 1000000000
28- else match assetInfo(fromBase58String(assetId)) {
29- case a: Asset =>
30- a.quantity
31- case _ =>
32- 1000000000
33- }
93+func getAddressIfValid (address) = toString(valueOrErrorMessage(addressFromString(address), (("Can't parse " + address) + " as address")))
3494
3595
36-func getIsReissuable (assetId) = if ((assetId == "GIC"))
37- then true
38- else match assetInfo(fromBase58String(assetId)) {
39- case a: Asset =>
40- a.reissuable
41- case _ =>
42- true
43- }
96+func validPoolK (key) = match getInteger(this, key) {
97+ case t: Int =>
98+ t
99+ case _ =>
100+ 0
101+}
102+
103+
104+func poolAA (key,asset) = match getInteger(this, ((key + "_") + asset)) {
105+ case t: Int =>
106+ t
107+ case _ =>
108+ 0
109+}
110+
111+
112+func userLiquidity (address,key,asset) = match getInteger(this, ((((key + "_") + address) + "_") + asset)) {
113+ case t: Int =>
114+ t
115+ case _ =>
116+ 0
117+}
118+
119+
120+func getAdminAddress () = if ((valueOrElse(getStringValue(adminAddress), "") == ""))
121+ then throw("Constructor has not been initialized yet!")
122+ else getStringValue(adminAddress)
123+
124+
125+@Callable(i)
126+func constructor (MainAssetId,AdminAddress,SwapFeeBps,acceptGic) = if ((i.caller != this))
127+ then throw("Only the contract itself can invoke this function")
128+ else if (!(isValidAsset(MainAssetId)))
129+ then throw("Invalid MainAssetId")
130+ else if ((AdminAddress == ""))
131+ then throw("Invalid AdminAddress")
132+ else if (if ((0 > SwapFeeBps))
133+ then true
134+ else (SwapFeeBps > 1000))
135+ then throw("SwapFeeBps must be between 0 and 1000")
136+ else [StringEntry(main, MainAssetId), StringEntry(adminAddress, AdminAddress), IntegerEntry(swapFee, SwapFeeBps), BooleanEntry(ac, true), BooleanEntry(poolAc, false), BooleanEntry(swapAc, false), BooleanEntry(liquidityAc, false), BooleanEntry(farmAc, false), BooleanEntry("accept_gic", acceptGic)]
137+
44138
45139
46140 @Callable(i)
47141 func changeAdmin (address) = {
48142 let admin = getAdminAddress()
49143 if ((toBase58String(i.caller.bytes) != admin))
50144 then throw("Only the Admin itself can invoke this function.")
51145 else [StringEntry(adminAddress, address)]
52146 }
53147
54148
55149
56150 @Callable(i)
57-func setNFTFee (feeBps) = {
151+func activate (v) = {
152+ let admin = getAdminAddress()
153+ if ((toBase58String(i.caller.bytes) != admin))
154+ then throw("Only the Admin itself can invoke this function")
155+ else [BooleanEntry(ac, v)]
156+ }
157+
158+
159+
160+@Callable(i)
161+func maintenance (pools,swap,stake,farms) = {
162+ let admin = getAdminAddress()
163+ if ((toBase58String(i.caller.bytes) != admin))
164+ then throw("Only the Admin itself can invoke this function")
165+ else [BooleanEntry(poolAc, pools), BooleanEntry(swapAc, swap), BooleanEntry(liquidityAc, stake), BooleanEntry(farmAc, farms)]
166+ }
167+
168+
169+
170+@Callable(i)
171+func setSwapFee (feeBps) = {
58172 let admin = getAdminAddress()
59173 if ((toBase58String(i.caller.bytes) != admin))
60174 then throw("Only the Admin itself can invoke this function")
61175 else if (if ((0 > feeBps))
62176 then true
63177 else (feeBps > 1000))
64178 then throw("Fee must be between 0 and 1000 basis points")
65- else [IntegerEntry(fee, feeBps)]
179+ else [IntegerEntry(swapFee, feeBps)]
66180 }
67181
68182
69183
70184 @Callable(i)
71-func createNFT (name,ipfs) = if ((100000000 > i.fee))
72- then throw("Error: NFT fee amount to be paid is less than 1 GIC")
73- else if ((ipfs == ""))
74- then throw("Error: Put a valid IPFS string.")
75- else if ((name == ""))
76- then throw("Error: Put a valid NFT name string.")
77- else {
78- let nftCreate = Issue(name, ipfs, 1, 0, false)
79- let idNft = calculateAssetId(nftCreate)
80-[nftCreate, ScriptTransfer(i.caller, 1, idNft)]
81- }
185+func blacklistToken (assetId,blacklist) = {
186+ let admin = getAdminAddress()
187+ if ((toBase58String(i.caller.bytes) != admin))
188+ then throw("Only the Admin itself can invoke this function")
189+ else if (if (!(isValidAsset(assetId)))
190+ then (assetId != "GIC")
191+ else false)
192+ then throw("Invalid assetId")
193+ else [BooleanEntry((blacklistedTokens + assetId), blacklist)]
194+ }
82195
83196
84197
85198 @Callable(i)
86-func sellNFT (ValueToSell) = if ((100 > ValueToSell))
87- then throw("Error: set a minimum value (0.00000100) to sell your NFT")
88- else {
89- let nft = i.payments[0]
90- let pmtAssetId = match nft.assetId {
91- case a: ByteVector =>
92- toBase58String(a)
93- case _ =>
94- "GIC"
95- }
96- let isRessuable = getIsReissuable(pmtAssetId)
97- let totalQuantity = getAssetquantity(pmtAssetId)
98- let totalDecimals = getAssetDecimals(pmtAssetId)
99- let OldNumberH = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
100- let numberH = (OldNumberH + 1)
101- let OldNumberHUser = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
102- let numberHUser = (OldNumberH + 1)
103- if ((pmtAssetId == "GIC"))
104- then throw("Error: Invalid NFT.")
105- else if (if (if (!(isRessuable))
106- then (totalQuantity == 1)
107- else false)
108- then (totalDecimals == 0)
109- else false)
110- then [BooleanEntry(("is_sale_" + pmtAssetId), true), IntegerEntry(("price_sale_" + pmtAssetId), ValueToSell), StringEntry(("coin_accepted_" + pmtAssetId), "GIC"), StringEntry(("saler_" + pmtAssetId), toBase58String(i.caller.bytes)), IntegerEntry(("history_" + toBase58String(i.caller.bytes)), numberHUser), StringEntry(((("history_" + toBase58String(i.caller.bytes)) + "_") + toString(numberHUser)), pmtAssetId), IntegerEntry(("history_" + pmtAssetId), numberH), StringEntry((("history_" + pmtAssetId) + toString(numberH)), "sell")]
111- else throw("Error: this is an asset, not an NFT.")
112- }
199+func createFarm (asset1,asset2,apr,lockBlocks,rewardAmount) = {
200+ let admin = getAdminAddress()
201+ if ((toBase58String(i.caller.bytes) != admin))
202+ then throw("Only the Admin itself can invoke this function")
203+ else if (!(isValidAsset(asset1)))
204+ then throw("Invalid asset1")
205+ else if (!(isValidAsset(asset2)))
206+ then throw("Invalid asset2")
207+ else if ((asset1 == asset2))
208+ then throw("Asset1 cannot be equal to Asset2")
209+ else if ((0 >= apr))
210+ then throw("APR must be positive")
211+ else if ((0 >= lockBlocks))
212+ then throw("Lock blocks must be positive")
213+ else if ((0 >= rewardAmount))
214+ then throw("Reward amount must be positive")
215+ else {
216+ let poolKey = kp(asset1, asset2)
217+ if ((validPoolK(poolKey) == 0))
218+ then throw("Pool does not exist")
219+ else {
220+ let farmKey = fk(asset1, asset2)
221+[IntegerEntry(farmKey, 1), IntegerEntry((farmApr + poolKey), apr), IntegerEntry((farmLockBlocks + poolKey), lockBlocks), IntegerEntry((farmRewardPool + poolKey), rewardAmount), IntegerEntry((farmTotalStaked + poolKey), 0), IntegerEntry((totalUsersFarm + poolKey), 0)]
222+ }
223+ }
224+ }
113225
114226
115227
116228 @Callable(i)
117-func buyNFT (nftId) = {
118- let pay = i.payments[0]
119- let pmtAssetId = match pay.assetId {
229+func fundFarm (asset1,asset2) = {
230+ let admin = getAdminAddress()
231+ if ((toBase58String(i.caller.bytes) != admin))
232+ then throw("Only the Admin itself can invoke this function")
233+ else {
234+ let pmt = i.payments[0]
235+ let pmtAssetId = match pmt.assetId {
236+ case a: ByteVector =>
237+ toBase58String(a)
238+ case _ =>
239+ "GIC"
240+ }
241+ if ((toBase58String(i.caller.bytes) != admin))
242+ then throw("Only the Admin itself can invoke this function")
243+ else if ((pmtAssetId != "GIC"))
244+ then throw("Rewards must be in GIC")
245+ else {
246+ let poolKey = kp(asset1, asset2)
247+ let farmKey = fk(asset1, asset2)
248+ if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
249+ then throw("Farm does not exist")
250+ else {
251+ let currentRewardPool = valueOrElse(getInteger(this, (farmRewardPool + poolKey)), 0)
252+ let newRewardPool = (currentRewardPool + pmt.amount)
253+[IntegerEntry((farmRewardPool + poolKey), newRewardPool)]
254+ }
255+ }
256+ }
257+ }
258+
259+
260+
261+@Callable(i)
262+func changeData (key,data) = {
263+ let admin = getAdminAddress()
264+ if ((toBase58String(i.caller.bytes) != admin))
265+ then throw("Only the Admin itself can invoke this function")
266+ else [IntegerEntry(key, data)]
267+ }
268+
269+
270+
271+@Callable(i)
272+func createPool (asset1,asset2,nameLp) = {
273+ let poolKey = kp(asset1, asset2)
274+ let admin = getAdminAddress()
275+ if ((toBase58String(i.caller.bytes) != admin))
276+ then throw("Only the Admin itself can invoke this function")
277+ else if (!(isValidAsset(asset1)))
278+ then throw("Invalid asset1")
279+ else if (!(isValidAsset(asset2)))
280+ then throw("Invalid asset2")
281+ else if ((asset1 == asset2))
282+ then throw("Asset1 cannot be equal to Asset2")
283+ else if (if (isBlacklisted(asset1))
284+ then true
285+ else isBlacklisted(asset2))
286+ then throw("One or both assets are blacklisted")
287+ else if ((validPoolK(poolKey) != 0))
288+ then throw("Pool already exists")
289+ else {
290+ let assetLpCreate = Issue(nameLp, ((("LP asset for pool " + asset1) + "_") + asset2), 1, 8, true)
291+ let idAssetLp = calculateAssetId(assetLpCreate)
292+[assetLpCreate, StringEntry((lp + poolKey), toBase58String(idAssetLp)), IntegerEntry(poolKey, 1)]
293+ }
294+ }
295+
296+
297+
298+@Callable(i)
299+func addLiquidity (asset1,asset2) = {
300+ let power = getBooleanValue(this, ac)
301+ let liquidityPower = getBooleanValue(this, liquidityAc)
302+ let acGic = getBooleanValue(this, "accept_gic")
303+ let poolKey = kp(asset1, asset2)
304+ let pmt1 = i.payments[0]
305+ let pmt2 = i.payments[1]
306+ let asset1Id = match pmt1.assetId {
120307 case a: ByteVector =>
121308 toBase58String(a)
122309 case _ =>
123310 "GIC"
124311 }
125- let feeBps = toBigInt(getIntegerValue(this, fee))
126- let isSale = valueOrElse(getBooleanValue(("is_sale_" + nftId)), false)
127- let valueSale = valueOrElse(getIntegerValue(("price_sale_" + nftId)), 0)
128- let feeToSend = ((toBigInt(pay.amount) * feeBps) / toBigInt(10000))
129- let adminAddr = getAdminAddress()
130- let OldNumberH = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
131- let numberH = (OldNumberH + 1)
132- let OldNumberHUser = valueOrElse(getIntegerValue(this, ("history_" + pmtAssetId)), 0)
133- let numberHUser = (OldNumberH + 1)
134- let toSendToSaler = (toBigInt(pay.amount) - feeToSend)
135- if ((pmtAssetId != "GIC"))
136- then throw("Error: Invalid Asset to buy.")
137- else if (!(isSale))
138- then throw("This NFT is no longer for sale.")
139- else if ((valueSale != pay.amount))
140- then throw("Insufficient value to make the purchase.")
141- else [ScriptTransfer(i.caller, 1, fromBase58String(nftId)), ScriptTransfer(addressFromStringValue(getStringValue(this, ("saler_" + pmtAssetId))), toInt(toSendToSaler), unit), ScriptTransfer(addressFromStringValue(adminAddr), toInt(feeToSend), unit), BooleanEntry(("is_sale_" + pmtAssetId), false), StringEntry(("saler_" + pmtAssetId), toBase58String(i.caller.bytes)), IntegerEntry(("history_" + pmtAssetId), numberH), StringEntry((("history_" + pmtAssetId) + toString(numberH)), "buy")]
312+ let asset2Id = match pmt2.assetId {
313+ case a: ByteVector =>
314+ toBase58String(a)
315+ case _ =>
316+ "GIC"
317+ }
318+ let decimals1 = getAssetDecimals(asset1Id)
319+ let decimals2 = getAssetDecimals(asset2Id)
320+ if (!(power))
321+ then throw("dApp is currently under maintenance")
322+ else if (liquidityPower)
323+ then throw("Liquidity is currently under maintenance")
324+ else if (if ((asset1Id != asset1))
325+ then true
326+ else (asset2Id != asset2))
327+ then throw("Payment assets do not match specified assets")
328+ else if ((asset1 == asset2))
329+ then throw("Asset1 cannot be equal to Asset2")
330+ else if (!(isValidAsset(asset1)))
331+ then throw("Invalid asset1")
332+ else if (!(isValidAsset(asset2)))
333+ then throw("Invalid asset2")
334+ else if (if (isBlacklisted(asset1))
335+ then true
336+ else isBlacklisted(asset2))
337+ then throw("One or both assets are blacklisted")
338+ else if (if (if ((asset1 == "GIC"))
339+ then true
340+ else (asset2 == "GIC"))
341+ then !(acGic)
342+ else false)
343+ then throw("GIC not allowed")
344+ else if ((validPoolK(poolKey) == 0))
345+ then throw("Pool does not exist. Create it first")
346+ else {
347+ let poolAmount1 = poolAA(poolKey, asset1)
348+ let poolAmount2 = poolAA(poolKey, asset2)
349+ let amount1 = pmt1.amount
350+ let amount2 = pmt2.amount
351+ let lpAmount = if (if ((poolAmount1 == 0))
352+ then true
353+ else (poolAmount2 == 0))
354+ then toBigInt(amount1)
355+ else ((toBigInt(amount1) * toBigInt(poolAmount2)) / toBigInt(poolAmount1))
356+ let currentKey = toBase58String(i.caller.bytes)
357+ let currentAmountAsset1 = userLiquidity(currentKey, poolKey, asset1)
358+ let currentAmountAsset2 = userLiquidity(currentKey, poolKey, asset2)
359+ let newAmountAsset1 = (toBigInt(currentAmountAsset1) + toBigInt(amount1))
360+ let newAmountAsset2 = (toBigInt(currentAmountAsset2) + toBigInt(amount2))
361+ let newPoolAmount1 = (toBigInt(poolAmount1) + toBigInt(amount1))
362+ let newPoolAmount2 = (toBigInt(poolAmount2) + toBigInt(amount2))
363+ let lpAssetId = getStringValue(this, (lp + poolKey))
364+ let oldAssetLpTotal = valueOrElse(getIntegerValue(poolKey), 0)
365+ let lpAsset = if ((lpAssetId == ""))
366+ then throw("Pool does not have a liquidity pair, contact admin")
367+ else fromBase58String(lpAssetId)
368+[IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset1), toInt(newAmountAsset1)), IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset2), toInt(newAmountAsset2)), IntegerEntry(((poolKey + "_") + asset1), toInt(newPoolAmount1)), IntegerEntry(((poolKey + "_") + asset2), toInt(newPoolAmount2)), IntegerEntry((((depositHeight + poolKey) + "_") + currentKey), height), IntegerEntry(poolKey, (oldAssetLpTotal + toInt(lpAmount))), Reissue(lpAsset, toInt(lpAmount), true), ScriptTransfer(i.caller, toInt(lpAmount), lpAsset)]
369+ }
370+ }
371+
372+
373+
374+@Callable(i)
375+func removeLiquidity (asset1,asset2,lpAmount) = {
376+ let power = getBooleanValue(this, ac)
377+ let liquidityPower = getBooleanValue(this, liquidityAc)
378+ let poolKey = kp(asset1, asset2)
379+ let lpAssetId = getStringValue(this, (lp + poolKey))
380+ let lpAsset = fromBase58String(lpAssetId)
381+ let pmt = i.payments[0]
382+ let pmtAssetId = match pmt.assetId {
383+ case a: ByteVector =>
384+ toBase58String(a)
385+ case _ =>
386+ "GIC"
387+ }
388+ let decimals1 = getAssetDecimals(asset1)
389+ let decimals2 = getAssetDecimals(asset2)
390+ if (!(power))
391+ then throw("dApp is currently under maintenance")
392+ else if (liquidityPower)
393+ then throw("Liquidity is currently under maintenance")
394+ else if ((lpAssetId == ""))
395+ then throw("Invalid LP asset")
396+ else if ((pmtAssetId != lpAssetId))
397+ then throw("Invalid LP token")
398+ else if ((validPoolK(poolKey) == 0))
399+ then throw("Pool does not exist")
400+ else if ((0 >= lpAmount))
401+ then throw("LP amount must be positive")
402+ else {
403+ let poolAmount1 = poolAA(poolKey, asset1)
404+ let poolAmount2 = poolAA(poolKey, asset2)
405+ let totalLpSupply = match assetInfo(fromBase58String(lpAssetId)) {
406+ case a: Asset =>
407+ a.quantity
408+ case _ =>
409+ 0
410+ }
411+ let amount1 = ((lpAmount * poolAmount1) / totalLpSupply)
412+ let amount2 = ((lpAmount * poolAmount2) / totalLpSupply)
413+ let currentKey = toBase58String(i.caller.bytes)
414+ let currentAmountAsset1 = userLiquidity(currentKey, poolKey, asset1)
415+ let currentAmountAsset2 = userLiquidity(currentKey, poolKey, asset2)
416+ let newAmountAsset1 = (currentAmountAsset1 - amount1)
417+ let newAmountAsset2 = (currentAmountAsset2 - amount2)
418+ if (if ((0 > newAmountAsset1))
419+ then true
420+ else (0 > newAmountAsset2))
421+ then throw("Insufficient liquidity balance")
422+ else {
423+ let newPoolAmount1 = (poolAmount1 - amount1)
424+ let newPoolAmount2 = (poolAmount2 - amount2)
425+ let asset1Bytes = if ((asset1 == "GIC"))
426+ then unit
427+ else fromBase58String(asset1)
428+ let asset2Bytes = if ((asset2 == "GIC"))
429+ then unit
430+ else fromBase58String(asset2)
431+ let oldAssetLpTotal = getIntegerValue(poolKey)
432+[IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset1), newAmountAsset1), IntegerEntry(((((poolKey + "_") + currentKey) + "_") + asset2), newAmountAsset2), IntegerEntry(((poolKey + "_") + asset1), newPoolAmount1), IntegerEntry(((poolKey + "_") + asset2), newPoolAmount2), IntegerEntry(poolKey, (oldAssetLpTotal - lpAmount)), Burn(lpAsset, lpAmount), ScriptTransfer(i.caller, amount1, asset1Bytes), ScriptTransfer(i.caller, amount2, asset2Bytes)]
433+ }
434+ }
435+ }
436+
437+
438+
439+@Callable(i)
440+func swap (assetIn,assetOut,maxSlippage) = {
441+ let power = getBooleanValue(this, ac)
442+ let swapActivate = getBooleanValue(this, swapAc)
443+ let acGic = getBooleanValue(this, "accept_gic")
444+ let poolKey = kp(assetIn, assetOut)
445+ let reversePoolKey = kp(assetOut, assetIn)
446+ let key = if ((validPoolK(poolKey) != 0))
447+ then poolKey
448+ else if ((validPoolK(reversePoolKey) != 0))
449+ then reversePoolKey
450+ else ""
451+ let pmt = i.payments[0]
452+ let pmtAssetId = match pmt.assetId {
453+ case a: ByteVector =>
454+ toBase58String(a)
455+ case _ =>
456+ "GIC"
457+ }
458+ let decimalsIn = getAssetDecimals(assetIn)
459+ let decimalsOut = getAssetDecimals(assetOut)
460+ let feeBps = toBigInt(getIntegerValue(this, swapFee))
461+ let adminAddr = addressFromStringValue(getStringValue(this, adminAddress))
462+ if (!(power))
463+ then throw("dApp is currently under maintenance")
464+ else if (swapActivate)
465+ then throw("Swap is currently under maintenance")
466+ else if ((pmtAssetId != assetIn))
467+ then throw("Payment asset does not match assetIn")
468+ else if (!(isValidAsset(assetIn)))
469+ then throw("Invalid assetIn")
470+ else if (!(isValidAsset(assetOut)))
471+ then throw("Invalid assetOut")
472+ else if (if (isBlacklisted(assetIn))
473+ then true
474+ else isBlacklisted(assetOut))
475+ then throw("One or both assets are blacklisted")
476+ else if ((assetIn == assetOut))
477+ then throw("assetIn cannot be equal to assetOut")
478+ else if (if (if ((assetIn == "GIC"))
479+ then true
480+ else (assetOut == "GIC"))
481+ then !(acGic)
482+ else false)
483+ then throw("GIC not allowed")
484+ else if ((key == ""))
485+ then throw("Pool does not exist")
486+ else {
487+ let isReverse = (key == reversePoolKey)
488+ let poolAsset1 = if (isReverse)
489+ then assetOut
490+ else assetIn
491+ let poolAsset2 = if (isReverse)
492+ then assetIn
493+ else assetOut
494+ let poolAmount1 = toBigInt(poolAA(key, poolAsset1))
495+ let poolAmount2 = toBigInt(poolAA(key, poolAsset2))
496+ let amountInAdjusted = toBigInt(pmt.amount)
497+ let amountOut = if (isReverse)
498+ then ((amountInAdjusted * poolAmount1) / poolAmount2)
499+ else ((amountInAdjusted * poolAmount2) / poolAmount1)
500+ let fee = ((amountOut * feeBps) / toBigInt(10000))
501+ let amountOutAfterFee = (amountOut - fee)
502+ let feeToSend = ((amountInAdjusted * feeBps) / toBigInt(10000))
503+ let minAmountOut = ((amountOutAfterFee * toBigInt((10000 - maxSlippage))) / toBigInt(10000))
504+ if ((minAmountOut > amountOutAfterFee))
505+ then throw("Slippage exceeded")
506+ else {
507+ let newPoolAmount1 = if (isReverse)
508+ then (poolAmount1 - amountOut)
509+ else (poolAmount1 + amountInAdjusted)
510+ let newPoolAmount2 = if (isReverse)
511+ then (poolAmount2 + amountInAdjusted)
512+ else (poolAmount2 - amountOut)
513+ let assettoReceive = if ((assetOut == "GIC"))
514+ then unit
515+ else fromBase58String(assetOut)
516+ let assetOutBytes = if ((assetIn == "GIC"))
517+ then unit
518+ else fromBase58String(assetIn)
519+[IntegerEntry(((key + "_") + poolAsset1), toInt(newPoolAmount1)), IntegerEntry(((key + "_") + poolAsset2), toInt(newPoolAmount2)), BooleanEntry("isReverse", isReverse), ScriptTransfer(i.caller, toInt(amountOutAfterFee), assettoReceive), ScriptTransfer(adminAddr, toInt(feeToSend), assetOutBytes)]
520+ }
521+ }
522+ }
523+
524+
525+
526+@Callable(i)
527+func farmsAddLP (asset1,asset2) = {
528+ let power = getBooleanValue(this, ac)
529+ let farmPower = getBooleanValue(this, farmAc)
530+ let poolKey = kp(asset1, asset2)
531+ let farmKey = fk(asset1, asset2)
532+ let pmt = i.payments[0]
533+ let lpAssetId = getStringValue(this, (lp + poolKey))
534+ let pmtAssetId = match pmt.assetId {
535+ case a: ByteVector =>
536+ toBase58String(a)
537+ case _ =>
538+ "GIC"
539+ }
540+ if (!(power))
541+ then throw("dApp is currently under maintenance")
542+ else if (farmPower)
543+ then throw("Farms are currently under maintenance")
544+ else if ((pmtAssetId != lpAssetId))
545+ then throw("Invalid LP token")
546+ else if ((validPoolK(poolKey) == 0))
547+ then throw("Pool does not exist")
548+ else if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
549+ then throw("Farm does not exist")
550+ else {
551+ let currentKey = toBase58String(i.caller.bytes)
552+ let stakedKey = sk(asset1, asset2, currentKey)
553+ let currentStaked = valueOrElse(getInteger(this, stakedKey), 0)
554+ let newStaked = (currentStaked + pmt.amount)
555+ let totalStakedKey = (farmTotalStaked + poolKey)
556+ let previousTotalStaked = valueOrElse(getInteger(this, totalStakedKey), 0)
557+ let newTotalStaked = (previousTotalStaked + pmt.amount)
558+ let totalUsers = valueOrElse(getInteger(this, (totalUsersFarm + poolKey)), 0)
559+ let newTotalUsers = if ((currentStaked == 0))
560+ then (totalUsers + 1)
561+ else totalUsers
562+[IntegerEntry(stakedKey, newStaked), IntegerEntry(totalStakedKey, newTotalStaked), IntegerEntry((totalUsersFarm + poolKey), newTotalUsers), IntegerEntry(fh(asset1, asset2, currentKey), height)]
563+ }
564+ }
565+
566+
567+
568+@Callable(i)
569+func farmsClaimLP (asset1,asset2) = {
570+ let power = getBooleanValue(this, ac)
571+ let farmPower = getBooleanValue(this, farmAc)
572+ let poolKey = kp(asset1, asset2)
573+ let farmKey = fk(asset1, asset2)
574+ let currentKey = toBase58String(i.caller.bytes)
575+ let stakedKey = sk(asset1, asset2, currentKey)
576+ let heightKey = fh(asset1, asset2, currentKey)
577+ let rewardKey = rk(asset1, asset2, currentKey)
578+ if (!(power))
579+ then throw("dApp is currently under maintenance")
580+ else if (farmPower)
581+ then throw("Farms are currently under maintenance")
582+ else if ((validPoolK(poolKey) == 0))
583+ then throw("Pool does not exist")
584+ else if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
585+ then throw("Farm does not exist")
586+ else {
587+ let currentStaked = valueOrElse(getInteger(this, stakedKey), 0)
588+ if ((0 >= currentStaked))
589+ then throw("No staked LP tokens")
590+ else {
591+ let rewardPool = valueOrElse(getInteger(this, (farmRewardPool + poolKey)), 0)
592+ let totalUsers = valueOrElse(getInteger(this, (totalUsersFarm + poolKey)), 1)
593+ let lockBlocks = getIntegerValue(this, (farmLockBlocks + poolKey))
594+ let apr = getIntegerValue(this, (farmApr + poolKey))
595+ let lastClaimHeight = valueOrElse(getInteger(this, heightKey), height)
596+ let blocksSinceLastClaim = (height - lastClaimHeight)
597+ if ((1 > blocksSinceLastClaim))
598+ then throw("Must wait at least 1 block since last claim")
599+ else {
600+ let rewardPerBlock = ((rewardPool * apr) / ((10000 * lockBlocks) * totalUsers))
601+ let reward = (((rewardPerBlock * blocksSinceLastClaim) * currentStaked) / D8)
602+ if ((reward > rewardPool))
603+ then throw("Insufficient reward pool")
604+ else {
605+ let newRewardPool = (rewardPool - reward)
606+[IntegerEntry(rewardKey, 0), IntegerEntry((farmRewardPool + poolKey), newRewardPool), IntegerEntry(heightKey, height), ScriptTransfer(i.caller, reward, unit)]
607+ }
608+ }
609+ }
610+ }
611+ }
612+
613+
614+
615+@Callable(i)
616+func farmsRemoveLP (asset1,asset2,amount) = {
617+ let power = getBooleanValue(this, ac)
618+ let farmPower = getBooleanValue(this, farmAc)
619+ let poolKey = kp(asset1, asset2)
620+ let farmKey = fk(asset1, asset2)
621+ let lpAssetId = getStringValue(this, (lp + poolKey))
622+ let lpAsset = if ((lpAssetId == ""))
623+ then unit
624+ else fromBase58String(lpAssetId)
625+ let currentKey = toBase58String(i.caller.bytes)
626+ let stakedKey = sk(asset1, asset2, currentKey)
627+ let heightKey = fh(asset1, asset2, currentKey)
628+ let rewardKey = rk(asset1, asset2, currentKey)
629+ if (!(power))
630+ then throw("dApp is currently under maintenance")
631+ else if (farmPower)
632+ then throw("Farms are currently under maintenance")
633+ else if ((validPoolK(poolKey) == 0))
634+ then throw("Pool does not exist")
635+ else if ((valueOrElse(getInteger(this, farmKey), 0) == 0))
636+ then throw("Farm does not exist")
637+ else {
638+ let currentStaked = valueOrElse(getInteger(this, stakedKey), 0)
639+ if ((0 >= amount))
640+ then throw("Amount must be positive")
641+ else if ((amount > currentStaked))
642+ then throw("Insufficient staked LP tokens")
643+ else {
644+ let rewardPool = valueOrElse(getInteger(this, (farmRewardPool + poolKey)), 0)
645+ let totalUsers = valueOrElse(getInteger(this, (totalUsersFarm + poolKey)), 1)
646+ let lockBlocks = getIntegerValue(this, (farmLockBlocks + poolKey))
647+ let apr = getIntegerValue(this, (farmApr + poolKey))
648+ let lastClaimHeight = valueOrElse(getInteger(this, heightKey), height)
649+ let blocksSinceLastClaim = (height - lastClaimHeight)
650+ let rewardPerBlock = ((rewardPool * apr) / ((10000 * lockBlocks) * totalUsers))
651+ let reward = if ((blocksSinceLastClaim >= 1))
652+ then (((rewardPerBlock * blocksSinceLastClaim) * currentStaked) / D8)
653+ else 0
654+ let newRewardPool = (rewardPool - reward)
655+ let newStaked = (currentStaked - amount)
656+ let totalStakedKey = (farmTotalStaked + poolKey)
657+ let previousTotalStaked = valueOrElse(getInteger(this, totalStakedKey), 0)
658+ let newTotalStaked = (previousTotalStaked - amount)
659+ let newTotalUsers = if ((newStaked == 0))
660+ then (totalUsers - 1)
661+ else totalUsers
662+[IntegerEntry(stakedKey, newStaked), IntegerEntry(totalStakedKey, newTotalStaked), IntegerEntry((totalUsersFarm + poolKey), newTotalUsers), IntegerEntry(rewardKey, 0), IntegerEntry((farmRewardPool + poolKey), newRewardPool), IntegerEntry(heightKey, height), ScriptTransfer(i.caller, amount, lpAsset), ScriptTransfer(i.caller, reward, unit)]
663+ }
664+ }
142665 }
143666
144667
145668 @Verifier(tx)
146-func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
669+func verify () = match tx {
670+ case _ =>
671+ sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
672+}
147673

github/gicsportsofficial/g8-explorer
144.41 ms