2021.07.08 14:25 [2669336] smart account 3PK7Xe5BiedRyxHLuMQx5ey9riUQqvUths2 > SELF 0.00000000 Waves

{ "type": 13, "id": "29zC8oFAePoLxo8rX6msDmVG2Mc3NQrf6FyTyyjTAXjF", "fee": 1400000, "feeAssetId": null, "timestamp": 1625743365848, "version": 2, "chainId": 87, "sender": "3PK7Xe5BiedRyxHLuMQx5ey9riUQqvUths2", "senderPublicKey": "7hEH4UN8YsnvBCS3YmjieqS332S69yTNjxNp83SmtDrQ", "proofs": [ "", "2HGRdag77nxpJ5qhuWVgz3RFUfwSnBqbJT3oLXf3t9LvtpQeQN8iu1tKSW3sAuALZb3neb8TTrrkKmArPcDTt51e", "37bJGd74gtNCpr1mq48HMhhgeUoWE3kcX2vXSiz3H9PAeRQtaqEndURP8ebmfP6x4W3Kq5dNHuBY8Nq9JjryZWDQ" ], "script": "base64:AAIEAAAAAAAAACoIAhIDCgEEEgcKBQEBCAgEEgMKAQESAwoBARIAEgMKAQESABIAEgMKAQEAAABGAAAAAAd2ZXJzaW9uAgAAAAUxLjAuMAAAAAAKa2V5VmVyc2lvbgIAAAAHdmVyc2lvbgAAAAAJa2V5QWN0aXZlAgAAAAZhY3RpdmUAAAAAC2tleUFzc2V0SWRBAgAAAApBX2Fzc2V0X2lkAAAAAAtrZXlBc3NldElkQgIAAAAKQl9hc3NldF9pZAAAAAALa2V5QmFsYW5jZUECAAAAD0FfYXNzZXRfYmFsYW5jZQAAAAALa2V5QmFsYW5jZUICAAAAD0JfYXNzZXRfYmFsYW5jZQAAAAAPa2V5QmFsYW5jZUluaXRBAgAAAAxBX2Fzc2V0X2luaXQAAAAAD2tleUJhbGFuY2VJbml0QgIAAAAMQl9hc3NldF9pbml0AAAAAA9rZXlTaGFyZUFzc2V0SWQCAAAADnNoYXJlX2Fzc2V0X2lkAAAAABNrZXlTaGFyZUFzc2V0U3VwcGx5AgAAABJzaGFyZV9hc3NldF9zdXBwbHkAAAAADWtleUNvbW1pc3Npb24CAAAACmNvbW1pc3Npb24AAAAAG2tleUNvbW1pc3Npb25TY2FsZURlbGltaXRlcgIAAAAaY29tbWlzc2lvbl9zY2FsZV9kZWxpbWl0ZXIAAAAACGtleUNhdXNlAgAAAA5zaHV0ZG93bl9jYXVzZQAAAAAPa2V5Rmlyc3RIYXJ2ZXN0AgAAAA1maXJzdF9oYXJ2ZXN0AAAAABVrZXlGaXJzdEhhcnZlc3RIZWlnaHQCAAAAFGZpcnN0X2hhcnZlc3RfaGVpZ2h0AAAAAAtrU2hhcmVMaW1pdAIAAAAcc2hhcmVfbGltaXRfb25fZmlyc3RfaGFydmVzdAAAAAALa0Jhc2VQZXJpb2QCAAAAC2Jhc2VfcGVyaW9kAAAAAA1rUGVyaW9kTGVuZ3RoAgAAAA1wZXJpb2RfbGVuZ3RoAAAAAAxrU3RhcnRIZWlnaHQCAAAADHN0YXJ0X2hlaWdodAAAAAATa0ZpcnN0SGFydmVzdEhlaWdodAIAAAAUZmlyc3RfaGFydmVzdF9oZWlnaHQAAAAADGFkbWluUHViS2V5MQEAAAAgugnXO4lZpNFdoP/xuzPQquE5aQ/L4r2R1jvuT16o4BMAAAAA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AAAAEnNoYXJlSW5pdGlhbFN1cHBseQUAAAANc2hhcmVEZWNpbWFscwYEAAAADHNoYXJlSXNzdWVJZAkABDgAAAABBQAAAApzaGFyZUlzc3VlBAAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAKa2V5VmVyc2lvbgUAAAAHdmVyc2lvbgkABEwAAAACCQEAAAAMQm9vbGVhbkVudHJ5AAAAAgUAAAAJa2V5QWN0aXZlBgkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAAtrZXlBc3NldElkQQUAAAAOcG10U3RyQXNzZXRJZEEJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAALa2V5QXNzZXRJZEIFAAAADnBtdFN0ckFzc2V0SWRCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAAKcG10QW1vdW50QQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAALa2V5QmFsYW5jZUIFAAAACnBtdEFtb3VudEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAADWtleUNvbW1pc3Npb24FAAAACmNvbW1pc3Npb24JAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAG2tleUNvbW1pc3Npb25TY2FsZURlbGltaXRlcgUAAAAYY29tbWlzc2lvblNjYWxlRGVsaW1pdGVyCQAETAAAAAIFAAAACnNoYXJlSXNzdWUJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAPa2V5U2hhcmVBc3NldElkCQACWAAAAAEFAAAADHNoYXJlSXNzdWVJZAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAATa2V5U2hhcmVBc3NldFN1cHBseQUAAAASc2hhcmVJbml0aWFsU3VwcGx5CQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAASc2hhcmVJbml0aWFsU3VwcGx5BQAAAAxzaGFyZUlzc3VlSWQFAAAAA25pbAMFAAAADGZpcnN0SGFydmVzdAkABE4AAAACBQAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAADEJvb2xlYW5FbnRyeQAAAAIFAAAAD2tleUZpcnN0SGFydmVzdAUAAAAMZmlyc3RIYXJ2ZXN0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABVrZXlGaXJzdEhhcnZlc3RIZWlnaHQJAABkAAAAAgUAAAALc3RhcnRIZWlnaHQJAABoAAAAAgUAAAAVZmlyc3RIYXJ2ZXN0RW5kUGVyaW9kBQAAAAxwZXJpb2RMZW5ndGgFAAAAA25pbAUAAAAJYmFzZUVudHJ5AAAAAWkBAAAAEWluaXRXaXRoSW5pdFJhdGlvAAAABQAAAAlhbXRBc3NldEEAAAAJYW10QXNzZXRCAAAAC3N0ckFzc2V0SWRBAAAAC3N0ckFzc2V0SWRCAAAADGZpcnN0SGFydmVzdAQAAAALJHQwODkzNDkwMjEJAQAAABZnZXRBc3NldEluZm9Gcm9tU3RyaW5nAAAAAQUAAAALc3RyQXNzZXRJZEEEAAAADnBtdFN0ckFzc2V0SWRBCAUAAAALJHQwODkzNDkwMjEAAAACXzEEAAAADXBtdEFzc2V0TmFtZUEIBQAAAAskdDA4OTM0OTAyMQAAAAJfMgQAAAAMcG10RGVjaW1hbHNBCAUAAAALJHQwODkzNDkwMjEAAAACXzMEAAAACyR0MDkwMjY5MTEzCQEAAAAWZ2V0QXNzZXRJbmZvRnJvbVN0cmluZwAAAAEFAAAAC3N0ckFzc2V0SWRCBAAAAA5wbXRTdHJBc3NldElkQggFAAAACyR0MDkwMjY5MTEzAAAAAl8xBAAAAA1wbXRBc3NldE5hbWVCCAUAAAALJHQwOTAyNjkxMTMAAAACXzIEAAAADHBtdERlY2ltYWxzQggFAAAACyR0MDkwMjY5MTEzAAAAAl8zAwkBAAAAASEAAAABCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgkABEwAAAACBQAAAAxhZG1pblB1YktleTEJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkyCQAETAAAAAIFAAAADGFkbWluUHViS2V5MwkABEwAAAACBQAAABJhZG1pblB1YktleVN0YWtpbmcFAAAAA25pbAgFAAAAAWkAAAAPY2FsbGVyUHVibGljS2V5CQAAAgAAAAECAAAAIU9ubHkgYWRtaW4gY2FuIGNhbGwgdGhpcyBmdW5jdGlvbgMJAQAAAAlpc0RlZmluZWQAAAABCQAEGwAAAAIFAAAABHRoaXMFAAAACWtleUFjdGl2ZQkAAAIAAAABAgAAABZEQXBwIGlzIGFscmVhZHkgYWN0aXZlAwkAAAAAAAACBQAAAAtzdHJBc3NldElkQQUAAAALc3RyQXNzZXRJZEIJAAACAAAAAQIAAAAYQXNzZXRzIG11c3QgYmUgZGlmZmVyZW50BAAAAAlzaGFyZU5hbWUJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAAXMJAAEvAAAAAgUAAAANcG10QXNzZXROYW1lQQAAAAAAAAAABwIAAAABXwkAAS8AAAACBQAAAA1wbXRBc3NldE5hbWVCAAAAAAAAAAAHBAAAABBzaGFyZURlc2NyaXB0aW9uCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAiU2hhcmVUb2tlbiBvZiBTd29wRmkgcHJvdG9jb2wgZm9yIAUAAAANcG10QXNzZXROYW1lQQIAAAAFIGFuZCAFAAAADXBtdEFzc2V0TmFtZUICAAAADCBhdCBhZGRyZXNzIAkABCUAAAABBQAAAAR0aGlzBAAAAA1zaGFyZURlY2ltYWxzCQAAaQAAAAIJAABkAAAAAgUAAAAMcG10RGVjaW1hbHNBBQAAAAxwbXREZWNpbWFsc0IAAAAAAAAAAAIEAAAAEnNoYXJlSW5pdGlhbFN1cHBseQAAAAAAAAAAAAQAAAAKc2hhcmVJc3N1ZQkABEIAAAAFBQAAAAlzaGFyZU5hbWUFAAAAEHNoYXJlRGVzY3JpcHRpb24FAAAAEnNoYXJlSW5pdGlhbFN1cHBseQUAAAANc2hhcmVEZWNpbWFscwYEAAAADHNoYXJlSXNzdWVJZAkABDgAAAABBQAAAApzaGFyZUlzc3VlBAAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAKa2V5VmVyc2lvbgUAAAAHdmVyc2lvbgkABEwAAAACCQEAAAAMQm9vbGVhbkVudHJ5AAAAAgUAAAAJa2V5QWN0aXZlBgkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAAtrZXlBc3NldElkQQUAAAAOcG10U3RyQXNzZXRJZEEJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAALa2V5QXNzZXRJZEIFAAAADnBtdFN0ckFzc2V0SWRCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAA9rZXlCYWxhbmNlSW5pdEEFAAAACWFtdEFzc2V0QQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAPa2V5QmFsYW5jZUluaXRCBQAAAAlhbXRBc3NldEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VBAAAAAAAAAAAACQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQgAAAAAAAAAAAAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAANa2V5Q29tbWlzc2lvbgUAAAAKY29tbWlzc2lvbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAba2V5Q29tbWlzc2lvblNjYWxlRGVsaW1pdGVyBQAAABhjb21taXNzaW9uU2NhbGVEZWxpbWl0ZXIJAARMAAAAAgUAAAAKc2hhcmVJc3N1ZQkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAA9rZXlTaGFyZUFzc2V0SWQJAAJYAAAAAQUAAAAMc2hhcmVJc3N1ZUlkCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABNrZXlTaGFyZUFzc2V0U3VwcGx5BQAAABJzaGFyZUluaXRpYWxTdXBwbHkFAAAAA25pbAMFAAAADGZpcnN0SGFydmVzdAkABE4AAAACBQAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAADEJvb2xlYW5FbnRyeQAAAAIFAAAAD2tleUZpcnN0SGFydmVzdAUAAAAMZmlyc3RIYXJ2ZXN0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABVrZXlGaXJzdEhhcnZlc3RIZWlnaHQJAABkAAAAAgUAAAALc3RhcnRIZWlnaHQJAABoAAAAAgUAAAAVZmlyc3RIYXJ2ZXN0RW5kUGVyaW9kBQAAAAxwZXJpb2RMZW5ndGgFAAAAA25pbAUAAAAJYmFzZUVudHJ5AAAAAWkBAAAAGGtlZXBMaW1pdEZvckZpcnN0SGFydmVzdAAAAAEAAAAKc2hhcmVMaW1pdAMJAQAAAAEhAAAAAQUAAAAIaXNBY3RpdmUJAAACAAAAAQIAAAAfREFwcCBpcyBpbmFjdGl2ZSBhdCB0aGlzIG1vbWVudAMJAQAAAAEhAAAAAQkBAAAAD2NvbnRhaW5zRWxlbWVudAAAAAIJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkxCQAETAAAAAIFAAAADGFkbWluUHViS2V5MgkABEwAAAACBQAAAAxhZG1pblB1YktleTMJAARMAAAAAgUAAAASYWRtaW5QdWJLZXlTdGFraW5nBQAAAANuaWwIBQAAAAFpAAAAD2NhbGxlclB1YmxpY0tleQkAAAIAAAABAgAAACFPbmx5IGFkbWluIGNhbiBjYWxsIHRoaXMgZnVuY3Rpb24JAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tTaGFyZUxpbWl0BQAAAApzaGFyZUxpbWl0BQAAAANuaWwAAAABaQEAAAAWcmVwbGVuaXNoV2l0aFR3b1Rva2VucwAAAAEAAAARc2xpcHBhZ2VUb2xlcmFuY2UEAAAAC3BtdEFzc2V0SWRBCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAdhc3NldElkBAAAAAtwbXRBc3NldElkQggJAAGRAAAAAggFAAAAAWkAAAAIcGF5bWVudHMAAAAAAAAAAAEAAAAHYXNzZXRJZAQAAAAKcG10QW1vdW50QQkBAAAAEGRlZHVjdFN0YWtpbmdGZWUAAAACCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAZhbW91bnQFAAAAC3BtdEFzc2V0SWRBBAAAAApwbXRBbW91bnRCCQEAAAAQZGVkdWN0U3Rha2luZ0ZlZQAAAAIICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAABAAAABmFtb3VudAUAAAALcG10QXNzZXRJZEIDAwkAAAAAAAACBQAAAAhiYWxhbmNlQQAAAAAAAAAAAAkAAAAAAAACBQAAAAhiYWxhbmNlQgAAAAAAAAAAAAcEAAAADSR0MDExNzk4MTE4NzUJAQAAAAxnZXRBc3NldEluZm8AAAABBQAAAAtwbXRBc3NldElkQQQAAAAOcG10U3RyQXNzZXRJZEEIBQAAAA0kdDAxMTc5ODExODc1AAAAAl8xBAAAAA1wbXRBc3NldE5hbWVBCAUAAAANJHQwMTE3OTgxMTg3NQAAAAJfMgQAAAAMcG10RGVjaW1hbHNBCAUAAAANJHQwMTE3OTgxMTg3NQAAAAJfMwQAAAANJHQwMTE4ODQxMTk2MQkBAAAADGdldEFzc2V0SW5mbwAAAAEFAAAAC3BtdEFzc2V0SWRCBAAAAA5wbXRTdHJBc3NldElkQggFAAAADSR0MDExODg0MTE5NjEAAAACXzEEAAAADXBtdEFzc2V0TmFtZUIIBQAAAA0kdDAxMTg4NDExOTYxAAAAAl8yBAAAAAxwbXREZWNpbWFsc0IIBQAAAA0kdDAxMTg4NDExOTYxAAAAAl8zBAAAAAp0b2tlblJhdGlvCQAAawAAAAMJAABrAAAAAwUAAAAKYXNzZXRJbml0QQUAAAALc2NhbGVWYWx1ZTgFAAAACnBtdEFtb3VudEEFAAAAC3NjYWxlVmFsdWUzCQAAawAAAAMFAAAACmFzc2V0SW5pdEIFAAAAC3NjYWxlVmFsdWU4BQAAAApwbXRBbW91bnRCAwkAAAAAAAACBQAAAAtwbXRBc3NldElkQQUAAAALcG10QXNzZXRJZEIJAAACAAAAAQIAAAAYQXNzZXRzIG11c3QgYmUgZGlmZmVyZW50BAAAAA1zaGFyZURlY2ltYWxzCQAAaQAAAAIJAABkAAAAAgUAAAAMcG10RGVjaW1hbHNBBQAAAAxwbXREZWNpbWFsc0IAAAAAAAAAAAIEAAAAEnNoYXJlSW5pdGlhbFN1cHBseQkAAGsAAAADCQAAbAAAAAYFAAAACnBtdEFtb3VudEEFAAAADHBtdERlY2ltYWxzQQAAAAAAAAAABQAAAAAAAAAAAQUAAAAMcG10RGVjaW1hbHNBBQAAAAhIQUxGRE9XTgkAAGwAAAAGBQAAAApwbXRBbW91bnRCBQAAAAxwbXREZWNpbWFsc0IAAAAAAAAAAAUAAAAAAAAAAAEFAAAADHBtdERlY2ltYWxzQgUAAAAISEFMRkRPV04JAABsAAAABgAAAAAAAAAACgAAAAAAAAAAAAUAAAANc2hhcmVEZWNpbWFscwAAAAAAAAAAAAAAAAAAAAAAAAUAAAAISEFMRkRPV04DCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDAwkAAGYAAAACAAAAAAAAAAAABQAAABFzbGlwcGFnZVRvbGVyYW5jZQYJAABmAAAAAgUAAAARc2xpcHBhZ2VUb2xlcmFuY2UFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAKVNsaXBwYWdlIHRvbGVyYW5jZSBtdXN0IGJlIGJldHdlZW4gMCBhbmQgCQABpAAAAAEFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyAgAAABYgaW5jbHVzaXZlbHkuIEFjdHVhbDogCQABpAAAAAEFAAAAEXNsaXBwYWdlVG9sZXJhbmNlAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAgkAAAIAAAABAgAAABxUd28gYXR0YWNoZWQgYXNzZXRzIGV4cGVjdGVkAwMJAABmAAAAAgkAAGkAAAACCQAAaAAAAAIFAAAAC3NjYWxlVmFsdWUzCQAAZQAAAAIFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIFAAAACnRva2VuUmF0aW8GCQAAZgAAAAIFAAAACnRva2VuUmF0aW8JAABpAAAAAgkAAGgAAAACBQAAAAtzY2FsZVZhbHVlMwkAAGQAAAACBQAAABpzbGlwcGFnZVRvbGVyYW5jZURlbGltaXRlcgUAAAARc2xpcHBhZ2VUb2xlcmFuY2UFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyCQAAAgAAAAECAAAAPUluY29ycmVjdCBhc3NldHMgYW1vdW50OiBhbW91bnRzIG11c3QgaGF2ZSB0aGUgY29udHJhY3QgcmF0aW8DAwkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEEFAAAACGFzc2V0SWRBBgkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEIFAAAACGFzc2V0SWRCCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAJUluY29ycmVjdCBhc3NldHMgYXR0YWNoZWQuIEV4cGVjdGVkOiAFAAAAC3N0ckFzc2V0SWRBAgAAAAUgYW5kIAUAAAALc3RyQXNzZXRJZEIDCQAAAAAAAAIFAAAAEnNoYXJlSW5pdGlhbFN1cHBseQAAAAAAAAAAAAkAAAIAAAABAgAAAB1Ub28gc21hbGwgYW1vdW50IHRvIHJlcGxlbmlzaAMJAQAAAAEhAAAAAQUAAAAQaGFzRW5vdWdoQmFsYW5jZQkABE4AAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAKcG10QW1vdW50QQUAAAALcG10QXNzZXRJZEEJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAAApwbXRBbW91bnRCBQAAAAtwbXRBc3NldElkQgUAAAADbmlsCQEAAAARc3VzcGVuZFN1c3BpY2lvdXMAAAAACQAETAAAAAIJAQAAAAdSZWlzc3VlAAAAAwUAAAAMc2hhcmVBc3NldElkBQAAABJzaGFyZUluaXRpYWxTdXBwbHkGCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAAKcG10QW1vdW50QQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAALa2V5QmFsYW5jZUIFAAAACnBtdEFtb3VudEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAE2tleVNoYXJlQXNzZXRTdXBwbHkFAAAAEnNoYXJlSW5pdGlhbFN1cHBseQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAAEnNoYXJlSW5pdGlhbFN1cHBseQUAAAAMc2hhcmVBc3NldElkBQAAAANuaWwEAAAACnRva2VuUmF0aW8JAABrAAAAAwkAAGsAAAADBQAAAAhiYWxhbmNlQQUAAAALc2NhbGVWYWx1ZTgFAAAACnBtdEFtb3VudEEFAAAAC3NjYWxlVmFsdWUzCQAAawAAAAMFAAAACGJhbGFuY2VCBQAAAAtzY2FsZVZhbHVlOAUAAAAKcG10QW1vdW50QgQAAAATcmF0aW9TaGFyZVRva2Vuc0luQQkAAGsAAAADBQAAAApwbXRBbW91bnRBBQAAAAtzY2FsZVZhbHVlOAUAAAAIYmFsYW5jZUEEAAAAE3JhdGlvU2hhcmVUb2tlbnNJbkIJAABrAAAAAwUAAAAKcG10QW1vdW50QgUAAAALc2NhbGVWYWx1ZTgFAAAACGJhbGFuY2VCBAAAABVzaGFyZVRva2VuVG9QYXlBbW91bnQJAABrAAAAAwkAAZcAAAABCQAETAAAAAIFAAAAE3JhdGlvU2hhcmVUb2tlbnNJbkEJAARMAAAAAgUAAAATcmF0aW9TaGFyZVRva2Vuc0luQgUAAAADbmlsBQAAABBzaGFyZUFzc2V0U3VwcGx5BQAAAAtzY2FsZVZhbHVlOAMJAQAAAAEhAAAAAQUAAAAIaXNBY3RpdmUJAAACAAAAAQIAAAAfREFwcCBpcyBpbmFjdGl2ZSBhdCB0aGlzIG1vbWVudAMDCQAAZgAAAAIAAAAAAAAAAAAFAAAAEXNsaXBwYWdlVG9sZXJhbmNlBgkAAGYAAAACBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAApU2xpcHBhZ2UgdG9sZXJhbmNlIG11c3QgYmUgYmV0d2VlbiAwIGFuZCAJAAGkAAAAAQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXICAAAAFiBpbmNsdXNpdmVseS4gQWN0dWFsOiAJAAGkAAAAAQUAAAARc2xpcHBhZ2VUb2xlcmFuY2UDCQEAAAACIT0AAAACCQABkAAAAAEIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAACCQAAAgAAAAECAAAAHFR3byBhdHRhY2hlZCBhc3NldHMgZXhwZWN0ZWQDAwkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEEFAAAACGFzc2V0SWRBBgkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEIFAAAACGFzc2V0SWRCCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAJUluY29ycmVjdCBhc3NldHMgYXR0YWNoZWQuIEV4cGVjdGVkOiAFAAAAC3N0ckFzc2V0SWRBAgAAAAUgYW5kIAUAAAALc3RyQXNzZXRJZEIDAwkAAGYAAAACCQAAaQAAAAIJAABoAAAAAgUAAAALc2NhbGVWYWx1ZTMJAABlAAAAAgUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIFAAAAEXNsaXBwYWdlVG9sZXJhbmNlBQAAABpzbGlwcGFnZVRvbGVyYW5jZURlbGltaXRlcgUAAAAKdG9rZW5SYXRpbwYJAABmAAAAAgUAAAAKdG9rZW5SYXRpbwkAAGkAAAACCQAAaAAAAAIFAAAAC3NjYWxlVmFsdWUzCQAAZAAAAAIFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIJAAACAAAAAQIAAAA9SW5jb3JyZWN0IGFzc2V0cyBhbW91bnQ6IGFtb3VudHMgbXVzdCBoYXZlIHRoZSBjb250cmFjdCByYXRpbwMJAAAAAAAAAgUAAAAVc2hhcmVUb2tlblRvUGF5QW1vdW50AAAAAAAAAAAACQAAAgAAAAECAAAAHVRvbyBzbWFsbCBhbW91bnQgdG8gcmVwbGVuaXNoAwkBAAAAASEAAAABBQAAABBoYXNFbm91Z2hCYWxhbmNlCQAETgAAAAIJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAAApwbXRBbW91bnRBBQAAAAtwbXRBc3NldElkQQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAACnBtdEFtb3VudEIFAAAAC3BtdEFzc2V0SWRCBQAAAANuaWwJAQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAAJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VBCQAAZAAAAAIFAAAACGJhbGFuY2VBBQAAAApwbXRBbW91bnRBCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQgkAAGQAAAACBQAAAAhiYWxhbmNlQgUAAAAKcG10QW1vdW50QgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAATa2V5U2hhcmVBc3NldFN1cHBseQkAAGQAAAACBQAAABBzaGFyZUFzc2V0U3VwcGx5BQAAABVzaGFyZVRva2VuVG9QYXlBbW91bnQJAARMAAAAAgkBAAAAB1JlaXNzdWUAAAADBQAAAAxzaGFyZUFzc2V0SWQFAAAAFXNoYXJlVG9rZW5Ub1BheUFtb3VudAYJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAABVzaGFyZVRva2VuVG9QYXlBbW91bnQFAAAADHNoYXJlQXNzZXRJZAUAAAADbmlsAAAAAWkBAAAACHdpdGhkcmF3AAAAAAQAAAANJHQwMTYzNzMxNjUyMwkABRQAAAACCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAZhbW91bnQICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAAAAB2Fzc2V0SWQEAAAACXBtdEFtb3VudAgFAAAADSR0MDE2MzczMTY1MjMAAAACXzEEAAAACnBtdEFzc2V0SWQIBQAAAA0kdDAxNjM3MzE2NTIzAAAAAl8yBAAAAAxhbW91bnRUb1BheUEJAQAAABBkZWR1Y3RTdGFraW5nRmVlAAAAAgkAAGsAAAADBQAAAAlwbXRBbW91bnQFAAAACGJhbGFuY2VBBQAAABBzaGFyZUFzc2V0U3VwcGx5BQAAAAhhc3NldElkQQQAAAAMYW1vdW50VG9QYXlCCQEAAAAQZGVkdWN0U3Rha2luZ0ZlZQAAAAIJAABrAAAAAwUAAAAJcG10QW1vdW50BQAAAAhiYWxhbmNlQgUAAAAQc2hhcmVBc3NldFN1cHBseQUAAAAIYXNzZXRJZEIDCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDCQEAAAACIT0AAAACCQABkAAAAAEIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAABCQAAAgAAAAECAAAAHU9uZSBhdHRhY2hlZCBwYXltZW50IGV4cGVjdGVkAwkBAAAAAiE9AAAAAgUAAAAKcG10QXNzZXRJZAUAAAAMc2hhcmVBc3NldElkCQAAAgAAAAEJAAEsAAAAAgIAAAAkSW5jb3JyZWN0IGFzc2V0IGF0dGFjaGVkLiBFeHBlY3RlZDogCQACWAAAAAEFAAAADHNoYXJlQXNzZXRJZAMJAQAAAAEhAAAAAQUAAAAQaGFzRW5vdWdoQmFsYW5jZQkABE4AAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAJcG10QW1vdW50BQAAAApwbXRBc3NldElkBQAAAANuaWwJAQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAADAwkAAGYAAAACBQAAAAxhbW91bnRUb1BheUEFAAAAEWF2YWlsYWJsZUJhbGFuY2VBBgkAAGYAAAACBQAAAAxhbW91bnRUb1BheUIFAAAAEWF2YWlsYWJsZUJhbGFuY2VCCQEAAAAidGhyb3dJbnN1ZmZpY2llbnRBdmFpbGFibGVCYWxhbmNlcwAAAAIFAAAADGFtb3VudFRvUGF5QQUAAAAMYW1vdW50VG9QYXlCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQkAAGUAAAACBQAAAAhiYWxhbmNlQQUAAAAMYW1vdW50VG9QYXlBCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQgkAAGUAAAACBQAAAAhiYWxhbmNlQgUAAAAMYW1vdW50VG9QYXlCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABNrZXlTaGFyZUFzc2V0U3VwcGx5CQAAZQAAAAIFAAAAEHNoYXJlQXNzZXRTdXBwbHkFAAAACXBtdEFtb3VudAkABEwAAAACCQEAAAAEQnVybgAAAAIFAAAADHNoYXJlQXNzZXRJZAUAAAAJcG10QW1vdW50CQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAMYW1vdW50VG9QYXlBBQAAAAhhc3NldElkQQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAADGFtb3VudFRvUGF5QgUAAAAIYXNzZXRJZEIFAAAAA25pbAAAAAFpAQAAAAhleGNoYW5nZQAAAAEAAAASbWluQW1vdW50VG9SZWNlaXZlBAAAAA0kdDAxNzcyOTE3ODA0CQAFFAAAAAIICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAAAABmFtb3VudAgJAAGRAAAAAggFAAAAAWkAAAAIcGF5bWVudHMAAAAAAAAAAAAAAAAHYXNzZXRJZAQAAAAJcG10QW1vdW50CAUAAAANJHQwMTc3MjkxNzgwNAAAAAJfMQQAAAAKcG10QXNzZXRJZAgFAAAADSR0MDE3NzI5MTc4MDQAAAACXzIKAQAAAA1jYWxjdWxhdGVGZWVzAAAAAgAAAAl0b2tlbkZyb20AAAAHdG9rZW5UbwQAAAAQYW1vdW50V2l0aG91dEZlZQkAAGsAAAADBQAAAAd0b2tlblRvBQAAAAlwbXRBbW91bnQJAABkAAAAAgUAAAAJcG10QW1vdW50BQAAAAl0b2tlbkZyb20EAAAADWFtb3VudFdpdGhGZWUJAABrAAAAAwUAAAAQYW1vdW50V2l0aG91dEZlZQkAAGUAAAACBQAAABhjb21taXNzaW9uU2NhbGVEZWxpbWl0ZXIFAAAACmNvbW1pc3Npb24FAAAAGGNvbW1pc3Npb25TY2FsZURlbGltaXRlcgQAAAAQZ292ZXJuYW5jZVJld2FyZAkAAGsAAAADBQAAABBhbW91bnRXaXRob3V0RmVlBQAAABRjb21taXNzaW9uR292ZXJuYW5jZQUAAAAYY29tbWlzc2lvblNjYWxlRGVsaW1pdGVyAwkAAGYAAAACBQAAABJtaW5BbW91bnRUb1JlY2VpdmUFAAAADWFtb3VudFdpdGhGZWUJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAdQ2FsY3VsYXRlZCBhbW91bnQgdG8gcmVjZWl2ZSAJAAGkAAAAAQUAAAANYW1vdW50V2l0aEZlZQIAAAAgIGlzIGxlc3MgdGhhbiBzcGVjaWZpZWQgbWluaW11bSAJAAGkAAAAAQUAAAASbWluQW1vdW50VG9SZWNlaXZlCQAFFQAAAAMFAAAAEGFtb3VudFdpdGhvdXRGZWUFAAAADWFtb3VudFdpdGhGZWUFAAAAEGdvdmVybmFuY2VSZXdhcmQDCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDAwkAAAAAAAACBQAAAAhiYWxhbmNlQQAAAAAAAAAAAAYJAAAAAAAAAgUAAAAIYmFsYW5jZUIAAAAAAAAAAAAJAAACAAAAAQIAAAAgQ2FuJ3QgZXhjaGFuZ2Ugd2l0aCB6ZXJvIGJhbGFuY2UDCQAAZwAAAAIAAAAAAAAAAAAFAAAAEm1pbkFtb3VudFRvUmVjZWl2ZQkAAAIAAAABCQABLAAAAAICAAAANE1pbmltYWwgYW1vdW50IHRvIHJlY2VpdmUgbXVzdCBiZSBwb3NpdGl2ZS4gQWN0dWFsOiAJAAGkAAAAAQUAAAASbWluQW1vdW50VG9SZWNlaXZlAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAQkAAAIAAAABAgAAAB1PbmUgYXR0YWNoZWQgcGF5bWVudCBleHBlY3RlZAMJAQAAAAEhAAAAAQUAAAAQaGFzRW5vdWdoQmFsYW5jZQkABE4AAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAJcG10QW1vdW50BQAAAApwbXRBc3NldElkBQAAAANuaWwJAQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAADCQAAAAAAAAIFAAAACnBtdEFzc2V0SWQFAAAACGFzc2V0SWRBBAAAAAthc3NldElkU2VuZAUAAAAIYXNzZXRJZEIEAAAADSR0MDE5MDc4MTkxNjkJAQAAAA1jYWxjdWxhdGVGZWVzAAAAAgUAAAAIYmFsYW5jZUEFAAAACGJhbGFuY2VCBAAAABBhbW91bnRXaXRob3V0RmVlCAUAAAANJHQwMTkwNzgxOTE2OQAAAAJfMQQAAAANYW1vdW50V2l0aEZlZQgFAAAADSR0MDE5MDc4MTkxNjkAAAACXzIEAAAAEGdvdmVybmFuY2VSZXdhcmQIBQAAAA0kdDAxOTA3ODE5MTY5AAAAAl8zBAAAAAtuZXdCYWxhbmNlQQkAAGQAAAACBQAAAAhiYWxhbmNlQQUAAAAJcG10QW1vdW50BAAAAAtuZXdCYWxhbmNlQgkAAGUAAAACCQAAZQAAAAIFAAAACGJhbGFuY2VCBQAAAA1hbW91bnRXaXRoRmVlBQAAABBnb3Zlcm5hbmNlUmV3YXJkAwMDCQAAAAAAAAIFAAAACGFzc2V0SWRBBQAAAARFVVJOCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudEVVUk4FAAAAC25ld0JhbGFuY2VBBwYDCQAAAAAAAAIFAAAACGFzc2V0SWRCBQAAAARFVVJOCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudEVVUk4FAAAAC25ld0JhbGFuY2VCBwkBAAAAIXRocm93SW5zdWZmaWNpZW50QXZhaWxhYmxlQmFsYW5jZQAAAAMFAAAADWFtb3VudFdpdGhGZWUFAAAAEWF2YWlsYWJsZUJhbGFuY2VCBQAAAAphc3NldE5hbWVCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAALbmV3QmFsYW5jZUEJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VCBQAAAAtuZXdCYWxhbmNlQgkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAADWFtb3VudFdpdGhGZWUFAAAAC2Fzc2V0SWRTZW5kCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMFAAAADXdhbGxldEFkZHJlc3MFAAAAEGdvdmVybmFuY2VSZXdhcmQFAAAAC2Fzc2V0SWRTZW5kBQAAAANuaWwDCQAAAAAAAAIFAAAACnBtdEFzc2V0SWQFAAAACGFzc2V0SWRCBAAAAAthc3NldElkU2VuZAUAAAAIYXNzZXRJZEEEAAAADSR0MDIwMDM4MjAxMjkJAQAAAA1jYWxjdWxhdGVGZWVzAAAAAgUAAAAIYmFsYW5jZUIFAAAACGJhbGFuY2VBBAAAABBhbW91bnRXaXRob3V0RmVlCAUAAAANJHQwMjAwMzgyMDEyOQAAAAJfMQQAAAANYW1vdW50V2l0aEZlZQgFAAAADSR0MDIwMDM4MjAxMjkAAAACXzIEAAAAEGdvdmVybmFuY2VSZXdhcmQIBQAAAA0kdDAyMDAzODIwMTI5AAAAAl8zBAAAAAtuZXdCYWxhbmNlQQkAAGUAAAACCQAAZQAAAAIFAAAACGJhbGFuY2VBBQAAAA1hbW91bnRXaXRoRmVlBQAAABBnb3Zlcm5hbmNlUmV3YXJkBAAAAAtuZXdCYWxhbmNlQgkAAGQAAAACBQAAAAhiYWxhbmNlQgUAAAAJcG10QW1vdW50AwMDCQAAAAAAAAIFAAAACGFzc2V0SWRBBQAAAARFVVJOCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudEVVUk4FAAAAC25ld0JhbGFuY2VBBwYDCQAAAAAAAAIFAAAACGFzc2V0SWRCBQAAAARFVVJOCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudEVVUk4FAAAAC25ld0JhbGFuY2VCBwkBAAAAIXRocm93SW5zdWZmaWNpZW50QXZhaWxhYmxlQmFsYW5jZQAAAAMFAAAADWFtb3VudFdpdGhGZWUFAAAAEWF2YWlsYWJsZUJhbGFuY2VBBQAAAAphc3NldE5hbWVBCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAALbmV3QmFsYW5jZUEJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VCBQAAAAtuZXdCYWxhbmNlQgkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAADWFtb3VudFdpdGhGZWUFAAAAC2Fzc2V0SWRTZW5kCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMFAAAADXdhbGxldEFkZHJlc3MFAAAAEGdvdmVybmFuY2VSZXdhcmQFAAAAC2Fzc2V0SWRTZW5kBQAAAANuaWwJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAkSW5jb3JyZWN0IGFzc2V0IGF0dGFjaGVkLiBFeHBlY3RlZDogBQAAAAtzdHJBc3NldElkQQIAAAAEIG9yIAUAAAALc3RyQXNzZXRJZEIAAAABaQEAAAAIc2h1dGRvd24AAAAAAwkBAAAAASEAAAABBQAAAAhpc0FjdGl2ZQkAAAIAAAABCQABLAAAAAICAAAAIkRBcHAgaXMgYWxyZWFkeSBzdXNwZW5kZWQuIENhdXNlOiAJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQdAAAAAgUAAAAEdGhpcwUAAAAIa2V5Q2F1c2UCAAAAGnRoZSBjYXVzZSB3YXNuJ3Qgc3BlY2lmaWVkAwkBAAAAASEAAAABCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgkABEwAAAACBQAAAAxhZG1pblB1YktleTEJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkyCQAETAAAAAIFAAAADGFkbWluUHViS2V5MwkABEwAAAACBQAAABRhZG1pblB1YktleVN0YXJ0U3RvcAUAAAADbmlsCAUAAAABaQAAAA9jYWxsZXJQdWJsaWNLZXkJAAACAAAAAQIAAAAhT25seSBhZG1pbiBjYW4gY2FsbCB0aGlzIGZ1bmN0aW9uCQEAAAAHc3VzcGVuZAAAAAECAAAAD1BhdXNlZCBieSBhZG1pbgAAAAFpAQAAAAhhY3RpdmF0ZQAAAAADBQAAAAhpc0FjdGl2ZQkAAAIAAAABAgAAABZEQXBwIGlzIGFscmVhZHkgYWN0aXZlAwkBAAAAASEAAAABCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgkABEwAAAACBQAAAAxhZG1pblB1YktleTEJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkyCQAETAAAAAIFAAAADGFkbWluUHViS2V5MwkABEwAAAACBQAAABRhZG1pblB1YktleVN0YXJ0U3RvcAUAAAADbmlsCAUAAAABaQAAAA9jYWxsZXJQdWJsaWNLZXkJAAACAAAAAQIAAAAhT25seSBhZG1pbiBjYW4gY2FsbCB0aGlzIGZ1bmN0aW9uCQAETAAAAAIJAQAAAAxCb29sZWFuRW50cnkAAAACBQAAAAlrZXlBY3RpdmUGCQAETAAAAAIJAQAAAAtEZWxldGVFbnRyeQAAAAEFAAAACGtleUNhdXNlBQAAAANuaWwAAAABaQEAAAAZdGFrZUludG9BY2NvdW50RXh0cmFGdW5kcwAAAAEAAAALYW1vdW50TGVhdmUEAAAAHXVuY291bnRhYmxlQW1vdW50RW5yb2xsQXNzZXRBCQAAZQAAAAIFAAAAGWFjY291bnRCYWxhbmNlV2l0aFN0YWtlZEEFAAAACGJhbGFuY2VBBAAAAB11bmNvdW50YWJsZUFtb3VudEVucm9sbEFzc2V0QgkAAGUAAAACBQAAABlhY2NvdW50QmFsYW5jZVdpdGhTdGFrZWRCBQAAAAhiYWxhbmNlQgQAAAANYW1vdW50RW5yb2xsQQkAAGUAAAACBQAAAB11bmNvdW50YWJsZUFtb3VudEVucm9sbEFzc2V0QQMJAAAAAAAAAgUAAAAIYXNzZXRJZEEFAAAABHVuaXQFAAAAC2Ftb3VudExlYXZlAAAAAAAAAAAABAAAAA1hbW91bnRFbnJvbGxCCQAAZQAAAAIFAAAAHXVuY291bnRhYmxlQW1vdW50RW5yb2xsQXNzZXRCAwkAAAAAAAACBQAAAAhhc3NldElkQgUAAAAEdW5pdAUAAAALYW1vdW50TGVhdmUAAAAAAAAAAAADCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDCQEAAAACIT0AAAACCAUAAAABaQAAAAZjYWxsZXIFAAAABHRoaXMJAAACAAAAAQIAAAArT25seSB0aGUgREFwcCBpdHNlbGYgY2FuIGNhbGwgdGhpcyBmdW5jdGlvbgMJAABmAAAAAgAAAAAAAAAAAAUAAAALYW1vdW50TGVhdmUJAAACAAAAAQkAASwAAAACAgAAADNBcmd1bWVudCAnYW1vdW50TGVhdmUnIGNhbm5vdCBiZSBuZWdhdGl2ZS4gQWN0dWFsOiAJAAGkAAAAAQUAAAALYW1vdW50TGVhdmUDAwkAAGYAAAACAAAAAAAAAAAABQAAAB11bmNvdW50YWJsZUFtb3VudEVucm9sbEFzc2V0QQYJAABmAAAAAgAAAAAAAAAAAAUAAAAddW5jb3VudGFibGVBbW91bnRFbnJvbGxBc3NldEIJAQAAAAdzdXNwZW5kAAAAAQIAAAAWRW5yb2xsIGFtb3VudCBuZWdhdGl2ZQMDCQAAZgAAAAIAAAAAAAAAAAAFAAAADWFtb3VudEVucm9sbEEGCQAAZgAAAAIAAAAAAAAAAAAFAAAADWFtb3VudEVucm9sbEIJ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", "height": 2669336, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: AxJdnPHcmdxufdRSxzPvfBMtt1tKRpxLsiBpjEP3xMNg Next: A5u81KG7iGRnu8HUpDE1ydvMrdkhqEAwyTF88AooGKD Diff:
OldNewDifferences
1515
1616 let keyBalanceB = "B_asset_balance"
1717
18+let keyBalanceInitA = "A_asset_init"
19+
20+let keyBalanceInitB = "B_asset_init"
21+
1822 let keyShareAssetId = "share_asset_id"
1923
2024 let keyShareAssetSupply = "share_asset_supply"
2428 let keyCommissionScaleDelimiter = "commission_scale_delimiter"
2529
2630 let keyCause = "shutdown_cause"
31+
32+let keyFirstHarvest = "first_harvest"
33+
34+let keyFirstHarvestHeight = "first_harvest_height"
35+
36+let kShareLimit = "share_limit_on_first_harvest"
37+
38+let kBasePeriod = "base_period"
39+
40+let kPeriodLength = "period_length"
41+
42+let kStartHeight = "start_height"
43+
44+let kFirstHarvestHeight = "first_harvest_height"
2745
2846 let adminPubKey1 = base58'DXDY2itiEcYBtGkVLnkpHtDFyWQUkoLJz79uJ7ECbMrA'
2947
3553
3654 let adminPubKeyStaking = base58'Czn4yoAuUZCVCLJDRfskn8URfkwpknwBTZDbs1wFrY7h'
3755
38-let governanceAddress = Address(base58'3P6J84oH51DzY6xk2mT5TheXRbrCwBMxonp')
56+let walletAddress = Address(base58'3P6J84oH51DzY6xk2mT5TheXRbrCwBMxonp')
3957
40-let stakingAddressUsdnAndNsbt = Address(base58'3PNikM6yp4NqcSU8guxQtmR5onr2D4e8yTJ')
58+let votingAddress = Address(base58'3PQZWxShKGRgBN1qoJw6B4s9YWS9FneZTPg')
4159
42-let stakingAddressDeFo = Address(base58'3PFhcMmEZoQTQ6ohA844c7C9M8ZJ18P8dDj')
60+let stakingAddress = Address(base58'3PFhcMmEZoQTQ6ohA844c7C9M8ZJ18P8dDj')
4361
44-let USDN = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
62+let EURNToWavesExchanger = Address(base58'3PK7Xe5BiedRyxHLuMQx5ey9riUQqvUths2')
4563
46-let NSBT = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
64+let basePeriod = valueOrErrorMessage(getInteger(votingAddress, kBasePeriod), "Empty kBasePeriod")
65+
66+let startHeight = valueOrErrorMessage(getInteger(votingAddress, kStartHeight), "Empty kStartHeight")
67+
68+let periodLength = valueOrErrorMessage(getInteger(votingAddress, kPeriodLength), "Empty kPeriodLength")
69+
70+let firstHarvestEndPeriod = ((basePeriod + ((height - startHeight) / periodLength)) + 3)
4771
4872 let EURN = base58'DUk2YTxhRoAqMJLus4G2b3fR8hMHVh6eiyFx5r29VR6t'
4973
50-let stakingAssets = [USDN, NSBT, EURN]
74+let stakingFeeInEURN = 234000
5175
5276 let isActive = getBooleanValue(this, keyActive)
5377
113137 }
114138
115139
116-func stakedAmount (assetId) = match assetId {
117- case id: ByteVector =>
118- let stakingAmount = if (containsElement([USDN, NSBT], id))
119- then getInteger(stakingAddressUsdnAndNsbt, ((("rpd_balance_" + toBase58String(id)) + "_") + toString(this)))
120- else getInteger(stakingAddressDeFo, ((("%s%s%s__stakingBalance__" + toBase58String(id)) + "__") + toString(this)))
121- match stakingAmount {
122- case staked: Int =>
123- staked
124- case nothing: Unit =>
125- 0
126- case _ =>
127- throw("Match error")
128- }
129- case waves =>
130- let balances = wavesBalance(this)
131- (balances.regular - balances.available)
140+let stakedAmountEURN = match getInteger(stakingAddress, ((("%s%s%s__stakingBalance__" + toBase58String(EURN)) + "__") + toString(this))) {
141+ case staked: Int =>
142+ staked
143+ case nothing: Unit =>
144+ 0
145+ case _ =>
146+ throw("Match error")
132147 }
133148
149+let assetInitA = getIntegerValue(this, keyBalanceInitA)
134150
135-let stakedAmountA = stakedAmount(assetIdA)
151+let assetInitB = getIntegerValue(this, keyBalanceInitB)
136152
137-let stakedAmountB = stakedAmount(assetIdB)
153+let availableBalanceA = (balanceA - (if ((assetIdA == EURN))
154+ then stakedAmountEURN
155+ else 0))
138156
139-let availableBalanceA = (balanceA - stakedAmountA)
157+let availableBalanceB = (balanceB - (if ((assetIdB == EURN))
158+ then stakedAmountEURN
159+ else 0))
140160
141-let availableBalanceB = (balanceB - stakedAmountB)
161+let accountBalanceWithStakedA = (accountBalance(assetIdA) + (if ((assetIdA == EURN))
162+ then stakedAmountEURN
163+ else 0))
142164
143-let accountBalanceWithStakedA = (accountBalance(assetIdA) + stakedAmountA)
144-
145-let accountBalanceWithStakedB = (accountBalance(assetIdB) + stakedAmountB)
165+let accountBalanceWithStakedB = (accountBalance(assetIdB) + (if ((assetIdB == EURN))
166+ then stakedAmountEURN
167+ else 0))
146168
147169 let hasEnoughBalance = if ((accountBalanceWithStakedA >= balanceA))
148170 then (accountBalanceWithStakedB >= balanceB)
160182 }
161183
162184
185+func getAssetInfoFromString (assetStr) = if ((assetStr == "WAVES"))
186+ then $Tuple3("WAVES", "WAVES", 8)
187+ else {
188+ let stringId = assetStr
189+ let id = fromBase58String(assetStr)
190+ let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist"))
191+ $Tuple3(stringId, info.name, info.decimals)
192+ }
193+
194+
163195 func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)]
164196
165197
166-func stakingFee (assetId) = (9 * value(value(assetInfo(assetId)).minSponsoredFee))
167-
168-
169-func deductStakingFee (amtA,amtB) = if (containsElement(stakingAssets, assetIdB))
198+func deductStakingFee (amount,assetId) = if ((assetId == EURN))
170199 then {
171- let deduct = stakingFee(value(assetIdB))
172- let resB = (amtB - deduct)
173- if ((0 >= resB))
174- then throw(((((("Insufficient amount " + toString(amtB)) + " ") + assetNameB) + " to deduct staking fee ") + toString(deduct)))
175- else $Tuple2(amtA, resB)
200+ let result = (amount - stakingFeeInEURN)
201+ if ((0 >= result))
202+ then throw((((("Insufficient amount " + toString(amount)) + " to deduct staking fee ") + toString(stakingFeeInEURN)) + " USD-N"))
203+ else result
176204 }
177- else if (containsElement(stakingAssets, assetIdA))
178- then {
179- let deduct = stakingFee(value(assetIdA))
180- let resA = (amtA - deduct)
181- if ((0 >= resA))
182- then throw(((((("Insufficient amount " + toString(amtA)) + " ") + assetNameA) + " to deduct staking fee ") + toString(deduct)))
183- else $Tuple2(resA, amtB)
184- }
185- else $Tuple2(amtA, amtB)
205+ else amount
186206
187207
188208 func throwInsufficientAvailableBalance (amount,available,assetName) = throw((((((((("Insufficient DApp balance to pay " + toString(amount)) + " ") + assetName) + " due to staking. Available: ") + toString(available)) + " ") + assetName) + ". Please contact support in Telegram: https://t.me/swopfisupport"))
191211 func throwInsufficientAvailableBalances (amountA,amountB) = throw((((((((((((((((("Insufficient DApp balance to pay " + toString(amountA)) + " ") + assetNameA) + " and ") + toString(amountB)) + " ") + assetNameB) + " due to staking. Available: ") + toString(availableBalanceA)) + " ") + assetNameA) + " and ") + toString(availableBalanceB)) + " ") + assetNameB) + ". Please contact support in Telegram: https://t.me/swopfisupport"))
192212
193213
214+func suspendSuspicious () = suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(accountBalanceWithStakedA)) + " ") + assetNameA) + ", ") + toString(accountBalanceWithStakedB)) + " ") + assetNameB) + ". State: ") + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB))
215+
216+
194217 @Callable(i)
195-func init () = {
196- let $t060166093 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
197- let pmtAmountA = $t060166093._1
198- let pmtAssetIdA = $t060166093._2
199- let $t060986175 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
200- let pmtAmountB = $t060986175._1
201- let pmtAssetIdB = $t060986175._2
202- let $t061806257 = getAssetInfo(pmtAssetIdA)
203- let pmtStrAssetIdA = $t061806257._1
204- let pmtAssetNameA = $t061806257._2
205- let pmtDecimalsA = $t061806257._3
206- let $t062626339 = getAssetInfo(pmtAssetIdB)
207- let pmtStrAssetIdB = $t062626339._1
208- let pmtAssetNameB = $t062626339._2
209- let pmtDecimalsB = $t062626339._3
210- if (isDefined(getBoolean(this, keyActive)))
211- then throw("DApp is already active")
212- else if ((pmtAssetIdA == pmtAssetIdB))
213- then throw("Assets must be different")
214- else {
215- let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
216- let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
217- let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
218- let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, HALFDOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, HALFDOWN), pow(10, 0, shareDecimals, 0, 0, HALFDOWN))
219- let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
220- let shareIssueId = calculateAssetId(shareIssue)
221-[StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
222- }
218+func init (firstHarvest) = {
219+ let $t064036480 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
220+ let pmtAmountA = $t064036480._1
221+ let pmtAssetIdA = $t064036480._2
222+ let $t064856562 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
223+ let pmtAmountB = $t064856562._1
224+ let pmtAssetIdB = $t064856562._2
225+ let $t065676644 = getAssetInfo(pmtAssetIdA)
226+ let pmtStrAssetIdA = $t065676644._1
227+ let pmtAssetNameA = $t065676644._2
228+ let pmtDecimalsA = $t065676644._3
229+ let $t066496726 = getAssetInfo(pmtAssetIdB)
230+ let pmtStrAssetIdB = $t066496726._1
231+ let pmtAssetNameB = $t066496726._2
232+ let pmtDecimalsB = $t066496726._3
233+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
234+ then throw("Only admin can call this function")
235+ else if (isDefined(getBoolean(this, keyActive)))
236+ then throw("DApp is already active")
237+ else if ((pmtAssetIdA == pmtAssetIdB))
238+ then throw("Assets must be different")
239+ else {
240+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
241+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
242+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
243+ let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, HALFDOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, HALFDOWN), pow(10, 0, shareDecimals, 0, 0, HALFDOWN))
244+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
245+ let shareIssueId = calculateAssetId(shareIssue)
246+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
247+ if (firstHarvest)
248+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
249+ else baseEntry
250+ }
223251 }
252+
253+
254+
255+@Callable(i)
256+func initWithInitRatio (amtAssetA,amtAssetB,strAssetIdA,strAssetIdB,firstHarvest) = {
257+ let $t089349021 = getAssetInfoFromString(strAssetIdA)
258+ let pmtStrAssetIdA = $t089349021._1
259+ let pmtAssetNameA = $t089349021._2
260+ let pmtDecimalsA = $t089349021._3
261+ let $t090269113 = getAssetInfoFromString(strAssetIdB)
262+ let pmtStrAssetIdB = $t090269113._1
263+ let pmtAssetNameB = $t090269113._2
264+ let pmtDecimalsB = $t090269113._3
265+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
266+ then throw("Only admin can call this function")
267+ else if (isDefined(getBoolean(this, keyActive)))
268+ then throw("DApp is already active")
269+ else if ((strAssetIdA == strAssetIdB))
270+ then throw("Assets must be different")
271+ else {
272+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
273+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
274+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
275+ let shareInitialSupply = 0
276+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
277+ let shareIssueId = calculateAssetId(shareIssue)
278+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceInitA, amtAssetA), IntegerEntry(keyBalanceInitB, amtAssetB), IntegerEntry(keyBalanceA, 0), IntegerEntry(keyBalanceB, 0), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply)]
279+ if (firstHarvest)
280+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
281+ else baseEntry
282+ }
283+ }
284+
285+
286+
287+@Callable(i)
288+func keepLimitForFirstHarvest (shareLimit) = if (!(isActive))
289+ then throw("DApp is inactive at this moment")
290+ else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
291+ then throw("Only admin can call this function")
292+ else [IntegerEntry(kShareLimit, shareLimit)]
224293
225294
226295
228297 func replenishWithTwoTokens (slippageTolerance) = {
229298 let pmtAssetIdA = i.payments[0].assetId
230299 let pmtAssetIdB = i.payments[1].assetId
231- let $t081628370 = deductStakingFee(i.payments[0].amount, i.payments[1].amount)
232- let pmtAmountA = $t081628370._1
233- let pmtAmountB = $t081628370._2
234- let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
235- let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
236- let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
237- let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
238- if (!(isActive))
239- then throw("DApp is inactive at this moment")
240- else if (if ((0 > slippageTolerance))
241- then true
242- else (slippageTolerance > slippageToleranceDelimiter))
243- then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
244- else if ((size(i.payments) != 2))
245- then throw("Two attached assets expected")
246- else if (if ((pmtAssetIdA != assetIdA))
300+ let pmtAmountA = deductStakingFee(i.payments[0].amount, pmtAssetIdA)
301+ let pmtAmountB = deductStakingFee(i.payments[1].amount, pmtAssetIdB)
302+ if (if ((balanceA == 0))
303+ then (balanceB == 0)
304+ else false)
305+ then {
306+ let $t01179811875 = getAssetInfo(pmtAssetIdA)
307+ let pmtStrAssetIdA = $t01179811875._1
308+ let pmtAssetNameA = $t01179811875._2
309+ let pmtDecimalsA = $t01179811875._3
310+ let $t01188411961 = getAssetInfo(pmtAssetIdB)
311+ let pmtStrAssetIdB = $t01188411961._1
312+ let pmtAssetNameB = $t01188411961._2
313+ let pmtDecimalsB = $t01188411961._3
314+ let tokenRatio = fraction(fraction(assetInitA, scaleValue8, pmtAmountA), scaleValue3, fraction(assetInitB, scaleValue8, pmtAmountB))
315+ if ((pmtAssetIdA == pmtAssetIdB))
316+ then throw("Assets must be different")
317+ else {
318+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
319+ let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, HALFDOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, HALFDOWN), pow(10, 0, shareDecimals, 0, 0, HALFDOWN))
320+ if (!(isActive))
321+ then throw("DApp is inactive at this moment")
322+ else if (if ((0 > slippageTolerance))
323+ then true
324+ else (slippageTolerance > slippageToleranceDelimiter))
325+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
326+ else if ((size(i.payments) != 2))
327+ then throw("Two attached assets expected")
328+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
329+ then true
330+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
331+ then throw("Incorrect assets amount: amounts must have the contract ratio")
332+ else if (if ((pmtAssetIdA != assetIdA))
333+ then true
334+ else (pmtAssetIdB != assetIdB))
335+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
336+ else if ((shareInitialSupply == 0))
337+ then throw("Too small amount to replenish")
338+ else if (!(hasEnoughBalance))
339+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
340+ else [Reissue(shareAssetId, shareInitialSupply, true), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareAssetId)]
341+ }
342+ }
343+ else {
344+ let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
345+ let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
346+ let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
347+ let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
348+ if (!(isActive))
349+ then throw("DApp is inactive at this moment")
350+ else if (if ((0 > slippageTolerance))
247351 then true
248- else (pmtAssetIdB != assetIdB))
249- then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
250- else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
251- then true
252- else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
253- then throw("Incorrect assets amount: amounts must have the contract ratio")
254- else if ((shareTokenToPayAmount == 0))
255- then throw("Too small amount to replenish")
256- else if (!(hasEnoughBalance))
257- then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB) + ". State: ") + toString(accountBalance(assetIdA))) + " ") + assetNameA) + ", ") + toString(accountBalance(assetIdB))) + " ") + assetNameB)))
258- else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
352+ else (slippageTolerance > slippageToleranceDelimiter))
353+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
354+ else if ((size(i.payments) != 2))
355+ then throw("Two attached assets expected")
356+ else if (if ((pmtAssetIdA != assetIdA))
357+ then true
358+ else (pmtAssetIdB != assetIdB))
359+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
360+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
361+ then true
362+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
363+ then throw("Incorrect assets amount: amounts must have the contract ratio")
364+ else if ((shareTokenToPayAmount == 0))
365+ then throw("Too small amount to replenish")
366+ else if (!(hasEnoughBalance))
367+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
368+ else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
369+ }
259370 }
260371
261372
262373
263374 @Callable(i)
264375 func withdraw () = {
265- let $t01062710777 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
266- let pmtAmount = $t01062710777._1
267- let pmtAssetId = $t01062710777._2
268- let $t01078210929 = deductStakingFee(fraction(pmtAmount, balanceA, shareAssetSupply), fraction(pmtAmount, balanceB, shareAssetSupply))
269- let amountToPayA = $t01078210929._1
270- let amountToPayB = $t01078210929._2
376+ let $t01637316523 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
377+ let pmtAmount = $t01637316523._1
378+ let pmtAssetId = $t01637316523._2
379+ let amountToPayA = deductStakingFee(fraction(pmtAmount, balanceA, shareAssetSupply), assetIdA)
380+ let amountToPayB = deductStakingFee(fraction(pmtAmount, balanceB, shareAssetSupply), assetIdB)
271381 if (!(isActive))
272382 then throw("DApp is inactive at this moment")
273383 else if ((size(i.payments) != 1))
275385 else if ((pmtAssetId != shareAssetId))
276386 then throw(("Incorrect asset attached. Expected: " + toBase58String(shareAssetId)))
277387 else if (!(hasEnoughBalance))
278- then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB) + ". State: ") + toString(accountBalance(assetIdA))) + " ") + assetNameA) + ", ") + toString(accountBalance(assetIdB))) + " ") + assetNameB)))
388+ then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
279389 else if (if ((amountToPayA > availableBalanceA))
280390 then true
281391 else (amountToPayB > availableBalanceB))
287397
288398 @Callable(i)
289399 func exchange (minAmountToReceive) = {
290- let $t01220412279 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
291- let pmtAmount = $t01220412279._1
292- let pmtAssetId = $t01220412279._2
400+ let $t01772917804 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
401+ let pmtAmount = $t01772917804._1
402+ let pmtAssetId = $t01772917804._2
293403 func calculateFees (tokenFrom,tokenTo) = {
294404 let amountWithoutFee = fraction(tokenTo, pmtAmount, (pmtAmount + tokenFrom))
295405 let amountWithFee = fraction(amountWithoutFee, (commissionScaleDelimiter - commission), commissionScaleDelimiter)
301411
302412 if (!(isActive))
303413 then throw("DApp is inactive at this moment")
304- else if ((0 >= minAmountToReceive))
305- then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
306- else if ((size(i.payments) != 1))
307- then throw("One attached payment expected")
308- else if (!(hasEnoughBalance))
309- then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB) + ". State: ") + toString(accountBalance(assetIdA))) + " ") + assetNameA) + ", ") + toString(accountBalance(assetIdB))) + " ") + assetNameB)))
310- else if ((pmtAssetId == assetIdA))
311- then {
312- let assetIdSend = assetIdB
313- let $t01371713808 = calculateFees(balanceA, balanceB)
314- let amountWithoutFee = $t01371713808._1
315- let amountWithFee = $t01371713808._2
316- let governanceReward = $t01371713808._3
317- let newBalanceA = (balanceA + pmtAmount)
318- let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
319- if (if ((stakedAmountA >= newBalanceA))
320- then true
321- else (stakedAmountB >= newBalanceB))
322- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
323- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
324- }
325- else if ((pmtAssetId == assetIdB))
414+ else if (if ((balanceA == 0))
415+ then true
416+ else (balanceB == 0))
417+ then throw("Can't exchange with zero balance")
418+ else if ((0 >= minAmountToReceive))
419+ then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
420+ else if ((size(i.payments) != 1))
421+ then throw("One attached payment expected")
422+ else if (!(hasEnoughBalance))
423+ then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
424+ else if ((pmtAssetId == assetIdA))
326425 then {
327- let assetIdSend = assetIdA
328- let $t01463114722 = calculateFees(balanceB, balanceA)
329- let amountWithoutFee = $t01463114722._1
330- let amountWithFee = $t01463114722._2
331- let governanceReward = $t01463114722._3
332- let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
333- let newBalanceB = (balanceB + pmtAmount)
334- if (if ((stakedAmountA >= newBalanceA))
426+ let assetIdSend = assetIdB
427+ let $t01907819169 = calculateFees(balanceA, balanceB)
428+ let amountWithoutFee = $t01907819169._1
429+ let amountWithFee = $t01907819169._2
430+ let governanceReward = $t01907819169._3
431+ let newBalanceA = (balanceA + pmtAmount)
432+ let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
433+ if (if (if ((assetIdA == EURN))
434+ then (stakedAmountEURN >= newBalanceA)
435+ else false)
335436 then true
336- else (stakedAmountB >= newBalanceB))
337- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
338- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
437+ else if ((assetIdB == EURN))
438+ then (stakedAmountEURN >= newBalanceB)
439+ else false)
440+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
441+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
339442 }
340- else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
443+ else if ((pmtAssetId == assetIdB))
444+ then {
445+ let assetIdSend = assetIdA
446+ let $t02003820129 = calculateFees(balanceB, balanceA)
447+ let amountWithoutFee = $t02003820129._1
448+ let amountWithFee = $t02003820129._2
449+ let governanceReward = $t02003820129._3
450+ let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
451+ let newBalanceB = (balanceB + pmtAmount)
452+ if (if (if ((assetIdA == EURN))
453+ then (stakedAmountEURN >= newBalanceA)
454+ else false)
455+ then true
456+ else if ((assetIdB == EURN))
457+ then (stakedAmountEURN >= newBalanceB)
458+ else false)
459+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
460+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
461+ }
462+ else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
341463 }
342464
343465
362484
363485 @Callable(i)
364486 func takeIntoAccountExtraFunds (amountLeave) = {
365- let uncountableAmountEnrollA = (accountBalanceWithStakedA - balanceA)
366- let uncountableAmountEnrollB = (accountBalanceWithStakedB - balanceB)
367- let amountEnrollA = (uncountableAmountEnrollA - (if ((assetIdA == EURN))
487+ let uncountableAmountEnrollAssetA = (accountBalanceWithStakedA - balanceA)
488+ let uncountableAmountEnrollAssetB = (accountBalanceWithStakedB - balanceB)
489+ let amountEnrollA = (uncountableAmountEnrollAssetA - (if ((assetIdA == unit))
368490 then amountLeave
369491 else 0))
370- let amountEnrollB = (uncountableAmountEnrollB - (if ((assetIdB == EURN))
492+ let amountEnrollB = (uncountableAmountEnrollAssetB - (if ((assetIdB == unit))
371493 then amountLeave
372494 else 0))
373495 if (!(isActive))
376498 then throw("Only the DApp itself can call this function")
377499 else if ((0 > amountLeave))
378500 then throw(("Argument 'amountLeave' cannot be negative. Actual: " + toString(amountLeave)))
379- else if (if ((0 > uncountableAmountEnrollA))
501+ else if (if ((0 > uncountableAmountEnrollAssetA))
380502 then true
381- else (0 > uncountableAmountEnrollB))
503+ else (0 > uncountableAmountEnrollAssetB))
382504 then suspend("Enroll amount negative")
383505 else if (if ((0 > amountEnrollA))
384506 then true
394516 let callTakeIntoAccount = if ((invoke.dApp == this))
395517 then (invoke.function == "takeIntoAccountExtraFunds")
396518 else false
397- let callStakingUsdnOrNsbt = if ((invoke.dApp == stakingAddressUsdnAndNsbt))
398- then if (if (containsElement(["lockNeutrino", "lockNsbt"], invoke.function))
519+ let callStaking = if ((invoke.dApp == stakingAddress))
520+ then if (if (if ((invoke.function == "startStaking"))
399521 then (size(invoke.payments) == 1)
400522 else false)
401- then true
402- else if (containsElement(["unlockNeutrino", "unlockNsbt"], invoke.function))
403- then (size(invoke.payments) == 0)
404- else false
405- else false
406- let callStakingDeFo = if ((invoke.dApp == stakingAddressDeFo))
407- then if (if ((invoke.function == "startStaking"))
408- then (size(invoke.payments) == 1)
523+ then (invoke.payments[0].assetId == EURN)
409524 else false)
410525 then true
411526 else if ((invoke.function == "stopStaking"))
412527 then (size(invoke.payments) == 0)
413528 else false
414529 else false
530+ let exchangeToWaves = if (if (if ((invoke.dApp == EURNToWavesExchanger))
531+ then (invoke.function == "exchange")
532+ else false)
533+ then (assetIdA == EURN)
534+ else false)
535+ then true
536+ else if ((assetIdB == EURN))
537+ then (invoke.payments[0].assetId == EURN)
538+ else false
415539 let signedByAdmin = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1))
416540 then true
417541 else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey2))
421545 else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKeyStaking)
422546 if (if (if (callTakeIntoAccount)
423547 then true
424- else callStakingUsdnOrNsbt)
548+ else callStaking)
425549 then true
426- else callStakingDeFo)
550+ else exchangeToWaves)
427551 then signedByAdmin
428552 else false
429553 case _ =>
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 4 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let version = "1.0.0"
55
66 let keyVersion = "version"
77
88 let keyActive = "active"
99
1010 let keyAssetIdA = "A_asset_id"
1111
1212 let keyAssetIdB = "B_asset_id"
1313
1414 let keyBalanceA = "A_asset_balance"
1515
1616 let keyBalanceB = "B_asset_balance"
1717
18+let keyBalanceInitA = "A_asset_init"
19+
20+let keyBalanceInitB = "B_asset_init"
21+
1822 let keyShareAssetId = "share_asset_id"
1923
2024 let keyShareAssetSupply = "share_asset_supply"
2125
2226 let keyCommission = "commission"
2327
2428 let keyCommissionScaleDelimiter = "commission_scale_delimiter"
2529
2630 let keyCause = "shutdown_cause"
31+
32+let keyFirstHarvest = "first_harvest"
33+
34+let keyFirstHarvestHeight = "first_harvest_height"
35+
36+let kShareLimit = "share_limit_on_first_harvest"
37+
38+let kBasePeriod = "base_period"
39+
40+let kPeriodLength = "period_length"
41+
42+let kStartHeight = "start_height"
43+
44+let kFirstHarvestHeight = "first_harvest_height"
2745
2846 let adminPubKey1 = base58'DXDY2itiEcYBtGkVLnkpHtDFyWQUkoLJz79uJ7ECbMrA'
2947
3048 let adminPubKey2 = base58'E6Wa1SGoktYcjHjsKrvjMiqJY3SWmGKcD8Q5L8kxSPS7'
3149
3250 let adminPubKey3 = base58'AZmWJtuy4GeVrMmJH4hfFBRApe1StvhJSk4jcbT6bArQ'
3351
3452 let adminPubKeyStartStop = base58'EtVkT6ed8GtbUiVVEqdmEqsp2J4qbb3rre2HFgxeVYdg'
3553
3654 let adminPubKeyStaking = base58'Czn4yoAuUZCVCLJDRfskn8URfkwpknwBTZDbs1wFrY7h'
3755
38-let governanceAddress = Address(base58'3P6J84oH51DzY6xk2mT5TheXRbrCwBMxonp')
56+let walletAddress = Address(base58'3P6J84oH51DzY6xk2mT5TheXRbrCwBMxonp')
3957
40-let stakingAddressUsdnAndNsbt = Address(base58'3PNikM6yp4NqcSU8guxQtmR5onr2D4e8yTJ')
58+let votingAddress = Address(base58'3PQZWxShKGRgBN1qoJw6B4s9YWS9FneZTPg')
4159
42-let stakingAddressDeFo = Address(base58'3PFhcMmEZoQTQ6ohA844c7C9M8ZJ18P8dDj')
60+let stakingAddress = Address(base58'3PFhcMmEZoQTQ6ohA844c7C9M8ZJ18P8dDj')
4361
44-let USDN = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
62+let EURNToWavesExchanger = Address(base58'3PK7Xe5BiedRyxHLuMQx5ey9riUQqvUths2')
4563
46-let NSBT = base58'6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g'
64+let basePeriod = valueOrErrorMessage(getInteger(votingAddress, kBasePeriod), "Empty kBasePeriod")
65+
66+let startHeight = valueOrErrorMessage(getInteger(votingAddress, kStartHeight), "Empty kStartHeight")
67+
68+let periodLength = valueOrErrorMessage(getInteger(votingAddress, kPeriodLength), "Empty kPeriodLength")
69+
70+let firstHarvestEndPeriod = ((basePeriod + ((height - startHeight) / periodLength)) + 3)
4771
4872 let EURN = base58'DUk2YTxhRoAqMJLus4G2b3fR8hMHVh6eiyFx5r29VR6t'
4973
50-let stakingAssets = [USDN, NSBT, EURN]
74+let stakingFeeInEURN = 234000
5175
5276 let isActive = getBooleanValue(this, keyActive)
5377
5478 let strAssetIdA = getStringValue(this, keyAssetIdA)
5579
5680 let strAssetIdB = getStringValue(this, keyAssetIdB)
5781
5882 let assetIdA = if ((strAssetIdA == "WAVES"))
5983 then unit
6084 else fromBase58String(strAssetIdA)
6185
6286 let assetIdB = if ((strAssetIdB == "WAVES"))
6387 then unit
6488 else fromBase58String(strAssetIdB)
6589
6690 let assetNameA = match assetIdA {
6791 case id: ByteVector =>
6892 value(assetInfo(id)).name
6993 case waves: Unit =>
7094 "WAVES"
7195 case _ =>
7296 throw("Match error")
7397 }
7498
7599 let assetNameB = match assetIdB {
76100 case id: ByteVector =>
77101 value(assetInfo(id)).name
78102 case waves: Unit =>
79103 "WAVES"
80104 case _ =>
81105 throw("Match error")
82106 }
83107
84108 let balanceA = getIntegerValue(this, keyBalanceA)
85109
86110 let balanceB = getIntegerValue(this, keyBalanceB)
87111
88112 let shareAssetId = fromBase58String(getStringValue(this, keyShareAssetId))
89113
90114 let shareAssetSupply = getIntegerValue(this, keyShareAssetSupply)
91115
92116 let commission = 3000
93117
94118 let commissionGovernance = 1200
95119
96120 let commissionScaleDelimiter = 1000000
97121
98122 let scaleValue3 = 1000
99123
100124 let scaleValue8 = 100000000
101125
102126 let slippageToleranceDelimiter = 1000
103127
104128 let scaleValue8Digits = 8
105129
106130 func accountBalance (assetId) = match assetId {
107131 case id: ByteVector =>
108132 assetBalance(this, id)
109133 case waves: Unit =>
110134 wavesBalance(this).available
111135 case _ =>
112136 throw("Match error")
113137 }
114138
115139
116-func stakedAmount (assetId) = match assetId {
117- case id: ByteVector =>
118- let stakingAmount = if (containsElement([USDN, NSBT], id))
119- then getInteger(stakingAddressUsdnAndNsbt, ((("rpd_balance_" + toBase58String(id)) + "_") + toString(this)))
120- else getInteger(stakingAddressDeFo, ((("%s%s%s__stakingBalance__" + toBase58String(id)) + "__") + toString(this)))
121- match stakingAmount {
122- case staked: Int =>
123- staked
124- case nothing: Unit =>
125- 0
126- case _ =>
127- throw("Match error")
128- }
129- case waves =>
130- let balances = wavesBalance(this)
131- (balances.regular - balances.available)
140+let stakedAmountEURN = match getInteger(stakingAddress, ((("%s%s%s__stakingBalance__" + toBase58String(EURN)) + "__") + toString(this))) {
141+ case staked: Int =>
142+ staked
143+ case nothing: Unit =>
144+ 0
145+ case _ =>
146+ throw("Match error")
132147 }
133148
149+let assetInitA = getIntegerValue(this, keyBalanceInitA)
134150
135-let stakedAmountA = stakedAmount(assetIdA)
151+let assetInitB = getIntegerValue(this, keyBalanceInitB)
136152
137-let stakedAmountB = stakedAmount(assetIdB)
153+let availableBalanceA = (balanceA - (if ((assetIdA == EURN))
154+ then stakedAmountEURN
155+ else 0))
138156
139-let availableBalanceA = (balanceA - stakedAmountA)
157+let availableBalanceB = (balanceB - (if ((assetIdB == EURN))
158+ then stakedAmountEURN
159+ else 0))
140160
141-let availableBalanceB = (balanceB - stakedAmountB)
161+let accountBalanceWithStakedA = (accountBalance(assetIdA) + (if ((assetIdA == EURN))
162+ then stakedAmountEURN
163+ else 0))
142164
143-let accountBalanceWithStakedA = (accountBalance(assetIdA) + stakedAmountA)
144-
145-let accountBalanceWithStakedB = (accountBalance(assetIdB) + stakedAmountB)
165+let accountBalanceWithStakedB = (accountBalance(assetIdB) + (if ((assetIdB == EURN))
166+ then stakedAmountEURN
167+ else 0))
146168
147169 let hasEnoughBalance = if ((accountBalanceWithStakedA >= balanceA))
148170 then (accountBalanceWithStakedB >= balanceB)
149171 else false
150172
151173 func getAssetInfo (assetId) = match assetId {
152174 case id: ByteVector =>
153175 let stringId = toBase58String(id)
154176 let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist"))
155177 $Tuple3(stringId, info.name, info.decimals)
156178 case waves: Unit =>
157179 $Tuple3("WAVES", "WAVES", 8)
158180 case _ =>
159181 throw("Match error")
160182 }
161183
162184
185+func getAssetInfoFromString (assetStr) = if ((assetStr == "WAVES"))
186+ then $Tuple3("WAVES", "WAVES", 8)
187+ else {
188+ let stringId = assetStr
189+ let id = fromBase58String(assetStr)
190+ let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist"))
191+ $Tuple3(stringId, info.name, info.decimals)
192+ }
193+
194+
163195 func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)]
164196
165197
166-func stakingFee (assetId) = (9 * value(value(assetInfo(assetId)).minSponsoredFee))
167-
168-
169-func deductStakingFee (amtA,amtB) = if (containsElement(stakingAssets, assetIdB))
198+func deductStakingFee (amount,assetId) = if ((assetId == EURN))
170199 then {
171- let deduct = stakingFee(value(assetIdB))
172- let resB = (amtB - deduct)
173- if ((0 >= resB))
174- then throw(((((("Insufficient amount " + toString(amtB)) + " ") + assetNameB) + " to deduct staking fee ") + toString(deduct)))
175- else $Tuple2(amtA, resB)
200+ let result = (amount - stakingFeeInEURN)
201+ if ((0 >= result))
202+ then throw((((("Insufficient amount " + toString(amount)) + " to deduct staking fee ") + toString(stakingFeeInEURN)) + " USD-N"))
203+ else result
176204 }
177- else if (containsElement(stakingAssets, assetIdA))
178- then {
179- let deduct = stakingFee(value(assetIdA))
180- let resA = (amtA - deduct)
181- if ((0 >= resA))
182- then throw(((((("Insufficient amount " + toString(amtA)) + " ") + assetNameA) + " to deduct staking fee ") + toString(deduct)))
183- else $Tuple2(resA, amtB)
184- }
185- else $Tuple2(amtA, amtB)
205+ else amount
186206
187207
188208 func throwInsufficientAvailableBalance (amount,available,assetName) = throw((((((((("Insufficient DApp balance to pay " + toString(amount)) + " ") + assetName) + " due to staking. Available: ") + toString(available)) + " ") + assetName) + ". Please contact support in Telegram: https://t.me/swopfisupport"))
189209
190210
191211 func throwInsufficientAvailableBalances (amountA,amountB) = throw((((((((((((((((("Insufficient DApp balance to pay " + toString(amountA)) + " ") + assetNameA) + " and ") + toString(amountB)) + " ") + assetNameB) + " due to staking. Available: ") + toString(availableBalanceA)) + " ") + assetNameA) + " and ") + toString(availableBalanceB)) + " ") + assetNameB) + ". Please contact support in Telegram: https://t.me/swopfisupport"))
192212
193213
214+func suspendSuspicious () = suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(accountBalanceWithStakedA)) + " ") + assetNameA) + ", ") + toString(accountBalanceWithStakedB)) + " ") + assetNameB) + ". State: ") + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB))
215+
216+
194217 @Callable(i)
195-func init () = {
196- let $t060166093 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
197- let pmtAmountA = $t060166093._1
198- let pmtAssetIdA = $t060166093._2
199- let $t060986175 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
200- let pmtAmountB = $t060986175._1
201- let pmtAssetIdB = $t060986175._2
202- let $t061806257 = getAssetInfo(pmtAssetIdA)
203- let pmtStrAssetIdA = $t061806257._1
204- let pmtAssetNameA = $t061806257._2
205- let pmtDecimalsA = $t061806257._3
206- let $t062626339 = getAssetInfo(pmtAssetIdB)
207- let pmtStrAssetIdB = $t062626339._1
208- let pmtAssetNameB = $t062626339._2
209- let pmtDecimalsB = $t062626339._3
210- if (isDefined(getBoolean(this, keyActive)))
211- then throw("DApp is already active")
212- else if ((pmtAssetIdA == pmtAssetIdB))
213- then throw("Assets must be different")
214- else {
215- let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
216- let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
217- let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
218- let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, HALFDOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, HALFDOWN), pow(10, 0, shareDecimals, 0, 0, HALFDOWN))
219- let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
220- let shareIssueId = calculateAssetId(shareIssue)
221-[StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
222- }
218+func init (firstHarvest) = {
219+ let $t064036480 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
220+ let pmtAmountA = $t064036480._1
221+ let pmtAssetIdA = $t064036480._2
222+ let $t064856562 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
223+ let pmtAmountB = $t064856562._1
224+ let pmtAssetIdB = $t064856562._2
225+ let $t065676644 = getAssetInfo(pmtAssetIdA)
226+ let pmtStrAssetIdA = $t065676644._1
227+ let pmtAssetNameA = $t065676644._2
228+ let pmtDecimalsA = $t065676644._3
229+ let $t066496726 = getAssetInfo(pmtAssetIdB)
230+ let pmtStrAssetIdB = $t066496726._1
231+ let pmtAssetNameB = $t066496726._2
232+ let pmtDecimalsB = $t066496726._3
233+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
234+ then throw("Only admin can call this function")
235+ else if (isDefined(getBoolean(this, keyActive)))
236+ then throw("DApp is already active")
237+ else if ((pmtAssetIdA == pmtAssetIdB))
238+ then throw("Assets must be different")
239+ else {
240+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
241+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
242+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
243+ let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, HALFDOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, HALFDOWN), pow(10, 0, shareDecimals, 0, 0, HALFDOWN))
244+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
245+ let shareIssueId = calculateAssetId(shareIssue)
246+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
247+ if (firstHarvest)
248+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
249+ else baseEntry
250+ }
223251 }
252+
253+
254+
255+@Callable(i)
256+func initWithInitRatio (amtAssetA,amtAssetB,strAssetIdA,strAssetIdB,firstHarvest) = {
257+ let $t089349021 = getAssetInfoFromString(strAssetIdA)
258+ let pmtStrAssetIdA = $t089349021._1
259+ let pmtAssetNameA = $t089349021._2
260+ let pmtDecimalsA = $t089349021._3
261+ let $t090269113 = getAssetInfoFromString(strAssetIdB)
262+ let pmtStrAssetIdB = $t090269113._1
263+ let pmtAssetNameB = $t090269113._2
264+ let pmtDecimalsB = $t090269113._3
265+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
266+ then throw("Only admin can call this function")
267+ else if (isDefined(getBoolean(this, keyActive)))
268+ then throw("DApp is already active")
269+ else if ((strAssetIdA == strAssetIdB))
270+ then throw("Assets must be different")
271+ else {
272+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
273+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
274+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
275+ let shareInitialSupply = 0
276+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
277+ let shareIssueId = calculateAssetId(shareIssue)
278+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceInitA, amtAssetA), IntegerEntry(keyBalanceInitB, amtAssetB), IntegerEntry(keyBalanceA, 0), IntegerEntry(keyBalanceB, 0), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply)]
279+ if (firstHarvest)
280+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
281+ else baseEntry
282+ }
283+ }
284+
285+
286+
287+@Callable(i)
288+func keepLimitForFirstHarvest (shareLimit) = if (!(isActive))
289+ then throw("DApp is inactive at this moment")
290+ else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
291+ then throw("Only admin can call this function")
292+ else [IntegerEntry(kShareLimit, shareLimit)]
224293
225294
226295
227296 @Callable(i)
228297 func replenishWithTwoTokens (slippageTolerance) = {
229298 let pmtAssetIdA = i.payments[0].assetId
230299 let pmtAssetIdB = i.payments[1].assetId
231- let $t081628370 = deductStakingFee(i.payments[0].amount, i.payments[1].amount)
232- let pmtAmountA = $t081628370._1
233- let pmtAmountB = $t081628370._2
234- let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
235- let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
236- let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
237- let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
238- if (!(isActive))
239- then throw("DApp is inactive at this moment")
240- else if (if ((0 > slippageTolerance))
241- then true
242- else (slippageTolerance > slippageToleranceDelimiter))
243- then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
244- else if ((size(i.payments) != 2))
245- then throw("Two attached assets expected")
246- else if (if ((pmtAssetIdA != assetIdA))
300+ let pmtAmountA = deductStakingFee(i.payments[0].amount, pmtAssetIdA)
301+ let pmtAmountB = deductStakingFee(i.payments[1].amount, pmtAssetIdB)
302+ if (if ((balanceA == 0))
303+ then (balanceB == 0)
304+ else false)
305+ then {
306+ let $t01179811875 = getAssetInfo(pmtAssetIdA)
307+ let pmtStrAssetIdA = $t01179811875._1
308+ let pmtAssetNameA = $t01179811875._2
309+ let pmtDecimalsA = $t01179811875._3
310+ let $t01188411961 = getAssetInfo(pmtAssetIdB)
311+ let pmtStrAssetIdB = $t01188411961._1
312+ let pmtAssetNameB = $t01188411961._2
313+ let pmtDecimalsB = $t01188411961._3
314+ let tokenRatio = fraction(fraction(assetInitA, scaleValue8, pmtAmountA), scaleValue3, fraction(assetInitB, scaleValue8, pmtAmountB))
315+ if ((pmtAssetIdA == pmtAssetIdB))
316+ then throw("Assets must be different")
317+ else {
318+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
319+ let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, HALFDOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, HALFDOWN), pow(10, 0, shareDecimals, 0, 0, HALFDOWN))
320+ if (!(isActive))
321+ then throw("DApp is inactive at this moment")
322+ else if (if ((0 > slippageTolerance))
323+ then true
324+ else (slippageTolerance > slippageToleranceDelimiter))
325+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
326+ else if ((size(i.payments) != 2))
327+ then throw("Two attached assets expected")
328+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
329+ then true
330+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
331+ then throw("Incorrect assets amount: amounts must have the contract ratio")
332+ else if (if ((pmtAssetIdA != assetIdA))
333+ then true
334+ else (pmtAssetIdB != assetIdB))
335+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
336+ else if ((shareInitialSupply == 0))
337+ then throw("Too small amount to replenish")
338+ else if (!(hasEnoughBalance))
339+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
340+ else [Reissue(shareAssetId, shareInitialSupply, true), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareAssetId)]
341+ }
342+ }
343+ else {
344+ let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
345+ let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
346+ let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
347+ let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
348+ if (!(isActive))
349+ then throw("DApp is inactive at this moment")
350+ else if (if ((0 > slippageTolerance))
247351 then true
248- else (pmtAssetIdB != assetIdB))
249- then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
250- else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
251- then true
252- else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
253- then throw("Incorrect assets amount: amounts must have the contract ratio")
254- else if ((shareTokenToPayAmount == 0))
255- then throw("Too small amount to replenish")
256- else if (!(hasEnoughBalance))
257- then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB) + ". State: ") + toString(accountBalance(assetIdA))) + " ") + assetNameA) + ", ") + toString(accountBalance(assetIdB))) + " ") + assetNameB)))
258- else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
352+ else (slippageTolerance > slippageToleranceDelimiter))
353+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
354+ else if ((size(i.payments) != 2))
355+ then throw("Two attached assets expected")
356+ else if (if ((pmtAssetIdA != assetIdA))
357+ then true
358+ else (pmtAssetIdB != assetIdB))
359+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
360+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
361+ then true
362+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
363+ then throw("Incorrect assets amount: amounts must have the contract ratio")
364+ else if ((shareTokenToPayAmount == 0))
365+ then throw("Too small amount to replenish")
366+ else if (!(hasEnoughBalance))
367+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
368+ else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
369+ }
259370 }
260371
261372
262373
263374 @Callable(i)
264375 func withdraw () = {
265- let $t01062710777 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
266- let pmtAmount = $t01062710777._1
267- let pmtAssetId = $t01062710777._2
268- let $t01078210929 = deductStakingFee(fraction(pmtAmount, balanceA, shareAssetSupply), fraction(pmtAmount, balanceB, shareAssetSupply))
269- let amountToPayA = $t01078210929._1
270- let amountToPayB = $t01078210929._2
376+ let $t01637316523 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
377+ let pmtAmount = $t01637316523._1
378+ let pmtAssetId = $t01637316523._2
379+ let amountToPayA = deductStakingFee(fraction(pmtAmount, balanceA, shareAssetSupply), assetIdA)
380+ let amountToPayB = deductStakingFee(fraction(pmtAmount, balanceB, shareAssetSupply), assetIdB)
271381 if (!(isActive))
272382 then throw("DApp is inactive at this moment")
273383 else if ((size(i.payments) != 1))
274384 then throw("One attached payment expected")
275385 else if ((pmtAssetId != shareAssetId))
276386 then throw(("Incorrect asset attached. Expected: " + toBase58String(shareAssetId)))
277387 else if (!(hasEnoughBalance))
278- then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB) + ". State: ") + toString(accountBalance(assetIdA))) + " ") + assetNameA) + ", ") + toString(accountBalance(assetIdB))) + " ") + assetNameB)))
388+ then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
279389 else if (if ((amountToPayA > availableBalanceA))
280390 then true
281391 else (amountToPayB > availableBalanceB))
282392 then throwInsufficientAvailableBalances(amountToPayA, amountToPayB)
283393 else [IntegerEntry(keyBalanceA, (balanceA - amountToPayA)), IntegerEntry(keyBalanceB, (balanceB - amountToPayB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply - pmtAmount)), Burn(shareAssetId, pmtAmount), ScriptTransfer(i.caller, amountToPayA, assetIdA), ScriptTransfer(i.caller, amountToPayB, assetIdB)]
284394 }
285395
286396
287397
288398 @Callable(i)
289399 func exchange (minAmountToReceive) = {
290- let $t01220412279 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
291- let pmtAmount = $t01220412279._1
292- let pmtAssetId = $t01220412279._2
400+ let $t01772917804 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
401+ let pmtAmount = $t01772917804._1
402+ let pmtAssetId = $t01772917804._2
293403 func calculateFees (tokenFrom,tokenTo) = {
294404 let amountWithoutFee = fraction(tokenTo, pmtAmount, (pmtAmount + tokenFrom))
295405 let amountWithFee = fraction(amountWithoutFee, (commissionScaleDelimiter - commission), commissionScaleDelimiter)
296406 let governanceReward = fraction(amountWithoutFee, commissionGovernance, commissionScaleDelimiter)
297407 if ((minAmountToReceive > amountWithFee))
298408 then throw(((("Calculated amount to receive " + toString(amountWithFee)) + " is less than specified minimum ") + toString(minAmountToReceive)))
299409 else $Tuple3(amountWithoutFee, amountWithFee, governanceReward)
300410 }
301411
302412 if (!(isActive))
303413 then throw("DApp is inactive at this moment")
304- else if ((0 >= minAmountToReceive))
305- then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
306- else if ((size(i.payments) != 1))
307- then throw("One attached payment expected")
308- else if (!(hasEnoughBalance))
309- then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB) + ". State: ") + toString(accountBalance(assetIdA))) + " ") + assetNameA) + ", ") + toString(accountBalance(assetIdB))) + " ") + assetNameB)))
310- else if ((pmtAssetId == assetIdA))
311- then {
312- let assetIdSend = assetIdB
313- let $t01371713808 = calculateFees(balanceA, balanceB)
314- let amountWithoutFee = $t01371713808._1
315- let amountWithFee = $t01371713808._2
316- let governanceReward = $t01371713808._3
317- let newBalanceA = (balanceA + pmtAmount)
318- let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
319- if (if ((stakedAmountA >= newBalanceA))
320- then true
321- else (stakedAmountB >= newBalanceB))
322- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
323- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
324- }
325- else if ((pmtAssetId == assetIdB))
414+ else if (if ((balanceA == 0))
415+ then true
416+ else (balanceB == 0))
417+ then throw("Can't exchange with zero balance")
418+ else if ((0 >= minAmountToReceive))
419+ then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
420+ else if ((size(i.payments) != 1))
421+ then throw("One attached payment expected")
422+ else if (!(hasEnoughBalance))
423+ then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
424+ else if ((pmtAssetId == assetIdA))
326425 then {
327- let assetIdSend = assetIdA
328- let $t01463114722 = calculateFees(balanceB, balanceA)
329- let amountWithoutFee = $t01463114722._1
330- let amountWithFee = $t01463114722._2
331- let governanceReward = $t01463114722._3
332- let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
333- let newBalanceB = (balanceB + pmtAmount)
334- if (if ((stakedAmountA >= newBalanceA))
426+ let assetIdSend = assetIdB
427+ let $t01907819169 = calculateFees(balanceA, balanceB)
428+ let amountWithoutFee = $t01907819169._1
429+ let amountWithFee = $t01907819169._2
430+ let governanceReward = $t01907819169._3
431+ let newBalanceA = (balanceA + pmtAmount)
432+ let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
433+ if (if (if ((assetIdA == EURN))
434+ then (stakedAmountEURN >= newBalanceA)
435+ else false)
335436 then true
336- else (stakedAmountB >= newBalanceB))
337- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
338- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
437+ else if ((assetIdB == EURN))
438+ then (stakedAmountEURN >= newBalanceB)
439+ else false)
440+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
441+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
339442 }
340- else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
443+ else if ((pmtAssetId == assetIdB))
444+ then {
445+ let assetIdSend = assetIdA
446+ let $t02003820129 = calculateFees(balanceB, balanceA)
447+ let amountWithoutFee = $t02003820129._1
448+ let amountWithFee = $t02003820129._2
449+ let governanceReward = $t02003820129._3
450+ let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
451+ let newBalanceB = (balanceB + pmtAmount)
452+ if (if (if ((assetIdA == EURN))
453+ then (stakedAmountEURN >= newBalanceA)
454+ else false)
455+ then true
456+ else if ((assetIdB == EURN))
457+ then (stakedAmountEURN >= newBalanceB)
458+ else false)
459+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
460+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
461+ }
462+ else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
341463 }
342464
343465
344466
345467 @Callable(i)
346468 func shutdown () = if (!(isActive))
347469 then throw(("DApp is already suspended. Cause: " + valueOrElse(getString(this, keyCause), "the cause wasn't specified")))
348470 else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStartStop], i.callerPublicKey)))
349471 then throw("Only admin can call this function")
350472 else suspend("Paused by admin")
351473
352474
353475
354476 @Callable(i)
355477 func activate () = if (isActive)
356478 then throw("DApp is already active")
357479 else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStartStop], i.callerPublicKey)))
358480 then throw("Only admin can call this function")
359481 else [BooleanEntry(keyActive, true), DeleteEntry(keyCause)]
360482
361483
362484
363485 @Callable(i)
364486 func takeIntoAccountExtraFunds (amountLeave) = {
365- let uncountableAmountEnrollA = (accountBalanceWithStakedA - balanceA)
366- let uncountableAmountEnrollB = (accountBalanceWithStakedB - balanceB)
367- let amountEnrollA = (uncountableAmountEnrollA - (if ((assetIdA == EURN))
487+ let uncountableAmountEnrollAssetA = (accountBalanceWithStakedA - balanceA)
488+ let uncountableAmountEnrollAssetB = (accountBalanceWithStakedB - balanceB)
489+ let amountEnrollA = (uncountableAmountEnrollAssetA - (if ((assetIdA == unit))
368490 then amountLeave
369491 else 0))
370- let amountEnrollB = (uncountableAmountEnrollB - (if ((assetIdB == EURN))
492+ let amountEnrollB = (uncountableAmountEnrollAssetB - (if ((assetIdB == unit))
371493 then amountLeave
372494 else 0))
373495 if (!(isActive))
374496 then throw("DApp is inactive at this moment")
375497 else if ((i.caller != this))
376498 then throw("Only the DApp itself can call this function")
377499 else if ((0 > amountLeave))
378500 then throw(("Argument 'amountLeave' cannot be negative. Actual: " + toString(amountLeave)))
379- else if (if ((0 > uncountableAmountEnrollA))
501+ else if (if ((0 > uncountableAmountEnrollAssetA))
380502 then true
381- else (0 > uncountableAmountEnrollB))
503+ else (0 > uncountableAmountEnrollAssetB))
382504 then suspend("Enroll amount negative")
383505 else if (if ((0 > amountEnrollA))
384506 then true
385507 else (0 > amountEnrollB))
386508 then throw("Too large amountLeave")
387509 else [IntegerEntry(keyBalanceA, (balanceA + amountEnrollA)), IntegerEntry(keyBalanceB, (balanceB + amountEnrollB)), IntegerEntry(("last_income_" + strAssetIdA), amountEnrollA), IntegerEntry(("last_income_" + strAssetIdB), amountEnrollB)]
388510 }
389511
390512
391513 @Verifier(tx)
392514 func verify () = match tx {
393515 case invoke: InvokeScriptTransaction =>
394516 let callTakeIntoAccount = if ((invoke.dApp == this))
395517 then (invoke.function == "takeIntoAccountExtraFunds")
396518 else false
397- let callStakingUsdnOrNsbt = if ((invoke.dApp == stakingAddressUsdnAndNsbt))
398- then if (if (containsElement(["lockNeutrino", "lockNsbt"], invoke.function))
519+ let callStaking = if ((invoke.dApp == stakingAddress))
520+ then if (if (if ((invoke.function == "startStaking"))
399521 then (size(invoke.payments) == 1)
400522 else false)
401- then true
402- else if (containsElement(["unlockNeutrino", "unlockNsbt"], invoke.function))
403- then (size(invoke.payments) == 0)
404- else false
405- else false
406- let callStakingDeFo = if ((invoke.dApp == stakingAddressDeFo))
407- then if (if ((invoke.function == "startStaking"))
408- then (size(invoke.payments) == 1)
523+ then (invoke.payments[0].assetId == EURN)
409524 else false)
410525 then true
411526 else if ((invoke.function == "stopStaking"))
412527 then (size(invoke.payments) == 0)
413528 else false
414529 else false
530+ let exchangeToWaves = if (if (if ((invoke.dApp == EURNToWavesExchanger))
531+ then (invoke.function == "exchange")
532+ else false)
533+ then (assetIdA == EURN)
534+ else false)
535+ then true
536+ else if ((assetIdB == EURN))
537+ then (invoke.payments[0].assetId == EURN)
538+ else false
415539 let signedByAdmin = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1))
416540 then true
417541 else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey2))
418542 then true
419543 else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey3))
420544 then true
421545 else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKeyStaking)
422546 if (if (if (callTakeIntoAccount)
423547 then true
424- else callStakingUsdnOrNsbt)
548+ else callStaking)
425549 then true
426- else callStakingDeFo)
550+ else exchangeToWaves)
427551 then signedByAdmin
428552 else false
429553 case _ =>
430554 let adminPubKey1Signed = if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1))
431555 then 1
432556 else 0
433557 let adminPubKey2Signed = if (sigVerify(tx.bodyBytes, tx.proofs[1], adminPubKey2))
434558 then 1
435559 else 0
436560 let adminPubKey3Signed = if (sigVerify(tx.bodyBytes, tx.proofs[2], adminPubKey3))
437561 then 1
438562 else 0
439563 (((adminPubKey1Signed + adminPubKey2Signed) + adminPubKey3Signed) >= 2)
440564 }
441565

github/deemru/w8io/786bc32 
95.88 ms