tx · BRTJwTHcVKR7RPDQ6xsKZmT8aWmzE93r5CA68kTTysan

3Mzbx34Ex5rRbn39vUHx9kiwh3nneF8S5we:  -0.02000000 Waves

2022.02.24 16:06 [1937765] smart account 3Mzbx34Ex5rRbn39vUHx9kiwh3nneF8S5we > SELF 0.00000000 Waves

{ "type": 13, "id": "BRTJwTHcVKR7RPDQ6xsKZmT8aWmzE93r5CA68kTTysan", "fee": 2000000, "feeAssetId": null, "timestamp": 1645707987651, "version": 1, "sender": "3Mzbx34Ex5rRbn39vUHx9kiwh3nneF8S5we", "senderPublicKey": "BxuvXsg8sFXf7MNECYrdsEgrMMMkWVKKhbcFdoormLWF", "proofs": [ "5UpHneXMKnN9ogwMdsWUaJ9dmZYKFMCsyzdjAE5b5hMP6ZHfUgCsudA4mBy26upVJ7BsyDbQo3BrqXAWPYGoKNQ7", "BaquSQ6sc1LNrdMdXBtu9FGm9skmriY2Xhxc4x8MXWu3KcuoLSc3JBvTbqhFgAprY48jZJgvGHQw6sjbKgodHLX", "5crEtXkRwjZj9jhytPr3NsNaBWvXyodC5PudDz8BwT4wmVfv8SmMUeLkCwRGdFo4wbwPJd4NgybAe2ZypKSnqrD2" ], "script": "base64:AAIFAAAAAAAAABkIAhIGCgQIAQgIEgASABIFCgMIAQgSABIAAAAAcgEAAAAOZ2V0TnVtYmVyQnlLZXkAAAABAAAAA2tleQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABBoAAAACBQAAAAR0aGlzBQAAAANrZXkAAAAAAAAAAAABAAAADmdldFN0cmluZ0J5S2V5AAAAAQAAAANrZXkJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQdAAAAAgUAAAAEdGhpcwUAAAADa2V5AgAAAAABAAAADGdldEJvb2xCeUtleQAAAAEAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEGwAAAAIFAAAABHRoaXMFAAAAA2tleQcBAAAAGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQAAAAIAAAAHYWRkcmVzcwAAAANrZXkJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgkBAAAAEUBleHRyTmF0aXZlKDEwNjIpAAAAAQUAAAAHYWRkcmVzcwUAAAADa2V5AAAAAAAAAAAAAQAAABhnZXRTdHJpbmdCeUFkZHJlc3NBbmRLZXkAAAACAAAAB2FkZHJlc3MAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEHQAAAAIJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEFAAAAB2FkZHJlc3MFAAAAA2tleQIAAAAAAQAAABZnZXRCb29sQnlBZGRyZXNzQW5kS2V5AAAAAgAAAAdhZGRyZXNzAAAAA2tleQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABBsAAAACCQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABBQAAAAdhZGRyZXNzBQAAAANrZXkHAQAAAAlhc0FueUxpc3QAAAABAAAAA3ZhbAQAAAAHJG1hdGNoMAUAAAADdmFsAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAAlMaXN0W0FueV0EAAAACnZhbEFueUx5c3QFAAAAByRtYXRjaDAFAAAACnZhbEFueUx5c3QJAAACAAAAAQIAAAAbZmFpbCB0byBjYXN0IGludG8gTGlzdFtBbnldAQAAAAhhc1N0cmluZwAAAAEAAAADdmFsBAAAAAckbWF0Y2gwBQAAAAN2YWwDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABlN0cmluZwQAAAAGdmFsU3RyBQAAAAckbWF0Y2gwBQAAAAZ2YWxTdHIJAAACAAAAAQIAAAAYZmFpbCB0byBjYXN0IGludG8gU3RyaW5nAQAAAAVhc0ludAAAAAEAAAADdmFsBAAAAAckbWF0Y2gwBQAAAAN2YWwDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAAA0ludAQAAAAGdmFsSW50BQAAAAckbWF0Y2gwBQAAAAZ2YWxJbnQJAAACAAAAAQIAAAAVZmFpbCB0byBjYXN0IGludG8gSW50AAAAABBwdWJLZXlBZG1pbnNMaXN0CQAETAAAAAICAAAALEdYeG1aYUNpZ1BnSnNUNmQxY0xyOG5VWEE2NW5HWDhWN0Z1QXp6clk4d1FFCQAETAAAAAICAAAALDN4dW14NGhyUHBlZW9vNzlITEdiWkNnZno5NUwzOVpGZHh4d1dlU1VDMlB1CQAETAAAAAICAAAALEQ0Q3F0SzVmQnNuelJvMzNaTWhyTTduaUxzZURIMnVSb0s2d2VCcTJKUTNBCQAETAAAAAICAAAALDVXUlhGU2p3Y1RiTmZLY0pzOFpxWG1TU1dZc1NWSlV0TXZNcVpqNWhINE5jBQAAAANuaWwAAAAAA1NFUAIAAAACX18AAAAAB1dBVkVMRVQAAAAAAAX14QAAAAAABVBBVUxJAAAAAAAAD0JAAAAAAAhQUklDRUxFVAAAAAAAAA9CQAAAAAAOREVGQVVMVFNXQVBGRUUAAAAAAAAATiAAAAAAD1RXRU5UWUZPVVJIT1VSUwAAAAAAAAAFoAAAAAAMSWR4TmV0QW1vdW50AAAAAAAAAAAAAAAAAAxJZHhGZWVBbW91bnQAAAAAAAAAAAEAAAAADklkeEdyb3NzQW1vdW50AAAAAAAAAAACAAAAABJOZXV0cmlub0Fzc2V0SWRLZXkCAAAAEW5ldXRyaW5vX2Fzc2V0X2lkAAAAAA5Cb25kQXNzZXRJZEtleQIAAAANYm9uZF9hc3NldF9pZAAAAAASQXVjdGlvbkNvbnRyYWN0S2V5AgAAABBhdWN0aW9uX2NvbnRyYWN0AAAAABZMaXF1aWRhdGlvbkNvbnRyYWN0S2V5AgAAABRsaXF1aWRhdGlvbl9jb250cmFjdAAAAAAOUlBEQ29udHJhY3RLZXkCAAAADHJwZF9jb250cmFjdAAAAAARQ29udG9sQ29udHJhY3RLZXkCAAAAEGNvbnRyb2xfY29udHJhY3QAAAAAD1Jlc3RDb250cmFjdEtleQIAAAANcmVzdF9jb250cmFjdAAAAAAPTWF0aENvbnRyYWN0S2V5AgAAAA1tYXRoX2NvbnRyYWN0AAAAABtCYWxhbmNlV2F2ZXNMb2NrSW50ZXJ2YWxLZXkCAAAAG2JhbGFuY2Vfd2F2ZXNfbG9ja19pbnRlcnZhbAAAAAAeQmFsYW5jZU5ldXRyaW5vTG9ja0ludGVydmFsS2V5AgAAAB5iYWxhbmNlX25ldXRyaW5vX2xvY2tfaW50ZXJ2YWwAAAAAFU1pbldhdmVzU3dhcEFtb3VudEtleQIAAAAVbWluX3dhdmVzX3N3YXBfYW1vdW50AAAAABhNaW5OZXV0cmlub1N3YXBBbW91bnRLZXkCAAAAGG1pbl9uZXV0cmlub19zd2FwX2Ftb3VudAAAAAAbTm9kZU9yYWNsZVByb3ZpZGVyUHViS2V5S2V5AgAAABRub2RlX29yYWNsZV9wcm92aWRlcgAAAAAVTmV1dHJpbm9PdXRGZWVQYXJ0S2V5AgAAABhuZXV0cmlub091dF9zd2FwX2ZlZVBhcnQAAAAAEldhdmVzT3V0RmVlUGFydEtleQIAAAAVd2F2ZXNPdXRfc3dhcF9mZWVQYXJ0AAAAABVGZWVzTWFuYWdlckFkZHJlc3NLZXkCAAAAFGZlZXNfbWFuYWdlcl9hZGRyZXNzAQAAABlrZXlRdWlja1N3YXBMaW1pdER1cmF0aW9uAAAAAAIAAAAaJXNfX3F1aWNrU3dhcExpbWl0RHVyYXRpb24AAAAACFByaWNlS2V5AgAAAAVwcmljZQAAAAANUHJpY2VJbmRleEtleQIAAAALcHJpY2VfaW5kZXgAAAAADElzQmxvY2tlZEtleQIAAAAKaXNfYmxvY2tlZAEAAAASZ2V0UHJpY2VIaXN0b3J5S2V5AAAAAQAAAAVibG9jawkAASwAAAACCQABLAAAAAIFAAAACFByaWNlS2V5AgAAAAFfCQABpAAAAAEFAAAABWJsb2NrAQAAABhnZXRIZWlnaHRQcmljZUJ5SW5kZXhLZXkAAAABAAAABWluZGV4CQABLAAAAAIJAAEsAAAAAgUAAAANUHJpY2VJbmRleEtleQIAAAABXwkAAaQAAAABBQAAAAVpbmRleAEAAAAVZ2V0U3Rha2luZ05vZGVCeUluZGV4AAAAAQAAAANpZHgJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEJAAS5AAAAAgkABEwAAAACAgAAAAYlcyVkJXMJAARMAAAAAgIAAAAFbGVhc2UJAARMAAAAAgkAAaQAAAABBQAAAANpZHgJAARMAAAAAgIAAAALbm9kZUFkZHJlc3MFAAAAA25pbAUAAAADU0VQAQAAABxnZXRTdGFraW5nTm9kZUFkZHJlc3NCeUluZGV4AAAAAQAAAANpZHgJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEJAQAAABVnZXRTdGFraW5nTm9kZUJ5SW5kZXgAAAABBQAAAANpZHgBAAAAH2dldFJlc2VydmVkQW1vdW50Rm9yU3BvbnNvcnNoaXAAAAAACQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAABHRoaXMJAAS5AAAAAgkABEwAAAACAgAAAAQlcyVzCQAETAAAAAICAAAABWxlYXNlCQAETAAAAAICAAAAF3Nwb25zb3JzaGlwV2F2ZXNSZXNlcnZlBQAAAANuaWwFAAAAA1NFUAkAAGgAAAACAAAAAAAAAAPoBQAAAAdXQVZFTEVUAQAAABhnZXRCYWxhbmNlVW5sb2NrQmxvY2tLZXkAAAABAAAABW93bmVyCQABLAAAAAICAAAAFWJhbGFuY2VfdW5sb2NrX2Jsb2NrXwUAAAAFb3duZXIBAAAADWdldExlYXNlSWRLZXkAAAABAAAACW5vZGVJbmRleAkABLkAAAACCQAETAAAAAICAAAABiVzJWQlcwkABEwAAAACAgAAAAVsZWFzZQkABEwAAAACCQABpAAAAAEFAAAACW5vZGVJbmRleAkABEwAAAACAgAAAAJpZAUAAAADbmlsBQAAAANTRVABAAAAEWdldExlYXNlQW1vdW50S2V5AAAAAQAAAAlub2RlSW5kZXgJAAS5AAAAAgkABEwAAAACAgAAAAYlcyVkJXMJAARMAAAAAgIAAAAFbGVhc2UJAARMAAAAAgkAAaQAAAABBQAAAAlub2RlSW5kZXgJAARMAAAAAgIAAAAGYW1vdW50BQAAAANuaWwFAAAAA1NFUAEAAAAQbWluU3dhcEFtb3VudEtFWQAAAAEAAAAIc3dhcFR5cGUJAAEsAAAAAgkAASwAAAACAgAAAARtaW5fBQAAAAhzd2FwVHlwZQIAAAAMX3N3YXBfYW1vdW50AQAAAA50b3RhbExvY2tlZEtFWQAAAAEAAAAIc3dhcFR5cGUJAAEsAAAAAgIAAAANYmFsYW5jZV9sb2NrXwUAAAAIc3dhcFR5cGUBAAAAFHRvdGFsTG9ja2VkQnlVc2VyS0VZAAAAAgAAAAhzd2FwVHlwZQAAAAVvd25lcgkABLkAAAACCQAETAAAAAICAAAADGJhbGFuY2VfbG9jawkABEwAAAACBQAAAAhzd2FwVHlwZQkABEwAAAACBQAAAAVvd25lcgUAAAADbmlsAgAAAAFfAQAAABZiYWxhbmNlTG9ja0ludGVydmFsS0VZAAAAAQAAAAhzd2FwVHlwZQkAASwAAAACCQABLAAAAAICAAAACGJhbGFuY2VfBQAAAAhzd2FwVHlwZQIAAAAOX2xvY2tfaW50ZXJ2YWwBAAAAGW1pbkJhbGFuY2VMb2NrSW50ZXJ2YWxLRVkAAAABAAAACHN3YXBUeXBlCQABLAAAAAIJAAEsAAAAAgIAAAAIYmFsYW5jZV8FAAAACHN3YXBUeXBlAgAAABZfbG9ja19pbnRlcnZhbF9taW5pbXVtAQAAABpub2RlQmFsYW5jZUxvY2tJbnRlcnZhbEtFWQAAAAACAAAAGmJhbGFuY2Vfbm9kZV9sb2NrX2ludGVydmFsAQAAAA1vdXRGZWVQYXJ0S0VZAAAAAQAAAAhzd2FwVHlwZQkAASwAAAACBQAAAAhzd2FwVHlwZQIAAAAQT3V0X3N3YXBfZmVlUGFydAEAAAARbWluU3dhcEFtb3VudFJFQUQAAAABAAAACHN3YXBUeXBlCQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAABHRoaXMJAQAAABBtaW5Td2FwQW1vdW50S0VZAAAAAQUAAAAIc3dhcFR5cGUAAAAAAAAAAAABAAAAD3RvdGFsTG9ja2VkUkVBRAAAAAEAAAAIc3dhcFR5cGUJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAEdGhpcwkBAAAADnRvdGFsTG9ja2VkS0VZAAAAAQUAAAAIc3dhcFR5cGUAAAAAAAAAAAABAAAAFXRvdGFsTG9ja2VkQnlVc2VyUkVBRAAAAAIAAAAIc3dhcFR5cGUAAAAFb3duZXIJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAEdGhpcwkBAAAAFHRvdGFsTG9ja2VkQnlVc2VyS0VZAAAAAgUAAAAIc3dhcFR5cGUFAAAABW93bmVyAAAAAAAAAAAAAQAAABdiYWxhbmNlTG9ja0ludGVydmFsUkVBRAAAAAEAAAAIc3dhcFR5cGUJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAEdGhpcwkBAAAAFmJhbGFuY2VMb2NrSW50ZXJ2YWxLRVkAAAABBQAAAAhzd2FwVHlwZQAAAAAAAAAFoAEAAAAabWluQmFsYW5jZUxvY2tJbnRlcnZhbFJFQUQAAAABAAAACHN3YXBUeXBlCQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAABHRoaXMJAQAAABltaW5CYWxhbmNlTG9ja0ludGVydmFsS0VZAAAAAQUAAAAIc3dhcFR5cGUAAAAAAAAAADwBAAAAG25vZGVCYWxhbmNlTG9ja0ludGVydmFsUkVBRAAAAAAJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAEdGhpcwkBAAAAGm5vZGVCYWxhbmNlTG9ja0ludGVydmFsS0VZAAAAAAAAAAAAAAAAAQEAAAAda2V5UXVpY2tTd2FwVXNlclNwZW50SW5QZXJpb2QAAAABAAAAC3VzZXJBZGRyZXNzCQAEuQAAAAIJAARMAAAAAgIAAAAEJXMlcwkABEwAAAACAgAAABpxdWlja1N3YXBVc2VyU3BlbnRJblBlcmlvZAkABEwAAAACBQAAAAt1c2VyQWRkcmVzcwUAAAADbmlsBQAAAANTRVABAAAAGmtleVVzZXJMYXN0UXVpY2tTd2FwSGVpZ2h0AAAAAQAAAAt1c2VyQWRkcmVzcwkABLkAAAACCQAETAAAAAICAAAABCVzJXMJAARMAAAAAgIAAAAXdXNlckxhc3RRdWlja1N3YXBIZWlnaHQJAARMAAAAAgUAAAALdXNlckFkZHJlc3MFAAAAA25pbAUAAAADU0VQAQAAABVmZWVNYW5hZ2VyQWRkcmVzc1JFQUQAAAAACQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQmAAAAAQkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEHQAAAAIFAAAABHRoaXMFAAAAFUZlZXNNYW5hZ2VyQWRkcmVzc0tleQkAASwAAAACBQAAABVGZWVzTWFuYWdlckFkZHJlc3NLZXkCAAAAESBpcyBub3Qgc3BlY2lmaWVkCQABLAAAAAIFAAAAFUZlZXNNYW5hZ2VyQWRkcmVzc0tleQIAAAAXIGludmFsaWQgYWRkcmVzcyBmb3JtYXQBAAAAFmNvbnZlcnROZXV0cmlub1RvV2F2ZXMAAAACAAAABmFtb3VudAAAAAVwcmljZQkAAGsAAAADCQAAawAAAAMFAAAABmFtb3VudAUAAAAIUFJJQ0VMRVQFAAAABXByaWNlBQAAAAdXQVZFTEVUBQAAAAVQQVVMSQEAAAAWY29udmVydFdhdmVzVG9OZXV0cmlubwAAAAIAAAAGYW1vdW50AAAABXByaWNlCQAAawAAAAMJAABrAAAAAwUAAAAGYW1vdW50BQAAAAVwcmljZQUAAAAIUFJJQ0VMRVQFAAAABVBBVUxJBQAAAAdXQVZFTEVUAQAAABJjb252ZXJ0V2F2ZXNUb0JvbmQAAAACAAAABmFtb3VudAAAAAVwcmljZQkBAAAAFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8AAAACBQAAAAZhbW91bnQFAAAABXByaWNlAQAAABZjb252ZXJ0SnNvbkFycmF5VG9MaXN0AAAAAQAAAAlqc29uQXJyYXkJAAS1AAAAAgUAAAAJanNvbkFycmF5AgAAAAEsAQAAABFtaW5Td2FwQW1vdW50RkFJTAAAAAIAAAAIc3dhcFR5cGUAAAANbWluU3dhcEFtb3VudAkAAAIAAAABCQABLAAAAAIJAAEsAAAAAgkAASwAAAACAgAAABhUaGUgc3BlY2lmaWVkIGFtb3VudCBpbiAFAAAACHN3YXBUeXBlAgAAACsgc3dhcCBpcyBsZXNzIHRoYW4gdGhlIHJlcXVpcmVkIG1pbmltdW0gb2YgCQABpAAAAAEFAAAADW1pblN3YXBBbW91bnQBAAAAFWVtZXJnZW5jeVNodXRkb3duRkFJTAAAAAAJAAACAAAAAQIAAABaY29udHJhY3QgaXMgYmxvY2tlZCBieSBFTUVSR0VOQ1kgU0hVVERPV04gYWN0aW9ucyB1bnRpbGwgcmVhY3RpdmF0aW9uIGJ5IGVtZXJnZW5jeSBvcmFjbGVzAQAAAA5wcmljZUluZGV4RkFJTAAAAAUAAAAFaW5kZXgAAAAKcHJpY2VJbmRleAAAAAtpbmRleEhlaWdodAAAAAx1bmxvY2tIZWlnaHQAAAAPcHJldkluZGV4SGVpZ2h0CQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAI2ludmFsaWQgcHJpY2UgaGlzdG9yeSBpbmRleDogaW5kZXg9CQABpAAAAAEFAAAABWluZGV4AgAAAAwgcHJpY2VJbmRleD0JAAGkAAAAAQUAAAAKcHJpY2VJbmRleAIAAAANIGluZGV4SGVpZ2h0PQkAAaQAAAABBQAAAAtpbmRleEhlaWdodAIAAAAOIHVubG9ja0hlaWdodD0JAAGkAAAAAQUAAAAMdW5sb2NrSGVpZ2h0AgAAABEgcHJldkluZGV4SGVpZ2h0PQkAAaQAAAABBQAAAA9wcmV2SW5kZXhIZWlnaHQAAAAAE2xpcXVpZGF0aW9uQ29udHJhY3QJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEFAAAAFkxpcXVpZGF0aW9uQ29udHJhY3RLZXkAAAAAD25ldXRyaW5vQXNzZXRJZAkAAlkAAAABCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABBQAAABJOZXV0cmlub0Fzc2V0SWRLZXkAAAAAD2F1Y3Rpb25Db250cmFjdAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAASQXVjdGlvbkNvbnRyYWN0S2V5AAAAAAtycGRDb250cmFjdAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAAOUlBEQ29udHJhY3RLZXkAAAAAD2NvbnRyb2xDb250cmFjdAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAARQ29udG9sQ29udHJhY3RLZXkAAAAADHJlc3RDb250cmFjdAkBAAAADmdldFN0cmluZ0J5S2V5AAAAAQUAAAAPUmVzdENvbnRyYWN0S2V5AAAAABNtYXRoQ29udHJhY3RBZGRyZXNzCQEAAAAOZ2V0U3RyaW5nQnlLZXkAAAABBQAAAA9NYXRoQ29udHJhY3RLZXkAAAAACnByaWNlSW5kZXgJAQAAABhnZXROdW1iZXJCeUFkZHJlc3NBbmRLZXkAAAACBQAAAA9jb250cm9sQ29udHJhY3QFAAAADVByaWNlSW5kZXhLZXkAAAAACWlzQmxvY2tlZAkBAAAAFmdldEJvb2xCeUFkZHJlc3NBbmRLZXkAAAACBQAAAA9jb250cm9sQ29udHJhY3QFAAAADElzQmxvY2tlZEtleQAAAAAYbm9kZU9yYWNsZVByb3ZpZGVyUHViS2V5CQACWQAAAAEJAQAAAA5nZXRTdHJpbmdCeUtleQAAAAEFAAAAG05vZGVPcmFjbGVQcm92aWRlclB1YktleUtleQAAAAALYm9uZEFzc2V0SWQJAAJZAAAAAQIAAAAsQkdoUFlxdVhTSzZVWWpuUExFbjk0Z0ZoZXRwNEp2Zjd0b2VrRVk0eDFWREgAAAAAFWRlcHJlY2F0ZWRCb25kQXNzZXRJZAkAAlkAAAABAgAAACw5NzVha1pCZm5NajUxM1U3TVphSEt6UXJtc0V4NWFFM3dkV0tUckhCaGJqRgAAAAAQbmV1dHJpbm9Db250cmFjdAUAAAAEdGhpcwAAAAAMbWF0aENvbnRyYWN0CQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABBQAAABNtYXRoQ29udHJhY3RBZGRyZXNzAAAAAAxjdXJyZW50UHJpY2UJAQAAABhnZXROdW1iZXJCeUFkZHJlc3NBbmRLZXkAAAACBQAAAA9jb250cm9sQ29udHJhY3QFAAAACFByaWNlS2V5AQAAABtjaGVja0lzVmFsaWRNaW5TcG9uc29yZWRGZWUAAAABAAAAAnR4BAAAAA5NSU5UUkFOU0ZFUkZFRQAAAAAAAAGGoAQAAAAWU3BvbnNvcmVkRmVlVXBwZXJCb3VuZAAAAAAAAAAD6AQAAAAPcmVhbE5ldXRyaW5vRmVlCQEAAAAWY29udmVydFdhdmVzVG9OZXV0cmlubwAAAAIFAAAADk1JTlRSQU5TRkVSRkVFBQAAAAxjdXJyZW50UHJpY2UEAAAADm1pbk5ldXRyaW5vRmVlCQAAaAAAAAIFAAAAD3JlYWxOZXV0cmlub0ZlZQAAAAAAAAAAAgQAAAAObWF4TmV1dHJpbm9GZWUJAABrAAAAAwUAAAAPcmVhbE5ldXRyaW5vRmVlBQAAABZTcG9uc29yZWRGZWVVcHBlckJvdW5kAAAAAAAAAABkBAAAAAhpbnB1dEZlZQkBAAAABXZhbHVlAAAAAQgFAAAAAnR4AAAAFG1pblNwb25zb3JlZEFzc2V0RmVlAwMJAABnAAAAAgUAAAAIaW5wdXRGZWUFAAAADm1pbk5ldXRyaW5vRmVlCQAAZwAAAAIFAAAADm1heE5ldXRyaW5vRmVlBQAAAAhpbnB1dEZlZQcJAAAAAAAAAggFAAAAAnR4AAAAB2Fzc2V0SWQFAAAAD25ldXRyaW5vQXNzZXRJZAcBAAAAD2dldFByaWNlSGlzdG9yeQAAAAEAAAAFYmxvY2sJAQAAABhnZXROdW1iZXJCeUFkZHJlc3NBbmRLZXkAAAACBQAAAA9jb250cm9sQ29udHJhY3QJAQAAABJnZXRQcmljZUhpc3RvcnlLZXkAAAABBQAAAAVibG9jawEAAAAVZ2V0SGVpZ2h0UHJpY2VCeUluZGV4AAAAAQAAAAVpbmRleAkBAAAAGGdldE51bWJlckJ5QWRkcmVzc0FuZEtleQAAAAIFAAAAD2NvbnRyb2xDb250cmFjdAkBAAAAGGdldEhlaWdodFByaWNlQnlJbmRleEtleQAAAAEFAAAABWluZGV4AAAAAAxzSWR4U3dhcFR5cGUAAAAAAAAAAAEAAAAACnNJZHhTdGF0dXMAAAAAAAAAAAIAAAAADHNJZHhJbkFtb3VudAAAAAAAAAAAAwAAAAAJc0lkeFByaWNlAAAAAAAAAAAEAAAAABBzSWR4T3V0TmV0QW1vdW50AAAAAAAAAAAFAAAAABBzSWR4T3V0RmVlQW1vdW50AAAAAAAAAAAGAAAAAA9zSWR4U3RhcnRIZWlnaHQAAAAAAAAAAAcAAAAAEnNJZHhTdGFydFRpbWVzdGFtcAAAAAAAAAAACAAAAAANc0lkeEVuZEhlaWdodAAAAAAAAAAACQAAAAAQc0lkeEVuZFRpbWVzdGFtcAAAAAAAAAAACgAAAAAUc0lkeFNlbGZVbmxvY2tIZWlnaHQAAAAAAAAAAAsAAAAAFHNJZHhSYW5kVW5sb2NrSGVpZ2h0AAAAAAAAAAAMAAAAAAlzSWR4SW5kZXgAAAAAAAAAAA0AAAAAEHNJZHhXaXRoZHJhd1R4SWQAAAAAAAAAAA4AAAAAC3NJZHhNaW5SYW5kAAAAAAAAAAAPAAAAAAtzSWR4TWF4UmFuZAAAAAAAAAAAEAEAAAAHc3dhcEtFWQAAAAIAAAALdXNlckFkZHJlc3MAAAAEdHhJZAkABLkAAAACCQAETAAAAAICAAAABCVzJXMJAARMAAAAAgUAAAALdXNlckFkZHJlc3MJAARMAAAAAgUAAAAEdHhJZAUAAAADbmlsBQAAAANTRVABAAAAC3N0clN3YXBEQVRBAAAAEAAAAAhzd2FwVHlwZQAAAAZzdGF0dXMAAAAIaW5BbW91bnQAAAAFcHJpY2UAAAAMb3V0TmV0QW1vdW50AAAADG91dEZlZUFtb3VudAAAAAtzdGFydEhlaWdodAAAAA5zdGFydFRpbWVzdGFtcAAAAAllbmRIZWlnaHQAAAAMZW5kVGltZXN0YW1wAAAAEHNlbGZVbmxvY2tIZWlnaHQAAAAQcmFuZFVubG9ja0hlaWdodAAAAAVpbmRleAAAAAx3aXRoZHJhd1R4SWQAAAAHcmFuZE1pbgAAAAdyYW5kTWF4CQAEuQAAAAIJAARMAAAAAgIAAAAcJXMlcyVkJWQlZCVkJWQlZCVkJWQlZCVkJWQlcwkABEwAAAACBQAAAAhzd2FwVHlwZQkABEwAAAACBQAAAAZzdGF0dXMJAARMAAAAAgUAAAAIaW5BbW91bnQJAARMAAAAAgUAAAAFcHJpY2UJAARMAAAAAgUAAAAMb3V0TmV0QW1vdW50CQAETAAAAAIFAAAADG91dEZlZUFtb3VudAkABEwAAAACBQAAAAtzdGFydEhlaWdodAkABEwAAAACBQAAAA5zdGFydFRpbWVzdGFtcAkABEwAAAACBQAAAAllbmRIZWlnaHQJAARMAAAAAgUAAAAMZW5kVGltZXN0YW1wCQAETAAAAAIFAAAAEHNlbGZVbmxvY2tIZWlnaHQJAARMAAAAAgUAAAAQcmFuZFVubG9ja0hlaWdodAkABEwAAAACBQAAAAVpbmRleAkABEwAAAACBQAAAAx3aXRoZHJhd1R4SWQJAARMAAAAAgUAAAAHcmFuZE1pbgkABEwAAAACBQAAAAdyYW5kTWF4BQAAAANuaWwFAAAAA1NFUAEAAAAPcGVuZGluZ1N3YXBEQVRBAAAAAwAAAAhzd2FwVHlwZQAAAA1pbkFzc2V0QW1vdW50AAAAEHNlbGZVbmxvY2tIZWlnaHQJAQAAAAtzdHJTd2FwREFUQQAAABAFAAAACHN3YXBUeXBlAgAAAAdQRU5ESU5HCQABpAAAAAEFAAAADWluQXNzZXRBbW91bnQCAAAAATACAAAAATACAAAAATAJAAGkAAAAAQUAAAAGaGVpZ2h0CQABpAAAAAEIBQAAAAlsYXN0QmxvY2sAAAAJdGltZXN0YW1wAgAAAAEwAgAAAAEwCQABpAAAAAEFAAAAEHNlbGZVbmxvY2tIZWlnaHQCAAAAATACAAAAATACAAAABE5VTEwCAAAAATACAAAAATABAAAADmZpbmlzaFN3YXBEQVRBAAAABwAAAAlkYXRhQXJyYXkAAAAFcHJpY2UAAAAMb3V0TmV0QW1vdW50AAAADG91dEZlZUFtb3VudAAAABByYW5kVW5sb2NrSGVpZ2h0AAAABWluZGV4AAAADHdpdGhkcmF3VHhJZAkBAAAAC3N0clN3YXBEQVRBAAAAEAkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAADHNJZHhTd2FwVHlwZQIAAAAIRklOSVNIRUQJAAGRAAAAAgUAAAAJZGF0YUFycmF5BQAAAAxzSWR4SW5BbW91bnQJAAGkAAAAAQUAAAAFcHJpY2UJAAGkAAAAAQUAAAAMb3V0TmV0QW1vdW50CQABpAAAAAEFAAAADG91dEZlZUFtb3VudAkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAAD3NJZHhTdGFydEhlaWdodAkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAAEnNJZHhTdGFydFRpbWVzdGFtcAkAAaQAAAABBQAAAAZoZWlnaHQJAAGkAAAAAQgFAAAACWxhc3RCbG9jawAAAAl0aW1lc3RhbXAJAAGRAAAAAgUAAAAJZGF0YUFycmF5BQAAABRzSWR4U2VsZlVubG9ja0hlaWdodAkAAaQAAAABBQAAABByYW5kVW5sb2NrSGVpZ2h0CQABpAAAAAEFAAAABWluZGV4BQAAAAx3aXRoZHJhd1R4SWQJAAGRAAAAAgUAAAAJZGF0YUFycmF5BQAAAAtzSWR4TWluUmFuZAkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAAC3NJZHhNYXhSYW5kAQAAABJzd2Fw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AA5uZXdMZWFzZUFtb3VudAkABE4AAAACBQAAAA51bmxlYXNlT3JFbXB0eQkABEwAAAACBQAAAAVsZWFzZQkABEwAAAACCQEAAAALQmluYXJ5RW50cnkAAAACBQAAAApsZWFzZUlkS2V5CQAEOQAAAAEFAAAABWxlYXNlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAARZ2V0TGVhc2VBbW91bnRLZXkAAAABBQAAAAlub2RlSW5kZXgFAAAADm5ld0xlYXNlQW1vdW50BQAAAANuaWwFAAAAA25pbAEAAAAOY29tbW9uV2l0aGRyYXcAAAAEAAAAB2FjY291bnQAAAAFaW5kZXgAAAAIc3dhcFR4SWQAAAABaQQAAAALdXNlckFkZHJlc3MJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEFAAAAB2FjY291bnQEAAAAEWZlZU1hbmFnZXJBZGRyZXNzCQEAAAAVZmVlTWFuYWdlckFkZHJlc3NSRUFEAAAAAAQAAAAJZGF0YUFycmF5CQEAAAASc3dhcERhdGFGYWlsT3JSRUFEAAAAAgUAAAAHYWNjb3VudAUAAAAIc3dhcFR4SWQEAAAAEHNlbGZVbmxvY2tIZWlnaHQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAAFHNJZHhTZWxmVW5sb2NrSGVpZ2h0BAAAAAhzd2FwVHlwZQkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAADHNJZHhTd2FwVHlwZQQAAAAIaW5BbW91bnQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAADHNJZHhJbkFtb3VudAQAAAAKc3dhcFN0YXR1cwkAAZEAAAACBQAAAAlkYXRhQXJyYXkFAAAACnNJZHhTdGF0dXMEAAAAC3N0YXJ0SGVpZ2h0CQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAJZGF0YUFycmF5BQAAAA9zSWR4U3RhcnRIZWlnaHQEAAAACm91dEZlZVBhcnQJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAEdGhpcwkBAAAADW91dEZlZVBhcnRLRVkAAAABBQAAAAhzd2FwVHlwZQUAAAAOREVGQVVMVFNXQVBGRUUEAAAAC3RvdGFsTG9ja2VkCQEAAAAPdG90YWxMb2NrZWRSRUFEAAAAAQUAAAAIc3dhcFR5cGUEAAAAEXRvdGFsTG9ja2VkQnlVc2VyCQEAAAAVdG90YWxMb2NrZWRCeVVzZXJSRUFEAAAAAgUAAAAIc3dhcFR5cGUFAAAAB2FjY291bnQEAAAADHVubG9ja0hlaWdodAkAAGQAAAACBQAAAAtzdGFydEhlaWdodAMJAAAAAAAAAgUAAAAIc3dhcFR5cGUCAAAABXdhdmVzCQEAAAARQGV4dHJOYXRpdmUoMTA1MCkAAAACBQAAAAR0aGlzBQAAABtCYWxhbmNlV2F2ZXNMb2NrSW50ZXJ2YWxLZXkJAQAAABFAZXh0ck5hdGl2ZSgxMDUwKQAAAAIFAAAABHRoaXMFAAAAHkJhbGFuY2VOZXV0cmlub0xvY2tJbnRlcnZhbEtleQQAAAALaW5kZXhIZWlnaHQJAQAAABVnZXRIZWlnaHRQcmljZUJ5SW5kZXgAAAABBQAAAAVpbmRleAQAAAAPcHJldkluZGV4SGVpZ2h0CQEAAAAVZ2V0SGVpZ2h0UHJpY2VCeUluZGV4AAAAAQkAAGUAAAACBQAAAAVpbmRleAAAAAAAAAAAAQQAAAAMcHJpY2VCeUluZGV4CQEAAAAPZ2V0UHJpY2VIaXN0b3J5AAAAAQUAAAALaW5kZXhIZWlnaHQEAAAAE291dEFtb3VudEdyb3NzVHVwbGUDCQAAAAAAAAIFAAAACHN3YXBUeXBlAgAAAAV3YXZlcwkABRQAAAACCQEAAAAWY29udmVydFdhdmVzVG9OZXV0cmlubwAAAAIFAAAACGluQW1vdW50BQAAAAxwcmljZUJ5SW5kZXgFAAAAD25ldXRyaW5vQXNzZXRJZAMJAAAAAAAAAgUAAAAIc3dhcFR5cGUCAAAACG5ldXRyaW5vCQAFFAAAAAIJAQAAABZjb252ZXJ0TmV1dHJpbm9Ub1dhdmVzAAAAAgUAAAAIaW5BbW91bnQFAAAADHByaWNlQnlJbmRleAUAAAAEdW5pdAkAAAIAAAABCQABLAAAAAICAAAAFlVuc3VwcG9ydGVkIHN3YXAgdHlwZSAFAAAACHN3YXBUeXBlBAAAAAxwYXlvdXRzQXJyYXkJAQAAAAlhcHBseUZlZXMAAAACCAUAAAATb3V0QW1vdW50R3Jvc3NUdXBsZQAAAAJfMQUAAAAKb3V0RmVlUGFydAQAAAAMb3V0TmV0QW1vdW50CQABkQAAAAIFAAAADHBheW91dHNBcnJheQUAAAAMSWR4TmV0QW1vdW50BAAAAAxvdXRGZWVBbW91bnQJAAGRAAAAAgUAAAAMcGF5b3V0c0FycmF5BQAAAAxJZHhGZWVBbW91bnQDBQAAAAlpc0Jsb2NrZWQJAQAAABVlbWVyZ2VuY3lTaHV0ZG93bkZBSUwAAAAAAwkBAAAAAiE9AAAAAgUAAAAKc3dhcFN0YXR1cwIAAAAHUEVORElORwkAAAIAAAABAgAAAB9zd2FwIGhhcyBiZWVuIGFscmVhZHkgcHJvY2Vzc2VkAwkAAGYAAAACBQAAAAx1bmxvY2tIZWlnaHQFAAAABmhlaWdodAkAAAIAAAABCQABLAAAAAIJAAEsAAAAAgIAAAARcGxlYXNlIHdhaXQgZm9yOiAJAAGkAAAAAQUAAAAMdW5sb2NrSGVpZ2h0AgAAAB8gYmxvY2sgaGVpZ2h0IHRvIHdpdGhkcmF3IGZ1bmRzAwMDCQAAZgAAAAIFAAAABWluZGV4BQAAAApwcmljZUluZGV4BgkAAGYAAAACBQAAAAx1bmxvY2tIZWlnaHQFAAAAC2luZGV4SGVpZ2h0BgMJAQAAAAIhPQAAAAIFAAAAD3ByZXZJbmRleEhlaWdodAAAAAAAAAAAAAkAAGcAAAACBQAAAA9wcmV2SW5kZXhIZWlnaHQFAAAADHVubG9ja0hlaWdodAcJAQAAAA5wcmljZUluZGV4RkFJTAAAAAUFAAAABWluZGV4BQAAAApwcmljZUluZGV4BQAAAAtpbmRleEhlaWdodAUAAAAMdW5sb2NrSGVpZ2h0BQAAAA9wcmV2SW5kZXhIZWlnaHQDCQAAZwAAAAIAAAAAAAAAAAAJAAGRAAAAAgUAAAAMcGF5b3V0c0FycmF5BQAAAA5JZHhHcm9zc0Ftb3VudAkAAAIAAAABAgAAABNiYWxhbmNlIGVxdWFscyB6ZXJvAwMJAABmAAAAAgAAAAAAAAAAAAUAAAAKb3V0RmVlUGFydAYJAABnAAAAAgUAAAAKb3V0RmVlUGFydAUAAAAFUEFVTEkJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAeaW52YWxpZCBvdXRGZWVQYXJ0IGNvbmZpZyBmb3IgBQAAAAhzd2FwVHlwZQIAAAASIHN3YXA6IG91dEZlZVBhcnQ9CQABpAAAAAEFAAAACm91dEZlZVBhcnQEAAAACWxlYXNlUGFydAMDCQAAAAAAAAIFAAAACHN3YXBUeXBlAgAAAAhuZXV0cmlubwkAAGYAAAACCAUAAAATb3V0QW1vdW50R3Jvc3NUdXBsZQAAAAJfMQAAAAAAAAAAAAcJAQAAABZwcmVwYXJlVW5sZWFzZUFuZExlYXNlAAAAAQgFAAAAE291dEFtb3VudEdyb3NzVHVwbGUAAAACXzEFAAAAA25pbAkABRQAAAACCQAETgAAAAIFAAAACWxlYXNlUGFydAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkBAAAAFHRvdGFsTG9ja2VkQnlVc2VyS0VZAAAAAgUAAAAIc3dhcFR5cGUFAAAAB2FjY291bnQJAABlAAAAAgUAAAARdG90YWxMb2NrZWRCeVVzZXIFAAAACGluQW1vdW50CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAOdG90YWxMb2NrZWRLRVkAAAABBQAAAAhzd2FwVHlwZQkAAGUAAAACBQAAAAt0b3RhbExvY2tlZAUAAAAIaW5BbW91bnQJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwUAAAALdXNlckFkZHJlc3MFAAAADG91dE5ldEFtb3VudAgFAAAAE291dEFtb3VudEdyb3NzVHVwbGUAAAACXzIJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwUAAAARZmVlTWFuYWdlckFkZHJlc3MFAAAADG91dEZlZUFtb3VudAgFAAAAE291dEFtb3VudEdyb3NzVHVwbGUAAAACXzIJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAB3N3YXBLRVkAAAACBQAAAAdhY2NvdW50BQAAAAhzd2FwVHhJZAkBAAAADmZpbmlzaFN3YXBEQVRBAAAABwUAAAAJZGF0YUFycmF5BQAAAAxwcmljZUJ5SW5kZXgFAAAADG91dE5ldEFtb3VudAUAAAAMb3V0RmVlQW1vdW50BQAAAAx1bmxvY2tIZWlnaHQFAAAABWluZGV4CQACWAAAAAEIBQAAAAFpAAAADXRyYW5zYWN0aW9uSWQFAAAAA25pbAUAAAAEdW5pdAAAAAYAAAABaQEAAAAKY29tbW9uU3dhcAAAAAQAAAAIc3dhcFR5cGUAAAAJcG10QW1vdW50AAAADnVzZXJBZGRyZXNzU3RyAAAABnR4SWQ1OAQAAAAVY2hlY2tUaGlzT3JpZ2luQ2FsbGVyAwkBAAAAAiE9AAAAAggFAAAAAWkAAAAGY2FsbGVyBQAAAAR0aGlzCQAAAgAAAAECAAAAIFBlcm1pc3Npb24gZGVuaWVkLiBXcm9uZyBjYWxsZXIuBgMJAAAAAAAAAgUAAAAVY2hlY2tUaGlzT3JpZ2luQ2FsbGVyBQAAABVjaGVja1RoaXNPcmlnaW5DYWxsZXIEAAAAC2NoZWNrQ2FsbGVyAwkBAAAAAiE9AAAAAggFAAAAAWkAAAAMb3JpZ2luQ2FsbGVyCQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABBQAAAA51c2VyQWRkcmVzc1N0cgkAAAIAAAABAgAAACdQZXJtaXNzaW9uIGRlbmllZC4gV3Jvbmcgb3JpZ2luIGNhbGxlci4GAwkAAAAAAAACBQAAAAtjaGVja0NhbGxlcgUAAAALY2hlY2tDYWxsZXIEAAAADW1pblN3YXBBbW91bnQJAQAAABFtaW5Td2FwQW1vdW50UkVBRAAAAAEFAAAACHN3YXBUeXBlBAAAAAt0b3RhbExvY2tlZAkBAAAAD3RvdGFsTG9ja2VkUkVBRAAAAAEFAAAACHN3YXBUeXBlBAAAABF0b3RhbExvY2tlZEJ5VXNlcgkBAAAAFXRvdGFsTG9ja2VkQnlVc2VyUkVBRAAAAAIFAAAACHN3YXBUeXBlBQAAAA51c2VyQWRkcmVzc1N0cgQAAAALbm9kZUFkZHJlc3MJAQAAABVnZXRTdGFraW5nTm9kZUJ5SW5kZXgAAAABAAAAAAAAAAAABAAAABZiYWxhbmNlTG9ja01heEludGVydmFsAwkAAAAAAAACBQAAAAtub2RlQWRkcmVzcwUAAAAOdXNlckFkZHJlc3NTdHIJAQAAABtub2RlQmFsYW5jZUxvY2tJbnRlcnZhbFJFQUQAAAAAAwkAAAAAAAACBQAAAAhzd2FwVHlwZQIAAAAFd2F2ZXMJAQAAABFAZXh0ck5hdGl2ZSgxMDUwKQAAAAIFAAAABHRoaXMFAAAAG0JhbGFuY2VXYXZlc0xvY2tJbnRlcnZhbEtleQkBAAAAEUBleHRyTmF0aXZlKDEwNTApAAAAAgUAAAAEdGhpcwUAAAAeQmFsYW5jZU5ldXRyaW5vTG9ja0ludGVydmFsS2V5BAAAABBzZWxmVW5sb2NrSGVpZ2h0CQAAZAAAAAIFAAAABmhlaWdodAUAAAAWYmFsYW5jZUxvY2tNYXhJbnRlcnZhbAQAAAALYW1vdW50Q2hlY2sDCQAAZgAAAAIFAAAADW1pblN3YXBBbW91bnQFAAAACXBtdEFtb3VudAkBAAAAEW1pblN3YXBBbW91bnRGQUlMAAAAAgUAAAAIc3dhcFR5cGUFAAAADW1pblN3YXBBbW91bnQGAwkAAAAAAAACBQAAAAthbW91bnRDaGVjawUAAAALYW1vdW50Q2hlY2sEAAAADmxhc3RTd2FwSGVpZ2h0CQEAAAAOZ2V0TnVtYmVyQnlLZXkAAAABCQEAAAAaa2V5VXNlckxhc3RRdWlja1N3YXBIZWlnaHQAAAABBQAAAA51c2VyQWRkcmVzc1N0cgMJAABmAAAAAgUAAAAPVFdFTlRZRk9VUkhPVVJTCQAAZQAAAAIFAAAABmhlaWdodAUAAAAObGFzdFN3YXBIZWlnaHQJAAACAAAAAQkAASwAAAACAgAAAEJZb3UgaGF2ZSBleGNlZWRlZCBzd2FwIGxpbWl0IGZvciAyNGghIE5leHQgYWxsb3dlZCBzd2FwIGhlaWdodCBpcyAJAAGkAAAAAQkAAGQAAAACBQAAAA5sYXN0U3dhcEhlaWdodAUAAAAPVFdFTlRZRk9VUkhPVVJTBAAAABJ1c2VyU3dhcExpbWl0c0luZm8JAQAAAAlhc0FueUxpc3QAAAABCQAD/AAAAAQFAAAADG1hdGhDb250cmFjdAIAAAAWY2FsY1VzZXJTd2FwTGltaXRzTUFUSAkABEwAAAACBQAAAA51c2VyQWRkcmVzc1N0cgUAAAADbmlsBQAAAANuaWwDCQAAAAAAAAIFAAAAEnVzZXJTd2FwTGltaXRzSW5mbwUAAAASdXNlclN3YXBMaW1pdHNJbmZvBAAAABNxdWlja1N3YXBMaW1pdFRvdGFsCQEAAAAFYXNJbnQAAAABCQABkQAAAAIFAAAAEnVzZXJTd2FwTGltaXRzSW5mbwAAAAAAAAAAAQQAAAAMcHJpY2VCeUluZGV4CQEAAAAPZ2V0UHJpY2VIaXN0b3J5AAAAAQkBAAAAFWdldEhlaWdodFByaWNlQnlJbmRleAAAAAEFAAAACnByaWNlSW5kZXgEAAAADnN3YXBVc2RuVm9sdW1lAwkAAAAAAAACBQAAAAhzd2FwVHlwZQIAAAAIbmV1dHJpbm8FAAAACXBtdEFtb3VudAkBAAAAFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8AAAACBQAAAAlwbXRBbW91bnQFAAAADHByaWNlQnlJbmRleAMJAABmAAAAAgUAAAAOc3dhcFVzZG5Wb2x1bWUFAAAAE3F1aWNrU3dhcExpbWl0VG90YWwJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAuWW91IGhhdmUgZXhjZWVkZWQgeW91ciBzd2FwIGxpbWl0ISBSZXF1ZXN0ZWQ6IAkAAaQAAAABBQAAAA5zd2FwVXNkblZvbHVtZQIAAAANLCBhdmFpbGFibGU6IAkAAaQAAAABBQAAABNxdWlja1N3YXBMaW1pdFRvdGFsAwUAAAAJaXNCbG9ja2VkCQEAAAAVZW1lcmdlbmN5U2h1dGRvd25GQUlMAAAAAAQAAAAJbGVhc2VQYXJ0AwkAAAAAAAACBQAAAAhzd2FwVHlwZQIAAAAFd2F2ZXMJAQAAABZwcmVwYXJlVW5sZWFzZUFuZExlYXNlAAAAAQAAAAAAAAAAAAUAAAADbmlsCQAFFAAAAAIJAAROAAAAAgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkBAAAAHWtleVF1aWNrU3dhcFVzZXJTcGVudEluUGVyaW9kAAAAAQUAAAAOdXNlckFkZHJlc3NTdHIFAAAADnN3YXBVc2RuVm9sdW1lCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAaa2V5VXNlckxhc3RRdWlja1N3YXBIZWlnaHQAAAABBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAGaGVpZ2h0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAUdG90YWxMb2NrZWRCeVVzZXJLRVkAAAACBQAAAAhzd2FwVHlwZQUAAAAOdXNlckFkZHJlc3NTdHIJAABkAAAAAgUAAAARdG90YWxMb2NrZWRCeVVzZXIFAAAACXBtdEFtb3VudAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkBAAAAGGdldEJhbGFuY2VVbmxvY2tCbG9ja0tleQAAAAEFAAAADnVzZXJBZGRyZXNzU3RyBQAAABBzZWxmVW5sb2NrSGVpZ2h0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAOdG90YWxMb2NrZWRLRVkAAAABBQAAAAhzd2FwVHlwZQkAAGQAAAACBQAAAAt0b3RhbExvY2tlZAUAAAAJcG10QW1vdW50CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAAAdzd2FwS0VZAAAAAgUAAAAOdXNlckFkZHJlc3NTdHIFAAAABnR4SWQ1OAkBAAAAD3BlbmRpbmdTd2FwREFUQQAAAAMFAAAACHN3YXBUeXBlBQAAAAlwbXRBbW91bnQFAAAAEHNlbGZVbmxvY2tIZWlnaHQFAAAAA25pbAUAAAAJbGVhc2VQYXJ0BQAAAAR1bml0CQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgAAAAFpAQAAABNzd2FwV2F2ZXNUb05ldXRyaW5vAAAAAAQAAAADcG10CQEAAAAFdmFsdWUAAAABCQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAwkBAAAACWlzRGVmaW5lZAAAAAEIBQAAAANwbXQAAAAHYXNzZXRJZAkAAAIAAAABAgAAAClPbmx5IFdhdmVzIHRva2VuIGlzIGFsbG93ZWQgZm9yIHN3YXBwaW5nLgQAAAALdXNlckFkZHJlc3MJAAQlAAAAAQgFAAAAAWkAAAAGY2FsbGVyBAAAAAZ0eElkNTgJAAJYAAAAAQgFAAAAAWkAAAANdHJhbnNhY3Rpb25JZAQAAAANY29tbW9uU3dhcEludgkAA/wAAAAEBQAAAAR0aGlzAgAAAApjb21tb25Td2FwCQAETAAAAAICAAAABXdhdmVzCQAETAAAAAIIBQAAAANwbXQAAAAGYW1vdW50CQAETAAAAAIFAAAAC3VzZXJBZGRyZXNzCQAETAAAAAIFAAAABnR4SWQ1OAUAAAADbmlsBQAAAANuaWwDCQAAAAAAAAIFAAAADWNvbW1vblN3YXBJbnYFAAAADWNvbW1vblN3YXBJbnYFAAAAA25pbAkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4AAAABaQEAAAATc3dhcE5ldXRyaW5vVG9XYXZlcwAAAAAEAAAAA3BtdAkBAAAABXZhbHVlAAAAAQkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAMJAQAAAAIhPQAAAAIIBQAAAANwbXQAAAAHYXNzZXRJZAUAAAAPbmV1dHJpbm9Bc3NldElkCQAAAgAAAAECAAAAOk9ubHkgYXBwcm9wcmlhdGUgTmV1dHJpbm8gdG9rZW5zIGFyZSBhbGxvd2VkIGZvciBzd2FwcGluZy4EAAAAC3VzZXJBZGRyZXNzCQAEJQAAAAEIBQAAAAFpAAAABmNhbGxlcgQAAAAGdHhJZDU4CQACWAAAAAEIBQAAAAFpAAAADXRyYW5zYWN0aW9uSWQEAAAADWNvbW1vblN3YXBJbnYJAAP8AAAABAUAAAAEdGhpcwIAAAAKY29tbW9uU3dhcAkABEwAAAACAgAAAAhuZXV0cmlubwkABEwAAAACCAUAAAADcG10AAAABmFtb3VudAkABEwAAAACBQAAAAt1c2VyQWRkcmVzcwkABEwAAAACBQAAAAZ0eElkNTgFAAAAA25pbAUAAAADbmlsAwkAAAAAAAACBQAAAA1jb21tb25Td2FwSW52BQAAAA1jb21tb25Td2FwSW52BQAAAANuaWwJAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAAAAAWkBAAAACHdpdGhkcmF3AAAAAwAAAAdhY2NvdW50AAAABWluZGV4AAAACHN3YXBUeElkCQEAAAAOY29tbW9uV2l0aGRyYXcAAAAEBQAAAAdhY2NvdW50BQAAAAVpbmRleAUAAAAIc3dhcFR4SWQFAAAAAWkAAAABaQEAAAARdHJhbnNmZXJUb0F1Y3Rpb24AAAAABAAAAA9uZXV0cmlub01ldHJpY3MJAQAAAAlhc0FueUxpc3QAAAABCQAD/AAAAAQFAAAADG1hdGhDb250cmFjdAIAAAAWY2FsY05ldXRpbm9NZXRyaWNzTUFUSAUAAAADbmlsBQAAAANuaWwEAAAAB3Jlc2VydmUJAQAAAAVhc0ludAAAAAEJAAGRAAAAAgUAAAAPbmV1dHJpbm9NZXRyaWNzAAAAAAAAAAAEBAAAAA5uZXV0cmlub1N1cHBseQkBAAAABWFzSW50AAAAAQkAAZEAAAACBQAAAA9uZXV0cmlub01ldHJpY3MAAAAAAAAAAAYEAAAAB3N1cnBsdXMJAQAAAAVhc0ludAAAAAEJAAGRAAAAAgUAAAAPbmV1dHJpbm9NZXRyaWNzAAAAAAAAAAAHBAAAAApuc2J0U3VwcGx5CQEAAAAFYXNJbnQAAAABCQABkQAAAAIFAAAAD25ldXRyaW5vTWV0cmljcwAAAAAAAAAACgQAAAAPYXVjdGlvbk5CQW1vdW50CQAAZQAAAAIFAAAADm5ldXRyaW5vU3VwcGx5CQAD8AAAAAIJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEFAAAAD2F1Y3Rpb25Db250cmFjdAUAAAALYm9uZEFzc2V0SWQEAAAAFnN1cnBsdXNXaXRoTGlxdWlkYXRpb24JAABlAAAAAgUAAAAHc3VycGx1cwkAA/AAAAACCQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABBQAAABNsaXF1aWRhdGlvbkNvbnRyYWN0BQAAAA9uZXV0cmlub0Fzc2V0SWQDBQAAAAlpc0Jsb2NrZWQJAAACAAAAAQIAAABaY29udHJhY3QgaXMgYmxvY2tlZCBieSBFTUVSR0VOQ1kgU0hVVERPV04gYWN0aW9ucyB1bnRpbGwgcmVhY3RpdmF0aW9uIGJ5IGVtZXJnZW5jeSBvcmFjbGVzAwkAAGYAAAACBQAAAA9hdWN0aW9uTkJBbW91bnQJAABoAAAAAgAAAAAAAAAAAQUAAAAFUEFVTEkJAARMAAAAAgkB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", "chainId": 84, "height": 1937765, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 5XeZwk7EdGCUhBmXJMqW8Z2mEZj41XWFp7qTpFRHTSjQ Next: B5qicd3hAzDL88bq2rJkjWx3HXhofN7UTk51WNqUDk6P Diff:
OldNewDifferences
1919 func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(addressFromStringValue(address), key), false)
2020
2121
22-let pubKeyAdminsList = ["GXxmZaCigPgJsT6d1cLr8nUXA65nGX8V7FuAzzrY8wQE", "3xumx4hrPpeeoo79HLGbZCgfz95L39ZFdxxwWeSUC2Pu", "D4CqtK5fBsnzRo33ZMhrM7niLseDH2uRoK6weBq2JQ3A", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
22+func asAnyList (val) = match val {
23+ case valAnyLyst: List[Any] =>
24+ valAnyLyst
25+ case _ =>
26+ throw("fail to cast into List[Any]")
27+}
28+
29+
30+func asString (val) = match val {
31+ case valStr: String =>
32+ valStr
33+ case _ =>
34+ throw("fail to cast into String")
35+}
36+
2337
2438 func asInt (val) = match val {
2539 case valInt: Int =>
2640 valInt
2741 case _ =>
28- throw("Failed to cast into Int")
42+ throw("fail to cast into Int")
2943 }
3044
45+
46+let pubKeyAdminsList = ["GXxmZaCigPgJsT6d1cLr8nUXA65nGX8V7FuAzzrY8wQE", "3xumx4hrPpeeoo79HLGbZCgfz95L39ZFdxxwWeSUC2Pu", "D4CqtK5fBsnzRo33ZMhrM7niLseDH2uRoK6weBq2JQ3A", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
3147
3248 let SEP = "__"
3349
3955
4056 let DEFAULTSWAPFEE = 20000
4157
58+let TWENTYFOURHOURS = 1440
59+
4260 let IdxNetAmount = 0
4361
4462 let IdxFeeAmount = 1
4563
4664 let IdxGrossAmount = 2
47-
48-let dora2NsbtSymbol = "NSBT-USDT"
49-
50-let minRand = 60
51-
52-let maxRand = 1440
5365
5466 let NeutrinoAssetIdKey = "neutrino_asset_id"
5567
6173
6274 let RPDContractKey = "rpd_contract"
6375
64-let ControlContractKey = "control_contract"
76+let ContolContractKey = "control_contract"
77+
78+let RestContractKey = "rest_contract"
79+
80+let MathContractKey = "math_contract"
6581
6682 let BalanceWavesLockIntervalKey = "balance_waves_lock_interval"
6783
7793
7894 let WavesOutFeePartKey = "wavesOut_swap_feePart"
7995
80-let RsaRandPublic58Key = "rand_rsa_public"
96+let FeesManagerAddressKey = "fees_manager_address"
8197
82-let keyGNsbtContract = "%s__gNsbtContract"
98+func keyQuickSwapLimitDuration () = "%s__quickSwapLimitDuration"
8399
84-let keyNsbtLockContract = "%s__nsbtLockContract"
85-
86-let keyDora2Contract = "%s__dora2Contract"
87-
88-let keyQuickSwapDelay = "%s__quickSwapDelay"
89-
90-let keyQuickSwapLimitDuration = "%s__quickSwapLimitDuration"
91-
92-let keyDora2LastHeightLimit = "%s__dora2LastHeightLimit"
93100
94101 let PriceKey = "price"
95102
142149 func outFeePartKEY (swapType) = (swapType + "Out_swap_feePart")
143150
144151
145-func keyQuickSwapUserSpentInPeriod (userAddress) = makeString(["%s%s", "quickSwapUserSpentInPeriod", toString(userAddress)], SEP)
146-
147-
148-func keyUserLastQuickSwapHeight (userAddress) = makeString(["%s%s", "userLastQuickSwapHeight", toString(userAddress)], SEP)
149-
150-
151-func keyDora2Price (symbol) = makeString(["%s%s", "price", symbol], SEP)
152-
153-
154-func keyDora2LastHeight (symbol) = makeString(["%s%s", "lastHeight", symbol], SEP)
155-
156-
157152 func minSwapAmountREAD (swapType) = valueOrElse(getInteger(this, minSwapAmountKEY(swapType)), 0)
158153
159154
163158 func totalLockedByUserREAD (swapType,owner) = valueOrElse(getInteger(this, totalLockedByUserKEY(swapType, owner)), 0)
164159
165160
166-func balanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, balanceLockIntervalKEY(swapType)), maxRand)
161+func balanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, balanceLockIntervalKEY(swapType)), 1440)
167162
168163
169-func minBalanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, minBalanceLockIntervalKEY(swapType)), minRand)
164+func minBalanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, minBalanceLockIntervalKEY(swapType)), 60)
170165
171166
172167 func nodeBalanceLockIntervalREAD () = valueOrElse(getInteger(this, nodeBalanceLockIntervalKEY()), 1)
168+
169+
170+func keyQuickSwapUserSpentInPeriod (userAddress) = makeString(["%s%s", "quickSwapUserSpentInPeriod", userAddress], SEP)
171+
172+
173+func keyUserLastQuickSwapHeight (userAddress) = makeString(["%s%s", "userLastQuickSwapHeight", userAddress], SEP)
174+
175+
176+func feeManagerAddressREAD () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(this, FeesManagerAddressKey), (FeesManagerAddressKey + " is not specified"))), (FeesManagerAddressKey + " invalid address format"))
173177
174178
175179 func convertNeutrinoToWaves (amount,price) = fraction(fraction(amount, PRICELET, price), WAVELET, PAULI)
201205
202206 let rpdContract = getStringByKey(RPDContractKey)
203207
204-let controlContract = getStringByKey(ControlContractKey)
208+let controlContract = getStringByKey(ContolContractKey)
209+
210+let restContract = getStringByKey(RestContractKey)
211+
212+let mathContractAddress = getStringByKey(MathContractKey)
205213
206214 let priceIndex = getNumberByAddressAndKey(controlContract, PriceIndexKey)
207215
213221
214222 let deprecatedBondAssetId = fromBase58String("975akZBfnMj513U7MZaHKzQrmsEx5aE3wdWKTrHBhbjF")
215223
216-let rsaPub = fromBase64String(valueOrErrorMessage(getString(this, RsaRandPublic58Key), "RSA public key has not been specified"))
217-
218224 let neutrinoContract = this
219225
226+let mathContract = addressFromStringValue(mathContractAddress)
227+
220228 let currentPrice = getNumberByAddressAndKey(controlContract, PriceKey)
221-
222-let neutrinoLockedBalance = totalLockedREAD("neutrino")
223-
224-let wavesLockedBalance = totalLockedREAD("waves")
225-
226-let reserve = (wavesBalance(neutrinoContract).regular - wavesLockedBalance)
227-
228-let neutrinoSupply = (((neutrinoLockedBalance + value(assetInfo(neutrinoAssetId)).quantity) - assetBalance(neutrinoContract, neutrinoAssetId)) - assetBalance(addressFromStringValue(liquidationContract), neutrinoAssetId))
229-
230-let surplus = (convertWavesToNeutrino(reserve, currentPrice) - neutrinoSupply)
231-
232-let deficit = (neutrinoSupply - convertWavesToNeutrino(reserve, currentPrice))
233229
234230 func checkIsValidMinSponsoredFee (tx) = {
235231 let MINTRANSFERFEE = 100000
284280
285281 let sIdxMaxRand = 16
286282
287-let sIdxIsQuick = 17
288-
289283 func swapKEY (userAddress,txId) = makeString(["%s%s", userAddress, txId], SEP)
290284
291285
292-func strSwapDATA (swapType,status,inAmount,price,outNetAmount,outFeeAmount,startHeight,startTimestamp,endHeight,endTimestamp,selfUnlockHeight,randUnlockHeight,index,withdrawTxId,randMin,randMax,isQuick) = makeString(["%s%s%d%d%d%d%d%d%d%d%d%d%d%s", swapType, status, inAmount, price, outNetAmount, outFeeAmount, startHeight, startTimestamp, endHeight, endTimestamp, selfUnlockHeight, randUnlockHeight, index, withdrawTxId, randMin, randMax, isQuick], SEP)
286+func strSwapDATA (swapType,status,inAmount,price,outNetAmount,outFeeAmount,startHeight,startTimestamp,endHeight,endTimestamp,selfUnlockHeight,randUnlockHeight,index,withdrawTxId,randMin,randMax) = makeString(["%s%s%d%d%d%d%d%d%d%d%d%d%d%s", swapType, status, inAmount, price, outNetAmount, outFeeAmount, startHeight, startTimestamp, endHeight, endTimestamp, selfUnlockHeight, randUnlockHeight, index, withdrawTxId, randMin, randMax], SEP)
293287
294288
295-func pendingSwapDATA (swapType,inAssetAmount,selfUnlockHeight) = strSwapDATA(swapType, "PENDING", toString(inAssetAmount), "0", "0", "0", toString(height), toString(lastBlock.timestamp), "0", "0", toString(selfUnlockHeight), "0", "0", "NULL", toString(minBalanceLockIntervalREAD(swapType)), toString(balanceLockIntervalREAD(swapType)), toString(false))
289+func pendingSwapDATA (swapType,inAssetAmount,selfUnlockHeight) = strSwapDATA(swapType, "PENDING", toString(inAssetAmount), "0", "0", "0", toString(height), toString(lastBlock.timestamp), "0", "0", toString(selfUnlockHeight), "0", "0", "NULL", "0", "0")
296290
297291
298-func accelerateSwapDATA (dataArray) = strSwapDATA(dataArray[sIdxSwapType], dataArray[sIdxStatus], dataArray[sIdxInAmount], dataArray[sIdxPrice], dataArray[sIdxOutNetAmount], dataArray[sIdxOutFeeAmount], dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], dataArray[sIdxEndHeight], dataArray[sIdxEndTimestamp], dataArray[sIdxSelfUnlockHeight], dataArray[sIdxRandUnlockHeight], dataArray[sIdxIndex], dataArray[sIdxWithdrawTxId], dataArray[sIdxMinRand], dataArray[sIdxMaxRand], toString(true))
299-
300-
301-func finishSwapDATA (dataArray,price,outNetAmount,outFeeAmount,randUnlockHeight,index,withdrawTxId) = strSwapDATA(dataArray[sIdxSwapType], "FINISHED", dataArray[sIdxInAmount], toString(price), toString(outNetAmount), toString(outFeeAmount), dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], toString(height), toString(lastBlock.timestamp), dataArray[sIdxSelfUnlockHeight], toString(randUnlockHeight), toString(index), withdrawTxId, dataArray[sIdxMinRand], dataArray[sIdxMaxRand], dataArray[sIdxIsQuick])
292+func finishSwapDATA (dataArray,price,outNetAmount,outFeeAmount,randUnlockHeight,index,withdrawTxId) = strSwapDATA(dataArray[sIdxSwapType], "FINISHED", dataArray[sIdxInAmount], toString(price), toString(outNetAmount), toString(outFeeAmount), dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], toString(height), toString(lastBlock.timestamp), dataArray[sIdxSelfUnlockHeight], toString(randUnlockHeight), toString(index), withdrawTxId, dataArray[sIdxMinRand], dataArray[sIdxMaxRand])
302293
303294
304295 func swapDataFailOrREAD (userAddress,swapTxId) = {
310301 func applyFees (amountGross,feePart) = {
311302 let feeAmount = fraction(amountGross, feePart, PAULI)
312303 [(amountGross - feeAmount), feeAmount, amountGross]
313- }
314-
315-
316-func randUnlockHeightOrFail (txId,rsaSig,swapType,startHeight,minMaxRandsTuple) = {
317- let isRsaValid = rsaVerify_16Kb(SHA256, toBytes(txId), rsaSig, rsaPub)
318- if (!(isRsaValid))
319- then throw("invalid RSA signature")
320- else {
321- let minBalanceLockInterval = minMaxRandsTuple._1
322- let maxBalanceLockInterval = minMaxRandsTuple._2
323- let rand = (toInt(sha256_16Kb(rsaSig)) % (maxBalanceLockInterval - minBalanceLockInterval))
324- let randLockInterval = (minBalanceLockInterval + (if ((0 > rand))
325- then -(rand)
326- else rand))
327- (startHeight + randLockInterval)
328- }
329304 }
330305
331306
355330
356331
357332 func prepareUnleaseAndLease (unleaseAmount) = {
358- let $t01621216300 = selectNode(unleaseAmount)
359- let nodeIndex = $t01621216300._1
360- let newLeaseAmount = $t01621216300._2
333+ let nodeTuple = selectNode(unleaseAmount)
334+ let nodeIndex = nodeTuple._1
335+ let newLeaseAmount = nodeTuple._2
361336 if ((newLeaseAmount > 0))
362337 then {
363338 let leaseIdKey = getLeaseIdKey(nodeIndex)
373348 }
374349
375350
376-func getDora2NSBTInfo () = {
377- let dora2Contract = addressFromStringValue(getStringValue(this, keyDora2Contract))
378- $Tuple2(valueOrErrorMessage(getInteger(dora2Contract, keyDora2Price(dora2NsbtSymbol)), "NSBT price is undefined"), valueOrElse(getInteger(dora2Contract, keyDora2LastHeight(dora2NsbtSymbol)), 0))
351+func commonWithdraw (account,index,swapTxId,i) = {
352+ let userAddress = addressFromStringValue(account)
353+ let feeManagerAddress = feeManagerAddressREAD()
354+ let dataArray = swapDataFailOrREAD(account, swapTxId)
355+ let selfUnlockHeight = parseIntValue(dataArray[sIdxSelfUnlockHeight])
356+ let swapType = dataArray[sIdxSwapType]
357+ let inAmount = parseIntValue(dataArray[sIdxInAmount])
358+ let swapStatus = dataArray[sIdxStatus]
359+ let startHeight = parseIntValue(dataArray[sIdxStartHeight])
360+ let outFeePart = valueOrElse(getInteger(this, outFeePartKEY(swapType)), DEFAULTSWAPFEE)
361+ let totalLocked = totalLockedREAD(swapType)
362+ let totalLockedByUser = totalLockedByUserREAD(swapType, account)
363+ let unlockHeight = (startHeight + (if ((swapType == "waves"))
364+ then getIntegerValue(this, BalanceWavesLockIntervalKey)
365+ else getIntegerValue(this, BalanceNeutrinoLockIntervalKey)))
366+ let indexHeight = getHeightPriceByIndex(index)
367+ let prevIndexHeight = getHeightPriceByIndex((index - 1))
368+ let priceByIndex = getPriceHistory(indexHeight)
369+ let outAmountGrossTuple = if ((swapType == "waves"))
370+ then $Tuple2(convertWavesToNeutrino(inAmount, priceByIndex), neutrinoAssetId)
371+ else if ((swapType == "neutrino"))
372+ then $Tuple2(convertNeutrinoToWaves(inAmount, priceByIndex), unit)
373+ else throw(("Unsupported swap type " + swapType))
374+ let payoutsArray = applyFees(outAmountGrossTuple._1, outFeePart)
375+ let outNetAmount = payoutsArray[IdxNetAmount]
376+ let outFeeAmount = payoutsArray[IdxFeeAmount]
377+ if (isBlocked)
378+ then emergencyShutdownFAIL()
379+ else if ((swapStatus != "PENDING"))
380+ then throw("swap has been already processed")
381+ else if ((unlockHeight > height))
382+ then throw((("please wait for: " + toString(unlockHeight)) + " block height to withdraw funds"))
383+ else if (if (if ((index > priceIndex))
384+ then true
385+ else (unlockHeight > indexHeight))
386+ then true
387+ else if ((prevIndexHeight != 0))
388+ then (prevIndexHeight >= unlockHeight)
389+ else false)
390+ then priceIndexFAIL(index, priceIndex, indexHeight, unlockHeight, prevIndexHeight)
391+ else if ((0 >= payoutsArray[IdxGrossAmount]))
392+ then throw("balance equals zero")
393+ else if (if ((0 > outFeePart))
394+ then true
395+ else (outFeePart >= PAULI))
396+ then throw(((("invalid outFeePart config for " + swapType) + " swap: outFeePart=") + toString(outFeePart)))
397+ else {
398+ let leasePart = if (if ((swapType == "neutrino"))
399+ then (outAmountGrossTuple._1 > 0)
400+ else false)
401+ then prepareUnleaseAndLease(outAmountGrossTuple._1)
402+ else nil
403+ $Tuple2((leasePart ++ [IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser - inAmount)), IntegerEntry(totalLockedKEY(swapType), (totalLocked - inAmount)), ScriptTransfer(userAddress, outNetAmount, outAmountGrossTuple._2), ScriptTransfer(feeManagerAddress, outFeeAmount, outAmountGrossTuple._2), StringEntry(swapKEY(account, swapTxId), finishSwapDATA(dataArray, priceByIndex, outNetAmount, outFeeAmount, unlockHeight, index, toBase58String(i.transactionId)))]), unit)
404+ }
379405 }
380406
381407
382-func thisOnly (i) = if ((i.caller != this))
383- then throw("Permission denied: this contract only allowed")
384- else true
385-
386-
387408 @Callable(i)
388-func prepareUnleaseAndLeaseWrapper (unleaseAmount) = {
389- let checkCaller = [thisOnly(i)]
390- if ((checkCaller == checkCaller))
391- then prepareUnleaseAndLease(unleaseAmount)
392- else throw("Strict value is not equal to itself.")
393- }
394-
395-
396-
397-@Callable(i)
398-func constructor (neutrinoAssetIdPrm,bondAssetIdPrm,auctionContractPrm,liquidationContractPrm,rpdContractPrm,controlContractPrm,nodeOracleProviderPubKeyPrm,rsaRandPublicKeyPrm,balanceWavesLockIntervalPrm,balanceNeutrinoLockIntervalPrm,minWavesSwapAmountPrm,minNeutrinoSwapAmountPrm,neutrinoOutFeePartPrm,wavesOutFeePartPrm) = {
399- let checkCaller = thisOnly(i)
400- if ((checkCaller == checkCaller))
401- then [StringEntry(NeutrinoAssetIdKey, neutrinoAssetIdPrm), StringEntry(BondAssetIdKey, bondAssetIdPrm), StringEntry(AuctionContractKey, auctionContractPrm), StringEntry(LiquidationContractKey, liquidationContractPrm), StringEntry(RPDContractKey, rpdContractPrm), StringEntry(ControlContractKey, controlContractPrm), StringEntry(NodeOracleProviderPubKeyKey, nodeOracleProviderPubKeyPrm), StringEntry(RsaRandPublic58Key, rsaRandPublicKeyPrm), IntegerEntry(BalanceWavesLockIntervalKey, balanceWavesLockIntervalPrm), IntegerEntry(BalanceNeutrinoLockIntervalKey, balanceNeutrinoLockIntervalPrm), IntegerEntry(MinWavesSwapAmountKey, minWavesSwapAmountPrm), IntegerEntry(MinNeutrinoSwapAmountKey, minNeutrinoSwapAmountPrm), IntegerEntry(NeutrinoOutFeePartKey, neutrinoOutFeePartPrm), IntegerEntry(WavesOutFeePartKey, wavesOutFeePartPrm)]
402- else throw("Strict value is not equal to itself.")
403- }
404-
405-
406-
407-@Callable(i)
408-func constructorV2 (gNsbtContract,nsbtLockContract,dora2Contract,quickSwapDelay,quickSwapLimitDuration,dora2LastHeightLimit) = {
409- let checkCaller = thisOnly(i)
410- if ((checkCaller == checkCaller))
411- then [StringEntry(keyGNsbtContract, gNsbtContract), StringEntry(keyNsbtLockContract, nsbtLockContract), StringEntry(keyDora2Contract, dora2Contract), IntegerEntry(keyQuickSwapDelay, quickSwapDelay), IntegerEntry(keyQuickSwapLimitDuration, quickSwapLimitDuration), IntegerEntry(keyDora2LastHeightLimit, dora2LastHeightLimit)]
412- else throw("Strict value is not equal to itself.")
413- }
414-
415-
416-
417-@Callable(i)
418-func commonSwap (swapType,amount,account,txId) = {
419- let checkCaller = thisOnly(i)
420- if ((checkCaller == checkCaller))
409+func commonSwap (swapType,pmtAmount,userAddressStr,txId58) = {
410+ let checkThisOriginCaller = if ((i.caller != this))
411+ then throw("Permission denied. Wrong caller.")
412+ else true
413+ if ((checkThisOriginCaller == checkThisOriginCaller))
421414 then {
422- let minSwapAmount = minSwapAmountREAD(swapType)
423- let totalLocked = totalLockedREAD(swapType)
424- let totalLockedByUser = totalLockedByUserREAD(swapType, account)
425- let nodeAddress = getStakingNodeByIndex(0)
426- let balanceLockMaxInterval = if ((nodeAddress == account))
427- then nodeBalanceLockIntervalREAD()
428- else balanceLockIntervalREAD(swapType)
429- let selfUnlockHeight = (height + balanceLockMaxInterval)
430- if ((minSwapAmount > amount))
431- then minSwapAmountFAIL(swapType, minSwapAmount)
432- else if (isBlocked)
433- then emergencyShutdownFAIL()
434- else {
435- let leasePart = if ((swapType == "waves"))
436- then prepareUnleaseAndLease(0)
437- else nil
438- ([IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser + amount)), IntegerEntry(getBalanceUnlockBlockKey(account), selfUnlockHeight), IntegerEntry(totalLockedKEY(swapType), (totalLocked + amount)), StringEntry(swapKEY(account, txId), pendingSwapDATA(swapType, amount, selfUnlockHeight))] ++ leasePart)
439- }
440- }
441- else throw("Strict value is not equal to itself.")
442- }
443-
444-
445-
446-@Callable(i)
447-func commonWithdraw (account,index,swapTxId,rsaSig) = {
448- let userAddress = addressFromStringValue(account)
449- let checkCaller = thisOnly(i)
450- if ((checkCaller == checkCaller))
451- then {
452- let dataArray = swapDataFailOrREAD(account, swapTxId)
453- let selfUnlockHeight = parseIntValue(dataArray[sIdxSelfUnlockHeight])
454- let swapType = dataArray[sIdxSwapType]
455- let inAmount = parseIntValue(dataArray[sIdxInAmount])
456- let swapStatus = dataArray[sIdxStatus]
457- let startHeight = parseIntValue(dataArray[sIdxStartHeight])
458- let isQuick = (dataArray[sIdxIsQuick] == "true")
459- let checkSwapType = if (if ((swapType != "waves"))
460- then (swapType != "neutrino")
461- else false)
462- then throw(("Unsupported swap type " + swapType))
415+ let checkCaller = if ((i.originCaller != addressFromStringValue(userAddressStr)))
416+ then throw("Permission denied. Wrong origin caller.")
463417 else true
464- if ((checkSwapType == checkSwapType))
418+ if ((checkCaller == checkCaller))
465419 then {
466- let outFeePart = valueOrElse(getInteger(this, outFeePartKEY(swapType)), DEFAULTSWAPFEE)
420+ let minSwapAmount = minSwapAmountREAD(swapType)
467421 let totalLocked = totalLockedREAD(swapType)
468- let totalLockedByUser = totalLockedByUserREAD(swapType, account)
469- let unlockHeight = if (isQuick)
470- then (startHeight + getIntegerValue(this, keyQuickSwapDelay))
471- else if ((rsaSig == base58''))
472- then selfUnlockHeight
473- else randUnlockHeightOrFail(swapTxId, rsaSig, swapType, startHeight, $Tuple2(parseIntValue(dataArray[sIdxMinRand]), parseIntValue(dataArray[sIdxMaxRand])))
474- let indexHeight = getHeightPriceByIndex(index)
475- let prevIndexHeight = getHeightPriceByIndex((index - 1))
476- let priceByIndex = getPriceHistory(indexHeight)
477- let $t02234322539 = if ((swapType == "waves"))
478- then $Tuple2(convertWavesToNeutrino(inAmount, priceByIndex), neutrinoAssetId)
479- else $Tuple2(convertNeutrinoToWaves(inAmount, priceByIndex), unit)
480- let outAmountGross = $t02234322539._1
481- let outAsset = $t02234322539._2
482- let payoutsArray = applyFees(outAmountGross, outFeePart)
483- let outNetAmount = payoutsArray[IdxNetAmount]
484- let outFeeAmount = payoutsArray[IdxFeeAmount]
485- let checks = [if (isBlocked)
486- then emergencyShutdownFAIL()
487- else true, if ((swapStatus != "PENDING"))
488- then throw("swap has been already processed")
489- else true, if ((unlockHeight > height))
490- then throw((("please wait for: " + toString(unlockHeight)) + " block height to withdraw funds"))
491- else true, if (if (if ((index > priceIndex))
492- then true
493- else (unlockHeight > indexHeight))
494- then true
495- else if ((prevIndexHeight != 0))
496- then (prevIndexHeight >= unlockHeight)
497- else false)
498- then priceIndexFAIL(index, priceIndex, indexHeight, unlockHeight, prevIndexHeight)
499- else true, if ((0 >= payoutsArray[IdxGrossAmount]))
500- then throw("balance equals zero")
501- else true, if (if ((0 > outFeePart))
502- then true
503- else (outFeePart >= PAULI))
504- then throw(((("invalid outFeePart config for " + swapType) + " swap: outFeePart=") + toString(outFeePart)))
505- else true]
506- if ((checks == checks))
422+ let totalLockedByUser = totalLockedByUserREAD(swapType, userAddressStr)
423+ let nodeAddress = getStakingNodeByIndex(0)
424+ let balanceLockMaxInterval = if ((nodeAddress == userAddressStr))
425+ then nodeBalanceLockIntervalREAD()
426+ else if ((swapType == "waves"))
427+ then getIntegerValue(this, BalanceWavesLockIntervalKey)
428+ else getIntegerValue(this, BalanceNeutrinoLockIntervalKey)
429+ let selfUnlockHeight = (height + balanceLockMaxInterval)
430+ let amountCheck = if ((minSwapAmount > pmtAmount))
431+ then minSwapAmountFAIL(swapType, minSwapAmount)
432+ else true
433+ if ((amountCheck == amountCheck))
507434 then {
508- let leasePart = if (if ((swapType == "neutrino"))
509- then (outAmountGross > 0)
510- else false)
511- then invoke(this, "prepareUnleaseAndLeaseWrapper", [outAmountGross], nil)
512- else unit
513- if ((leasePart == leasePart))
514- then {
515- let sendFee = invoke(addressFromStringValue(getStringValue(this, keyGNsbtContract)), "deposit", nil, [AttachedPayment(outAsset, outFeeAmount)])
516- if ((sendFee == sendFee))
517- then [IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser - inAmount)), IntegerEntry(totalLockedKEY(swapType), (totalLocked - inAmount)), ScriptTransfer(userAddress, outNetAmount, outAsset), StringEntry(swapKEY(account, swapTxId), finishSwapDATA(dataArray, priceByIndex, outNetAmount, outFeeAmount, unlockHeight, index, toBase58String(i.transactionId)))]
435+ let lastSwapHeight = getNumberByKey(keyUserLastQuickSwapHeight(userAddressStr))
436+ if ((TWENTYFOURHOURS > (height - lastSwapHeight)))
437+ then throw(("You have exceeded swap limit for 24h! Next allowed swap height is " + toString((lastSwapHeight + TWENTYFOURHOURS))))
438+ else {
439+ let userSwapLimitsInfo = asAnyList(invoke(mathContract, "calcUserSwapLimitsMATH", [userAddressStr], nil))
440+ if ((userSwapLimitsInfo == userSwapLimitsInfo))
441+ then {
442+ let quickSwapLimitTotal = asInt(userSwapLimitsInfo[1])
443+ let priceByIndex = getPriceHistory(getHeightPriceByIndex(priceIndex))
444+ let swapUsdnVolume = if ((swapType == "neutrino"))
445+ then pmtAmount
446+ else convertWavesToNeutrino(pmtAmount, priceByIndex)
447+ if ((swapUsdnVolume > quickSwapLimitTotal))
448+ then throw(((("You have exceeded your swap limit! Requested: " + toString(swapUsdnVolume)) + ", available: ") + toString(quickSwapLimitTotal)))
449+ else if (isBlocked)
450+ then emergencyShutdownFAIL()
451+ else {
452+ let leasePart = if ((swapType == "waves"))
453+ then prepareUnleaseAndLease(0)
454+ else nil
455+ $Tuple2(([IntegerEntry(keyQuickSwapUserSpentInPeriod(userAddressStr), swapUsdnVolume), IntegerEntry(keyUserLastQuickSwapHeight(userAddressStr), height), IntegerEntry(totalLockedByUserKEY(swapType, userAddressStr), (totalLockedByUser + pmtAmount)), IntegerEntry(getBalanceUnlockBlockKey(userAddressStr), selfUnlockHeight), IntegerEntry(totalLockedKEY(swapType), (totalLocked + pmtAmount)), StringEntry(swapKEY(userAddressStr, txId58), pendingSwapDATA(swapType, pmtAmount, selfUnlockHeight))] ++ leasePart), unit)
456+ }
457+ }
518458 else throw("Strict value is not equal to itself.")
519459 }
520- else throw("Strict value is not equal to itself.")
521460 }
522461 else throw("Strict value is not equal to itself.")
523462 }
529468
530469
531470 @Callable(i)
532-func swap (swapType) = {
533- let accelerate = true
534- let wavesToNeutrino = (swapType == "waves")
535- let neutrinoToWaves = (swapType == "neutrino")
471+func swapWavesToNeutrino () = {
536472 let pmt = value(i.payments[0])
537- let account = toString(i.caller)
538- let swapTxId = toBase58String(i.transactionId)
539- if (if (wavesToNeutrino)
540- then (pmt.assetId != unit)
541- else false)
473+ if (isDefined(pmt.assetId))
542474 then throw("Only Waves token is allowed for swapping.")
543- else if (if (neutrinoToWaves)
544- then (pmt.assetId != neutrinoAssetId)
545- else false)
546- then throw("Only appropriate Neutrino tokens are allowed for swapping.")
547- else if (if (!(wavesToNeutrino))
548- then !(neutrinoToWaves)
549- else false)
550- then throw("Invalid swap type")
551- else {
552- let commonSwapInv = invoke(this, "commonSwap", [swapType, pmt.amount, account, swapTxId], nil)
553- if ((commonSwapInv == commonSwapInv))
554- then {
555- let accelerateInv = if (accelerate)
556- then invoke(this, "accelerate", [account, swapTxId], nil)
557- else true
558- if ((accelerateInv == accelerateInv))
559- then nil
560- else throw("Strict value is not equal to itself.")
561- }
562- else throw("Strict value is not equal to itself.")
563- }
475+ else {
476+ let userAddress = toString(i.caller)
477+ let txId58 = toBase58String(i.transactionId)
478+ let commonSwapInv = invoke(this, "commonSwap", ["waves", pmt.amount, userAddress, txId58], nil)
479+ if ((commonSwapInv == commonSwapInv))
480+ then nil
481+ else throw("Strict value is not equal to itself.")
482+ }
564483 }
565484
566485
567486
568487 @Callable(i)
569-func accelerate (account,swapTxId) = {
570- let checkCaller = thisOnly(i)
571- if ((checkCaller == checkCaller))
572- then {
573- let userAddress = addressFromStringValue(account)
574- let dataArray = swapDataFailOrREAD(account, swapTxId)
575- let swapType = dataArray[sIdxSwapType]
576- let inAmount = parseIntValue(dataArray[sIdxInAmount])
577- let swapStatus = dataArray[sIdxStatus]
578- let isQuick = (dataArray[sIdxIsQuick] == "true")
579- let $t02544625654 = match invoke(this, "quickSwapLimitREADONLY", [toString(userAddress)], nil) {
580- case r: (Int, Int) =>
581- $Tuple2(r._1, r._2)
582- case _ =>
583- throw("Type casting error")
584- }
585- let quickSwapLimitTotal = $t02544625654._1
586- let quickSwapUserSpent = $t02544625654._2
587- let quickSwapLimit = (quickSwapLimitTotal - quickSwapUserSpent)
588- let priceByIndex = getPriceHistory(getHeightPriceByIndex(priceIndex))
589- let swapUsdnVolume = if ((swapType == "neutrino"))
590- then inAmount
591- else convertWavesToNeutrino(inAmount, priceByIndex)
592- let checks = [if ((swapStatus != "PENDING"))
593- then throw("Swap has been already processed")
594- else true, if (isQuick)
595- then throw("Swap has been already accelerated")
596- else true, if ((swapUsdnVolume > quickSwapLimit))
597- then throw(((("You have exceeded the quick swaps limit! Requested: " + toString(swapUsdnVolume)) + ", available: ") + toString(quickSwapLimit)))
598- else true]
599- if ((checks == checks))
600- then [IntegerEntry(keyQuickSwapUserSpentInPeriod(userAddress), (swapUsdnVolume + quickSwapUserSpent)), IntegerEntry(keyUserLastQuickSwapHeight(userAddress), height), StringEntry(swapKEY(account, swapTxId), accelerateSwapDATA(dataArray))]
488+func swapNeutrinoToWaves () = {
489+ let pmt = value(i.payments[0])
490+ if ((pmt.assetId != neutrinoAssetId))
491+ then throw("Only appropriate Neutrino tokens are allowed for swapping.")
492+ else {
493+ let userAddress = toString(i.caller)
494+ let txId58 = toBase58String(i.transactionId)
495+ let commonSwapInv = invoke(this, "commonSwap", ["neutrino", pmt.amount, userAddress, txId58], nil)
496+ if ((commonSwapInv == commonSwapInv))
497+ then nil
601498 else throw("Strict value is not equal to itself.")
602499 }
603- else throw("Strict value is not equal to itself.")
604500 }
605501
606502
607503
608504 @Callable(i)
609-func withdraw (account,index,swapTxId) = {
610- let result = invoke(this, "commonWithdraw", [account, index, swapTxId, base58''], nil)
611- if ((result == result))
612- then nil
613- else throw("Strict value is not equal to itself.")
614- }
615-
616-
617-
618-@Callable(i)
619-func withdrawRand (account,index,swapTxId,rsaSig) = {
620- let result = invoke(this, "commonWithdraw", [account, index, swapTxId, rsaSig], nil)
621- if ((result == result))
622- then nil
623- else throw("Strict value is not equal to itself.")
624- }
505+func withdraw (account,index,swapTxId) = commonWithdraw(account, index, swapTxId, i)
625506
626507
627508
628509 @Callable(i)
629510 func transferToAuction () = {
511+ let neutrinoMetrics = asAnyList(invoke(mathContract, "calcNeutinoMetricsMATH", nil, nil))
512+ let reserve = asInt(neutrinoMetrics[4])
513+ let neutrinoSupply = asInt(neutrinoMetrics[6])
514+ let surplus = asInt(neutrinoMetrics[7])
515+ let nsbtSupply = asInt(neutrinoMetrics[10])
630516 let auctionNBAmount = (neutrinoSupply - assetBalance(addressFromStringValue(auctionContract), bondAssetId))
631517 let surplusWithLiquidation = (surplus - assetBalance(addressFromStringValue(liquidationContract), neutrinoAssetId))
632518 if (isBlocked)
646532 else $Tuple2(prepareUnleaseAndLease(0), "success")
647533
648534
649-
650-@Callable(i)
651-func calcTotalQuickSwapLimitREADONLY (gNsbtAmount) = {
652- let $t02870528762 = getDora2NSBTInfo()
653- let nsbtPrice = $t02870528762._1
654- let nsbtPriceLastHeight = $t02870528762._2
655- let check = if (((height - nsbtPriceLastHeight) > getIntegerValue(this, keyDora2LastHeightLimit)))
656- then throw("NSBT price is outdated")
657- else true
658- if ((check == check))
659- then $Tuple2(nil, fraction(gNsbtAmount, nsbtPrice, WAVELET))
660- else throw("Strict value is not equal to itself.")
661- }
662-
663-
664-
665-@Callable(i)
666-func quickSwapLimitREADONLY (userAddressStr) = {
667- let limitAvalableWaves = toString((10000 * 100000000))
668- let limitAvalableUsdn = toString((100000 * 1000000))
669- let limitRemainingBlocksToReset = toString(940)
670- $Tuple2(nil, makeString(["%d%d%d", limitAvalableWaves, limitAvalableUsdn, limitRemainingBlocksToReset], SEP))
671- }
672-
673-
674535 @Verifier(tx)
675536 func verify () = {
676537 let id = toBase58String(tx.id)
677- let minSignaturesNumber = 3
678538 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
679539 then 1
680540 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
684544 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
685545 then 2
686546 else 0))
687- let isSignaturesValid = (count >= minSignaturesNumber)
688547 match tx {
689548 case sponsorTx: SponsorFeeTransaction =>
690- if (isSignaturesValid)
691- then checkIsValidMinSponsoredFee(sponsorTx)
549+ if (checkIsValidMinSponsoredFee(sponsorTx))
550+ then (count >= 3)
692551 else false
693552 case _ =>
694- isSignaturesValid
553+ (count >= 3)
695554 }
696555 }
697556
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 func getNumberByKey (key) = valueOrElse(getInteger(this, key), 0)
55
66
77 func getStringByKey (key) = valueOrElse(getString(this, key), "")
88
99
1010 func getBoolByKey (key) = valueOrElse(getBoolean(this, key), false)
1111
1212
1313 func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(addressFromStringValue(address), key), 0)
1414
1515
1616 func getStringByAddressAndKey (address,key) = valueOrElse(getString(addressFromStringValue(address), key), "")
1717
1818
1919 func getBoolByAddressAndKey (address,key) = valueOrElse(getBoolean(addressFromStringValue(address), key), false)
2020
2121
22-let pubKeyAdminsList = ["GXxmZaCigPgJsT6d1cLr8nUXA65nGX8V7FuAzzrY8wQE", "3xumx4hrPpeeoo79HLGbZCgfz95L39ZFdxxwWeSUC2Pu", "D4CqtK5fBsnzRo33ZMhrM7niLseDH2uRoK6weBq2JQ3A", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
22+func asAnyList (val) = match val {
23+ case valAnyLyst: List[Any] =>
24+ valAnyLyst
25+ case _ =>
26+ throw("fail to cast into List[Any]")
27+}
28+
29+
30+func asString (val) = match val {
31+ case valStr: String =>
32+ valStr
33+ case _ =>
34+ throw("fail to cast into String")
35+}
36+
2337
2438 func asInt (val) = match val {
2539 case valInt: Int =>
2640 valInt
2741 case _ =>
28- throw("Failed to cast into Int")
42+ throw("fail to cast into Int")
2943 }
3044
45+
46+let pubKeyAdminsList = ["GXxmZaCigPgJsT6d1cLr8nUXA65nGX8V7FuAzzrY8wQE", "3xumx4hrPpeeoo79HLGbZCgfz95L39ZFdxxwWeSUC2Pu", "D4CqtK5fBsnzRo33ZMhrM7niLseDH2uRoK6weBq2JQ3A", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"]
3147
3248 let SEP = "__"
3349
3450 let WAVELET = 100000000
3551
3652 let PAULI = 1000000
3753
3854 let PRICELET = 1000000
3955
4056 let DEFAULTSWAPFEE = 20000
4157
58+let TWENTYFOURHOURS = 1440
59+
4260 let IdxNetAmount = 0
4361
4462 let IdxFeeAmount = 1
4563
4664 let IdxGrossAmount = 2
47-
48-let dora2NsbtSymbol = "NSBT-USDT"
49-
50-let minRand = 60
51-
52-let maxRand = 1440
5365
5466 let NeutrinoAssetIdKey = "neutrino_asset_id"
5567
5668 let BondAssetIdKey = "bond_asset_id"
5769
5870 let AuctionContractKey = "auction_contract"
5971
6072 let LiquidationContractKey = "liquidation_contract"
6173
6274 let RPDContractKey = "rpd_contract"
6375
64-let ControlContractKey = "control_contract"
76+let ContolContractKey = "control_contract"
77+
78+let RestContractKey = "rest_contract"
79+
80+let MathContractKey = "math_contract"
6581
6682 let BalanceWavesLockIntervalKey = "balance_waves_lock_interval"
6783
6884 let BalanceNeutrinoLockIntervalKey = "balance_neutrino_lock_interval"
6985
7086 let MinWavesSwapAmountKey = "min_waves_swap_amount"
7187
7288 let MinNeutrinoSwapAmountKey = "min_neutrino_swap_amount"
7389
7490 let NodeOracleProviderPubKeyKey = "node_oracle_provider"
7591
7692 let NeutrinoOutFeePartKey = "neutrinoOut_swap_feePart"
7793
7894 let WavesOutFeePartKey = "wavesOut_swap_feePart"
7995
80-let RsaRandPublic58Key = "rand_rsa_public"
96+let FeesManagerAddressKey = "fees_manager_address"
8197
82-let keyGNsbtContract = "%s__gNsbtContract"
98+func keyQuickSwapLimitDuration () = "%s__quickSwapLimitDuration"
8399
84-let keyNsbtLockContract = "%s__nsbtLockContract"
85-
86-let keyDora2Contract = "%s__dora2Contract"
87-
88-let keyQuickSwapDelay = "%s__quickSwapDelay"
89-
90-let keyQuickSwapLimitDuration = "%s__quickSwapLimitDuration"
91-
92-let keyDora2LastHeightLimit = "%s__dora2LastHeightLimit"
93100
94101 let PriceKey = "price"
95102
96103 let PriceIndexKey = "price_index"
97104
98105 let IsBlockedKey = "is_blocked"
99106
100107 func getPriceHistoryKey (block) = ((PriceKey + "_") + toString(block))
101108
102109
103110 func getHeightPriceByIndexKey (index) = ((PriceIndexKey + "_") + toString(index))
104111
105112
106113 func getStakingNodeByIndex (idx) = getStringByKey(makeString(["%s%d%s", "lease", toString(idx), "nodeAddress"], SEP))
107114
108115
109116 func getStakingNodeAddressByIndex (idx) = addressFromStringValue(getStakingNodeByIndex(idx))
110117
111118
112119 func getReservedAmountForSponsorship () = valueOrElse(getInteger(this, makeString(["%s%s", "lease", "sponsorshipWavesReserve"], SEP)), (1000 * WAVELET))
113120
114121
115122 func getBalanceUnlockBlockKey (owner) = ("balance_unlock_block_" + owner)
116123
117124
118125 func getLeaseIdKey (nodeIndex) = makeString(["%s%d%s", "lease", toString(nodeIndex), "id"], SEP)
119126
120127
121128 func getLeaseAmountKey (nodeIndex) = makeString(["%s%d%s", "lease", toString(nodeIndex), "amount"], SEP)
122129
123130
124131 func minSwapAmountKEY (swapType) = (("min_" + swapType) + "_swap_amount")
125132
126133
127134 func totalLockedKEY (swapType) = ("balance_lock_" + swapType)
128135
129136
130137 func totalLockedByUserKEY (swapType,owner) = makeString(["balance_lock", swapType, owner], "_")
131138
132139
133140 func balanceLockIntervalKEY (swapType) = (("balance_" + swapType) + "_lock_interval")
134141
135142
136143 func minBalanceLockIntervalKEY (swapType) = (("balance_" + swapType) + "_lock_interval_minimum")
137144
138145
139146 func nodeBalanceLockIntervalKEY () = "balance_node_lock_interval"
140147
141148
142149 func outFeePartKEY (swapType) = (swapType + "Out_swap_feePart")
143150
144151
145-func keyQuickSwapUserSpentInPeriod (userAddress) = makeString(["%s%s", "quickSwapUserSpentInPeriod", toString(userAddress)], SEP)
146-
147-
148-func keyUserLastQuickSwapHeight (userAddress) = makeString(["%s%s", "userLastQuickSwapHeight", toString(userAddress)], SEP)
149-
150-
151-func keyDora2Price (symbol) = makeString(["%s%s", "price", symbol], SEP)
152-
153-
154-func keyDora2LastHeight (symbol) = makeString(["%s%s", "lastHeight", symbol], SEP)
155-
156-
157152 func minSwapAmountREAD (swapType) = valueOrElse(getInteger(this, minSwapAmountKEY(swapType)), 0)
158153
159154
160155 func totalLockedREAD (swapType) = valueOrElse(getInteger(this, totalLockedKEY(swapType)), 0)
161156
162157
163158 func totalLockedByUserREAD (swapType,owner) = valueOrElse(getInteger(this, totalLockedByUserKEY(swapType, owner)), 0)
164159
165160
166-func balanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, balanceLockIntervalKEY(swapType)), maxRand)
161+func balanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, balanceLockIntervalKEY(swapType)), 1440)
167162
168163
169-func minBalanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, minBalanceLockIntervalKEY(swapType)), minRand)
164+func minBalanceLockIntervalREAD (swapType) = valueOrElse(getInteger(this, minBalanceLockIntervalKEY(swapType)), 60)
170165
171166
172167 func nodeBalanceLockIntervalREAD () = valueOrElse(getInteger(this, nodeBalanceLockIntervalKEY()), 1)
168+
169+
170+func keyQuickSwapUserSpentInPeriod (userAddress) = makeString(["%s%s", "quickSwapUserSpentInPeriod", userAddress], SEP)
171+
172+
173+func keyUserLastQuickSwapHeight (userAddress) = makeString(["%s%s", "userLastQuickSwapHeight", userAddress], SEP)
174+
175+
176+func feeManagerAddressREAD () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(this, FeesManagerAddressKey), (FeesManagerAddressKey + " is not specified"))), (FeesManagerAddressKey + " invalid address format"))
173177
174178
175179 func convertNeutrinoToWaves (amount,price) = fraction(fraction(amount, PRICELET, price), WAVELET, PAULI)
176180
177181
178182 func convertWavesToNeutrino (amount,price) = fraction(fraction(amount, price, PRICELET), PAULI, WAVELET)
179183
180184
181185 func convertWavesToBond (amount,price) = convertWavesToNeutrino(amount, price)
182186
183187
184188 func convertJsonArrayToList (jsonArray) = split(jsonArray, ",")
185189
186190
187191 func minSwapAmountFAIL (swapType,minSwapAmount) = throw(((("The specified amount in " + swapType) + " swap is less than the required minimum of ") + toString(minSwapAmount)))
188192
189193
190194 func emergencyShutdownFAIL () = throw("contract is blocked by EMERGENCY SHUTDOWN actions untill reactivation by emergency oracles")
191195
192196
193197 func priceIndexFAIL (index,priceIndex,indexHeight,unlockHeight,prevIndexHeight) = throw(((((((((("invalid price history index: index=" + toString(index)) + " priceIndex=") + toString(priceIndex)) + " indexHeight=") + toString(indexHeight)) + " unlockHeight=") + toString(unlockHeight)) + " prevIndexHeight=") + toString(prevIndexHeight)))
194198
195199
196200 let liquidationContract = getStringByKey(LiquidationContractKey)
197201
198202 let neutrinoAssetId = fromBase58String(getStringByKey(NeutrinoAssetIdKey))
199203
200204 let auctionContract = getStringByKey(AuctionContractKey)
201205
202206 let rpdContract = getStringByKey(RPDContractKey)
203207
204-let controlContract = getStringByKey(ControlContractKey)
208+let controlContract = getStringByKey(ContolContractKey)
209+
210+let restContract = getStringByKey(RestContractKey)
211+
212+let mathContractAddress = getStringByKey(MathContractKey)
205213
206214 let priceIndex = getNumberByAddressAndKey(controlContract, PriceIndexKey)
207215
208216 let isBlocked = getBoolByAddressAndKey(controlContract, IsBlockedKey)
209217
210218 let nodeOracleProviderPubKey = fromBase58String(getStringByKey(NodeOracleProviderPubKeyKey))
211219
212220 let bondAssetId = fromBase58String("BGhPYquXSK6UYjnPLEn94gFhetp4Jvf7toekEY4x1VDH")
213221
214222 let deprecatedBondAssetId = fromBase58String("975akZBfnMj513U7MZaHKzQrmsEx5aE3wdWKTrHBhbjF")
215223
216-let rsaPub = fromBase64String(valueOrErrorMessage(getString(this, RsaRandPublic58Key), "RSA public key has not been specified"))
217-
218224 let neutrinoContract = this
219225
226+let mathContract = addressFromStringValue(mathContractAddress)
227+
220228 let currentPrice = getNumberByAddressAndKey(controlContract, PriceKey)
221-
222-let neutrinoLockedBalance = totalLockedREAD("neutrino")
223-
224-let wavesLockedBalance = totalLockedREAD("waves")
225-
226-let reserve = (wavesBalance(neutrinoContract).regular - wavesLockedBalance)
227-
228-let neutrinoSupply = (((neutrinoLockedBalance + value(assetInfo(neutrinoAssetId)).quantity) - assetBalance(neutrinoContract, neutrinoAssetId)) - assetBalance(addressFromStringValue(liquidationContract), neutrinoAssetId))
229-
230-let surplus = (convertWavesToNeutrino(reserve, currentPrice) - neutrinoSupply)
231-
232-let deficit = (neutrinoSupply - convertWavesToNeutrino(reserve, currentPrice))
233229
234230 func checkIsValidMinSponsoredFee (tx) = {
235231 let MINTRANSFERFEE = 100000
236232 let SponsoredFeeUpperBound = 1000
237233 let realNeutrinoFee = convertWavesToNeutrino(MINTRANSFERFEE, currentPrice)
238234 let minNeutrinoFee = (realNeutrinoFee * 2)
239235 let maxNeutrinoFee = fraction(realNeutrinoFee, SponsoredFeeUpperBound, 100)
240236 let inputFee = value(tx.minSponsoredAssetFee)
241237 if (if ((inputFee >= minNeutrinoFee))
242238 then (maxNeutrinoFee >= inputFee)
243239 else false)
244240 then (tx.assetId == neutrinoAssetId)
245241 else false
246242 }
247243
248244
249245 func getPriceHistory (block) = getNumberByAddressAndKey(controlContract, getPriceHistoryKey(block))
250246
251247
252248 func getHeightPriceByIndex (index) = getNumberByAddressAndKey(controlContract, getHeightPriceByIndexKey(index))
253249
254250
255251 let sIdxSwapType = 1
256252
257253 let sIdxStatus = 2
258254
259255 let sIdxInAmount = 3
260256
261257 let sIdxPrice = 4
262258
263259 let sIdxOutNetAmount = 5
264260
265261 let sIdxOutFeeAmount = 6
266262
267263 let sIdxStartHeight = 7
268264
269265 let sIdxStartTimestamp = 8
270266
271267 let sIdxEndHeight = 9
272268
273269 let sIdxEndTimestamp = 10
274270
275271 let sIdxSelfUnlockHeight = 11
276272
277273 let sIdxRandUnlockHeight = 12
278274
279275 let sIdxIndex = 13
280276
281277 let sIdxWithdrawTxId = 14
282278
283279 let sIdxMinRand = 15
284280
285281 let sIdxMaxRand = 16
286282
287-let sIdxIsQuick = 17
288-
289283 func swapKEY (userAddress,txId) = makeString(["%s%s", userAddress, txId], SEP)
290284
291285
292-func strSwapDATA (swapType,status,inAmount,price,outNetAmount,outFeeAmount,startHeight,startTimestamp,endHeight,endTimestamp,selfUnlockHeight,randUnlockHeight,index,withdrawTxId,randMin,randMax,isQuick) = makeString(["%s%s%d%d%d%d%d%d%d%d%d%d%d%s", swapType, status, inAmount, price, outNetAmount, outFeeAmount, startHeight, startTimestamp, endHeight, endTimestamp, selfUnlockHeight, randUnlockHeight, index, withdrawTxId, randMin, randMax, isQuick], SEP)
286+func strSwapDATA (swapType,status,inAmount,price,outNetAmount,outFeeAmount,startHeight,startTimestamp,endHeight,endTimestamp,selfUnlockHeight,randUnlockHeight,index,withdrawTxId,randMin,randMax) = makeString(["%s%s%d%d%d%d%d%d%d%d%d%d%d%s", swapType, status, inAmount, price, outNetAmount, outFeeAmount, startHeight, startTimestamp, endHeight, endTimestamp, selfUnlockHeight, randUnlockHeight, index, withdrawTxId, randMin, randMax], SEP)
293287
294288
295-func pendingSwapDATA (swapType,inAssetAmount,selfUnlockHeight) = strSwapDATA(swapType, "PENDING", toString(inAssetAmount), "0", "0", "0", toString(height), toString(lastBlock.timestamp), "0", "0", toString(selfUnlockHeight), "0", "0", "NULL", toString(minBalanceLockIntervalREAD(swapType)), toString(balanceLockIntervalREAD(swapType)), toString(false))
289+func pendingSwapDATA (swapType,inAssetAmount,selfUnlockHeight) = strSwapDATA(swapType, "PENDING", toString(inAssetAmount), "0", "0", "0", toString(height), toString(lastBlock.timestamp), "0", "0", toString(selfUnlockHeight), "0", "0", "NULL", "0", "0")
296290
297291
298-func accelerateSwapDATA (dataArray) = strSwapDATA(dataArray[sIdxSwapType], dataArray[sIdxStatus], dataArray[sIdxInAmount], dataArray[sIdxPrice], dataArray[sIdxOutNetAmount], dataArray[sIdxOutFeeAmount], dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], dataArray[sIdxEndHeight], dataArray[sIdxEndTimestamp], dataArray[sIdxSelfUnlockHeight], dataArray[sIdxRandUnlockHeight], dataArray[sIdxIndex], dataArray[sIdxWithdrawTxId], dataArray[sIdxMinRand], dataArray[sIdxMaxRand], toString(true))
299-
300-
301-func finishSwapDATA (dataArray,price,outNetAmount,outFeeAmount,randUnlockHeight,index,withdrawTxId) = strSwapDATA(dataArray[sIdxSwapType], "FINISHED", dataArray[sIdxInAmount], toString(price), toString(outNetAmount), toString(outFeeAmount), dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], toString(height), toString(lastBlock.timestamp), dataArray[sIdxSelfUnlockHeight], toString(randUnlockHeight), toString(index), withdrawTxId, dataArray[sIdxMinRand], dataArray[sIdxMaxRand], dataArray[sIdxIsQuick])
292+func finishSwapDATA (dataArray,price,outNetAmount,outFeeAmount,randUnlockHeight,index,withdrawTxId) = strSwapDATA(dataArray[sIdxSwapType], "FINISHED", dataArray[sIdxInAmount], toString(price), toString(outNetAmount), toString(outFeeAmount), dataArray[sIdxStartHeight], dataArray[sIdxStartTimestamp], toString(height), toString(lastBlock.timestamp), dataArray[sIdxSelfUnlockHeight], toString(randUnlockHeight), toString(index), withdrawTxId, dataArray[sIdxMinRand], dataArray[sIdxMaxRand])
302293
303294
304295 func swapDataFailOrREAD (userAddress,swapTxId) = {
305296 let swapKey = swapKEY(userAddress, swapTxId)
306297 split(valueOrErrorMessage(getString(this, swapKey), ("no swap data for " + swapKey)), SEP)
307298 }
308299
309300
310301 func applyFees (amountGross,feePart) = {
311302 let feeAmount = fraction(amountGross, feePart, PAULI)
312303 [(amountGross - feeAmount), feeAmount, amountGross]
313- }
314-
315-
316-func randUnlockHeightOrFail (txId,rsaSig,swapType,startHeight,minMaxRandsTuple) = {
317- let isRsaValid = rsaVerify_16Kb(SHA256, toBytes(txId), rsaSig, rsaPub)
318- if (!(isRsaValid))
319- then throw("invalid RSA signature")
320- else {
321- let minBalanceLockInterval = minMaxRandsTuple._1
322- let maxBalanceLockInterval = minMaxRandsTuple._2
323- let rand = (toInt(sha256_16Kb(rsaSig)) % (maxBalanceLockInterval - minBalanceLockInterval))
324- let randLockInterval = (minBalanceLockInterval + (if ((0 > rand))
325- then -(rand)
326- else rand))
327- (startHeight + randLockInterval)
328- }
329304 }
330305
331306
332307 func abs (x) = if ((0 > x))
333308 then -(x)
334309 else x
335310
336311
337312 func selectNode (unleaseAmount) = {
338313 let amountToLease = ((wavesBalance(neutrinoContract).available - unleaseAmount) - getReservedAmountForSponsorship())
339314 let oldLeased0 = getNumberByKey(getLeaseAmountKey(0))
340315 let oldLeased1 = getNumberByKey(getLeaseAmountKey(1))
341316 let newLeased0 = (amountToLease + oldLeased0)
342317 let newLeased1 = (amountToLease + oldLeased1)
343318 if (if ((newLeased0 > 0))
344319 then true
345320 else (newLeased1 > 0))
346321 then {
347322 let delta0 = abs((newLeased0 - oldLeased1))
348323 let delta1 = abs((newLeased1 - oldLeased0))
349324 if ((delta1 >= delta0))
350325 then $Tuple2(0, newLeased0)
351326 else $Tuple2(1, newLeased1)
352327 }
353328 else $Tuple2(-1, 0)
354329 }
355330
356331
357332 func prepareUnleaseAndLease (unleaseAmount) = {
358- let $t01621216300 = selectNode(unleaseAmount)
359- let nodeIndex = $t01621216300._1
360- let newLeaseAmount = $t01621216300._2
333+ let nodeTuple = selectNode(unleaseAmount)
334+ let nodeIndex = nodeTuple._1
335+ let newLeaseAmount = nodeTuple._2
361336 if ((newLeaseAmount > 0))
362337 then {
363338 let leaseIdKey = getLeaseIdKey(nodeIndex)
364339 let oldLease = getBinary(this, leaseIdKey)
365340 let unleaseOrEmpty = if (isDefined(oldLease))
366341 then [LeaseCancel(value(oldLease))]
367342 else nil
368343 let leaseAmountKey = getLeaseAmountKey(nodeIndex)
369344 let lease = Lease(getStakingNodeAddressByIndex(nodeIndex), newLeaseAmount)
370345 (unleaseOrEmpty ++ [lease, BinaryEntry(leaseIdKey, calculateLeaseId(lease)), IntegerEntry(getLeaseAmountKey(nodeIndex), newLeaseAmount)])
371346 }
372347 else nil
373348 }
374349
375350
376-func getDora2NSBTInfo () = {
377- let dora2Contract = addressFromStringValue(getStringValue(this, keyDora2Contract))
378- $Tuple2(valueOrErrorMessage(getInteger(dora2Contract, keyDora2Price(dora2NsbtSymbol)), "NSBT price is undefined"), valueOrElse(getInteger(dora2Contract, keyDora2LastHeight(dora2NsbtSymbol)), 0))
351+func commonWithdraw (account,index,swapTxId,i) = {
352+ let userAddress = addressFromStringValue(account)
353+ let feeManagerAddress = feeManagerAddressREAD()
354+ let dataArray = swapDataFailOrREAD(account, swapTxId)
355+ let selfUnlockHeight = parseIntValue(dataArray[sIdxSelfUnlockHeight])
356+ let swapType = dataArray[sIdxSwapType]
357+ let inAmount = parseIntValue(dataArray[sIdxInAmount])
358+ let swapStatus = dataArray[sIdxStatus]
359+ let startHeight = parseIntValue(dataArray[sIdxStartHeight])
360+ let outFeePart = valueOrElse(getInteger(this, outFeePartKEY(swapType)), DEFAULTSWAPFEE)
361+ let totalLocked = totalLockedREAD(swapType)
362+ let totalLockedByUser = totalLockedByUserREAD(swapType, account)
363+ let unlockHeight = (startHeight + (if ((swapType == "waves"))
364+ then getIntegerValue(this, BalanceWavesLockIntervalKey)
365+ else getIntegerValue(this, BalanceNeutrinoLockIntervalKey)))
366+ let indexHeight = getHeightPriceByIndex(index)
367+ let prevIndexHeight = getHeightPriceByIndex((index - 1))
368+ let priceByIndex = getPriceHistory(indexHeight)
369+ let outAmountGrossTuple = if ((swapType == "waves"))
370+ then $Tuple2(convertWavesToNeutrino(inAmount, priceByIndex), neutrinoAssetId)
371+ else if ((swapType == "neutrino"))
372+ then $Tuple2(convertNeutrinoToWaves(inAmount, priceByIndex), unit)
373+ else throw(("Unsupported swap type " + swapType))
374+ let payoutsArray = applyFees(outAmountGrossTuple._1, outFeePart)
375+ let outNetAmount = payoutsArray[IdxNetAmount]
376+ let outFeeAmount = payoutsArray[IdxFeeAmount]
377+ if (isBlocked)
378+ then emergencyShutdownFAIL()
379+ else if ((swapStatus != "PENDING"))
380+ then throw("swap has been already processed")
381+ else if ((unlockHeight > height))
382+ then throw((("please wait for: " + toString(unlockHeight)) + " block height to withdraw funds"))
383+ else if (if (if ((index > priceIndex))
384+ then true
385+ else (unlockHeight > indexHeight))
386+ then true
387+ else if ((prevIndexHeight != 0))
388+ then (prevIndexHeight >= unlockHeight)
389+ else false)
390+ then priceIndexFAIL(index, priceIndex, indexHeight, unlockHeight, prevIndexHeight)
391+ else if ((0 >= payoutsArray[IdxGrossAmount]))
392+ then throw("balance equals zero")
393+ else if (if ((0 > outFeePart))
394+ then true
395+ else (outFeePart >= PAULI))
396+ then throw(((("invalid outFeePart config for " + swapType) + " swap: outFeePart=") + toString(outFeePart)))
397+ else {
398+ let leasePart = if (if ((swapType == "neutrino"))
399+ then (outAmountGrossTuple._1 > 0)
400+ else false)
401+ then prepareUnleaseAndLease(outAmountGrossTuple._1)
402+ else nil
403+ $Tuple2((leasePart ++ [IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser - inAmount)), IntegerEntry(totalLockedKEY(swapType), (totalLocked - inAmount)), ScriptTransfer(userAddress, outNetAmount, outAmountGrossTuple._2), ScriptTransfer(feeManagerAddress, outFeeAmount, outAmountGrossTuple._2), StringEntry(swapKEY(account, swapTxId), finishSwapDATA(dataArray, priceByIndex, outNetAmount, outFeeAmount, unlockHeight, index, toBase58String(i.transactionId)))]), unit)
404+ }
379405 }
380406
381407
382-func thisOnly (i) = if ((i.caller != this))
383- then throw("Permission denied: this contract only allowed")
384- else true
385-
386-
387408 @Callable(i)
388-func prepareUnleaseAndLeaseWrapper (unleaseAmount) = {
389- let checkCaller = [thisOnly(i)]
390- if ((checkCaller == checkCaller))
391- then prepareUnleaseAndLease(unleaseAmount)
392- else throw("Strict value is not equal to itself.")
393- }
394-
395-
396-
397-@Callable(i)
398-func constructor (neutrinoAssetIdPrm,bondAssetIdPrm,auctionContractPrm,liquidationContractPrm,rpdContractPrm,controlContractPrm,nodeOracleProviderPubKeyPrm,rsaRandPublicKeyPrm,balanceWavesLockIntervalPrm,balanceNeutrinoLockIntervalPrm,minWavesSwapAmountPrm,minNeutrinoSwapAmountPrm,neutrinoOutFeePartPrm,wavesOutFeePartPrm) = {
399- let checkCaller = thisOnly(i)
400- if ((checkCaller == checkCaller))
401- then [StringEntry(NeutrinoAssetIdKey, neutrinoAssetIdPrm), StringEntry(BondAssetIdKey, bondAssetIdPrm), StringEntry(AuctionContractKey, auctionContractPrm), StringEntry(LiquidationContractKey, liquidationContractPrm), StringEntry(RPDContractKey, rpdContractPrm), StringEntry(ControlContractKey, controlContractPrm), StringEntry(NodeOracleProviderPubKeyKey, nodeOracleProviderPubKeyPrm), StringEntry(RsaRandPublic58Key, rsaRandPublicKeyPrm), IntegerEntry(BalanceWavesLockIntervalKey, balanceWavesLockIntervalPrm), IntegerEntry(BalanceNeutrinoLockIntervalKey, balanceNeutrinoLockIntervalPrm), IntegerEntry(MinWavesSwapAmountKey, minWavesSwapAmountPrm), IntegerEntry(MinNeutrinoSwapAmountKey, minNeutrinoSwapAmountPrm), IntegerEntry(NeutrinoOutFeePartKey, neutrinoOutFeePartPrm), IntegerEntry(WavesOutFeePartKey, wavesOutFeePartPrm)]
402- else throw("Strict value is not equal to itself.")
403- }
404-
405-
406-
407-@Callable(i)
408-func constructorV2 (gNsbtContract,nsbtLockContract,dora2Contract,quickSwapDelay,quickSwapLimitDuration,dora2LastHeightLimit) = {
409- let checkCaller = thisOnly(i)
410- if ((checkCaller == checkCaller))
411- then [StringEntry(keyGNsbtContract, gNsbtContract), StringEntry(keyNsbtLockContract, nsbtLockContract), StringEntry(keyDora2Contract, dora2Contract), IntegerEntry(keyQuickSwapDelay, quickSwapDelay), IntegerEntry(keyQuickSwapLimitDuration, quickSwapLimitDuration), IntegerEntry(keyDora2LastHeightLimit, dora2LastHeightLimit)]
412- else throw("Strict value is not equal to itself.")
413- }
414-
415-
416-
417-@Callable(i)
418-func commonSwap (swapType,amount,account,txId) = {
419- let checkCaller = thisOnly(i)
420- if ((checkCaller == checkCaller))
409+func commonSwap (swapType,pmtAmount,userAddressStr,txId58) = {
410+ let checkThisOriginCaller = if ((i.caller != this))
411+ then throw("Permission denied. Wrong caller.")
412+ else true
413+ if ((checkThisOriginCaller == checkThisOriginCaller))
421414 then {
422- let minSwapAmount = minSwapAmountREAD(swapType)
423- let totalLocked = totalLockedREAD(swapType)
424- let totalLockedByUser = totalLockedByUserREAD(swapType, account)
425- let nodeAddress = getStakingNodeByIndex(0)
426- let balanceLockMaxInterval = if ((nodeAddress == account))
427- then nodeBalanceLockIntervalREAD()
428- else balanceLockIntervalREAD(swapType)
429- let selfUnlockHeight = (height + balanceLockMaxInterval)
430- if ((minSwapAmount > amount))
431- then minSwapAmountFAIL(swapType, minSwapAmount)
432- else if (isBlocked)
433- then emergencyShutdownFAIL()
434- else {
435- let leasePart = if ((swapType == "waves"))
436- then prepareUnleaseAndLease(0)
437- else nil
438- ([IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser + amount)), IntegerEntry(getBalanceUnlockBlockKey(account), selfUnlockHeight), IntegerEntry(totalLockedKEY(swapType), (totalLocked + amount)), StringEntry(swapKEY(account, txId), pendingSwapDATA(swapType, amount, selfUnlockHeight))] ++ leasePart)
439- }
440- }
441- else throw("Strict value is not equal to itself.")
442- }
443-
444-
445-
446-@Callable(i)
447-func commonWithdraw (account,index,swapTxId,rsaSig) = {
448- let userAddress = addressFromStringValue(account)
449- let checkCaller = thisOnly(i)
450- if ((checkCaller == checkCaller))
451- then {
452- let dataArray = swapDataFailOrREAD(account, swapTxId)
453- let selfUnlockHeight = parseIntValue(dataArray[sIdxSelfUnlockHeight])
454- let swapType = dataArray[sIdxSwapType]
455- let inAmount = parseIntValue(dataArray[sIdxInAmount])
456- let swapStatus = dataArray[sIdxStatus]
457- let startHeight = parseIntValue(dataArray[sIdxStartHeight])
458- let isQuick = (dataArray[sIdxIsQuick] == "true")
459- let checkSwapType = if (if ((swapType != "waves"))
460- then (swapType != "neutrino")
461- else false)
462- then throw(("Unsupported swap type " + swapType))
415+ let checkCaller = if ((i.originCaller != addressFromStringValue(userAddressStr)))
416+ then throw("Permission denied. Wrong origin caller.")
463417 else true
464- if ((checkSwapType == checkSwapType))
418+ if ((checkCaller == checkCaller))
465419 then {
466- let outFeePart = valueOrElse(getInteger(this, outFeePartKEY(swapType)), DEFAULTSWAPFEE)
420+ let minSwapAmount = minSwapAmountREAD(swapType)
467421 let totalLocked = totalLockedREAD(swapType)
468- let totalLockedByUser = totalLockedByUserREAD(swapType, account)
469- let unlockHeight = if (isQuick)
470- then (startHeight + getIntegerValue(this, keyQuickSwapDelay))
471- else if ((rsaSig == base58''))
472- then selfUnlockHeight
473- else randUnlockHeightOrFail(swapTxId, rsaSig, swapType, startHeight, $Tuple2(parseIntValue(dataArray[sIdxMinRand]), parseIntValue(dataArray[sIdxMaxRand])))
474- let indexHeight = getHeightPriceByIndex(index)
475- let prevIndexHeight = getHeightPriceByIndex((index - 1))
476- let priceByIndex = getPriceHistory(indexHeight)
477- let $t02234322539 = if ((swapType == "waves"))
478- then $Tuple2(convertWavesToNeutrino(inAmount, priceByIndex), neutrinoAssetId)
479- else $Tuple2(convertNeutrinoToWaves(inAmount, priceByIndex), unit)
480- let outAmountGross = $t02234322539._1
481- let outAsset = $t02234322539._2
482- let payoutsArray = applyFees(outAmountGross, outFeePart)
483- let outNetAmount = payoutsArray[IdxNetAmount]
484- let outFeeAmount = payoutsArray[IdxFeeAmount]
485- let checks = [if (isBlocked)
486- then emergencyShutdownFAIL()
487- else true, if ((swapStatus != "PENDING"))
488- then throw("swap has been already processed")
489- else true, if ((unlockHeight > height))
490- then throw((("please wait for: " + toString(unlockHeight)) + " block height to withdraw funds"))
491- else true, if (if (if ((index > priceIndex))
492- then true
493- else (unlockHeight > indexHeight))
494- then true
495- else if ((prevIndexHeight != 0))
496- then (prevIndexHeight >= unlockHeight)
497- else false)
498- then priceIndexFAIL(index, priceIndex, indexHeight, unlockHeight, prevIndexHeight)
499- else true, if ((0 >= payoutsArray[IdxGrossAmount]))
500- then throw("balance equals zero")
501- else true, if (if ((0 > outFeePart))
502- then true
503- else (outFeePart >= PAULI))
504- then throw(((("invalid outFeePart config for " + swapType) + " swap: outFeePart=") + toString(outFeePart)))
505- else true]
506- if ((checks == checks))
422+ let totalLockedByUser = totalLockedByUserREAD(swapType, userAddressStr)
423+ let nodeAddress = getStakingNodeByIndex(0)
424+ let balanceLockMaxInterval = if ((nodeAddress == userAddressStr))
425+ then nodeBalanceLockIntervalREAD()
426+ else if ((swapType == "waves"))
427+ then getIntegerValue(this, BalanceWavesLockIntervalKey)
428+ else getIntegerValue(this, BalanceNeutrinoLockIntervalKey)
429+ let selfUnlockHeight = (height + balanceLockMaxInterval)
430+ let amountCheck = if ((minSwapAmount > pmtAmount))
431+ then minSwapAmountFAIL(swapType, minSwapAmount)
432+ else true
433+ if ((amountCheck == amountCheck))
507434 then {
508- let leasePart = if (if ((swapType == "neutrino"))
509- then (outAmountGross > 0)
510- else false)
511- then invoke(this, "prepareUnleaseAndLeaseWrapper", [outAmountGross], nil)
512- else unit
513- if ((leasePart == leasePart))
514- then {
515- let sendFee = invoke(addressFromStringValue(getStringValue(this, keyGNsbtContract)), "deposit", nil, [AttachedPayment(outAsset, outFeeAmount)])
516- if ((sendFee == sendFee))
517- then [IntegerEntry(totalLockedByUserKEY(swapType, account), (totalLockedByUser - inAmount)), IntegerEntry(totalLockedKEY(swapType), (totalLocked - inAmount)), ScriptTransfer(userAddress, outNetAmount, outAsset), StringEntry(swapKEY(account, swapTxId), finishSwapDATA(dataArray, priceByIndex, outNetAmount, outFeeAmount, unlockHeight, index, toBase58String(i.transactionId)))]
435+ let lastSwapHeight = getNumberByKey(keyUserLastQuickSwapHeight(userAddressStr))
436+ if ((TWENTYFOURHOURS > (height - lastSwapHeight)))
437+ then throw(("You have exceeded swap limit for 24h! Next allowed swap height is " + toString((lastSwapHeight + TWENTYFOURHOURS))))
438+ else {
439+ let userSwapLimitsInfo = asAnyList(invoke(mathContract, "calcUserSwapLimitsMATH", [userAddressStr], nil))
440+ if ((userSwapLimitsInfo == userSwapLimitsInfo))
441+ then {
442+ let quickSwapLimitTotal = asInt(userSwapLimitsInfo[1])
443+ let priceByIndex = getPriceHistory(getHeightPriceByIndex(priceIndex))
444+ let swapUsdnVolume = if ((swapType == "neutrino"))
445+ then pmtAmount
446+ else convertWavesToNeutrino(pmtAmount, priceByIndex)
447+ if ((swapUsdnVolume > quickSwapLimitTotal))
448+ then throw(((("You have exceeded your swap limit! Requested: " + toString(swapUsdnVolume)) + ", available: ") + toString(quickSwapLimitTotal)))
449+ else if (isBlocked)
450+ then emergencyShutdownFAIL()
451+ else {
452+ let leasePart = if ((swapType == "waves"))
453+ then prepareUnleaseAndLease(0)
454+ else nil
455+ $Tuple2(([IntegerEntry(keyQuickSwapUserSpentInPeriod(userAddressStr), swapUsdnVolume), IntegerEntry(keyUserLastQuickSwapHeight(userAddressStr), height), IntegerEntry(totalLockedByUserKEY(swapType, userAddressStr), (totalLockedByUser + pmtAmount)), IntegerEntry(getBalanceUnlockBlockKey(userAddressStr), selfUnlockHeight), IntegerEntry(totalLockedKEY(swapType), (totalLocked + pmtAmount)), StringEntry(swapKEY(userAddressStr, txId58), pendingSwapDATA(swapType, pmtAmount, selfUnlockHeight))] ++ leasePart), unit)
456+ }
457+ }
518458 else throw("Strict value is not equal to itself.")
519459 }
520- else throw("Strict value is not equal to itself.")
521460 }
522461 else throw("Strict value is not equal to itself.")
523462 }
524463 else throw("Strict value is not equal to itself.")
525464 }
526465 else throw("Strict value is not equal to itself.")
527466 }
528467
529468
530469
531470 @Callable(i)
532-func swap (swapType) = {
533- let accelerate = true
534- let wavesToNeutrino = (swapType == "waves")
535- let neutrinoToWaves = (swapType == "neutrino")
471+func swapWavesToNeutrino () = {
536472 let pmt = value(i.payments[0])
537- let account = toString(i.caller)
538- let swapTxId = toBase58String(i.transactionId)
539- if (if (wavesToNeutrino)
540- then (pmt.assetId != unit)
541- else false)
473+ if (isDefined(pmt.assetId))
542474 then throw("Only Waves token is allowed for swapping.")
543- else if (if (neutrinoToWaves)
544- then (pmt.assetId != neutrinoAssetId)
545- else false)
546- then throw("Only appropriate Neutrino tokens are allowed for swapping.")
547- else if (if (!(wavesToNeutrino))
548- then !(neutrinoToWaves)
549- else false)
550- then throw("Invalid swap type")
551- else {
552- let commonSwapInv = invoke(this, "commonSwap", [swapType, pmt.amount, account, swapTxId], nil)
553- if ((commonSwapInv == commonSwapInv))
554- then {
555- let accelerateInv = if (accelerate)
556- then invoke(this, "accelerate", [account, swapTxId], nil)
557- else true
558- if ((accelerateInv == accelerateInv))
559- then nil
560- else throw("Strict value is not equal to itself.")
561- }
562- else throw("Strict value is not equal to itself.")
563- }
475+ else {
476+ let userAddress = toString(i.caller)
477+ let txId58 = toBase58String(i.transactionId)
478+ let commonSwapInv = invoke(this, "commonSwap", ["waves", pmt.amount, userAddress, txId58], nil)
479+ if ((commonSwapInv == commonSwapInv))
480+ then nil
481+ else throw("Strict value is not equal to itself.")
482+ }
564483 }
565484
566485
567486
568487 @Callable(i)
569-func accelerate (account,swapTxId) = {
570- let checkCaller = thisOnly(i)
571- if ((checkCaller == checkCaller))
572- then {
573- let userAddress = addressFromStringValue(account)
574- let dataArray = swapDataFailOrREAD(account, swapTxId)
575- let swapType = dataArray[sIdxSwapType]
576- let inAmount = parseIntValue(dataArray[sIdxInAmount])
577- let swapStatus = dataArray[sIdxStatus]
578- let isQuick = (dataArray[sIdxIsQuick] == "true")
579- let $t02544625654 = match invoke(this, "quickSwapLimitREADONLY", [toString(userAddress)], nil) {
580- case r: (Int, Int) =>
581- $Tuple2(r._1, r._2)
582- case _ =>
583- throw("Type casting error")
584- }
585- let quickSwapLimitTotal = $t02544625654._1
586- let quickSwapUserSpent = $t02544625654._2
587- let quickSwapLimit = (quickSwapLimitTotal - quickSwapUserSpent)
588- let priceByIndex = getPriceHistory(getHeightPriceByIndex(priceIndex))
589- let swapUsdnVolume = if ((swapType == "neutrino"))
590- then inAmount
591- else convertWavesToNeutrino(inAmount, priceByIndex)
592- let checks = [if ((swapStatus != "PENDING"))
593- then throw("Swap has been already processed")
594- else true, if (isQuick)
595- then throw("Swap has been already accelerated")
596- else true, if ((swapUsdnVolume > quickSwapLimit))
597- then throw(((("You have exceeded the quick swaps limit! Requested: " + toString(swapUsdnVolume)) + ", available: ") + toString(quickSwapLimit)))
598- else true]
599- if ((checks == checks))
600- then [IntegerEntry(keyQuickSwapUserSpentInPeriod(userAddress), (swapUsdnVolume + quickSwapUserSpent)), IntegerEntry(keyUserLastQuickSwapHeight(userAddress), height), StringEntry(swapKEY(account, swapTxId), accelerateSwapDATA(dataArray))]
488+func swapNeutrinoToWaves () = {
489+ let pmt = value(i.payments[0])
490+ if ((pmt.assetId != neutrinoAssetId))
491+ then throw("Only appropriate Neutrino tokens are allowed for swapping.")
492+ else {
493+ let userAddress = toString(i.caller)
494+ let txId58 = toBase58String(i.transactionId)
495+ let commonSwapInv = invoke(this, "commonSwap", ["neutrino", pmt.amount, userAddress, txId58], nil)
496+ if ((commonSwapInv == commonSwapInv))
497+ then nil
601498 else throw("Strict value is not equal to itself.")
602499 }
603- else throw("Strict value is not equal to itself.")
604500 }
605501
606502
607503
608504 @Callable(i)
609-func withdraw (account,index,swapTxId) = {
610- let result = invoke(this, "commonWithdraw", [account, index, swapTxId, base58''], nil)
611- if ((result == result))
612- then nil
613- else throw("Strict value is not equal to itself.")
614- }
615-
616-
617-
618-@Callable(i)
619-func withdrawRand (account,index,swapTxId,rsaSig) = {
620- let result = invoke(this, "commonWithdraw", [account, index, swapTxId, rsaSig], nil)
621- if ((result == result))
622- then nil
623- else throw("Strict value is not equal to itself.")
624- }
505+func withdraw (account,index,swapTxId) = commonWithdraw(account, index, swapTxId, i)
625506
626507
627508
628509 @Callable(i)
629510 func transferToAuction () = {
511+ let neutrinoMetrics = asAnyList(invoke(mathContract, "calcNeutinoMetricsMATH", nil, nil))
512+ let reserve = asInt(neutrinoMetrics[4])
513+ let neutrinoSupply = asInt(neutrinoMetrics[6])
514+ let surplus = asInt(neutrinoMetrics[7])
515+ let nsbtSupply = asInt(neutrinoMetrics[10])
630516 let auctionNBAmount = (neutrinoSupply - assetBalance(addressFromStringValue(auctionContract), bondAssetId))
631517 let surplusWithLiquidation = (surplus - assetBalance(addressFromStringValue(liquidationContract), neutrinoAssetId))
632518 if (isBlocked)
633519 then throw("contract is blocked by EMERGENCY SHUTDOWN actions untill reactivation by emergency oracles")
634520 else if ((auctionNBAmount > (1 * PAULI)))
635521 then [ScriptTransfer(addressFromStringValue(auctionContract), auctionNBAmount, bondAssetId)]
636522 else if ((surplusWithLiquidation >= (1 * PAULI)))
637523 then [ScriptTransfer(addressFromStringValue(liquidationContract), surplusWithLiquidation, neutrinoAssetId)]
638524 else throw(((((((("bond were generated or do not need it. Deficit:" + toString(auctionNBAmount)) + "|") + toString(0)) + ". Surplus:") + toString(surplusWithLiquidation)) + "|") + toString(surplus)))
639525 }
640526
641527
642528
643529 @Callable(i)
644530 func acceptWaves () = if ((i.caller != addressFromStringValue(auctionContract)))
645531 then throw("Currently only auction contract is allowed to call")
646532 else $Tuple2(prepareUnleaseAndLease(0), "success")
647533
648534
649-
650-@Callable(i)
651-func calcTotalQuickSwapLimitREADONLY (gNsbtAmount) = {
652- let $t02870528762 = getDora2NSBTInfo()
653- let nsbtPrice = $t02870528762._1
654- let nsbtPriceLastHeight = $t02870528762._2
655- let check = if (((height - nsbtPriceLastHeight) > getIntegerValue(this, keyDora2LastHeightLimit)))
656- then throw("NSBT price is outdated")
657- else true
658- if ((check == check))
659- then $Tuple2(nil, fraction(gNsbtAmount, nsbtPrice, WAVELET))
660- else throw("Strict value is not equal to itself.")
661- }
662-
663-
664-
665-@Callable(i)
666-func quickSwapLimitREADONLY (userAddressStr) = {
667- let limitAvalableWaves = toString((10000 * 100000000))
668- let limitAvalableUsdn = toString((100000 * 1000000))
669- let limitRemainingBlocksToReset = toString(940)
670- $Tuple2(nil, makeString(["%d%d%d", limitAvalableWaves, limitAvalableUsdn, limitRemainingBlocksToReset], SEP))
671- }
672-
673-
674535 @Verifier(tx)
675536 func verify () = {
676537 let id = toBase58String(tx.id)
677- let minSignaturesNumber = 3
678538 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
679539 then 1
680540 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
681541 then 1
682542 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[2], fromBase58String(pubKeyAdminsList[2])))
683543 then 1
684544 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
685545 then 2
686546 else 0))
687- let isSignaturesValid = (count >= minSignaturesNumber)
688547 match tx {
689548 case sponsorTx: SponsorFeeTransaction =>
690- if (isSignaturesValid)
691- then checkIsValidMinSponsoredFee(sponsorTx)
549+ if (checkIsValidMinSponsoredFee(sponsorTx))
550+ then (count >= 3)
692551 else false
693552 case _ =>
694- isSignaturesValid
553+ (count >= 3)
695554 }
696555 }
697556

github/deemru/w8io/6500d08 
129.07 ms