tx · 6Wi8nwXfJ8vWkHvV3S3wmx4MFEJvLMgQuhtRec2AxwR2

3PG2vMhK5CPqsCDodvLGzQ84QkoHXCJ3oNP:  -0.14000000 Waves

2020.07.16 11:57 [2153038] smart account 3PG2vMhK5CPqsCDodvLGzQ84QkoHXCJ3oNP > SELF 0.00000000 Waves

{ "type": 13, "id": "6Wi8nwXfJ8vWkHvV3S3wmx4MFEJvLMgQuhtRec2AxwR2", "fee": 14000000, "feeAssetId": null, "timestamp": 1594887078061, "version": 1, "sender": "3PG2vMhK5CPqsCDodvLGzQ84QkoHXCJ3oNP", "senderPublicKey": "5RM3w4ysmDbtgfswnVNPx7DQkNwVAG3RoxNFHgt6ToNU", "proofs": [ "", "2gjiPPab7DMMB7bkizGN2HMWWDfmJ2ZdZSZuFX22VdQ76k5kfoi6JMrvXci4ryY1jfVKPMscB8hVvtPsC2mLteZe", "", "3MSbZvFwxtvs8ms6su5fBDP186bWbkZa1JJZ4y4ih5R5MEyeN2A9AWAqHorfRu3RmwoMhzZQQNNrjkLh3Jqz3atp" ], "script": "base64:AAIDAAAAAAAAAAQIARIAAAAAOAEAAAAOZ2V0TnVtYmVyQnlLZXkAAAABAAAAA2tleQQAAAAHJG1hdGNoMAkABBoAAAACBQAAAAR0aGlzBQAAAANrZXkDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAAA0ludAQAAAABYQUAAAAHJG1hdGNoMAUAAAABYQAAAAAAAAAAAAEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABAAAAA2tleQQAAAAHJG1hdGNoMAkABB0AAAACBQAAAAR0aGlzBQAAAANrZXkDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABlN0cmluZwQAAAABYQUAAAAHJG1hdGNoMAUAAAABYQIAAAAAAQAAABZnZXRCb29sQnlBZGRyZXNzQW5kS2V5AAAAAgAAAAdhZGRyZXNzAAAAA2tleQQAAAAHJG1hdGNoMAkABBsAAAACBQAAAAdhZGRyZXNzBQAAAANrZXkDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAAB0Jvb2xlYW4EAAAAAWEFAAAAByRtYXRjaDAFAAAAAWEHAQAAABhnZXRTdHJpbmdCeUFkZHJlc3NBbmRLZXkAAAACAAAAB2FkZHJlc3MAAAADa2V5BAAAAAckbWF0Y2gwCQAEHQAAAAIFAAAAB2FkZHJlc3MFAAAAA2tleQMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAGU3RyaW5nBAAAAAFhBQAAAAckbWF0Y2gwBQAAAAFhAgAAAAABAAAAGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQAAAAIAAAAHYWRkcmVzcwAAAANrZXkEAAAAByRtYXRjaDAJAAQaAAAAAgUAAAAHYWRkcmVzcwUAAAADa2V5AwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAANJbnQEAAAAAWEFAAAAByRtYXRjaDAFAAAAAWEAAAAAAAAAAAAAAAAAB1dBVkVMRVQAAAAAAAX14QAAAAAABVBBVUxJAAAAAAAAD0JAAAAAAA1NSU5PUkRFUlRPVEFMCQAAaAAAAAIAAAAAAAAAAAoFAAAAB1dBVkVMRVQAAAAABk1BWFJPSQAAAAAAAAAAXwAAAAAIQ0FOQ0VMRUQCAAAACGNhbmNlbGVkAAAAAANORVcCAAAAA25ldwAAAAAGRklMTEVEAgAAAAZmaWxsZWQAAAAAE05ldXRyaW5vQ29udHJhY3RLZXkCAAAAEW5ldXRyaW5vX2NvbnRyYWN0AAAAAAhQcmljZUtleQIAAAAFcHJpY2UAAAAADkJvbmRBc3NldElkS2V5AgAAAA1ib25kX2Fzc2V0X2lkAAAAABJOZXV0cmlub0Fzc2V0SWRLZXkCAAAAEW5ldXRyaW5vX2Fzc2V0X2lkAAAAABJDb250cm9sQ29udHJhY3RLZXkCAAAAEGNvbnRyb2xfY29udHJhY3QAAAAAEUJhbGFuY2VMb2NrZWRrS2V5AgAAAA1iYWxhbmNlX2xvY2tfAAAAABVXYXZlc0xvY2tlZEJhbGFuY2VLZXkJAAEsAAAAAgUAAAARQmFsYW5jZUxvY2tlZGtLZXkCAAAABXdhdmVzAAAAABhOZXV0cmlub0xvY2tlZEJhbGFuY2VLZXkJAAEsAAAAAgUAAAARQmFsYW5jZUxvY2tlZGtLZXkCAAAACG5ldXRyaW5vAAAAABZMaXF1aWRhdGlvbkNvbnRyYWN0S2V5AgAAABRsaXF1aWRhdGlvbl9jb250cmFjdAAAAAANRmlyc3RPcmRlcktleQIAAAALb3JkZXJfZmlyc3QBAAAAEmdldFJvaUJ5T3JkZXJJZEtleQAAAAEAAAAHb3JkZXJJZAkAASwAAAACAgAAABBkZWJ1Z19vcmRlcl9yb2lfBQAAAAdvcmRlcklkAQAAABBnZXRPcmRlclByaWNlS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAADG9yZGVyX3ByaWNlXwUAAAAHb3JkZXJJZAEAAAAVZ2V0RGVidWdPcmRlclByaWNlS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAAGWRlYnVnX29yZGVyX2N1cnJlbnRQcmljZV8FAAAAB29yZGVySWQBAAAAEGdldE9yZGVyVG90YWxLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAAMb3JkZXJfdG90YWxfBQAAAAdvcmRlcklkAQAAABBnZXRPcmRlck93bmVyS2V5AAAAAQAAAAdvcmRlcklkCQABLAAAAAICAAAADG9yZGVyX293bmVyXwUAAAAHb3JkZXJJZAEAAAARZ2V0T3JkZXJIZWlnaHRLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAANb3JkZXJfaGVpZ2h0XwUAAAAHb3JkZXJJZAEAAAARZ2V0T3JkZXJTdGF0dXNLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAANb3JkZXJfc3RhdHVzXwUAAAAHb3JkZXJJZAEAAAAWZ2V0T3JkZXJGaWxsZWRUb3RhbEtleQAAAAEAAAAHb3JkZXJJZAkAASwAAAACAgAAABNvcmRlcl9maWxsZWRfdG90YWxfBQAAAAdvcmRlcklkAQAAAA9nZXRQcmV2T3JkZXJLZXkAAAABAAAAB29yZGVySWQJAAEsAAAAAgIAAAALb3JkZXJfcHJldl8FAAAAB29yZGVySWQBAAAAD2dldE5leHRPcmRlcktleQAAAAEAAAAHb3JkZXJJZAkAASwAAAACAgAAAAtvcmRlcl9uZXh0XwUAAAAHb3JkZXJJZAEAAAAWY29udmVydE5ldXRyaW5vVG9XYXZlcwAAAAIAAAAGYW1vdW50AAAABXByaWNlCQAAawAAAAMJAABrAAAAAwUAAAAGYW1vdW50AAAAAAAAAABkBQAAAAVwcmljZQUAAAAHV0FWRUxFVAUAAAAFUEFVTEkBAAAAFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8AAAACAAAABmFtb3VudAAAAAVwcmljZQkAAGsAAAADCQAAawAAAAMFAAAABmFtb3VudAUAAAAFcHJpY2UAAAAAAAAAAGQFAAAABVBBVUxJBQAAAAdXQVZFTEVUAQAAABJjb252ZXJ0V2F2ZXNUb0JvbmQAAAACAAAABmFtb3VudAAAAAVwcmljZQkBAAAAFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8AAAACBQAAAAZhbW91bnQFAAAABXByaWNlAQAAABJjb252ZXJ0Qm9uZFRvV2F2ZXMAAAACAAAABmFtb3VudAAAAAVwcmljZQkBAAAAFmNvbnZlcnROZXV0cmlub1RvV2F2ZXMAAAACBQAAAAZhbW91bnQFAAAABXByaWNlAAAAABBuZXV0cmlub0NvbnRyYWN0CQEAAAAHQWRkcmVzcwAAAAEBAAAAGgFXcARipkeb6a1WaJTL74WMMIIgKJoIFJayAAAAAA9jb250cm9sQ29udHJhY3QJAQAAAAdBZGRyZXNzAAAAAQEAAAAaAVcjs60SXJOkyuw5/k9G1s1WTS37EPtjmHoAAAAAE2xpcXVpZGF0aW9uQ29udHJhY3QJAQAAAAdBZGRyZXNzAAAAAQEAAAAaAVca6knL+gp+ygh/KNWflY4Me2m1qTiRH0gAAAAAD25ldXRyaW5vQXNzZXRJZAEAAAAgtiYpwwT1zlORpA5LdSQvZIxRsfrfr1QpvUjSHSqyqtEAAAAAC2JvbmRBc3NldElkAQAAACBV7sO+qgvwUOhxyBuqbnCepLHI/kouucHxzMVrD3iXSwAAAAAJaXNCbG9ja2VkCQEAAAAWZ2V0Qm9vbEJ5QWRkcmVzc0FuZEtleQAAAAIFAAAAD2NvbnRyb2xDb250cmFjdAIAAAAKaXNfYmxvY2tlZAAAAAAMY3VycmVudFByaWNlCQEAAAAYZ2V0TnVtYmVyQnlBZGRyZXNzQW5kS2V5AAAAAgUAAAAPY29udHJvbENvbnRyYWN0BQAAAAhQcmljZUtleQAAAAAVbmV1dHJpbm9Mb2NrZWRCYWxhbmNlCQEAAAAYZ2V0TnVtYmVyQnlBZGRyZXNzQW5kS2V5AAAAAgUAAAAQbmV1dHJpbm9Db250cmFjdAUAAAAYTmV1dHJpbm9Mb2NrZWRCYWxhbmNlS2V5AAAAAAdyZXNlcnZlCQAAZQAAAAIJAQAAAAx3YXZlc0JhbGFuY2UAAAABBQAAABBuZXV0cmlub0NvbnRyYWN0CQEAAAAYZ2V0TnVtYmVyQnlBZGRyZXNzQW5kS2V5AAAAAgUAAAAQbmV1dHJpbm9Db250cmFjdAUAAAAVV2F2ZXNMb2NrZWRCYWxhbmNlS2V5AAAAAA5uZXV0cmlub1N1cHBseQkAAGUAAAACCQAAZQAAAAIJAABkAAAAAgUAAAAVbmV1dHJpbm9Mb2NrZWRCYWxhbmNlCAkBAAAAB2V4dHJhY3QAAAABCQAD7AAAAAEFAAAAD25ldXRyaW5vQXNzZXRJZAAAAAhxdWFudGl0eQkAA+sAAAACBQAAABBuZXV0cmlub0NvbnRyYWN0BQAAAA9uZXV0cmlub0Fzc2V0SWQJAAPrAAAAAgUAAAATbGlxdWlkYXRpb25Db250cmFjdAUAAAAPbmV1dHJpbm9Bc3NldElkAAAAAAdkZWZpY2l0CQAAZQAAAAIFAAAADm5ldXRyaW5vU3VwcGx5CQEAAAAWY29udmVydFdhdmVzVG9OZXV0cmlubwAAAAIFAAAAB3Jlc2VydmUFAAAADGN1cnJlbnRQcmljZQAAAAAKZmlyc3RPcmRlcgkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAANRmlyc3RPcmRlcktleQEAAAANZ2V0T3JkZXJQcmljZQAAAAEAAAACaWQJAQAAAA5nZXROdW1iZXJCeUtleQAAAAEJAQAAABBnZXRPcmRlclByaWNlS2V5AAAAAQUAAAACaWQBAAAAEmdldERlYnVnT3JkZXJQcmljZQAAAAEAAAACaWQJAQAAAA5nZXROdW1iZXJCeUtleQAAAAEJAQAAABVnZXREZWJ1Z09yZGVyUHJpY2VLZXkAAAABBQAAAAJpZAEAAAANZ2V0T3JkZXJUb3RhbAAAAAEAAAACaWQJAQAAAA5nZXROdW1iZXJCeUtleQAAAAEJAQAAABBnZXRPcmRlclRvdGFsS2V5AAAAAQUAAAACaWQBAAAADWdldE9yZGVyT3duZXIAAAABAAAAAmlkCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAQZ2V0T3JkZXJPd25lcktleQAAAAEFAAAAAmlkAQAAAA5nZXRPcmRlclN0YXR1cwAAAAEAAAACaWQJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAQAAABFnZXRPcmRlclN0YXR1c0tleQAAAAEFAAAAAmlkAQAAABNnZXRPcmRlckZpbGxlZFRvdGFsAAAAAQAAAAJpZAkBAAAADmdldE51bWJlckJ5S2V5AAAAAQkBAAAAFmdldE9yZGVyRmlsbGVkVG90YWxLZXkAAAABBQAAAAJpZAEAAAAMZ2V0UHJldk9yZGVyAAAAAQAAAAJpZAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQkBAAAAD2dldFByZXZPcmRlcktleQAAAAEFAAAAAmlkAQAAAAxnZXROZXh0T3JkZXIAAAABAAAAAmlkCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABCQEAAAAPZ2V0TmV4dE9yZGVyS2V5AAAAAQUAAAACaWQAAAABAAAAAWkBAAAAEW1pZ3JhdGVPcmRlclByaWNlAAAAAAQAAAAUbGFzdE1pZ3JhdGVkT3JkZXJLZXkCAAAAJ21pZ3JhdGlvbl82ZGVjaW1hbHNfbGFzdE1pZ3JhdGVkT3JkZXJJZAQAAAALZmluYWxTdGF0dXMCAAAAFW1pZ3JhdGVkX3RvXzZkZWNpbWFscwQAAAATbGFzdE1pZ3JhdGVkT3JkZXJJZAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAAUbGFzdE1pZ3JhdGVkT3JkZXJLZXkEAAAADmN1cnJlbnRPcmRlcklkAwkAAAAAAAACBQAAABNsYXN0TWlncmF0ZWRPcmRlcklkAgAAAAAFAAAACmZpcnN0T3JkZXIJAQAAAAxnZXROZXh0T3JkZXIAAAABBQAAABNsYXN0TWlncmF0ZWRPcmRlcklkBAAAABdvcmRlck1pZ3JhdGlvblN0YXR1c0tleQkAASwAAAACAgAAABttaWdyYXRpb25fNmRlY2ltYWxzX3N0YXR1c18FAAAADmN1cnJlbnRPcmRlcklkBAAAABRvcmRlck1pZ3JhdGlvblN0YXR1cwkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAAXb3JkZXJNaWdyYXRpb25TdGF0dXNLZXkDCQAAAAAAAAIFAAAAFG9yZGVyTWlncmF0aW9uU3RhdHVzBQAAAAtmaW5hbFN0YXR1cwkAAAIAAAABAgAAACZlcnJvcjogb3JkZXIgaGFzIGJlZW4gYWxyZWFkeSBtaWdyYXRlZAMJAAAAAAAAAgUAAAAOY3VycmVudE9yZGVySWQCAAAAAAkAAAIAAAABAgAAAEZzdGF0dXMgb2s6IGFsbCBvcmRlcnMgaGF2ZSBiZWVuIGFscmVhZHkgbWlncmF0ZWQgb3Igbm90aGluZyB0byBtaWdyYXRlCQEAAAAIV3JpdGVTZXQAAAABCQAETAAAAAIJAQAAAAlEYXRhRW50cnkAAAACBQAAABdvcmRlck1pZ3JhdGlvblN0YXR1c0tleQUAAAALZmluYWxTdGF0dXMJAARMAAAAAgkBAAAACURhdGFFbnRyeQAAAAIFAAAAFGxhc3RNaWdyYXRlZE9yZGVyS2V5BQAAAA5jdXJyZW50T3JkZXJJZAkABEwAAAACCQEAAAAJRGF0YUVudHJ5AAAAAgkBAAAAEGdldE9yZGVyUHJpY2VLZXkAAAABBQAAAA5jdXJyZW50T3JkZXJJZAkAAGgAAAACCQEAAAANZ2V0T3JkZXJQcmljZQAAAAEFAAAADmN1cnJlbnRPcmRlcklkAAAAAAAAACcQCQAETAAAAAIJAQAAAAlEYXRhRW50cnkAAAACCQEAAAAVZ2V0RGVidWdPcmRlclByaWNlS2V5AAAAAQUAAAAOY3VycmVudE9yZGVySWQJAABoAAAAAgkBAAAAEmdldERlYnVnT3JkZXJQcmljZQAAAAEFAAAADmN1cnJlbnRPcmRlcklkAAAAAAAAACcQBQAAAANuaWwAAAABAAAAAnR4AQAAAAZ2ZXJpZnkAAAAABAAAABBwdWJLZXlBZG1pbnNMaXN0CQAETAAAAAICAAAALEJMRW9ndXpQVktWVGZYeHhUM1c3UnFmOGFVbTJnZ0M5VmVtZDJNUWF3TTJHCQAETAAAAAICAAAALEZXVmZmWXIyQUxtSE1lalptM1dxZUx6NlNkeW0zZ0xGR3RKbjRLVHd5VTV4CQAETAAAAAICAAAALDNXaDJMYVdjYjVnZzdLMnBQY1czRXA2RUF1UkJ6WWtBZ3JkcHQ0M2pUREZhCQAETAAAAAICAAAALDVXUlhGU2p3Y1RiTmZLY0pzOFpxWG1TU1dZc1NWSlV0TXZNcVpqNWhINE5jBQAAAANuaWwEAAAABWNvdW50CQAAZAAAAAIJAABkAAAAAgkAAGQAAAACAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAAJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAAAAAAAAAAAAAABAAAAAAAAAAAAAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAEJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAABAAAAAAAAAAABAAAAAAAAAAAAAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAIJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAACAAAAAAAAAAABAAAAAAAAAAAAAwkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAMJAAJZAAAAAQkAAZEAAAACBQAAABBwdWJLZXlBZG1pbnNMaXN0AAAAAAAAAAADAAAAAAAAAAACAAAAAAAAAAAACQAAZwAAAAIFAAAABWNvdW50AAAAAAAAAAADL7egIA==", "chainId": 87, "height": 2153038, "spentComplexity": 0 } View: original | compacted Prev: 8T1wo7TX2VSUkXXt2AhZhWFVAQafiw8MQCoSmjN9A5e1 Next: Fc62ZY3vCzp7oKtUmhfxGcjScDeUYS4dB8cKmAHd8J9p Diff:
OldNewDifferences
8181 func getOrderPriceKey (orderId) = ("order_price_" + orderId)
8282
8383
84+func getDebugOrderPriceKey (orderId) = ("debug_order_currentPrice_" + orderId)
85+
86+
8487 func getOrderTotalKey (orderId) = ("order_total_" + orderId)
8588
8689
141144 func getOrderPrice (id) = getNumberByKey(getOrderPriceKey(id))
142145
143146
147+func getDebugOrderPrice (id) = getNumberByKey(getDebugOrderPriceKey(id))
148+
149+
144150 func getOrderTotal (id) = getNumberByKey(getOrderTotalKey(id))
145151
146152
159165 func getNextOrder (id) = getStringByKey(getNextOrderKey(id))
160166
161167
162-func orderData (orderId,totalWavelets,owner,status,roi,price) = [DataEntry(getOrderPriceKey(orderId), price), DataEntry(getOrderTotalKey(orderId), totalWavelets), DataEntry(getOrderOwnerKey(orderId), owner), DataEntry(getOrderHeightKey(orderId), height), DataEntry(getOrderStatusKey(orderId), status), DataEntry(("debug_order_currentPrice_" + orderId), currentPrice), DataEntry(getRoiByOrderIdKey(orderId), roi)]
163-
164-
165-func internalSellBond (currentMaxRoi,pFirstOrder,pNextOrder,pFilledTotal,pRoi,pPrice,pPaymentWavelets,orderOwnerAddress,instantOrder) = {
166- let priceWavesByBondCents = fraction((100 + pRoi), currentPrice, 100)
167- let remainedTotal = (pPaymentWavelets - pFilledTotal)
168- let fillableOrderAmount = convertWavesToBond(remainedTotal, priceWavesByBondCents)
169- let totalOrderWaveletesRequired = convertBondToWaves(fillableOrderAmount, priceWavesByBondCents)
170- let nbTokensSellCondition = (currentMaxRoi >= pRoi)
171- if (!(nbTokensSellCondition))
172- then throw(("innapropriate roi: " + toString(pRoi)))
173- else if ((totalOrderWaveletesRequired == 0))
174- then throw("cannot fill order at the moment")
175- else {
176- let changeWavelets = (remainedTotal - totalOrderWaveletesRequired)
177- let writeSetData = if (instantOrder)
178- then [DataEntry(getOrderPriceKey(pFirstOrder), pPrice), DataEntry(getOrderTotalKey(pFirstOrder), pPaymentWavelets), DataEntry(getOrderOwnerKey(pFirstOrder), toBase58String(orderOwnerAddress.bytes)), DataEntry(getOrderHeightKey(pFirstOrder), height), DataEntry(getOrderFilledTotalKey(pFirstOrder), (pFilledTotal + totalOrderWaveletesRequired)), DataEntry(getOrderStatusKey(pFirstOrder), FILLED), DataEntry(("debug_order_currentPrice_" + pFirstOrder), currentPrice), DataEntry(getRoiByOrderIdKey(pFirstOrder), pRoi)]
179- else [DataEntry(getOrderFilledTotalKey(pFirstOrder), (pFilledTotal + totalOrderWaveletesRequired)), DataEntry(getOrderStatusKey(pFirstOrder), FILLED), DataEntry(getOrderTotalKey(pFirstOrder), pPaymentWavelets), DataEntry(getPrevOrderKey(pNextOrder), ""), DataEntry(FirstOrderKey, pNextOrder)]
180- ScriptResult(WriteSet(writeSetData), TransferSet([ScriptTransfer(orderOwnerAddress, fillableOrderAmount, bondAssetId), ScriptTransfer(neutrinoContract, totalOrderWaveletesRequired, unit), ScriptTransfer(orderOwnerAddress, changeWavelets, unit)]))
181- }
168+@Callable(i)
169+func migrateOrderPrice () = {
170+ let lastMigratedOrderKey = "migration_6decimals_lastMigratedOrderId"
171+ let finalStatus = "migrated_to_6decimals"
172+ let lastMigratedOrderId = getStringByKey(lastMigratedOrderKey)
173+ let currentOrderId = if ((lastMigratedOrderId == ""))
174+ then firstOrder
175+ else getNextOrder(lastMigratedOrderId)
176+ let orderMigrationStatusKey = ("migration_6decimals_status_" + currentOrderId)
177+ let orderMigrationStatus = getStringByKey(orderMigrationStatusKey)
178+ if ((orderMigrationStatus == finalStatus))
179+ then throw("error: order has been already migrated")
180+ else if ((currentOrderId == ""))
181+ then throw("status ok: all orders have been already migrated or nothing to migrate")
182+ else WriteSet([DataEntry(orderMigrationStatusKey, finalStatus), DataEntry(lastMigratedOrderKey, currentOrderId), DataEntry(getOrderPriceKey(currentOrderId), (getOrderPrice(currentOrderId) * 10000)), DataEntry(getDebugOrderPriceKey(currentOrderId), (getDebugOrderPrice(currentOrderId) * 10000))])
182183 }
183-
184-
185-func internalAddBuyBondOrder (currentMaxRoi,roi,price,prevOrder,inv,instantOrder) = {
186- let pmt = extract(inv.payment)
187- let newOrderId = toBase58String(inv.transactionId)
188- if (isBlocked)
189- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
190- else if ((MINORDERTOTAL > pmt.amount))
191- then throw(("min order total equals " + toString(MINORDERTOTAL)))
192- else if ((roi > MAXROI))
193- then throw("max setOrder ROI is 95%")
194- else if (if ((deficit > 0))
195- then (0 > roi)
196- else false)
197- then throw("can't place order with negative roi during deficit")
198- else if ((-(MAXROI) > roi))
199- then throw("min setOrder ROI is -95%")
200- else if ((roi == 0))
201- then throw("roi should not be equal to 0")
202- else if (isDefined(pmt.assetId))
203- then throw("can use waves only")
204- else if ((getOrderOwner(newOrderId) != ""))
205- then throw("order exists")
206- else if (if ((prevOrder != ""))
207- then (getOrderStatus(prevOrder) != NEW)
208- else false)
209- then throw("prev order status is not new")
210- else {
211- let isNewOrderAtFirstPosition = (prevOrder == "")
212- let owner = toString(inv.caller)
213- let nextOrder = if (isNewOrderAtFirstPosition)
214- then firstOrder
215- else getNextOrder(prevOrder)
216- let nextOrderRoi = getNumberByKey(getRoiByOrderIdKey(nextOrder))
217- let isNextOrderError = if (if ((nextOrder != ""))
218- then (roi >= nextOrderRoi)
219- else false)
220- then true
221- else false
222- let prevOrderRoi = getNumberByKey(getRoiByOrderIdKey(prevOrder))
223- let isPrevOrderError = if (if ((prevOrder != ""))
224- then (prevOrderRoi > roi)
225- else false)
226- then true
227- else false
228- if (if (isNextOrderError)
229- then true
230- else isPrevOrderError)
231- then throw(((("invalid order isPrevOrderError:" + toString(isPrevOrderError)) + " isNextOrderError:") + toString(isNextOrderError)))
232- else if (if (isNewOrderAtFirstPosition)
233- then (currentMaxRoi >= roi)
234- else false)
235- then internalSellBond(currentMaxRoi, newOrderId, nextOrder, 0, roi, price, pmt.amount, inv.caller, true)
236- else if (instantOrder)
237- then throw("Instant order couldn't be added into waiting queue")
238- else WriteSet([DataEntry(getPrevOrderKey(newOrderId), prevOrder), DataEntry(getNextOrderKey(newOrderId), nextOrder), DataEntry(getNextOrderKey(prevOrder), if ((prevOrder == ""))
239- then ""
240- else newOrderId), DataEntry(getPrevOrderKey(nextOrder), if ((nextOrder == ""))
241- then ""
242- else newOrderId), DataEntry(FirstOrderKey, if (if ((firstOrder == ""))
243- then true
244- else (firstOrder == nextOrder))
245- then newOrderId
246- else firstOrder), orderData(newOrderId, pmt.amount, owner, NEW, roi, price)])
247- }
248- }
249-
250-
251-@Callable(i)
252-func instantBuyNsbtOrFail (noLessThenRoi) = {
253- let pmt = extract(i.payment)
254- let currentMaxRoi = fraction(deficit, 100, neutrinoSupply)
255- if ((noLessThenRoi > currentMaxRoi))
256- then throw(((("Current maxRoi=" + toString(currentMaxRoi)) + " is less then passed parameter noLessThenRoi=") + toString(noLessThenRoi)))
257- else {
258- let priceWavesByBondCents = fraction((100 + currentMaxRoi), currentPrice, 100)
259- internalAddBuyBondOrder(currentMaxRoi, currentMaxRoi, ((100 * 100) / priceWavesByBondCents), "", i, true)
260- }
261- }
262-
263-
264-
265-@Callable(i)
266-func addBuyBondOrder (price,prevOrder) = {
267- let pmt = extract(i.payment)
268- let priceWavesByBondCents = fraction(100, 100, price)
269- if ((0 >= price))
270- then throw("price less zero")
271- else internalAddBuyBondOrder(fraction(deficit, 100, neutrinoSupply), fraction((priceWavesByBondCents - currentPrice), 100, currentPrice), price, prevOrder, i, false)
272- }
273-
274-
275-
276-@Callable(i)
277-func cancelOrder (orderId) = {
278- let owner = getOrderOwner(orderId)
279- let amount = (getOrderTotal(orderId) - getOrderFilledTotal(orderId))
280- let caller = toString(i.caller)
281- let nextOrder = getNextOrder(orderId)
282- let prevOrder = getPrevOrder(orderId)
283- if (isBlocked)
284- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
285- else if ((owner != caller))
286- then throw("permission denied")
287- else if ((getOrderStatus(orderId) != NEW))
288- then throw("invalid order status")
289- else ScriptResult(WriteSet([DataEntry(FirstOrderKey, if ((firstOrder == orderId))
290- then nextOrder
291- else firstOrder), DataEntry(getNextOrderKey(prevOrder), nextOrder), DataEntry(getPrevOrderKey(nextOrder), prevOrder), DataEntry(getOrderStatusKey(orderId), CANCELED)]), TransferSet([ScriptTransfer(i.caller, amount, unit)]))
292- }
293-
294-
295-
296-@Callable(i)
297-func sellBond () = if (isBlocked)
298- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
299- else if ((firstOrder == ""))
300- then throw("empty orderbook")
301- else {
302- let nextOrder = getNextOrder(firstOrder)
303- let filledTotal = getOrderFilledTotal(firstOrder)
304- let orderPrice = getOrderPrice(firstOrder)
305- let roi = getNumberByKey(getRoiByOrderIdKey(firstOrder))
306- let paymentWavelets = getOrderTotal(firstOrder)
307- let orderOwnerAddress = Address(fromBase58String(getOrderOwner(firstOrder)))
308- internalSellBond(fraction(deficit, 100, neutrinoSupply), firstOrder, nextOrder, filledTotal, roi, orderPrice, paymentWavelets, orderOwnerAddress, false)
309- }
310184
311185
312186 @Verifier(tx)
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 3 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 func getNumberByKey (key) = match getInteger(this, key) {
55 case a: Int =>
66 a
77 case _ =>
88 0
99 }
1010
1111
1212 func getStringByKey (key) = match getString(this, key) {
1313 case a: String =>
1414 a
1515 case _ =>
1616 ""
1717 }
1818
1919
2020 func getBoolByAddressAndKey (address,key) = match getBoolean(address, key) {
2121 case a: Boolean =>
2222 a
2323 case _ =>
2424 false
2525 }
2626
2727
2828 func getStringByAddressAndKey (address,key) = match getString(address, key) {
2929 case a: String =>
3030 a
3131 case _ =>
3232 ""
3333 }
3434
3535
3636 func getNumberByAddressAndKey (address,key) = match getInteger(address, key) {
3737 case a: Int =>
3838 a
3939 case _ =>
4040 0
4141 }
4242
4343
4444 let WAVELET = 100000000
4545
4646 let PAULI = 1000000
4747
4848 let MINORDERTOTAL = (10 * WAVELET)
4949
5050 let MAXROI = 95
5151
5252 let CANCELED = "canceled"
5353
5454 let NEW = "new"
5555
5656 let FILLED = "filled"
5757
5858 let NeutrinoContractKey = "neutrino_contract"
5959
6060 let PriceKey = "price"
6161
6262 let BondAssetIdKey = "bond_asset_id"
6363
6464 let NeutrinoAssetIdKey = "neutrino_asset_id"
6565
6666 let ControlContractKey = "control_contract"
6767
6868 let BalanceLockedkKey = "balance_lock_"
6969
7070 let WavesLockedBalanceKey = (BalanceLockedkKey + "waves")
7171
7272 let NeutrinoLockedBalanceKey = (BalanceLockedkKey + "neutrino")
7373
7474 let LiquidationContractKey = "liquidation_contract"
7575
7676 let FirstOrderKey = "order_first"
7777
7878 func getRoiByOrderIdKey (orderId) = ("debug_order_roi_" + orderId)
7979
8080
8181 func getOrderPriceKey (orderId) = ("order_price_" + orderId)
8282
8383
84+func getDebugOrderPriceKey (orderId) = ("debug_order_currentPrice_" + orderId)
85+
86+
8487 func getOrderTotalKey (orderId) = ("order_total_" + orderId)
8588
8689
8790 func getOrderOwnerKey (orderId) = ("order_owner_" + orderId)
8891
8992
9093 func getOrderHeightKey (orderId) = ("order_height_" + orderId)
9194
9295
9396 func getOrderStatusKey (orderId) = ("order_status_" + orderId)
9497
9598
9699 func getOrderFilledTotalKey (orderId) = ("order_filled_total_" + orderId)
97100
98101
99102 func getPrevOrderKey (orderId) = ("order_prev_" + orderId)
100103
101104
102105 func getNextOrderKey (orderId) = ("order_next_" + orderId)
103106
104107
105108 func convertNeutrinoToWaves (amount,price) = fraction(fraction(amount, 100, price), WAVELET, PAULI)
106109
107110
108111 func convertWavesToNeutrino (amount,price) = fraction(fraction(amount, price, 100), PAULI, WAVELET)
109112
110113
111114 func convertWavesToBond (amount,price) = convertWavesToNeutrino(amount, price)
112115
113116
114117 func convertBondToWaves (amount,price) = convertNeutrinoToWaves(amount, price)
115118
116119
117120 let neutrinoContract = Address(base58'3PC9BfRwJWWiw9AREE2B3eWzCks3CYtg4yo')
118121
119122 let controlContract = Address(base58'3P5Bfd58PPfNvBM2Hy8QfbcDqMeNtzg7KfP')
120123
121124 let liquidationContract = Address(base58'3P4PCxsJqMzQBALo8zANHtBDZRRquobHQp7')
122125
123126 let neutrinoAssetId = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
124127
125128 let bondAssetId = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
126129
127130 let isBlocked = getBoolByAddressAndKey(controlContract, "is_blocked")
128131
129132 let currentPrice = getNumberByAddressAndKey(controlContract, PriceKey)
130133
131134 let neutrinoLockedBalance = getNumberByAddressAndKey(neutrinoContract, NeutrinoLockedBalanceKey)
132135
133136 let reserve = (wavesBalance(neutrinoContract) - getNumberByAddressAndKey(neutrinoContract, WavesLockedBalanceKey))
134137
135138 let neutrinoSupply = (((neutrinoLockedBalance + extract(assetInfo(neutrinoAssetId)).quantity) - assetBalance(neutrinoContract, neutrinoAssetId)) - assetBalance(liquidationContract, neutrinoAssetId))
136139
137140 let deficit = (neutrinoSupply - convertWavesToNeutrino(reserve, currentPrice))
138141
139142 let firstOrder = getStringByKey(FirstOrderKey)
140143
141144 func getOrderPrice (id) = getNumberByKey(getOrderPriceKey(id))
142145
143146
147+func getDebugOrderPrice (id) = getNumberByKey(getDebugOrderPriceKey(id))
148+
149+
144150 func getOrderTotal (id) = getNumberByKey(getOrderTotalKey(id))
145151
146152
147153 func getOrderOwner (id) = getStringByKey(getOrderOwnerKey(id))
148154
149155
150156 func getOrderStatus (id) = getStringByKey(getOrderStatusKey(id))
151157
152158
153159 func getOrderFilledTotal (id) = getNumberByKey(getOrderFilledTotalKey(id))
154160
155161
156162 func getPrevOrder (id) = getStringByKey(getPrevOrderKey(id))
157163
158164
159165 func getNextOrder (id) = getStringByKey(getNextOrderKey(id))
160166
161167
162-func orderData (orderId,totalWavelets,owner,status,roi,price) = [DataEntry(getOrderPriceKey(orderId), price), DataEntry(getOrderTotalKey(orderId), totalWavelets), DataEntry(getOrderOwnerKey(orderId), owner), DataEntry(getOrderHeightKey(orderId), height), DataEntry(getOrderStatusKey(orderId), status), DataEntry(("debug_order_currentPrice_" + orderId), currentPrice), DataEntry(getRoiByOrderIdKey(orderId), roi)]
163-
164-
165-func internalSellBond (currentMaxRoi,pFirstOrder,pNextOrder,pFilledTotal,pRoi,pPrice,pPaymentWavelets,orderOwnerAddress,instantOrder) = {
166- let priceWavesByBondCents = fraction((100 + pRoi), currentPrice, 100)
167- let remainedTotal = (pPaymentWavelets - pFilledTotal)
168- let fillableOrderAmount = convertWavesToBond(remainedTotal, priceWavesByBondCents)
169- let totalOrderWaveletesRequired = convertBondToWaves(fillableOrderAmount, priceWavesByBondCents)
170- let nbTokensSellCondition = (currentMaxRoi >= pRoi)
171- if (!(nbTokensSellCondition))
172- then throw(("innapropriate roi: " + toString(pRoi)))
173- else if ((totalOrderWaveletesRequired == 0))
174- then throw("cannot fill order at the moment")
175- else {
176- let changeWavelets = (remainedTotal - totalOrderWaveletesRequired)
177- let writeSetData = if (instantOrder)
178- then [DataEntry(getOrderPriceKey(pFirstOrder), pPrice), DataEntry(getOrderTotalKey(pFirstOrder), pPaymentWavelets), DataEntry(getOrderOwnerKey(pFirstOrder), toBase58String(orderOwnerAddress.bytes)), DataEntry(getOrderHeightKey(pFirstOrder), height), DataEntry(getOrderFilledTotalKey(pFirstOrder), (pFilledTotal + totalOrderWaveletesRequired)), DataEntry(getOrderStatusKey(pFirstOrder), FILLED), DataEntry(("debug_order_currentPrice_" + pFirstOrder), currentPrice), DataEntry(getRoiByOrderIdKey(pFirstOrder), pRoi)]
179- else [DataEntry(getOrderFilledTotalKey(pFirstOrder), (pFilledTotal + totalOrderWaveletesRequired)), DataEntry(getOrderStatusKey(pFirstOrder), FILLED), DataEntry(getOrderTotalKey(pFirstOrder), pPaymentWavelets), DataEntry(getPrevOrderKey(pNextOrder), ""), DataEntry(FirstOrderKey, pNextOrder)]
180- ScriptResult(WriteSet(writeSetData), TransferSet([ScriptTransfer(orderOwnerAddress, fillableOrderAmount, bondAssetId), ScriptTransfer(neutrinoContract, totalOrderWaveletesRequired, unit), ScriptTransfer(orderOwnerAddress, changeWavelets, unit)]))
181- }
168+@Callable(i)
169+func migrateOrderPrice () = {
170+ let lastMigratedOrderKey = "migration_6decimals_lastMigratedOrderId"
171+ let finalStatus = "migrated_to_6decimals"
172+ let lastMigratedOrderId = getStringByKey(lastMigratedOrderKey)
173+ let currentOrderId = if ((lastMigratedOrderId == ""))
174+ then firstOrder
175+ else getNextOrder(lastMigratedOrderId)
176+ let orderMigrationStatusKey = ("migration_6decimals_status_" + currentOrderId)
177+ let orderMigrationStatus = getStringByKey(orderMigrationStatusKey)
178+ if ((orderMigrationStatus == finalStatus))
179+ then throw("error: order has been already migrated")
180+ else if ((currentOrderId == ""))
181+ then throw("status ok: all orders have been already migrated or nothing to migrate")
182+ else WriteSet([DataEntry(orderMigrationStatusKey, finalStatus), DataEntry(lastMigratedOrderKey, currentOrderId), DataEntry(getOrderPriceKey(currentOrderId), (getOrderPrice(currentOrderId) * 10000)), DataEntry(getDebugOrderPriceKey(currentOrderId), (getDebugOrderPrice(currentOrderId) * 10000))])
182183 }
183-
184-
185-func internalAddBuyBondOrder (currentMaxRoi,roi,price,prevOrder,inv,instantOrder) = {
186- let pmt = extract(inv.payment)
187- let newOrderId = toBase58String(inv.transactionId)
188- if (isBlocked)
189- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
190- else if ((MINORDERTOTAL > pmt.amount))
191- then throw(("min order total equals " + toString(MINORDERTOTAL)))
192- else if ((roi > MAXROI))
193- then throw("max setOrder ROI is 95%")
194- else if (if ((deficit > 0))
195- then (0 > roi)
196- else false)
197- then throw("can't place order with negative roi during deficit")
198- else if ((-(MAXROI) > roi))
199- then throw("min setOrder ROI is -95%")
200- else if ((roi == 0))
201- then throw("roi should not be equal to 0")
202- else if (isDefined(pmt.assetId))
203- then throw("can use waves only")
204- else if ((getOrderOwner(newOrderId) != ""))
205- then throw("order exists")
206- else if (if ((prevOrder != ""))
207- then (getOrderStatus(prevOrder) != NEW)
208- else false)
209- then throw("prev order status is not new")
210- else {
211- let isNewOrderAtFirstPosition = (prevOrder == "")
212- let owner = toString(inv.caller)
213- let nextOrder = if (isNewOrderAtFirstPosition)
214- then firstOrder
215- else getNextOrder(prevOrder)
216- let nextOrderRoi = getNumberByKey(getRoiByOrderIdKey(nextOrder))
217- let isNextOrderError = if (if ((nextOrder != ""))
218- then (roi >= nextOrderRoi)
219- else false)
220- then true
221- else false
222- let prevOrderRoi = getNumberByKey(getRoiByOrderIdKey(prevOrder))
223- let isPrevOrderError = if (if ((prevOrder != ""))
224- then (prevOrderRoi > roi)
225- else false)
226- then true
227- else false
228- if (if (isNextOrderError)
229- then true
230- else isPrevOrderError)
231- then throw(((("invalid order isPrevOrderError:" + toString(isPrevOrderError)) + " isNextOrderError:") + toString(isNextOrderError)))
232- else if (if (isNewOrderAtFirstPosition)
233- then (currentMaxRoi >= roi)
234- else false)
235- then internalSellBond(currentMaxRoi, newOrderId, nextOrder, 0, roi, price, pmt.amount, inv.caller, true)
236- else if (instantOrder)
237- then throw("Instant order couldn't be added into waiting queue")
238- else WriteSet([DataEntry(getPrevOrderKey(newOrderId), prevOrder), DataEntry(getNextOrderKey(newOrderId), nextOrder), DataEntry(getNextOrderKey(prevOrder), if ((prevOrder == ""))
239- then ""
240- else newOrderId), DataEntry(getPrevOrderKey(nextOrder), if ((nextOrder == ""))
241- then ""
242- else newOrderId), DataEntry(FirstOrderKey, if (if ((firstOrder == ""))
243- then true
244- else (firstOrder == nextOrder))
245- then newOrderId
246- else firstOrder), orderData(newOrderId, pmt.amount, owner, NEW, roi, price)])
247- }
248- }
249-
250-
251-@Callable(i)
252-func instantBuyNsbtOrFail (noLessThenRoi) = {
253- let pmt = extract(i.payment)
254- let currentMaxRoi = fraction(deficit, 100, neutrinoSupply)
255- if ((noLessThenRoi > currentMaxRoi))
256- then throw(((("Current maxRoi=" + toString(currentMaxRoi)) + " is less then passed parameter noLessThenRoi=") + toString(noLessThenRoi)))
257- else {
258- let priceWavesByBondCents = fraction((100 + currentMaxRoi), currentPrice, 100)
259- internalAddBuyBondOrder(currentMaxRoi, currentMaxRoi, ((100 * 100) / priceWavesByBondCents), "", i, true)
260- }
261- }
262-
263-
264-
265-@Callable(i)
266-func addBuyBondOrder (price,prevOrder) = {
267- let pmt = extract(i.payment)
268- let priceWavesByBondCents = fraction(100, 100, price)
269- if ((0 >= price))
270- then throw("price less zero")
271- else internalAddBuyBondOrder(fraction(deficit, 100, neutrinoSupply), fraction((priceWavesByBondCents - currentPrice), 100, currentPrice), price, prevOrder, i, false)
272- }
273-
274-
275-
276-@Callable(i)
277-func cancelOrder (orderId) = {
278- let owner = getOrderOwner(orderId)
279- let amount = (getOrderTotal(orderId) - getOrderFilledTotal(orderId))
280- let caller = toString(i.caller)
281- let nextOrder = getNextOrder(orderId)
282- let prevOrder = getPrevOrder(orderId)
283- if (isBlocked)
284- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
285- else if ((owner != caller))
286- then throw("permission denied")
287- else if ((getOrderStatus(orderId) != NEW))
288- then throw("invalid order status")
289- else ScriptResult(WriteSet([DataEntry(FirstOrderKey, if ((firstOrder == orderId))
290- then nextOrder
291- else firstOrder), DataEntry(getNextOrderKey(prevOrder), nextOrder), DataEntry(getPrevOrderKey(nextOrder), prevOrder), DataEntry(getOrderStatusKey(orderId), CANCELED)]), TransferSet([ScriptTransfer(i.caller, amount, unit)]))
292- }
293-
294-
295-
296-@Callable(i)
297-func sellBond () = if (isBlocked)
298- then throw("contract is blocked by EMERGENCY SHUTDOWN actions until reactivation by emergency oracles")
299- else if ((firstOrder == ""))
300- then throw("empty orderbook")
301- else {
302- let nextOrder = getNextOrder(firstOrder)
303- let filledTotal = getOrderFilledTotal(firstOrder)
304- let orderPrice = getOrderPrice(firstOrder)
305- let roi = getNumberByKey(getRoiByOrderIdKey(firstOrder))
306- let paymentWavelets = getOrderTotal(firstOrder)
307- let orderOwnerAddress = Address(fromBase58String(getOrderOwner(firstOrder)))
308- internalSellBond(fraction(deficit, 100, neutrinoSupply), firstOrder, nextOrder, filledTotal, roi, orderPrice, paymentWavelets, orderOwnerAddress, false)
309- }
310184
311185
312186 @Verifier(tx)
313187 func verify () = {
314188 let pubKeyAdminsList = ["BLEoguzPVKVTfXxxT3W7Rqf8aUm2ggC9Vemd2MQawM2G", "FWVffYr2ALmHMejZm3WqeLz6Sdym3gLFGtJn4KTwyU5x", "3Wh2LaWcb5gg7K2pPcW3Ep6EAuRBzYkAgrdpt43jTDFa", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
315189 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
316190 then 1
317191 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
318192 then 1
319193 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[2], fromBase58String(pubKeyAdminsList[2])))
320194 then 1
321195 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
322196 then 2
323197 else 0))
324198 (count >= 3)
325199 }
326200

github/deemru/w8io/3ef1775 
60.01 ms