tx · 5yztQop6MHd879EdnPTmsMAy4dSVWYuuAiroMT8kVFYd 3N2rcyriCEdhEj4exvjYdSW3afoRPa54PJL: -0.01400000 Waves 2021.10.07 13:39 [1735703] smart account 3N2rcyriCEdhEj4exvjYdSW3afoRPa54PJL > SELF 0.00000000 Waves
{ "type": 13, "id": "5yztQop6MHd879EdnPTmsMAy4dSVWYuuAiroMT8kVFYd", "fee": 1400000, "feeAssetId": null, "timestamp": 1633603235857, "version": 1, "sender": "3N2rcyriCEdhEj4exvjYdSW3afoRPa54PJL", "senderPublicKey": "HcYUcARjSQ9s6sNtiBY85skwJEowirwR4zUZmdVFtQq3", "proofs": [ "5JNxQ6JpFG5wnL21JfBWcWJv2MASXa5Hx3ZysTdBBGQv8JNH7JhCb2Y87aNZebTneQ3pX7YAKxX3Gxcs2SZi5VMb" ], "script": "base64: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AAEFAAAAAWgJAARMAAAAAgkAAaQAAAABBQAAAAl0aW1lc3RhbXAFAAAAA25pbAUAAAADU0VQAQAAABBrZXlGYWN0b3J5Q29uZmlnAAAAAAIAAAARJXNfX2ZhY3RvcnlDb25maWcBAAAAKWtleU1hcHBpbmdQb29sQ29udHJhY3RBZGRyZXNzVG9Qb29sQXNzZXRzAAAAAQAAABNwb29sQ29udHJhY3RBZGRyZXNzCQABLAAAAAIJAAEsAAAAAgIAAAAIJXMlcyVzX18FAAAAE3Bvb2xDb250cmFjdEFkZHJlc3MCAAAAIF9fbWFwcGluZ3NfX3Bvb2xDb250cmFjdDJMcEFzc2V0AQAAAA1rZXlQb29sQ29uZmlnAAAAAgAAAAlpQW10QXNzZXQAAAALaVByaWNlQXNzZXQJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAIJWQlZCVzX18FAAAACWlBbXRBc3NldAIAAAACX18FAAAAC2lQcmljZUFzc2V0AgAAAAhfX2NvbmZpZwEAAAAfa2V5TWFwcGluZ3NCYXNlQXNzZXQyaW50ZXJuYWxJZAAAAAEAAAAMYmFzZUFzc2V0U3RyCQABLAAAAAICAAAAKCVzJXMlc19fbWFwcGluZ3NfX2Jhc2VBc3NldDJpbnRlcm5hbElkX18FAAAADGJhc2VBc3NldFN0cgEAAAATa2V5QWxsUG9vbHNTaHV0ZG93bgAAAAACAAAADCVzX19zaHV0ZG93bgEAAAANa2V5UG9vbFdlaWdodAAAAAEAAAAPY29udHJhY3RBZGRyZXNzCQABLAAAAAICAAAAEiVzJXNfX3Bvb2xXZWlnaHRfXwUAAAAPY29udHJhY3RBZGRyZXNzAQAAAA9nZXRTdHJpbmdPckZhaWwAAAACAAAAB2FkZHJlc3MAAAADa2V5CQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQdAAAAAgUAAAAHYWRkcmVzcwUAAAADa2V5CQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAACm1hbmRhdG9yeSAJAAQlAAAAAQUAAAAHYWRkcmVzcwIAAAABLgUAAAADa2V5AgAAAA8gaXMgbm90IGRlZmluZWQBAAAADGdldEludE9yRmFpbAAAAAIAAAAHYWRkcmVzcwAAAANrZXkJAQAAABN2YWx1ZU9yRXJyb3JNZXNzYWdlAAAAAgkABBoAAAACBQAAAAdhZGRyZXNzBQAAAANrZXkJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAKbWFuZGF0b3J5IAkABCUAAAABBQAAAAdhZGRyZXNzAgAAAAEuBQAAAANrZXkCAAAADyBpcyBub3QgZGVmaW5lZAAAAAAPZmFjdG9yeUNvbnRyYWN0CQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABCQEAAAAPZ2V0U3RyaW5nT3JGYWlsAAAAAgUAAAAEdGhpcwkBAAAAEmtleUZhY3RvcnlDb250cmFjdAAAAAABAAAAEGlzR2xvYmFsU2h1dGRvd24AAAAACQEAAAALdmFsdWVPckVsc2UAAAACCQAEGwAAAAIFAAAAD2ZhY3RvcnlDb250cmFjdAkBAAAAE2tleUFsbFBvb2xzU2h1dGRvd24AAAAABwEAAAANZ2V0UG9vbENvbmZpZwAAAAAEAAAACGFtdEFzc2V0CQEAAAAPZ2V0U3RyaW5nT3JGYWlsAAAAAgUAAAAEdGhpcwkBAAAAC2tleUFtdEFzc2V0AAAAAAQAAAAKcHJpY2VBc3NldAkBAAAAD2dldFN0cmluZ09yRmFpbAAAAAIFAAAABHRoaXMJAQAAAA1rZXlQcmljZUFzc2V0AAAAAAQAAAALaVByaWNlQXNzZXQJAQAAAAxnZXRJbnRPckZhaWwAAAACBQAAAA9mYWN0b3J5Q29udHJhY3QJAQAAAB9rZXlNYXBwaW5nc0Jhc2VBc3NldDJpbnRlcm5hbElkAAAAAQUAAAAKcHJpY2VBc3NldAQAAAAJaUFtdEFzc2V0CQEAAAAMZ2V0SW50T3JGYWlsAAAAAgUAAAAPZmFjdG9yeUNvbnRyYWN0CQEAAAAfa2V5TWFwcGluZ3NCYXNlQXNzZXQyaW50ZXJuYWxJZAAAAAEFAAAACGFtdEFzc2V0CQAEtQAAAAIJAQAAAA9nZXRTdHJpbmdPckZhaWwAAAACBQAAAA9mYWN0b3J5Q29udHJhY3QJAQAAAA1rZXlQb29sQ29uZmlnAAAAAgkAAaQAAAABBQAAAAlpQW10QXNzZXQJAAGkAAAAAQUAAAALaVByaWNlQXNzZXQFAAAAA1NFUAEAAAAQZ2V0RmFjdG9yeUNvbmZpZwAAAAAJAAS1AAAAAgkBAAAAD2dldFN0cmluZ09yRmFpbAAAAAIFAAAAD2ZhY3RvcnlDb250cmFjdAkBAAAAEGtleUZhY3RvcnlDb25maWcAAAAABQAAAANTRVABAAAAEWRhdGFQdXRBY3Rpb25JbmZvAAAACgAAAA1pbkFtdEFzc2V0QW10AAAAD2luUHJpY2VBc3NldEFtdAAAAAhvdXRMcEFtdAAAAAVwcmljZQAAAB1zbGlwcGFnZVRvbGVyYW5jZVBhc3NlZEJ5VXNlcgAAABVzbGlwcGFnZVRvbGVyYW5jZVJlYWwAAAAIdHhIZWlnaHQAAAALdHhUaW1lc3RhbXAAAAASc2xpcGFnZUFtdEFzc2V0QW10AAAAFHNsaXBhZ2VQcmljZUFzc2V0QW10CQAEuQAAAAIJAARMAAAAAgIAAAASJWQlZCVkJWQlZCVkJWQlZCVkCQAETAAAAAIJAAGkAAAAAQUAAAANaW5BbXRBc3NldEFtdAkABEwAAAACCQABpAAAAAEFAAAAD2luUHJpY2VBc3NldEFtdAkABEwAAAACCQABpAAAAAEFAAAACG91dExwQW10CQAETAAAAAIJAAGkAAAAAQUAAAAFcHJpY2UJAARMAAAAAgkAAaQAAAABBQAAAB1zbGlwcGFnZVRvbGVyYW5jZVBhc3NlZEJ5VXNlcgkABEwAAAACCQABpAAAAAEFAAAAFXNsaXBwYWdlVG9sZXJhbmNlUmVhbAkABEwAAAACCQABpAAAAAEFAAAACHR4SGVpZ2h0CQAETAAAAAIJAAGkAAAAAQUAAAALdHhUaW1lc3RhbXAJAARMAAAAAgkAAaQAAAABBQAAABJzbGlwYWdlQW10QXNzZXRBbXQJAARMAAAAAgkAAaQAAAABBQAAABRzbGlwYWdlUHJpY2VBc3NldEFtdAUAAAADbmlsBQAAAANTRVABAAAAEWRhdGFHZXRBY3Rpb25JbmZvAAAABgAAAA5vdXRBbXRBc3NldEFtdAAAABBvdXRQcmljZUFzc2V0QW10AAAAB2luTHBBbXQAAAAFcHJpY2UAAAAIdHhIZWlnaHQAAAALdHhUaW1lc3RhbXAJAAS5AAAAAgkABEwAAAACAgAAAAwlZCVkJWQlZCVkJWQJAARMAAAAAgkAAaQAAAABBQAAAA5vdXRBbXRBc3NldEFtdAkABEwAAAACCQABpAAAAAEFAAAAEG91dFByaWNlQXNzZXRBbXQJAARMAAAAAgkAAaQAAAABBQAAAAdpbkxwQW10CQAETAAAAAIJAAGkAAAAAQUAAAAFcHJpY2UJAARMAAAAAgkAAaQAAAABBQAAAAh0eEhlaWdodAkABEwAAAACCQABpAAAAAEFAAAAC3R4VGltZXN0YW1wBQAAAANuaWwFAAAAA1NFUAEAAAANZ2V0QWNjQmFsYW5jZQAAAAEAAAAHYXNzZXRJZAMJAAAAAAAAAgUAAAAHYXNzZXRJZAIAAAAFV0FWRVMICQAD7wAAAAEFAAAABHRoaXMAAAAJYXZhaWxhYmxlCQAD8AAAAAIFAAAABHRoaXMJAAJZAAAAAQUAAAAHYXNzZXRJZAEAAAAPY2FsY1ByaWNlQmlnSW50AAAAAwAAABNwcmljZUFzc2V0TG9ja2VkQW10AAAAEmFtdEFzc2V0bExvY2tlZEFtdAAAAAtyZXN1bHRTY2FsZQkAATwAAAADBQAAABNwcmljZUFzc2V0TG9ja2VkQW10BQAAAAtyZXN1bHRTY2FsZQUAAAASYW10QXNzZXRsTG9ja2VkQW10AQAAABBwcml2YXRlQ2FsY1ByaWNlAAAABAAAAAthbXRBc3NldERjbQAAAA1wcmljZUFzc2V0RGNtAAAAC2FtdEFzc2V0QW10AAAADXByaWNlQXNzZXRBbXQEAAAADmFtdEFzc2V0QW10WDE4CQEAAAAFdG9YMTgAAAACBQAAAAthbXRBc3NldEFtdAUAAAALYW10QXNzZXREY20EAAAAEHByaWNlQXNzZXRBbXRYMTgJAQAAAAV0b1gxOAAAAAIFAAAADXByaWNlQXNzZXRBbXQFAAAADXByaWNlQXNzZXREY20JAAGgAAAAAQkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMFAAAAEHByaWNlQXNzZXRBbXRYMTgFAAAADmFtdEFzc2V0QW10WDE4BQAAAAxzY2FsZThCaWdJbnQBAAAACmNhbGNQcmljZXMAAAADAAAAC2FtdEFzc2V0QW10AAAADXByaWNlQXNzZXRBbXQAAAAFbHBBbXQEAAAAA2NmZwkBAAAADWdldFBvb2xDb25maWcAAAAABAAAAAthbXRBc3NldERjbQkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAAA2NmZwUAAAAOaWR4QW10QXNzZXREY20EAAAADXByaWNlQXNzZXREY20JAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAANjZmcFAAAAEGlkeFByaWNlQXNzZXREY20EAAAACXBvb2xQcmljZQkBAAAAEHByaXZhdGVDYWxjUHJpY2UAAAAEBQAAAAthbXRBc3NldERjbQUAAAANcHJpY2VBc3NldERjbQUAAAALYW10QXNzZXRBbXQFAAAADXByaWNlQXNzZXRBbXQEAAAADmFtdEFzc2V0QW10WDE4CQEAAAAFdG9YMTgAAAACBQAAAAthbXRBc3NldEFtdAUAAAALYW10QXNzZXREY20EAAAAEHByaWNlQXNzZXRBbXRYMTgJAQAAAAV0b1gxOAAAAAIFAAAADXByaWNlQXNzZXRBbXQFAAAADXByaWNlQXNzZXREY20EAAAACGxwQW10WDE4CQEAAAAFdG9YMTgAAAACBQAAAAVscEFtdAUAAAAGc2NhbGU4BAAAABFscFByaWNlSW5BbXRBc3NldAkAAaAAAAABCQEAAAAPY2FsY1ByaWNlQmlnSW50AAAAAwUAAAAOYW10QXNzZXRBbXRYMTgFAAAACGxwQW10WDE4BQAAAAxzY2FsZThCaWdJbnQEAAAAE2xwUHJpY2VJblByaWNlQXNzZXQJAAGgAAAAAQkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMFAAAAEHByaWNlQXNzZXRBbXRYMTgFAAAACGxwQW10WDE4BQAAAAxzY2FsZThCaWdJbnQJAARMAAAAAgUAAAAJcG9vbFByaWNlCQAETAAAAAIFAAAAEWxwUHJpY2VJbkFtdEFzc2V0CQAETAAAAAIFAAAAE2xwUHJpY2VJblByaWNlQXNzZXQFAAAAA25pbAEAAAAUZXN0aW1hdGVHZXRPcGVyYXRpb24AAAADAAAAEHBheW1lbnRMcEFzc2V0SWQAAAAMcGF5bWVudExwQW10AAAAC3VzZXJBZGRyZXNzBAAAAA5wb29sQ29uZmlnTGlzdAkBAAAADWdldFBvb2xDb25maWcAAAAABAAAAAlscEFzc2V0SWQJAAGRAAAAAgUAAAAOcG9vbENvbmZpZ0xpc3QFAAAAEGlkeFBvb2xMUEFzc2V0SWQEAAAACmFtdEFzc2V0SWQJAAGRAAAAAgUAAAAOcG9vbENvbmZpZ0xpc3QFAAAADWlkeEFtdEFzc2V0SWQEAAAADHByaWNlQXNzZXRJZAkAAZEAAAACBQAAAA5wb29sQ29uZmlnTGlzdAUAAAAPaWR4UHJpY2VBc3NldElkBAAAAAtpQW10QXNzZXRJZAkAAZEAAAACBQAAAA5wb29sQ29uZmlnTGlzdAUAAAAOaWR4SUFtdEFzc2V0SWQEAAAADWlQcmljZUFzc2V0SWQJAAGRAAAAAgUAAAAOcG9vbENvbmZpZ0xpc3QFAAAAEGlkeElQcmljZUFzc2V0SWQEAAAAC2FtdEFzc2V0RGNtCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAOcG9vbENvbmZpZ0xpc3QFAAAADmlkeEFtdEFzc2V0RGNtBAAAAA1wcmljZUFzc2V0RGNtCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAOcG9vbENvbmZpZ0xpc3QFAAAAEGlkeFByaWNlQXNzZXREY20EAAAACnBvb2xTdGF0dXMJAAGRAAAAAgUAAAAOcG9vbENvbmZpZ0xpc3QFAAAADWlkeFBvb2xTdGF0dXMEAAAADXBvb2xMUEJhbGFuY2UICQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAPsAAAAAQkAAlkAAAABBQAAAAlscEFzc2V0SWQJAAEsAAAAAgkAASwAAAACAgAAAAZBc3NldCAFAAAACWxwQXNzZXRJZAIAAAAOIGRvZXNuJ3QgZXhpc3QAAAAIcXVhbnRpdHkDCQEAAAACIT0AAAACBQAAAAlscEFzc2V0SWQFAAAAEHBheW1lbnRMcEFzc2V0SWQJAAACAAAAAQIAAAAVSW52YWxpZCBhc3NldCBwYXNzZWQuBAAAAA9hbXRBc3NldEJhbGFuY2UJAQAAAA1nZXRBY2NCYWxhbmNlAAAAAQUAAAAKYW10QXNzZXRJZAQAAAASYW10QXNzZXRCYWxhbmNlWDE4CQEAAAAFdG9YMTgAAAACBQAAAA9hbXRBc3NldEJhbGFuY2UFAAAAC2FtdEFzc2V0RGNtBAAAABFwcmljZUFzc2V0QmFsYW5jZQkBAAAADWdldEFjY0JhbGFuY2UAAAABBQAAAAxwcmljZUFzc2V0SWQEAAAAFHByaWNlQXNzZXRCYWxhbmNlWDE4CQEAAAAFdG9YMTgAAAACBQAAABFwcmljZUFzc2V0QmFsYW5jZQUAAAANcHJpY2VBc3NldERjbQQAAAAIY3VyUHJpY2UJAAGgAAAAAQkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMFAAAAFHByaWNlQXNzZXRCYWxhbmNlWDE4BQAAABJhbXRBc3NldEJhbGFuY2VYMTgFAAAADHNjYWxlOEJpZ0ludAQAAAAPcGF5bWVudExwQW10WDE4CQEAAAAFdG9YMTgAAAACBQAAAAxwYXltZW50THBBbXQFAAAABnNjYWxlOAQAAAAQcG9vbExQQmFsYW5jZVgxOAkBAAAABXRvWDE4AAAAAgUAAAANcG9vbExQQmFsYW5jZQUAAAAGc2NhbGU4BAAAAA5vdXRBbXRBc3NldEFtdAkAATwAAAADBQAAABJhbXRBc3NldEJhbGFuY2VYMTgFAAAAD3BheW1lbnRMcEFtdFgxOAUAAAAQcG9vbExQQmFsYW5jZVgxOAQAAAAQb3V0UHJpY2VBc3NldEFtdAkAATwAAAADBQAAABRwcmljZUFzc2V0QmFsYW5jZVgxOAUAAAAPcGF5bWVudExwQW10WDE4BQAAABBwb29sTFBCYWxhbmNlWDE4BAAAABNvdXRBbXRBc3NldEFtdEZpbmFsCQEAAAAHZnJvbVgxOAAAAAIFAAAADm91dEFtdEFzc2V0QW10BQAAAAthbXRBc3NldERjbQQAAAAVb3V0UHJpY2VBc3NldEFtdEZpbmFsCQEAAAAHZnJvbVgxOAAAAAIFAAAAEG91dFByaWNlQXNzZXRBbXQFAAAADXByaWNlQXNzZXREY20JAAUdAAAACwUAAAATb3V0QW10QXNzZXRBbXRGaW5hbAUAAAAVb3V0UHJpY2VBc3NldEFtdEZpbmFsBQAAAAtpQW10QXNzZXRJZAUAAAANaVByaWNlQXNzZXRJZAUAAAAKYW10QXNzZXRJZAUAAAAMcHJpY2VBc3NldElkBQAAAA9hbXRBc3NldEJhbGFuY2UFAAAAEXByaWNlQXNzZXRCYWxhbmNlBQAAAA1wb29sTFBCYWxhbmNlBQAAAAhjdXJQcmljZQUAAAAKcG9vbFN0YXR1cwEAAAAUZXN0aW1hdGVQdXRPcGVyYXRpb24AAAAHAAAAEXNsaXBwYWdlVG9sZXJhbmNlAAAADWluQW10QXNzZXRBbXQAAAAMaW5BbXRBc3NldElkAAAAD2luUHJpY2VBc3NldEFtdAAAAA5pblByaWNlQXNzZXRJZAAAAAt1c2VyQWRkcmVzcwAAAAppc0V2YWx1YXRlBAAAAANjZmcJAQAAAA1nZXRQb29sQ29uZmlnAAAAAAQAAAAJbHBBc3NldElkCQACWQAAAAEJAAGRAAAAAgUAAAADY2ZnBQAAABBpZHhQb29sTFBBc3NldElkBAAAAAphbXRBc3NldElkCQABkQAAAAIFAAAAA2NmZwUAAAANaWR4QW10QXNzZXRJZAQAAAAMcHJpY2VBc3NldElkCQABkQAAAAIFAAAAA2NmZwUAAAAPaWR4UHJpY2VBc3NldElkBAAAAAtpQW10QXNzZXRJZAkAAZEAAAACBQAAAANjZmcFAAAADmlkeElBbXRBc3NldElkBAAAAA1pUHJpY2VBc3NldElkCQABkQAAAAIFAAAAA2NmZwUAAAAQaWR4SVByaWNlQXNzZXRJZAQAAAALYW10QXNzZXREY20JAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAANjZmcFAAAADmlkeEFtdEFzc2V0RGNtBAAAAA1wcmljZUFzc2V0RGNtCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAADY2ZnBQAAABBpZHhQcmljZUFzc2V0RGNtBAAAAApwb29sU3RhdHVzCQABkQAAAAIFAAAAA2NmZwUAAAANaWR4UG9vbFN0YXR1cwQAAAANcG9vbExQQmFsYW5jZQgJAQAAABN2YWx1ZU9yRXJyb3JNZXNzYWdlAAAAAgkAA+wAAAABBQAAAAlscEFzc2V0SWQJAAEsAAAAAgkAASwAAAACAgAAAAZBc3NldCAJAAJYAAAAAQUAAAAJbHBBc3NldElkAgAAAA4gZG9lc24ndCBleGlzdAAAAAhxdWFudGl0eQMDCQEAAAACIT0AAAACBQAAAAphbXRBc3NldElkBQAAAAxpbkFtdEFzc2V0SWQGCQEAAAACIT0AAAACBQAAAAxwcmljZUFzc2V0SWQFAAAADmluUHJpY2VBc3NldElkCQAAAgAAAAECAAAAIkludmFsaWQgYW10IG9yIHByaWNlIGFzc2V0IHBhc3NlZC4EAAAAEmFjY0FtdEFzc2V0QmFsYW5jZQMFAAAACmlzRXZhbHVhdGUJAQAAAA1nZXRBY2NCYWxhbmNlAAAAAQUAAAAKYW10QXNzZXRJZAkAAGUAAAACCQEAAAANZ2V0QWNjQmFsYW5jZQAAAAEFAAAACmFtdEFzc2V0SWQFAAAADWluQW10QXNzZXRBbXQEAAAAFGFjY1ByaWNlQXNzZXRCYWxhbmNlAwUAAAAKaXNFdmFsdWF0ZQkBAAAADWdldEFjY0JhbGFuY2UAAAABBQAAAAxwcmljZUFzc2V0SWQJAABlAAAAAgkBAAAADWdldEFjY0JhbGFuY2UAAAABBQAAAAxwcmljZUFzc2V0SWQFAAAAD2luUHJpY2VBc3NldEFtdAQAAAAQaW5BbXRBc3NldEFtdFgxOAkBAAAABXRvWDE4AAAAAgUAAAANaW5BbXRBc3NldEFtdAUAAAALYW10QXNzZXREY20EAAAAEmluUHJpY2VBc3NldEFtdFgxOAkBAAAABXRvWDE4AAAAAgUAAAAPaW5QcmljZUFzc2V0QW10BQAAAA1wcmljZUFzc2V0RGNtBAAAAAl1c2VyUHJpY2UJAAGgAAAAAQkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMFAAAAEmluUHJpY2VBc3NldEFtdFgxOAUAAAAQaW5BbXRBc3NldEFtdFgxOAUAAAAMc2NhbGU4QmlnSW50BAAAABVhY2NBbXRBc3NldEJhbGFuY2VYMTgJAQAAAAV0b1gxOAAAAAIFAAAAEmFjY0FtdEFzc2V0QmFsYW5jZQUAAAALYW10QXNzZXREY20EAAAAF2FjY1ByaWNlQXNzZXRCYWxhbmNlWDE4CQEAAAAFdG9YMTgAAAACBQAAABRhY2NQcmljZUFzc2V0QmFsYW5jZQUAAAANcHJpY2VBc3NldERjbQQAAAAIY3VyUHJpY2UDCQAAAAAAAAIFAAAADXBvb2xMUEJhbGFuY2UAAAAAAAAAAAAAAAAAAAAAAAAJAAGgAAAAAQkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMFAAAAF2FjY1ByaWNlQXNzZXRCYWxhbmNlWDE4BQAAABVhY2NBbXRBc3NldEJhbGFuY2VYMTgFAAAADHNjYWxlOEJpZ0ludAQAAAAIc2xpcHBhZ2UDCQAAAAAAAAIFAAAACGN1clByaWNlAAAAAAAAAAAAAAAAAAAAAAAAAwkAAGYAAAACBQAAAAhjdXJQcmljZQUAAAAJdXNlclByaWNlCQAAawAAAAMJAABlAAAAAgUAAAAIY3VyUHJpY2UFAAAACXVzZXJQcmljZQUAAAAGc2NhbGU4BQAAAAhjdXJQcmljZQkAAGsAAAADCQAAZQAAAAIFAAAACXVzZXJQcmljZQUAAAAIY3VyUHJpY2UFAAAABnNjYWxlOAUAAAAIY3VyUHJpY2UDAwkBAAAAAiE9AAAAAgUAAAAIY3VyUHJpY2UAAAAAAAAAAAAJAABmAAAAAgUAAAAIc2xpcHBhZ2UFAAAAEXNsaXBwYWdlVG9sZXJhbmNlBwkAAAIAAAABCQABLAAAAAIJAAEsAAAAAgkAASwAAAACAgAAAA9QcmljZSBzbGlwcGFnZSAJAAGkAAAAAQUAAAAIc2xpcHBhZ2UCAAAAHiBleGNlZWRlZCB0aGUgcGFzc2VkIGxpbWl0IG9mIAkAAaQAAAABBQAAABFzbGlwcGFnZVRvbGVyYW5jZQQAAAADcmVzAwkAAAAAAAACBQAAAA1wb29sTFBCYWxhbmNlAAAAAAAAAAAABAAAAAtscFJlc3VsdEJ1ZgkAAHYAAAAGCQABOQAAAAIFAAAAEGluQW10QXNzZXRBbXRYMTgFAAAAEmluUHJpY2VBc3NldEFtdFgxOAAAAAAAAAAAAAkAATYAAAABAAAAAAAAAAAFAAAAAAAAAAABAAAAAAAAAAAABQAAAARET1dOCQAFFgAAAAQJAQAAAAdmcm9tWDE4AAAAAgUAAAALbHBSZXN1bHRCdWYFAAAABnNjYWxlOAkBAAAAB2Zyb21YMTgAAAACBQAAABBpbkFtdEFzc2V0QW10WDE4BQAAAAthbXRBc3NldERjbQkBAAAAB2Zyb21YMTgAAAACBQAAABJpblByaWNlQXNzZXRBbXRYMTgFAAAADXByaWNlQXNzZXREY20JAAGgAAAAAQkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMJAAE3AAAAAgUAAAAXYWNjUHJpY2VBc3NldEJhbGFuY2VYMTgFAAAAEmluUHJpY2VBc3NldEFtdFgxOAkAATcAAAACBQAAABVhY2NBbXRBc3NldEJhbGFuY2VYMTgFAAAAEGluQW10QXNzZXRBbXRYMTgFAAAADHNjYWxlOEJpZ0ludAQAAAAQcG9vbExQQmFsYW5jZVgxOAkBAAAABXRvWDE4AAAAAgUAAAANcG9vbExQQmFsYW5jZQUAAAAGc2NhbGU4BAAAAAtjdXJQcmljZVgxOAkBAAAABXRvWDE4AAAAAgUAAAAIY3VyUHJpY2UFAAAABnNjYWxlOAQAAAAGcEVxQnlBCQABPAAAAAMFAAAAC2N1clByaWNlWDE4BQAAABBpbkFtdEFzc2V0QW10WDE4BQAAAAdzY2FsZTE4BAAAAA5yZXNBc3NldHNUb1B1dAMJAAE/AAAAAgUAAAAGcEVxQnlBBQAAABJpblByaWNlQXNzZXRBbXRYMTgJAAUUAAAAAgkAATwAAAADBQAAABJpblByaWNlQXNzZXRBbXRYMTgFAAAAB3NjYWxlMTgFAAAAC2N1clByaWNlWDE4BQAAABJpblByaWNlQXNzZXRBbXRYMTgJAAUUAAAAAgUAAAAQaW5BbXRBc3NldEFtdFgxOAUAAAAGcEVxQnlBBAAAAAhscFJlc3VsdAkBAAAAB2Zyb21YMTgAAAACCQABPAAAAAMFAAAAEHBvb2xMUEJhbGFuY2VYMTgIBQAAAA5yZXNBc3NldHNUb1B1dAAAAAJfMgUAAAAXYWNjUHJpY2VBc3NldEJhbGFuY2VYMTgFAAAABnNjYWxlOAkABRYAAAAEBQAAAAhscFJlc3VsdAkBAAAAB2Zyb21YMTgAAAACCAUAAAAOcmVzQXNzZXRzVG9QdXQAAAACXzEFAAAAC2FtdEFzc2V0RGNtCQEAAAAHZnJvbVgxOAAAAAIIBQAAAA5yZXNBc3NldHNUb1B1dAAAAAJfMgUAAAANcHJpY2VBc3NldERjbQkAAaAAAAABCQEAAAAPY2FsY1ByaWNlQmlnSW50AAAAAwkAATcAAAACBQAAABdhY2NQcmljZUFzc2V0QmFsYW5jZVgxOAgFAAAADnJlc0Fzc2V0c1RvUHV0AAAAAl8yCQABNwAAAAIFAAAAFWFjY0FtdEFzc2V0QmFsYW5jZVgxOAgFAAAADnJlc0Fzc2V0c1RvUHV0AAAAAl8xBQAAAAxzY2FsZThCaWdJbnQDCQAAZwAAAAIAAAAAAAAAAAAIBQAAAANyZXMAAAACXzEJAAACAAAAAQIAAAA2SW52YWxpZCBjYWxjdWxhdGlvbnMuIExQIGNhbGN1bGF0ZWQgaXMgbGVzcyB0aGFuIHplcm8uCQAFHgAAAAwIBQAAAANyZXMAAAACXzEIBQAAAANyZXMAAAACXzQFAAAAEmFjY0FtdEFzc2V0QmFsYW5jZQUAAAAUYWNjUHJpY2VBc3NldEJhbGFuY2UFAAAADXBvb2xMUEJhbGFuY2UFAAAAC2lBbXRBc3NldElkBQAAAA1pUHJpY2VBc3NldElkBQAAAAlscEFzc2V0SWQFAAAACHNsaXBwYWdlBQAAAApwb29sU3RhdHVzCAUAAAADcmVzAAAAAl8yCAUAAAADcmVzAAAAAl8zAQAAABt2YWxpZGF0ZU1hdGNoZXJPcmRlckFsbG93ZWQAAAABAAAABW9yZGVyBAAAAANjZmcJAQAAAA1nZXRQb29sQ29uZmlnAAAAAAQAAAAKYW10QXNzZXRJZAkAAZEAAAACBQAAAANjZmcFAAAADWlkeEFtdEFzc2V0SWQEAAAADHByaWNlQXNzZXRJZAkAAZEAAAACBQAAAANjZmcFAAAAD2lkeFByaWNlQXNzZXRJZAQAAAAKcG9vbFN0YXR1cwkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAAA2NmZwUAAAANaWR4UG9vbFN0YXR1cwQAAAALYW10QXNzZXREY20JAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAANjZmcFAAAADmlkeEFtdEFzc2V0RGNtBAAAAA1wcmljZUFzc2V0RGNtCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAADY2ZnBQAAABBpZHhQcmljZUFzc2V0RGNtBAAAABJhY2NBbXRBc3NldEJhbGFuY2UJAQAAAA1nZXRBY2NCYWxhbmNlAAAAAQUAAAAKYW10QXNzZXRJZAQAAAAUYWNjUHJpY2VBc3NldEJhbGFuY2UJAQAAAA1nZXRBY2NCYWxhbmNlAAAAAQUAAAAMcHJpY2VBc3NldElkBAAAAAhjdXJQcmljZQMJAAAAAAAAAggFAAAABW9yZGVyAAAACW9yZGVyVHlwZQUAAAADQnV5CQEAAAAQcHJpdmF0ZUNhbGNQcmljZQAAAAQFAAAAC2FtdEFzc2V0RGNtBQAAAA1wcmljZUFzc2V0RGNtCQAAZAAAAAIFAAAAEmFjY0FtdEFzc2V0QmFsYW5jZQgFAAAABW9yZGVyAAAABmFtb3VudAUAAAAUYWNjUHJpY2VBc3NldEJhbGFuY2UJAQAAABBwcml2YXRlQ2FsY1ByaWNlAAAABAUAAAALYW10QXNzZXREY20FAAAADXByaWNlQXNzZXREY20JAABlAAAAAgUAAAASYWNjQW10QXNzZXRCYWxhbmNlCAUAAAAFb3JkZXIAAAAGYW1vdW50BQAAABRhY2NQcmljZUFzc2V0QmFsYW5jZQMDAwkBAAAAEGlzR2xvYmFsU2h1dGRvd24AAAAABgkAAAAAAAACBQAAAApwb29sU3RhdHVzBQAAABNQb29sTWF0Y2hlckRpc2FibGVkBgkAAAAAAAACBQAAAApwb29sU3RhdHVzBQAAAAxQb29sU2h1dGRvd24JAAACAAAAAQIAAAAcRXhjaGFuZ2Ugb3BlcmF0aW9ucyBkaXNhYmxlZAQAAAANb3JkZXJBbXRBc3NldAgIBQAAAAVvcmRlcgAAAAlhc3NldFBhaXIAAAALYW1vdW50QXNzZXQEAAAAEG9yZGVyQW10QXNzZXRTdHIDCQAAAAAAAAIFAAAADW9yZGVyQW10QXNzZXQFAAAABHVuaXQCAAAABVdBVkVTCQACWAAAAAEJAQAAAAV2YWx1ZQAAAAEFAAAADW9yZGVyQW10QXNzZXQEAAAAD29yZGVyUHJpY2VBc3NldAgIBQAAAAVvcmRlcgAAAAlhc3NldFBhaXIAAAAKcHJpY2VBc3NldAQAAAASb3JkZXJQcmljZUFzc2V0U3RyAwkAAAAAAAACBQAAAA9vcmRlclByaWNlQXNzZXQFAAAABHVuaXQCAAAABVdBVkVTCQACWAAAAAEJAQAAAAV2YWx1ZQAAAAEFAAAAD29yZGVyUHJpY2VBc3NldAMDCQEAAAACIT0AAAACBQAAABBvcmRlckFtdEFzc2V0U3RyBQAAAAphbXRBc3NldElkBgkBAAAAAiE9AAAAAgUAAAASb3JkZXJQcmljZUFzc2V0U3RyBQAAAAxwcmljZUFzc2V0SWQJAAACAAAAAQIAAAATV3Jvbmcgb3JkZXIgYXNzZXRzLgQAAAAKb3JkZXJQcmljZQgFAAAABW9yZGVyAAAABXByaWNlBAAAAAhwcmljZURjbQkAAGsAAAADBQAAAAZzY2FsZTgFAAAADXByaWNlQXNzZXREY20FAAAAC2FtdEFzc2V0RGNtBAAAABBjYXN0ZWRPcmRlclByaWNlCQEAAAAHdG9TY2FsZQAAAAMFAAAACm9yZGVyUHJpY2UFAAAABnNjYWxlOAUAAAAIcHJpY2VEY20EAAAAEWlzT3JkZXJQcmljZVZhbGlkAwkAAAAAAAACCAUAAAAFb3JkZXIAAAAJb3JkZXJUeXBlBQAAAANCdXkJAABnAAAAAgUAAAAIY3VyUHJpY2UFAAAAEGNhc3RlZE9yZGVyUHJpY2UJAABnAAAAAgUAAAAQY2FzdGVkT3JkZXJQcmljZQUAAAAIY3VyUHJpY2UDCQEAAAABIQAAAAEFAAAAEWlzT3JkZXJQcmljZVZhbGlkCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAMk9yZGVyIHByaWNlIGxlYWRzIHRvIEsgZGVjcmVhc2UuIGNhc3RlZE9yZGVyUHJpY2U9CQABpAAAAAEFAAAAEGNhc3RlZE9yZGVyUHJpY2UCAAAACiBjdXJQcmljZT0JAAGkAAAAAQUAAAAIY3VyUHJpY2UGAQAAAAljb21tb25HZXQAAAABAAAAAWkEAAAAC3BtdEFtdEFzc2V0CQEAAAAFdmFsdWUAAAABCQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAABAAAAApwbXRBc3NldElkCQEAAAAFdmFsdWUAAAABCAUAAAALcG10QW10QXNzZXQAAAAHYXNzZXRJZAQAAAALcG10QXNzZXRBbXQIBQAAAAtwbXRBbXRBc3NldAAAAAZhbW91bnQEAAAAA3JlcwkBAAAAFGVzdGltYXRlR2V0T3BlcmF0aW9uAAAAAwkAAlgAAAABBQAAAApwbXRBc3NldElkBQAAAAtwbXRBc3NldEFtdAkABCUAAAABCAUAAAABaQAAAAZjYWxsZXIEAAAADm91dEFtdEFzc2V0QW10CAUAAAADcmVzAAAAAl8xBAAAABBvdXRQcmljZUFzc2V0QW10CAUAAAADcmVzAAAAAl8yBAAAAAtpQW10QXNzZXRJZAgFAAAAA3JlcwAAAAJfMwQAAAANaVByaWNlQXNzZXRJZAgFAAAA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aWNlQXNzZXQAAAAHYXNzZXRJZAQAAAAGZXN0UHV0CQEAAAAUZXN0aW1hdGVQdXRPcGVyYXRpb24AAAAHBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAANaW5BbXRBc3NldEFtdAkAAlgAAAABBQAAAAxpbkFtdEFzc2V0SWQFAAAAD2luUHJpY2VBc3NldEFtdAkAAlgAAAABBQAAAA5pblByaWNlQXNzZXRJZAkABCUAAAABCAUAAAABaQAAAAZjYWxsZXIHBAAAAAhvdXRMcEFtdAgFAAAABmVzdFB1dAAAAAJfMQQAAAAIY3VyUHJpY2UIBQAAAAZlc3RQdXQAAAACXzIEAAAAE3Bvb2xBbXRBc3NldEJhbGFuY2UIBQAAAAZlc3RQdXQAAAACXzMEAAAAFXBvb2xQcmljZUFzc2V0QmFsYW5jZQgFAAAABmVzdFB1dAAAAAJfNAQAAAANcG9vbExQQmFsYW5jZQgFAAAABmVzdFB1dAAAAAJfNQQAAAALaUFtdEFzc2V0SWQIBQAAAAZlc3RQdXQAAAACXzYEAAAADWlQcmljZUFzc2V0SWQIBQAAAAZlc3RQdXQAAAACXzcEAAAACWxwQXNzZXRJZAgFAAAABmVzdFB1dAAAAAJfOAQAAAAMc2xpcHBhZ2VDYWxjCAUAAAAGZXN0UHV0AAAAAl85BAAAAApwb29sU3RhdHVzCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEIBQAAAAZlc3RQdXQAAAADXzEwBAAAAA5yZWFsQW10QXNzZXRJbggFAAAABmVzdFB1dAAAAANfMTEEAAAAEHJlYWxQcmljZUFzc2V0SW4IBQAAAAZlc3RQdXQAAAADXzEyAwMDCQEAAAAQaXNHbG9iYWxTaHV0ZG93bgAAAAAGCQAAAAAAAAIFAAAACnBvb2xTdGF0dXMFAAAAD1Bvb2xQdXREaXNhYmxlZAYJAAAAAAAAAgUAAAAKcG9vbFN0YXR1cwUAAAAMUG9vbFNodXRkb3duCQAAAgAAAAEJAAEsAAAAAgIAAAAsUHV0IG9wZXJhdGlvbiBpcyBibG9ja2VkIGJ5IGFkbWluLiBTdGF0dXMgPSAJAAGkAAAAAQUAAAAKcG9vbFN0YXR1cwQAAAALbHBBbXRUb0VtaXQDCQEAAAABIQAAAAEFAAAABmVtaXRMcAAAAAAAAAAAAAUAAAAIb3V0THBBbXQEAAAADmRpZmZJbkFtdEFzc2V0CQAAZQAAAAIFAAAADWluQW10QXNzZXRBbXQFAAAADnJlYWxBbXRBc3NldEluBAAAABBkaWZmSW5QcmljZUFzc2V0CQAAZQAAAAIFAAAAD2luUHJpY2VBc3NldEFtdAUAAAAQcmVhbFByaWNlQXNzZXRJbgQAAAAOZGF0YVN0YXRlV3JpdGUJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAADGtleVByaWNlTGFzdAAAAAAJAAEsAAAAAgIAAAAEJXNfXwkAAaQAAAABBQAAAAhjdXJQcmljZQkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAAPa2V5UHJpY2VIaXN0b3J5AAAAAgUAAAAGaGVpZ2h0CAUAAAAJbGFzdEJsb2NrAAAACXRpbWVzdGFtcAkAASwAAAACAgAAAAQlc19fCQABpAAAAAEFAAAACGN1clByaWNlCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABJrZXlQdXRBY3Rpb25CeVVzZXIAAAACCQAEJQAAAAEIBQAAAAFpAAAABmNhbGxlcgkAAlgAAAABCAUAAAABaQAAAA10cmFuc2FjdGlvbklkCQEAAAARZGF0YVB1dEFjdGlvbkluZm8AAAAKBQAAAA5yZWFsQW10QXNzZXRJbgUAAAAQcmVhbFByaWNlQXNzZXRJbgUAAAALbHBBbXRUb0VtaXQFAAAACGN1clByaWNlBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAMc2xpcHBhZ2VDYWxjBQAAAAZoZWlnaHQIBQAAAAlsYXN0QmxvY2sAAAAJdGltZXN0YW1wBQAAAA5kaWZmSW5BbXRBc3NldAUAAAAQZGlmZkluUHJpY2VBc3NldAUAAAADbmlsCQAFGwAAAAkFAAAAC2xwQW10VG9FbWl0BQAAAAlscEFzc2V0SWQFAAAADmRhdGFTdGF0ZVdyaXRlBQAAAA1pbkFtdEFzc2V0QW10BQAAAA5yZWFsQW10QXNzZXRJbgUAAAAPaW5QcmljZUFzc2V0QW10BQAAABByZWFsUHJpY2VBc3NldEluCAUAAAALcG10QW10QXNzZXQAAAAHYXNzZXRJZAgFAAAADXBtdFByaWNlQXNzZXQAAAAHYXNzZXRJZAAAAAoAAAABaQEAAAALY29uc3RydWN0b3IAAAACAAAAD2ZhY3RvcnlDb250cmFjdAAAABBtYW5hZ2VyUHVibGljS2V5AwkBAAAAAiE9AAAAAggFAAAAAWkAAAAGY2FsbGVyBQAAAAR0aGlzCQAAAgAAAAECAAAAEnBlcm1pc3Npb25zIGRlbmllZAkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAASa2V5RmFjdG9yeUNvbnRyYWN0AAAAAAUAAAAPZmFjdG9yeUNvbnRyYWN0CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABNrZXlNYW5hZ2VyUHVibGljS2V5AAAAAAUAAAAQbWFuYWdlclB1YmxpY0tleQUAAAADbmlsAAAAAWkBAAAAA3B1dAAAAAIAAAARc2xpcHBhZ2VUb2xlcmFuY2UAAAAPc2hvdWxkQXV0b1N0YWtlBAAAAApmYWN0b3J5Q2ZnCQEAAAAQZ2V0RmFjdG9yeUNvbmZpZwAAAAAEAAAAD3N0YWtpbmdDb250cmFjdAkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEJgAAAAEJAAGRAAAAAgUAAAAKZmFjdG9yeUNmZwUAAAAZaWR4RmFjdG9yeVN0YWtpbmdDb250cmFjdAIAAAAhRXJyb3IuIEluY29ycmVjdCBzdGFraW5nIGFkZHJlc3MuBAAAABBzbGlwcGFnZUNvbnRyYWN0CQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQmAAAAAQkAAZEAAAACBQAAAApmYWN0b3J5Q2ZnBQAAABppZHhGYWN0b3J5U2xpcHBhZ2VDb250cmFjdAIAAAArRXJyb3IuIEluY29ycmVjdCBzbGlwcGFnZSBjb250cmFjdCBhZGRyZXNzLgQAAAAHYWN0aW9ucwkBAAAACWNvbW1vblB1dAAAAAMFAAAAAWkFAAAAEXNsaXBwYWdlVG9sZXJhbmNlBgQAAAAJYW10VG9FbWl0CAUAAAAHYWN0aW9ucwAAAAJfMQQAAAAJbHBBc3NldElkCAUAAAAHYWN0aW9ucwAAAAJfMgQAAAAFc3RhdGUIBQAAAAdhY3Rpb25zAAAAAl8zBAAAAAVpbkFtdAgFAAAAB2Fj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CQAETAAAAAIJAAGkAAAAAQUAAAAKcG9vbFdlaWdodAUAAAADbmlsBQAAAANTRVAAAAABaQEAAAAdZXZhbHVhdGVQdXRCeUFtdEFzc2V0UkVBRE9OTFkAAAADAAAAEXNsaXBwYWdlVG9sZXJhbmNlAAAADWluQW10QXNzZXRBbXQAAAALdXNlckFkZHJlc3MEAAAAA2NmZwkBAAAADWdldFBvb2xDb25maWcAAAAABAAAAAlscEFzc2V0SWQJAAJZAAAAAQkAAZEAAAACBQAAAANjZmcFAAAAEGlkeFBvb2xMUEFzc2V0SWQEAAAACmFtdEFzc2V0SWQJAAGRAAAAAgUAAAADY2ZnBQAAAA1pZHhBbXRBc3NldElkBAAAAAxwcmljZUFzc2V0SWQJAAGRAAAAAgUAAAADY2ZnBQAAAA9pZHhQcmljZUFzc2V0SWQEAAAAC2lBbXRBc3NldElkCQABkQAAAAIFAAAAA2NmZwUAAAAOaWR4SUFtdEFzc2V0SWQEAAAADWlQcmljZUFzc2V0SWQJAAGRAAAAAgUAAAADY2ZnBQAAABBpZHhJUHJpY2VBc3NldElkBAAAAAthbXRBc3NldERjbQkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAAA2NmZwUAAAAOaWR4QW10QXNzZXREY20EAAAADXByaWNlQXNzZXREY20JAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAANjZmcFAAAAEGlkeFByaWNlQXNzZXREY20EAAAACnBvb2xTdGF0dXMJAAGRAAAAAgUAAAADY2ZnBQAAAA1pZHhQb29sU3RhdHVzBAAAAA1wb29sTFBCYWxhbmNlCAkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAD7AAAAAEFAAAACWxwQXNzZXRJZAkAASwAAAACCQABLAAAAAICAAAABkFzc2V0IAkAAlgAAAABBQAAAAlscEFzc2V0SWQCAAAADiBkb2Vzbid0IGV4aXN0AAAACHF1YW50aXR5BAAAABJhY2NBbXRBc3NldEJhbGFuY2UJAQAAAA1nZXRBY2NCYWxhbmNlAAAAAQUAAAAKYW10QXNzZXRJZAQAAAAUYWNjUHJpY2VBc3NldEJhbGFuY2UJAQAAAA1nZXRBY2NCYWxhbmNlAAAAAQUAAAAMcHJpY2VBc3NldElkBAAAAA5hbXRBc3NldEFtdFgxOAkBAAAABXRvWDE4AAAAAgUAAAASYWNjQW10QXNzZXRCYWxhbmNlBQAAAAthbXRBc3NldERjbQQAAAAQcHJpY2VBc3NldEFtdFgxOAkBAAAABXRvWDE4AAAAAgUAAAAUYWNjUHJpY2VBc3NldEJhbGFuY2UFAAAADXByaWNlQXNzZXREY20EAAAAC2N1clByaWNlWDE4AwkAAAAAAAACBQAAAA1wb29sTFBCYWxhbmNlAAAAAAAAAAAACQABNgAAAAEAAAAAAAAAAAAJAQAAAA9jYWxjUHJpY2VCaWdJbnQAAAADBQAAABBwcmljZUFzc2V0QW10WDE4BQAAAA5hbXRBc3NldEFtdFgxOAUAAAAHc2NhbGUxOAQAAAAQaW5BbXRBc3NldEFtdFgxOAkBAAAABXRvWDE4AAAAAgUAAAANaW5BbXRBc3NldEFtdAUAAAALYW10QXNzZXREY20EAAAAEmluUHJpY2VBc3NldEFtdFgxOAkAATwAAAADBQAAABBpbkFtdEFzc2V0QW10WDE4BQAAAAtjdXJQcmljZVgxOAUAAAAHc2NhbGUxOAQAAAAPaW5QcmljZUFzc2V0QW10CQEAAAAHZnJvbVgxOAAAAAIFAAAAEmluUHJpY2VBc3NldEFtdFgxOAUAAAANcHJpY2VBc3NldERjbQQAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwkBAAAAFGVzdGltYXRlUHV0T3BlcmF0aW9uAAAABwUAAAARc2xpcHBhZ2VUb2xlcmFuY2UFAAAADWluQW10QXNzZXRBbXQFAAAACmFtdEFzc2V0SWQFAAAAD2luUHJpY2VBc3NldEFtdAUAAAAMcHJpY2VBc3NldElkBQAAAAt1c2VyQWRkcmVzcwYEAAAADG91dExwQW10Q2FsYwgFAAAAE2VzdGltYXRlZFB1dFJlc3VsdHMAAAACXzEEAAAADGN1clByaWNlQ2FsYwgFAAAAE2VzdGltYXRlZFB1dFJlc3VsdHMAAAACXzIEAAAAF3Bvb2xBbXRBc3NldEJhbGFuY2VDYWxjCAUAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwAAAAJfMwQAAAAZcG9vbFByaWNlQXNzZXRCYWxhbmNlQ0FsYwgFAAAAE2VzdGltYXRlZFB1dFJlc3VsdHMAAAACXzQEAAAAEXBvb2xMUEJhbGFuY2VDYWxjCAUAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwAAAAJfNQQAAAAOcG9vbFN0YXR1c0NhbGMJAQAAAA1wYXJzZUludFZhbHVlAAAAAQgFAAAAE2VzdGltYXRlZFB1dFJlc3VsdHMAAAADXzEwCQAFFAAAAAIFAAAAA25pbAkABLkAAAACCQAETAAAAAICAAAADCVkJWQlZCVkJWQlZAkABEwAAAACCQABpAAAAAEFAAAADG91dExwQW10Q2FsYwkABEwAAAACCQABpAAAAAEJAQAAAAdmcm9tWDE4AAAAAgUAAAALY3VyUHJpY2VYMTgFAAAABnNjYWxlOAkABEwAAAACCQABpAAAAAEFAAAAF3Bvb2xBbXRBc3NldEJhbGFuY2VDYWxjCQAETAAAAAIJAAGkAAAAAQUAAAAZcG9vbFByaWNlQXNzZXRCYWxhbmNlQ0FsYwkABEwAAAACCQABpAAAAAEFAAAAEXBvb2xMUEJhbGFuY2VDYWxjCQAETAAAAAIJAAGkAAAAAQUAAAAOcG9vbFN0YXR1c0NhbGMFAAAAA25pbAUAAAADU0VQAAAAAWkBAAAAH2V2YWx1YXRlUHV0QnlQcmljZUFzc2V0UkVBRE9OTFkAAAADAAAAEXNsaXBwYWdlVG9sZXJhbmNlAAAAD2luUHJpY2VBc3NldEFtdAAAAAt1c2VyQWRkcmVzcwQAAAADY2ZnCQEAAAANZ2V0UG9vbENvbmZpZwAAAAAEAAAACWxwQXNzZXRJZAkAAlkAAAABCQABkQAAAAIFAAAAA2NmZwUAAAAQaWR4UG9vbExQQXNzZXRJZAQAAAAKYW10QXNzZXRJZAkAAZEAAAACBQAAAANjZmcFAAAADWlkeEFtdEFzc2V0SWQEAAAADHByaWNlQXNzZXRJZAkAAZEAAAACBQAAAANjZmcFAAAAD2lkeFByaWNlQXNzZXRJZAQAAAALaUFtdEFzc2V0SWQJAAGRAAAAAgUAAAADY2ZnBQAAAA5pZHhJQW10QXNzZXRJZAQAAAANaVByaWNlQXNzZXRJZAkAAZEAAAACBQAAAANjZmcFAAAAEGlkeElQcmljZUFzc2V0SWQEAAAAC2FtdEFzc2V0RGNtCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAADY2ZnBQAAAA5pZHhBbXRBc3NldERjbQQAAAANcHJpY2VBc3NldERjbQkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAAA2NmZwUAAAAQaWR4UHJpY2VBc3NldERjbQQAAAAKcG9vbFN0YXR1cwkAAZEAAAACBQAAAANjZmcFAAAADWlkeFBvb2xTdGF0dXMEAAAADXBvb2xMUEJhbGFuY2UICQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAPsAAAAAQUAAAAJbHBBc3NldElkCQABLAAAAAIJAAEsAAAAAgIAAAAGQXNzZXQgCQACWAAAAAEFAAAACWxwQXNzZXRJZAIAAAAOIGRvZXNuJ3QgZXhpc3QAAAAIcXVhbnRpdHkEAAAAEmFjY0FtdEFzc2V0QmFsYW5jZQkBAAAADWdldEFjY0JhbGFuY2UAAAABBQAAAAphbXRBc3NldElkBAAAABRhY2NQcmljZUFzc2V0QmFsYW5jZQkBAAAADWdldEFjY0JhbGFuY2UAAAABBQAAAAxwcmljZUFzc2V0SWQEAAAADmFtdEFzc2V0QW10WDE4CQEAAAAFdG9YMTgAAAACBQAAABJhY2NBbXRBc3NldEJhbGFuY2UFAAAAC2FtdEFzc2V0RGNtBAAAABBwcmljZUFzc2V0QW10WDE4CQEAAAAFdG9YMTgAAAACBQAAABRhY2NQcmljZUFzc2V0QmFsYW5jZQUAAAANcHJpY2VBc3NldERjbQQAAAALY3VyUHJpY2VYMTgDCQAAAAAAAAIFAAAADXBvb2xMUEJhbGFuY2UAAAAAAAAAAAAJAAE2AAAAAQAAAAAAAAAAAAkBAAAAD2NhbGNQcmljZUJpZ0ludAAAAAMFAAAAEHByaWNlQXNzZXRBbXRYMTgFAAAADmFtdEFzc2V0QW10WDE4BQAAAAdzY2FsZTE4BAAAABJpblByaWNlQXNzZXRBbXRYMTgJAQAAAAV0b1gxOAAAAAIFAAAAD2luUHJpY2VBc3NldEFtdAUAAAANcHJpY2VBc3NldERjbQQAAAAQaW5BbXRBc3NldEFtdFgxOAkAATwAAAADBQAAABJpblByaWNlQXNzZXRBbXRYMTgFAAAAB3NjYWxlMTgFAAAAC2N1clByaWNlWDE4BAAAAA1pbkFtdEFzc2V0QW10CQEAAAAHZnJvbVgxOAAAAAIFAAAAEGluQW10QXNzZXRBbXRYMTgFAAAAC2FtdEFzc2V0RGNtBAAAABNlc3RpbWF0ZWRQdXRSZXN1bHRzCQEAAAAUZXN0aW1hdGVQdXRPcGVyYXRpb24AAAAHBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAANaW5BbXRBc3NldEFtdAUAAAAKYW10QXNzZXRJZAUAAAAPaW5QcmljZUFzc2V0QW10BQAAAAxwcmljZUFzc2V0SWQFAAAAC3VzZXJBZGRyZXNzBgQAAAAMb3V0THBBbXRDYWxjCAUAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwAAAAJfMQQAAAAMY3VyUHJpY2VDYWxjCAUAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwAAAAJfMgQAAAAXcG9vbEFtdEFzc2V0QmFsYW5jZUNhbGMIBQAAABNlc3RpbWF0ZWRQdXRSZXN1bHRzAAAAAl8zBAAAABlwb29sUHJpY2VBc3NldEJhbGFuY2VDQWxjCAUAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwAAAAJfNAQAAAARcG9vbExQQmFsYW5jZUNhbGMIBQAAABNlc3RpbWF0ZWRQdXRSZXN1bHRzAAAAAl81BAAAAA5wb29sU3RhdHVzQ2FsYwkBAAAADXBhcnNlSW50VmFsdWUAAAABCAUAAAATZXN0aW1hdGVkUHV0UmVzdWx0cwAAAANfMTAJAAUUAAAAAgUAAAADbmlsCQAEuQAAAAIJAARMAAAAAgIAAAAMJWQlZCVkJWQlZCVkCQAETAAAAAIJAAGkAAAAAQUAAAAMb3V0THBBbXRDYWxjCQAETAAAAAIJAAGkAAAAAQkBAAAAB2Zyb21YMTgAAAACBQAAAAtjdXJQcmljZVgxOAUAAAAGc2NhbGU4CQAETAAAAAIJAAGkAAAAAQUAAAAXcG9vbEFtdEFzc2V0QmFsYW5jZUNhbGMJAARMAAAAAgkAAaQAAAABBQAAABlwb29sUHJpY2VBc3NldEJhbGFuY2VDQWxjCQAETAAAAAIJAAGkAAAAAQUAAAARcG9vbExQQmFsYW5jZUNhbGMJAARMAAAAAgkAAaQAAAABBQAAAA5wb29sU3RhdHVzQ2FsYwUAAAADbmlsBQAAAANTRVAAAAABaQEAAAATZXZhbHVhdGVHZXRSRUFET05MWQAAAAMAAAAQcGF5bWVudExwQXNzZXRJZAAAAAxwYXltZW50THBBbXQAAAALdXNlckFkZHJlc3MEAAAAA3JlcwkBAAAAFGVzdGltYXRlR2V0T3BlcmF0aW9uAAAAAwUAAAAQcGF5bWVudExwQXNzZXRJZAUAAAAMcGF5bWVudExwQW10BQAAAAt1c2VyQWRkcmVzcwQAAAAOb3V0QW10QXNzZXRBbXQIBQAAAANyZXMAAAACXzEEAAAAEG91dFByaWNlQXNzZXRBbXQIBQAAAANyZXMAAAACXzIEAAAAE3Bvb2xBbXRBc3NldEJhbGFuY2UIBQAAAANyZXMAAAACXzcEAAAAFXBvb2xQcmljZUFzc2V0QmFsYW5jZQgFAAAAA3JlcwAAAAJfOAQAAAANcG9vbExQQmFsYW5jZQgFAAAAA3JlcwAAAAJfOQQAAAAIY3VyUHJpY2UIBQAAAANyZXMAAAADXzEwBAAAAApwb29sU3RhdHVzCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEIBQAAAANyZXMAAAADXzExCQAFFAAAAAIFAAAAA25pbAkABLkAAAACCQAETAAAAAICAAAADiVkJWQlZCVkJWQlZCVkCQAETAAAAAIJAAGkAAAAAQUAAAAOb3V0QW10QXNzZXRBbXQJAARMAAAAAgkAAaQAAAABBQAAABBvdXRQcmljZUFzc2V0QW10CQAETAAAAAIJAAGkAAAAAQUAAAATcG9vbEFtdEFzc2V0QmFsYW5jZQkABEwAAAACCQABpAAAAAEFAAAAFXBvb2xQcmljZUFzc2V0QmFsYW5jZQkABEwAAAACCQABpAAAAAEFAAAADXBvb2xMUEJhbGFuY2UJAARMAAAAAgkAAaQAAAABBQAAAAhjdXJQcmljZQkABEwAAAACCQABpAAAAAEFAAAACnBvb2xTdGF0dXMFAAAAA25pbAUAAAADU0VQAAAAAQAAAAJ0eAEAAAAGdmVyaWZ5AAAAAAQAAAAHJG1hdGNoMAUAAAACdHgDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABU9yZGVyBAAAAAVvcmRlcgUAAAAHJG1hdGNoMAMJAQAAABt2YWxpZGF0ZU1hdGNoZXJPcmRlckFsbG93ZWQAAAABBQAAAAVvcmRlcgkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAAIBQAAAAJ0eAAAAA9zZW5kZXJQdWJsaWNLZXkHBAAAAA1tYW5hZ2VyUHVibGljCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHQAAAAIFAAAABHRoaXMJAQAAABNrZXlNYW5hZ2VyUHVibGljS2V5AAAAAAUAAAAFRU1QVFkDCQAAAAAAAAIFAAAADW1hbmFnZXJQdWJsaWMFAAAABUVNUFRZCQAB9AAAAAMIBQAAAAJ0eAAAAAlib2R5Qnl0ZXMJAAGRAAAAAggFAAAAAnR4AAAABnByb29mcwAAAAAAAAAAAAgFAAAAAnR4AAAAD3NlbmRlclB1YmxpY0tleQkAAfQAAAADCAUAAAACdHgAAAAJYm9keUJ5dGVzCQABkQAAAAIIBQAAAAJ0eAAAAAZwcm9vZnMAAAAAAAAAAAAJAAJZAAAAAQUAAAANbWFuYWdlclB1YmxpY8sQnDc=", "chainId": 84, "height": 1735703, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: CeftPtPctkMfhkkuebUVxHcircebAfkRrZyWUfHJVLKQ Next: EPJr4sVVh1ruQsft5SxRuTzLhqb86JjNEUZ3uvmC3kuD Diff:
Old | New | Differences | |
---|---|---|---|
10 | 10 | let scale18 = toBigInt(1000000000000000000) | |
11 | 11 | ||
12 | 12 | let SEP = "__" | |
13 | + | ||
14 | + | let EMPTY = "" | |
13 | 15 | ||
14 | 16 | let PoolActive = 1 | |
15 | 17 | ||
25 | 27 | ||
26 | 28 | let idxPoolLPAssetId = 3 | |
27 | 29 | ||
28 | - | let | |
30 | + | let idxAmtAssetId = 4 | |
29 | 31 | ||
30 | 32 | let idxPriceAssetId = 5 | |
31 | 33 | ||
32 | - | let | |
34 | + | let idxAmtAssetDcm = 6 | |
33 | 35 | ||
34 | - | let | |
36 | + | let idxPriceAssetDcm = 7 | |
35 | 37 | ||
36 | - | let | |
38 | + | let idxIAmtAssetId = 8 | |
37 | 39 | ||
38 | - | let | |
40 | + | let idxIPriceAssetId = 9 | |
39 | 41 | ||
40 | - | let | |
42 | + | let idxLPAssetDcm = 10 | |
41 | 43 | ||
42 | - | let idxMatcherPublicKey = 11 | |
43 | - | ||
44 | - | let idxPoolAmountAssetAmt = 1 | |
44 | + | let idxPoolAmtAssetAmt = 1 | |
45 | 45 | ||
46 | 46 | let idxPoolPriceAssetAmt = 2 | |
47 | 47 | ||
49 | 49 | ||
50 | 50 | let idxFactoryStakingContract = 1 | |
51 | 51 | ||
52 | - | let idxFactoryBoostingConfig = 2 | |
52 | + | let idxFactorySlippageContract = 7 | |
53 | + | ||
54 | + | func toX18 (origVal,origScaleMult) = fraction(toBigInt(origVal), scale18, toBigInt(origScaleMult)) | |
55 | + | ||
56 | + | ||
57 | + | func fromX18 (val,resultScaleMult) = toInt(fraction(val, toBigInt(resultScaleMult), scale18)) | |
58 | + | ||
59 | + | ||
60 | + | func toScale (amt,resScale,curScale) = fraction(amt, resScale, curScale) | |
61 | + | ||
53 | 62 | ||
54 | 63 | func keyFactoryContract () = "%s__factoryContract" | |
64 | + | ||
65 | + | ||
66 | + | func keyManagerPublicKey () = "%s__managerPublicKey" | |
55 | 67 | ||
56 | 68 | ||
57 | 69 | func keyPriceLast () = "%s%s__price__last" | |
60 | 72 | func keyPriceHistory (h,timestamp) = makeString(["%s%s%d%d__price__history", toString(h), toString(timestamp)], SEP) | |
61 | 73 | ||
62 | 74 | ||
63 | - | func keyPoolLiquidity (internalAmountAsset,internalPriceAsset) = (((("%d%d%s__" + internalAmountAsset) + "__") + internalPriceAsset) + "__locked") | |
64 | - | ||
65 | - | ||
66 | 75 | func keyPutActionByUser (userAddress,txId) = ((("%s%s%s__P__" + userAddress) + "__") + txId) | |
67 | 76 | ||
68 | 77 | ||
69 | 78 | func keyGetActionByUser (userAddress,txId) = ((("%s%s%s__G__" + userAddress) + "__") + txId) | |
70 | 79 | ||
71 | 80 | ||
72 | - | func | |
81 | + | func keyAmtAsset () = "%s__amountAsset" | |
73 | 82 | ||
74 | 83 | ||
75 | 84 | func keyPriceAsset () = "%s__priceAsset" | |
76 | - | ||
77 | - | ||
78 | - | func keyKCurrent () = "%s%s__K_current" | |
79 | 85 | ||
80 | 86 | ||
81 | 87 | func keyKHistoric (h,timestamp) = makeString(["%s%s%d%d__K_history", toString(h), toString(timestamp)], SEP) | |
87 | 93 | func keyMappingPoolContractAddressToPoolAssets (poolContractAddress) = (("%s%s%s__" + poolContractAddress) + "__mappings__poolContract2LpAsset") | |
88 | 94 | ||
89 | 95 | ||
90 | - | func keyPoolConfig ( | |
96 | + | func keyPoolConfig (iAmtAsset,iPriceAsset) = (((("%d%d%s__" + iAmtAsset) + "__") + iPriceAsset) + "__config") | |
91 | 97 | ||
92 | 98 | ||
93 | 99 | func keyMappingsBaseAsset2internalId (baseAssetStr) = ("%s%s%s__mappings__baseAsset2internalId__" + baseAssetStr) | |
99 | 105 | func keyPoolWeight (contractAddress) = ("%s%s__poolWeight__" + contractAddress) | |
100 | 106 | ||
101 | 107 | ||
102 | - | let factoryContract = addressFromStringValue(valueOrErrorMessage(getString(this, keyFactoryContract()), "No Factory Account found.")) | |
108 | + | func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined")) | |
109 | + | ||
110 | + | ||
111 | + | func getIntOrFail (address,key) = valueOrErrorMessage(getInteger(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined")) | |
112 | + | ||
113 | + | ||
114 | + | let factoryContract = addressFromStringValue(getStringOrFail(this, keyFactoryContract())) | |
103 | 115 | ||
104 | 116 | func isGlobalShutdown () = valueOrElse(getBoolean(factoryContract, keyAllPoolsShutdown()), false) | |
105 | 117 | ||
106 | 118 | ||
107 | 119 | func getPoolConfig () = { | |
108 | - | let | |
109 | - | let | |
110 | - | let | |
111 | - | let | |
112 | - | split( | |
120 | + | let amtAsset = getStringOrFail(this, keyAmtAsset()) | |
121 | + | let priceAsset = getStringOrFail(this, keyPriceAsset()) | |
122 | + | let iPriceAsset = getIntOrFail(factoryContract, keyMappingsBaseAsset2internalId(priceAsset)) | |
123 | + | let iAmtAsset = getIntOrFail(factoryContract, keyMappingsBaseAsset2internalId(amtAsset)) | |
124 | + | split(getStringOrFail(factoryContract, keyPoolConfig(toString(iAmtAsset), toString(iPriceAsset))), SEP) | |
113 | 125 | } | |
114 | 126 | ||
115 | 127 | ||
116 | - | func getFactoryConfig () = split( | |
128 | + | func getFactoryConfig () = split(getStringOrFail(factoryContract, keyFactoryConfig()), SEP) | |
117 | 129 | ||
118 | 130 | ||
119 | - | func getPoolLiquidity (amountAssetInternalId,priceAssetInternalId) = { | |
120 | - | let currentPoolLiquidityValue = getString(this, keyPoolLiquidity(amountAssetInternalId, priceAssetInternalId)) | |
121 | - | if (isDefined(currentPoolLiquidityValue)) | |
122 | - | then split(value(currentPoolLiquidityValue), SEP) | |
123 | - | else ["", "0", "0", "0"] | |
124 | - | } | |
131 | + | func dataPutActionInfo (inAmtAssetAmt,inPriceAssetAmt,outLpAmt,price,slippageTolerancePassedByUser,slippageToleranceReal,txHeight,txTimestamp,slipageAmtAssetAmt,slipagePriceAssetAmt) = makeString(["%d%d%d%d%d%d%d%d%d", toString(inAmtAssetAmt), toString(inPriceAssetAmt), toString(outLpAmt), toString(price), toString(slippageTolerancePassedByUser), toString(slippageToleranceReal), toString(txHeight), toString(txTimestamp), toString(slipageAmtAssetAmt), toString(slipagePriceAssetAmt)], SEP) | |
125 | 132 | ||
126 | 133 | ||
127 | - | func | |
134 | + | func dataGetActionInfo (outAmtAssetAmt,outPriceAssetAmt,inLpAmt,price,txHeight,txTimestamp) = makeString(["%d%d%d%d%d%d", toString(outAmtAssetAmt), toString(outPriceAssetAmt), toString(inLpAmt), toString(price), toString(txHeight), toString(txTimestamp)], SEP) | |
128 | 135 | ||
129 | 136 | ||
130 | - | func dataPutActionInfo (inAmountAssetAmt,inPriceAssetAmt,outLpAmt,price,slippageTolerancePassedByUser,slippageToleranceReal,txHeight,txTimestamp) = makeString(["%d%d%d%d%d%d%d", toString(inAmountAssetAmt), toString(inPriceAssetAmt), toString(outLpAmt), toString(price), toString(slippageTolerancePassedByUser), toString(slippageToleranceReal), toString(txHeight), toString(txTimestamp)], SEP) | |
131 | - | ||
132 | - | ||
133 | - | func dataGetActionInfo (outAmountAssetAmt,outPriceAssetAmt,inLpAmt,price,txHeight,txTimestamp) = makeString(["%d%d%d%d%d%d", toString(outAmountAssetAmt), toString(outPriceAssetAmt), toString(inLpAmt), toString(price), toString(txHeight), toString(txTimestamp)], SEP) | |
134 | - | ||
135 | - | ||
136 | - | func privateCastToScaleOf (amount,resultScale,currentScale) = fraction(amount, resultScale, currentScale) | |
137 | - | ||
138 | - | ||
139 | - | func privateBigIntCastToScaleOf (amount,resultScale,currentScale) = fraction(amount, resultScale, currentScale) | |
140 | - | ||
141 | - | ||
142 | - | func calculatePrice (priceAssetLockedAmt,amountAssetlLockedAmt) = fraction(priceAssetLockedAmt, scale8, amountAssetlLockedAmt) | |
143 | - | ||
144 | - | ||
145 | - | func calculatePriceBigInt (priceAssetLockedAmt,amountAssetlLockedAmt) = toInt(fraction(priceAssetLockedAmt, scale8BigInt, amountAssetlLockedAmt)) | |
146 | - | ||
147 | - | ||
148 | - | func privateCalculatePrice (amoutAssetDecimals,priceAssetDecimals,amountAssetAmt,priceAssetAmt) = { | |
149 | - | let amountAssetAmtCasted = privateCastToScaleOf(amountAssetAmt, scale8, amoutAssetDecimals) | |
150 | - | let priceAssetAmtCasted = privateCastToScaleOf(priceAssetAmt, scale8, priceAssetDecimals) | |
151 | - | calculatePrice(priceAssetAmtCasted, amountAssetAmtCasted) | |
152 | - | } | |
153 | - | ||
154 | - | ||
155 | - | func privateGetAccountBalance (assetId) = if ((assetId == "WAVES")) | |
137 | + | func getAccBalance (assetId) = if ((assetId == "WAVES")) | |
156 | 138 | then wavesBalance(this).available | |
157 | 139 | else assetBalance(this, fromBase58String(assetId)) | |
158 | 140 | ||
159 | 141 | ||
160 | - | func calculatePrices (amountAssetAmt,priceAssetAmt,lpAmount) = { | |
161 | - | let poolConfigDataList = getPoolConfig() | |
162 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
163 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
164 | - | let poolPrice = privateCalculatePrice(amoutAssetDecimals, priceAssetDecimals, amountAssetAmt, priceAssetAmt) | |
165 | - | let amountAssetPoolLockedAmt = privateCastToScaleOf(amountAssetAmt, scale8, amoutAssetDecimals) | |
166 | - | let priceAssetPoolLockedAmt = privateCastToScaleOf(priceAssetAmt, scale8, priceAssetDecimals) | |
167 | - | let lpPriceInAmountAsset = calculatePrice(amountAssetPoolLockedAmt, lpAmount) | |
168 | - | let lpPriceInPriceAsset = calculatePrice(priceAssetPoolLockedAmt, lpAmount) | |
169 | - | [poolPrice, lpPriceInAmountAsset, lpPriceInPriceAsset] | |
142 | + | func calcPriceBigInt (priceAssetLockedAmt,amtAssetlLockedAmt,resultScale) = fraction(priceAssetLockedAmt, resultScale, amtAssetlLockedAmt) | |
143 | + | ||
144 | + | ||
145 | + | func privateCalcPrice (amtAssetDcm,priceAssetDcm,amtAssetAmt,priceAssetAmt) = { | |
146 | + | let amtAssetAmtX18 = toX18(amtAssetAmt, amtAssetDcm) | |
147 | + | let priceAssetAmtX18 = toX18(priceAssetAmt, priceAssetDcm) | |
148 | + | toInt(calcPriceBigInt(priceAssetAmtX18, amtAssetAmtX18, scale8BigInt)) | |
170 | 149 | } | |
171 | 150 | ||
172 | 151 | ||
173 | - | func privateEstimateGetOperation (paymentLpAssetId,paymentLpAmount,userAddress) = { | |
152 | + | func calcPrices (amtAssetAmt,priceAssetAmt,lpAmt) = { | |
153 | + | let cfg = getPoolConfig() | |
154 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
155 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
156 | + | let poolPrice = privateCalcPrice(amtAssetDcm, priceAssetDcm, amtAssetAmt, priceAssetAmt) | |
157 | + | let amtAssetAmtX18 = toX18(amtAssetAmt, amtAssetDcm) | |
158 | + | let priceAssetAmtX18 = toX18(priceAssetAmt, priceAssetDcm) | |
159 | + | let lpAmtX18 = toX18(lpAmt, scale8) | |
160 | + | let lpPriceInAmtAsset = toInt(calcPriceBigInt(amtAssetAmtX18, lpAmtX18, scale8BigInt)) | |
161 | + | let lpPriceInPriceAsset = toInt(calcPriceBigInt(priceAssetAmtX18, lpAmtX18, scale8BigInt)) | |
162 | + | [poolPrice, lpPriceInAmtAsset, lpPriceInPriceAsset] | |
163 | + | } | |
164 | + | ||
165 | + | ||
166 | + | func estimateGetOperation (paymentLpAssetId,paymentLpAmt,userAddress) = { | |
174 | 167 | let poolConfigList = getPoolConfig() | |
175 | 168 | let lpAssetId = poolConfigList[idxPoolLPAssetId] | |
176 | - | let | |
169 | + | let amtAssetId = poolConfigList[idxAmtAssetId] | |
177 | 170 | let priceAssetId = poolConfigList[idxPriceAssetId] | |
178 | - | let | |
179 | - | let | |
180 | - | let | |
181 | - | let | |
171 | + | let iAmtAssetId = poolConfigList[idxIAmtAssetId] | |
172 | + | let iPriceAssetId = poolConfigList[idxIPriceAssetId] | |
173 | + | let amtAssetDcm = parseIntValue(poolConfigList[idxAmtAssetDcm]) | |
174 | + | let priceAssetDcm = parseIntValue(poolConfigList[idxPriceAssetDcm]) | |
182 | 175 | let poolStatus = poolConfigList[idxPoolStatus] | |
183 | - | let poolLiquidityList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
184 | - | let poolLPBalance = parseIntValue(poolLiquidityList[idxPoolLPAssetAmt]) | |
185 | - | let poolAmountAssetBalance = parseIntValue(poolLiquidityList[idxPoolAmountAssetAmt]) | |
186 | - | let poolPriceAssetBalance = parseIntValue(poolLiquidityList[idxPoolPriceAssetAmt]) | |
176 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(fromBase58String(lpAssetId)), (("Asset " + lpAssetId) + " doesn't exist")).quantity | |
187 | 177 | if ((lpAssetId != paymentLpAssetId)) | |
188 | 178 | then throw("Invalid asset passed.") | |
189 | 179 | else { | |
190 | - | let amountAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(privateGetAccountBalance(amountAssetId)), scale18, toBigInt(amoutAssetDecimals)) | |
191 | - | let priceAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(privateGetAccountBalance(priceAssetId)), scale18, toBigInt(priceAssetDecimals)) | |
192 | - | let currentPrice = calculatePriceBigInt(priceAssetPoolLockedAmt, amountAssetPoolLockedAmt) | |
193 | - | let paymentLpAmountBuf = privateBigIntCastToScaleOf(toBigInt(paymentLpAmount), scale18, scale8BigInt) | |
194 | - | let poolLPBalanceBuf = privateBigIntCastToScaleOf(toBigInt(poolLPBalance), scale18, scale8BigInt) | |
195 | - | let outAmountAssetAmt = fraction(amountAssetPoolLockedAmt, paymentLpAmountBuf, poolLPBalanceBuf) | |
196 | - | let outPriceAssetAmt = fraction(priceAssetPoolLockedAmt, paymentLpAmountBuf, poolLPBalanceBuf) | |
197 | - | let outAmountAssetAmtFinal = toInt(privateBigIntCastToScaleOf(outAmountAssetAmt, toBigInt(amoutAssetDecimals), scale18)) | |
198 | - | let outPriceAssetAmtFinal = toInt(privateBigIntCastToScaleOf(outPriceAssetAmt, toBigInt(priceAssetDecimals), scale18)) | |
199 | - | $Tuple11(outAmountAssetAmtFinal, outPriceAssetAmtFinal, amountAssetInternalId, priceAssetInternalId, amountAssetId, priceAssetId, poolAmountAssetBalance, poolPriceAssetBalance, poolLPBalance, currentPrice, poolStatus) | |
180 | + | let amtAssetBalance = getAccBalance(amtAssetId) | |
181 | + | let amtAssetBalanceX18 = toX18(amtAssetBalance, amtAssetDcm) | |
182 | + | let priceAssetBalance = getAccBalance(priceAssetId) | |
183 | + | let priceAssetBalanceX18 = toX18(priceAssetBalance, priceAssetDcm) | |
184 | + | let curPrice = toInt(calcPriceBigInt(priceAssetBalanceX18, amtAssetBalanceX18, scale8BigInt)) | |
185 | + | let paymentLpAmtX18 = toX18(paymentLpAmt, scale8) | |
186 | + | let poolLPBalanceX18 = toX18(poolLPBalance, scale8) | |
187 | + | let outAmtAssetAmt = fraction(amtAssetBalanceX18, paymentLpAmtX18, poolLPBalanceX18) | |
188 | + | let outPriceAssetAmt = fraction(priceAssetBalanceX18, paymentLpAmtX18, poolLPBalanceX18) | |
189 | + | let outAmtAssetAmtFinal = fromX18(outAmtAssetAmt, amtAssetDcm) | |
190 | + | let outPriceAssetAmtFinal = fromX18(outPriceAssetAmt, priceAssetDcm) | |
191 | + | $Tuple11(outAmtAssetAmtFinal, outPriceAssetAmtFinal, iAmtAssetId, iPriceAssetId, amtAssetId, priceAssetId, amtAssetBalance, priceAssetBalance, poolLPBalance, curPrice, poolStatus) | |
200 | 192 | } | |
201 | 193 | } | |
202 | 194 | ||
203 | 195 | ||
204 | - | func privateEstimatePutOperation (slippageTolerance,inAmountAssetAmt,inAmountAssetId,inPriceAssetAmt,inPriceAssetId,userAddress,isEvaluate) = { | |
205 | - | let poolConfigDataList = getPoolConfig() | |
206 | - | let lpAssetId = fromBase58String(poolConfigDataList[idxPoolLPAssetId]) | |
207 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
208 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
209 | - | let amountAssetInternalId = poolConfigDataList[idxAmountAssetInternalId] | |
210 | - | let priceAssetInternalId = poolConfigDataList[idxPriceAssetInternalId] | |
211 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
212 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
213 | - | let poolStatus = poolConfigDataList[idxPoolStatus] | |
214 | - | let poolLiquidityDataList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
215 | - | let poolAmountAssetBalance = parseIntValue(poolLiquidityDataList[idxPoolAmountAssetAmt]) | |
216 | - | let poolPriceAssetBalance = parseIntValue(poolLiquidityDataList[idxPoolPriceAssetAmt]) | |
217 | - | let poolLPBalance = parseIntValue(poolLiquidityDataList[idxPoolLPAssetAmt]) | |
218 | - | if (if ((amountAssetId != inAmountAssetId)) | |
196 | + | func estimatePutOperation (slippageTolerance,inAmtAssetAmt,inAmtAssetId,inPriceAssetAmt,inPriceAssetId,userAddress,isEvaluate) = { | |
197 | + | let cfg = getPoolConfig() | |
198 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
199 | + | let amtAssetId = cfg[idxAmtAssetId] | |
200 | + | let priceAssetId = cfg[idxPriceAssetId] | |
201 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
202 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
203 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
204 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
205 | + | let poolStatus = cfg[idxPoolStatus] | |
206 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
207 | + | if (if ((amtAssetId != inAmtAssetId)) | |
219 | 208 | then true | |
220 | 209 | else (priceAssetId != inPriceAssetId)) | |
221 | - | then throw("Invalid | |
210 | + | then throw("Invalid amt or price asset passed.") | |
222 | 211 | else { | |
223 | - | let accountAmountAssetBalance = if (isEvaluate) | |
224 | - | then privateGetAccountBalance(amountAssetId) | |
225 | - | else (privateGetAccountBalance(amountAssetId) - inAmountAssetAmt) | |
226 | - | let accountPriceAssetBalance = if (isEvaluate) | |
227 | - | then privateGetAccountBalance(priceAssetId) | |
228 | - | else (privateGetAccountBalance(priceAssetId) - inPriceAssetAmt) | |
229 | - | let amoutAssetDecimalsBigInt = toBigInt(amoutAssetDecimals) | |
230 | - | let priceAssetDecimalsBigInt = toBigInt(priceAssetDecimals) | |
231 | - | let inAmountAssetAmtCalculated = privateBigIntCastToScaleOf(toBigInt(inAmountAssetAmt), scale18, amoutAssetDecimalsBigInt) | |
232 | - | let inPriceAssetAmtCalculated = privateBigIntCastToScaleOf(toBigInt(inPriceAssetAmt), scale18, priceAssetDecimalsBigInt) | |
233 | - | let userPrice = calculatePriceBigInt(inPriceAssetAmtCalculated, inAmountAssetAmtCalculated) | |
234 | - | let amountAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(accountAmountAssetBalance), scale18, amoutAssetDecimalsBigInt) | |
235 | - | let priceAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(accountPriceAssetBalance), scale18, priceAssetDecimalsBigInt) | |
236 | - | let curentPrice = if ((poolLPBalance == 0)) | |
212 | + | let accAmtAssetBalance = if (isEvaluate) | |
213 | + | then getAccBalance(amtAssetId) | |
214 | + | else (getAccBalance(amtAssetId) - inAmtAssetAmt) | |
215 | + | let accPriceAssetBalance = if (isEvaluate) | |
216 | + | then getAccBalance(priceAssetId) | |
217 | + | else (getAccBalance(priceAssetId) - inPriceAssetAmt) | |
218 | + | let inAmtAssetAmtX18 = toX18(inAmtAssetAmt, amtAssetDcm) | |
219 | + | let inPriceAssetAmtX18 = toX18(inPriceAssetAmt, priceAssetDcm) | |
220 | + | let userPrice = toInt(calcPriceBigInt(inPriceAssetAmtX18, inAmtAssetAmtX18, scale8BigInt)) | |
221 | + | let accAmtAssetBalanceX18 = toX18(accAmtAssetBalance, amtAssetDcm) | |
222 | + | let accPriceAssetBalanceX18 = toX18(accPriceAssetBalance, priceAssetDcm) | |
223 | + | let curPrice = if ((poolLPBalance == 0)) | |
237 | 224 | then 0 | |
238 | - | else | |
239 | - | let slippage = if (( | |
225 | + | else toInt(calcPriceBigInt(accPriceAssetBalanceX18, accAmtAssetBalanceX18, scale8BigInt)) | |
226 | + | let slippage = if ((curPrice == 0)) | |
240 | 227 | then 0 | |
241 | - | else if (( | |
242 | - | then fraction(( | |
243 | - | else fraction((userPrice - | |
244 | - | if (if (( | |
245 | - | then ( | |
228 | + | else if ((curPrice > userPrice)) | |
229 | + | then fraction((curPrice - userPrice), scale8, curPrice) | |
230 | + | else fraction((userPrice - curPrice), scale8, curPrice) | |
231 | + | if (if ((curPrice != 0)) | |
232 | + | then (slippage > slippageTolerance) | |
246 | 233 | else false) | |
247 | - | then throw(((("Price slippage " + toString( | |
234 | + | then throw(((("Price slippage " + toString(slippage)) + " exceeded the passed limit of ") + toString(slippageTolerance))) | |
248 | 235 | else { | |
249 | - | let finalPrice = calculatePriceBigInt((priceAssetPoolLockedAmt + inPriceAssetAmtCalculated), (amountAssetPoolLockedAmt + inAmountAssetAmtCalculated)) | |
250 | - | let outLpAmount = if ((poolLPBalance == 0)) | |
236 | + | let res = if ((poolLPBalance == 0)) | |
251 | 237 | then { | |
252 | - | let lpResultBuf = pow(( | |
253 | - | | |
238 | + | let lpResultBuf = pow((inAmtAssetAmtX18 * inPriceAssetAmtX18), 0, toBigInt(5), 1, 0, DOWN) | |
239 | + | $Tuple4(fromX18(lpResultBuf, scale8), fromX18(inAmtAssetAmtX18, amtAssetDcm), fromX18(inPriceAssetAmtX18, priceAssetDcm), toInt(calcPriceBigInt((accPriceAssetBalanceX18 + inPriceAssetAmtX18), (accAmtAssetBalanceX18 + inAmtAssetAmtX18), scale8BigInt))) | |
254 | 240 | } | |
255 | 241 | else { | |
256 | - | let poolLPBalanceBuf = privateBigIntCastToScaleOf(toBigInt(poolLPBalance), scale18, scale8BigInt) | |
257 | - | let lpAmtByAmountAsset = toInt(privateBigIntCastToScaleOf(fraction(poolLPBalanceBuf, inAmountAssetAmtCalculated, amountAssetPoolLockedAmt), scale8BigInt, scale18)) | |
258 | - | let lpAmtByPriceAsset = toInt(privateBigIntCastToScaleOf(fraction(poolLPBalanceBuf, inPriceAssetAmtCalculated, priceAssetPoolLockedAmt), scale8BigInt, scale18)) | |
259 | - | if ((lpAmtByPriceAsset > lpAmtByAmountAsset)) | |
260 | - | then lpAmtByAmountAsset | |
261 | - | else lpAmtByPriceAsset | |
242 | + | let poolLPBalanceX18 = toX18(poolLPBalance, scale8) | |
243 | + | let curPriceX18 = toX18(curPrice, scale8) | |
244 | + | let pEqByA = fraction(curPriceX18, inAmtAssetAmtX18, scale18) | |
245 | + | let resAssetsToPut = if ((pEqByA > inPriceAssetAmtX18)) | |
246 | + | then $Tuple2(fraction(inPriceAssetAmtX18, scale18, curPriceX18), inPriceAssetAmtX18) | |
247 | + | else $Tuple2(inAmtAssetAmtX18, pEqByA) | |
248 | + | let lpResult = fromX18(fraction(poolLPBalanceX18, resAssetsToPut._2, accPriceAssetBalanceX18), scale8) | |
249 | + | $Tuple4(lpResult, fromX18(resAssetsToPut._1, amtAssetDcm), fromX18(resAssetsToPut._2, priceAssetDcm), toInt(calcPriceBigInt((accPriceAssetBalanceX18 + resAssetsToPut._2), (accAmtAssetBalanceX18 + resAssetsToPut._1), scale8BigInt))) | |
262 | 250 | } | |
263 | - | if ((0 >= | |
251 | + | if ((0 >= res._1)) | |
264 | 252 | then throw("Invalid calculations. LP calculated is less than zero.") | |
265 | - | else $ | |
253 | + | else $Tuple12(res._1, res._4, accAmtAssetBalance, accPriceAssetBalance, poolLPBalance, iAmtAssetId, iPriceAssetId, lpAssetId, slippage, poolStatus, res._2, res._3) | |
266 | 254 | } | |
267 | 255 | } | |
268 | 256 | } | |
269 | 257 | ||
270 | 258 | ||
271 | 259 | func validateMatcherOrderAllowed (order) = { | |
272 | - | let poolConfigDataList = getPoolConfig() | |
273 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
274 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
275 | - | let poolStatus = parseIntValue(poolConfigDataList[idxPoolStatus]) | |
276 | - | let matcherPublicKeyStr = poolConfigDataList[idxMatcherPublicKey] | |
277 | - | let amountAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
278 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
279 | - | let accountAmountAssetBalance = privateGetAccountBalance(amountAssetId) | |
280 | - | let accountPriceAssetBalance = privateGetAccountBalance(priceAssetId) | |
281 | - | let currentPrice = if ((order.orderType == Buy)) | |
282 | - | then privateCalculatePrice(amountAssetDecimals, priceAssetDecimals, (accountAmountAssetBalance + order.amount), accountPriceAssetBalance) | |
283 | - | else privateCalculatePrice(amountAssetDecimals, priceAssetDecimals, (accountAmountAssetBalance + order.amount), accountPriceAssetBalance) | |
284 | - | let isPoolStatusValid = if (if (if (isGlobalShutdown()) | |
260 | + | let cfg = getPoolConfig() | |
261 | + | let amtAssetId = cfg[idxAmtAssetId] | |
262 | + | let priceAssetId = cfg[idxPriceAssetId] | |
263 | + | let poolStatus = parseIntValue(cfg[idxPoolStatus]) | |
264 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
265 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
266 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
267 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
268 | + | let curPrice = if ((order.orderType == Buy)) | |
269 | + | then privateCalcPrice(amtAssetDcm, priceAssetDcm, (accAmtAssetBalance + order.amount), accPriceAssetBalance) | |
270 | + | else privateCalcPrice(amtAssetDcm, priceAssetDcm, (accAmtAssetBalance - order.amount), accPriceAssetBalance) | |
271 | + | if (if (if (isGlobalShutdown()) | |
285 | 272 | then true | |
286 | 273 | else (poolStatus == PoolMatcherDisabled)) | |
287 | 274 | then true | |
288 | 275 | else (poolStatus == PoolShutdown)) | |
289 | - | then false | |
290 | - | else true | |
291 | - | let isValidMatcherSign = true | |
292 | - | let orderAmountAsset = order.assetPair.amountAsset | |
293 | - | let orderAmountAssetStr = if ((orderAmountAsset == unit)) | |
294 | - | then "WAVES" | |
295 | - | else toBase58String(value(orderAmountAsset)) | |
296 | - | let orderPriceAsset = order.assetPair.priceAsset | |
297 | - | let orderPriceAssetStr = if ((orderPriceAsset == unit)) | |
298 | - | then "WAVES" | |
299 | - | else toBase58String(value(orderPriceAsset)) | |
300 | - | let isValidAssetPair = if (if ((orderAmountAssetStr != amountAssetId)) | |
301 | - | then true | |
302 | - | else (orderPriceAssetStr != priceAssetId)) | |
303 | - | then false | |
304 | - | else true | |
305 | - | let orderPrice = order.price | |
306 | - | let priceDecimals = fraction(scale8, priceAssetDecimals, amountAssetDecimals) | |
307 | - | let castedOrderPrice = privateCastToScaleOf(orderPrice, scale8, priceDecimals) | |
308 | - | let isOrderPriceValid = if ((order.orderType == Buy)) | |
309 | - | then (currentPrice >= castedOrderPrice) | |
310 | - | else (castedOrderPrice >= currentPrice) | |
311 | - | if (if (if (isValidAssetPair) | |
312 | - | then isValidMatcherSign | |
313 | - | else false) | |
314 | - | then isPoolStatusValid | |
315 | - | else false) | |
316 | - | then isOrderPriceValid | |
317 | - | else false | |
276 | + | then throw("Exchange operations disabled") | |
277 | + | else { | |
278 | + | let orderAmtAsset = order.assetPair.amountAsset | |
279 | + | let orderAmtAssetStr = if ((orderAmtAsset == unit)) | |
280 | + | then "WAVES" | |
281 | + | else toBase58String(value(orderAmtAsset)) | |
282 | + | let orderPriceAsset = order.assetPair.priceAsset | |
283 | + | let orderPriceAssetStr = if ((orderPriceAsset == unit)) | |
284 | + | then "WAVES" | |
285 | + | else toBase58String(value(orderPriceAsset)) | |
286 | + | if (if ((orderAmtAssetStr != amtAssetId)) | |
287 | + | then true | |
288 | + | else (orderPriceAssetStr != priceAssetId)) | |
289 | + | then throw("Wrong order assets.") | |
290 | + | else { | |
291 | + | let orderPrice = order.price | |
292 | + | let priceDcm = fraction(scale8, priceAssetDcm, amtAssetDcm) | |
293 | + | let castedOrderPrice = toScale(orderPrice, scale8, priceDcm) | |
294 | + | let isOrderPriceValid = if ((order.orderType == Buy)) | |
295 | + | then (curPrice >= castedOrderPrice) | |
296 | + | else (castedOrderPrice >= curPrice) | |
297 | + | if (!(isOrderPriceValid)) | |
298 | + | then throw(((("Order price leads to K decrease. castedOrderPrice=" + toString(castedOrderPrice)) + " curPrice=") + toString(curPrice))) | |
299 | + | else true | |
300 | + | } | |
301 | + | } | |
318 | 302 | } | |
319 | 303 | ||
320 | 304 | ||
321 | - | @Callable(i) | |
322 | - | func constructor (factoryContract) = if ((i.caller != this)) | |
323 | - | then throw("permissions denied") | |
324 | - | else [StringEntry(keyFactoryContract(), factoryContract)] | |
305 | + | func commonGet (i) = { | |
306 | + | let pmtAmtAsset = value(i.payments[0]) | |
307 | + | let pmtAssetId = value(pmtAmtAsset.assetId) | |
308 | + | let pmtAssetAmt = pmtAmtAsset.amount | |
309 | + | let res = estimateGetOperation(toBase58String(pmtAssetId), pmtAssetAmt, toString(i.caller)) | |
310 | + | let outAmtAssetAmt = res._1 | |
311 | + | let outPriceAssetAmt = res._2 | |
312 | + | let iAmtAssetId = res._3 | |
313 | + | let iPriceAssetId = res._4 | |
314 | + | let amtAssetId = res._5 | |
315 | + | let priceAssetId = res._6 | |
316 | + | let poolAmtAssetBalance = res._7 | |
317 | + | let poolPriceAssetBalance = res._8 | |
318 | + | let poolLPBalance = res._9 | |
319 | + | let curPrice = res._10 | |
320 | + | let poolStatus = parseIntValue(res._11) | |
321 | + | if (if (isGlobalShutdown()) | |
322 | + | then true | |
323 | + | else (poolStatus == PoolShutdown)) | |
324 | + | then throw(("Get operation is blocked by admin. Status = " + toString(poolStatus))) | |
325 | + | else $Tuple5(outAmtAssetAmt, outPriceAssetAmt, pmtAssetAmt, pmtAssetId, [ScriptTransfer(i.caller, outAmtAssetAmt, if ((amtAssetId == "WAVES")) | |
326 | + | then unit | |
327 | + | else fromBase58String(amtAssetId)), ScriptTransfer(i.caller, outPriceAssetAmt, if ((priceAssetId == "WAVES")) | |
328 | + | then unit | |
329 | + | else fromBase58String(priceAssetId)), StringEntry(keyGetActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataGetActionInfo(outAmtAssetAmt, outPriceAssetAmt, pmtAssetAmt, curPrice, height, lastBlock.timestamp)), StringEntry(keyPriceLast(), ("%s__" + toString(curPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(curPrice)))]) | |
330 | + | } | |
325 | 331 | ||
326 | 332 | ||
327 | - | ||
328 | - | @Callable(i) | |
329 | - | func put (slippageTolerance,shouldAutoStake) = { | |
330 | - | let factoryConfigList = getFactoryConfig() | |
331 | - | let stakingContract = valueOrErrorMessage(addressFromString(factoryConfigList[idxFactoryStakingContract]), "Error. Incorrect staking address.") | |
332 | - | let pmtAmountAsset = value(i.payments[0]) | |
333 | - | let inAmountAssetAmt = pmtAmountAsset.amount | |
334 | - | let inAmountAssetId = if (!(isDefined(pmtAmountAsset.assetId))) | |
333 | + | func commonPut (i,slippageTolerance,emitLp) = { | |
334 | + | let pmtAmtAsset = value(i.payments[0]) | |
335 | + | let inAmtAssetAmt = pmtAmtAsset.amount | |
336 | + | let inAmtAssetId = if (!(isDefined(pmtAmtAsset.assetId))) | |
335 | 337 | then fromBase58String("WAVES") | |
336 | - | else value( | |
338 | + | else value(pmtAmtAsset.assetId) | |
337 | 339 | let pmtPriceAsset = value(i.payments[1]) | |
338 | 340 | let inPriceAssetAmt = pmtPriceAsset.amount | |
339 | 341 | let inPriceAssetId = if (!(isDefined(pmtPriceAsset.assetId))) | |
340 | 342 | then fromBase58String("WAVES") | |
341 | 343 | else value(pmtPriceAsset.assetId) | |
342 | - | let estimatedPutResults = privateEstimatePutOperation(slippageTolerance, inAmountAssetAmt, toBase58String(inAmountAssetId), inPriceAssetAmt, toBase58String(inPriceAssetId), toString(i.caller), false) | |
343 | - | let outLpAmount = estimatedPutResults._1 | |
344 | - | let curentPrice = estimatedPutResults._2 | |
345 | - | let poolAmountAssetBalance = estimatedPutResults._3 | |
346 | - | let poolPriceAssetBalance = estimatedPutResults._4 | |
347 | - | let poolLPBalance = estimatedPutResults._5 | |
348 | - | let amountAssetInternalId = estimatedPutResults._6 | |
349 | - | let priceAssetInternalId = estimatedPutResults._7 | |
350 | - | let lpAssetId = estimatedPutResults._8 | |
351 | - | let slippageCalculated = estimatedPutResults._9 | |
352 | - | let poolStatus = parseIntValue(estimatedPutResults._10) | |
344 | + | let estPut = estimatePutOperation(slippageTolerance, inAmtAssetAmt, toBase58String(inAmtAssetId), inPriceAssetAmt, toBase58String(inPriceAssetId), toString(i.caller), false) | |
345 | + | let outLpAmt = estPut._1 | |
346 | + | let curPrice = estPut._2 | |
347 | + | let poolAmtAssetBalance = estPut._3 | |
348 | + | let poolPriceAssetBalance = estPut._4 | |
349 | + | let poolLPBalance = estPut._5 | |
350 | + | let iAmtAssetId = estPut._6 | |
351 | + | let iPriceAssetId = estPut._7 | |
352 | + | let lpAssetId = estPut._8 | |
353 | + | let slippageCalc = estPut._9 | |
354 | + | let poolStatus = parseIntValue(estPut._10) | |
355 | + | let realAmtAssetIn = estPut._11 | |
356 | + | let realPriceAssetIn = estPut._12 | |
353 | 357 | if (if (if (isGlobalShutdown()) | |
354 | 358 | then true | |
355 | 359 | else (poolStatus == PoolPutDisabled)) | |
357 | 361 | else (poolStatus == PoolShutdown)) | |
358 | 362 | then throw(("Put operation is blocked by admin. Status = " + toString(poolStatus))) | |
359 | 363 | else { | |
360 | - | let emitLPAssetOnFactory = invoke(factoryContract, "emit", [outLpAmount], nil) | |
361 | - | if ((emitLPAssetOnFactory == emitLPAssetOnFactory)) | |
362 | - | then if ((emitLPAssetOnFactory != "success")) | |
363 | - | then throw("LP asset reissue failed during emit call on factory.") | |
364 | - | else { | |
365 | - | let results = [StringEntry(keyPriceLast(), ("%s__" + toString(curentPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(curentPrice))), StringEntry(keyPutActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataPutActionInfo(inAmountAssetAmt, inPriceAssetAmt, outLpAmount, curentPrice, slippageTolerance, slippageCalculated, height, lastBlock.timestamp)), StringEntry(keyPoolLiquidity(amountAssetInternalId, priceAssetInternalId), dataPoolLiquidity((poolAmountAssetBalance + inAmountAssetAmt), (poolPriceAssetBalance + inPriceAssetAmt), (poolLPBalance + outLpAmount)))] | |
366 | - | if (shouldAutoStake) | |
367 | - | then { | |
368 | - | let stake = invoke(stakingContract, "stake", nil, [AttachedPayment(lpAssetId, outLpAmount)]) | |
369 | - | if ((stake == stake)) | |
370 | - | then results | |
371 | - | else throw("Strict value is not equal to itself.") | |
372 | - | } | |
373 | - | else (results :+ ScriptTransfer(i.caller, outLpAmount, lpAssetId)) | |
374 | - | } | |
364 | + | let lpAmtToEmit = if (!(emitLp)) | |
365 | + | then 0 | |
366 | + | else outLpAmt | |
367 | + | let diffInAmtAsset = (inAmtAssetAmt - realAmtAssetIn) | |
368 | + | let diffInPriceAsset = (inPriceAssetAmt - realPriceAssetIn) | |
369 | + | let dataStateWrite = [StringEntry(keyPriceLast(), ("%s__" + toString(curPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(curPrice))), StringEntry(keyPutActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataPutActionInfo(realAmtAssetIn, realPriceAssetIn, lpAmtToEmit, curPrice, slippageTolerance, slippageCalc, height, lastBlock.timestamp, diffInAmtAsset, diffInPriceAsset))] | |
370 | + | $Tuple9(lpAmtToEmit, lpAssetId, dataStateWrite, inAmtAssetAmt, realAmtAssetIn, inPriceAssetAmt, realPriceAssetIn, pmtAmtAsset.assetId, pmtPriceAsset.assetId) | |
371 | + | } | |
372 | + | } | |
373 | + | ||
374 | + | ||
375 | + | @Callable(i) | |
376 | + | func constructor (factoryContract,managerPublicKey) = if ((i.caller != this)) | |
377 | + | then throw("permissions denied") | |
378 | + | else [StringEntry(keyFactoryContract(), factoryContract), StringEntry(keyManagerPublicKey(), managerPublicKey)] | |
379 | + | ||
380 | + | ||
381 | + | ||
382 | + | @Callable(i) | |
383 | + | func put (slippageTolerance,shouldAutoStake) = { | |
384 | + | let factoryCfg = getFactoryConfig() | |
385 | + | let stakingContract = valueOrErrorMessage(addressFromString(factoryCfg[idxFactoryStakingContract]), "Error. Incorrect staking address.") | |
386 | + | let slippageContract = valueOrErrorMessage(addressFromString(factoryCfg[idxFactorySlippageContract]), "Error. Incorrect slippage contract address.") | |
387 | + | let actions = commonPut(i, slippageTolerance, true) | |
388 | + | let amtToEmit = actions._1 | |
389 | + | let lpAssetId = actions._2 | |
390 | + | let state = actions._3 | |
391 | + | let inAmt = actions._4 | |
392 | + | let realInAmt = actions._5 | |
393 | + | let inPrice = actions._6 | |
394 | + | let realInPrice = actions._7 | |
395 | + | let amId = actions._8 | |
396 | + | let prId = actions._9 | |
397 | + | let diffInAmtAsset = (inAmt - realInAmt) | |
398 | + | let diffInPriceAsset = (inPrice - realInPrice) | |
399 | + | let emitInv = invoke(factoryContract, "emit", [amtToEmit], nil) | |
400 | + | if ((emitInv == emitInv)) | |
401 | + | then { | |
402 | + | let slippageAInv = if ((inAmt > realInAmt)) | |
403 | + | then invoke(slippageContract, "put", nil, [AttachedPayment(amId, diffInAmtAsset)]) | |
404 | + | else nil | |
405 | + | if ((slippageAInv == slippageAInv)) | |
406 | + | then { | |
407 | + | let slippagePInv = if ((inPrice > realInPrice)) | |
408 | + | then invoke(slippageContract, "put", nil, [AttachedPayment(prId, diffInPriceAsset)]) | |
409 | + | else nil | |
410 | + | if ((slippagePInv == slippagePInv)) | |
411 | + | then { | |
412 | + | let lpStakeInv = if (shouldAutoStake) | |
413 | + | then invoke(stakingContract, "stake", nil, [AttachedPayment(lpAssetId, amtToEmit)]) | |
414 | + | else nil | |
415 | + | if ((lpStakeInv == lpStakeInv)) | |
416 | + | then (state :+ ScriptTransfer(i.caller, amtToEmit, lpAssetId)) | |
417 | + | else throw("Strict value is not equal to itself.") | |
418 | + | } | |
419 | + | else throw("Strict value is not equal to itself.") | |
420 | + | } | |
375 | 421 | else throw("Strict value is not equal to itself.") | |
376 | 422 | } | |
423 | + | else throw("Strict value is not equal to itself.") | |
424 | + | } | |
425 | + | ||
426 | + | ||
427 | + | ||
428 | + | @Callable(i) | |
429 | + | func putForFree () = { | |
430 | + | let actions = commonPut(i, 0, false) | |
431 | + | actions._3 | |
377 | 432 | } | |
378 | 433 | ||
379 | 434 | ||
380 | 435 | ||
381 | 436 | @Callable(i) | |
382 | 437 | func get () = { | |
383 | - | let pmtAmountAsset = value(i.payments[0]) | |
384 | - | let pmtAssetId = value(pmtAmountAsset.assetId) | |
385 | - | let pmtAssetAmount = pmtAmountAsset.amount | |
386 | - | let results = privateEstimateGetOperation(toBase58String(pmtAssetId), pmtAssetAmount, toString(i.caller)) | |
387 | - | let outAmountAssetAmt = results._1 | |
388 | - | let outPriceAssetAmt = results._2 | |
389 | - | let amountAssetInternalId = results._3 | |
390 | - | let priceAssetInternalId = results._4 | |
391 | - | let amountAssetId = results._5 | |
392 | - | let priceAssetId = results._6 | |
393 | - | let poolAmountAssetBalance = results._7 | |
394 | - | let poolPriceAssetBalance = results._8 | |
395 | - | let poolLPBalance = results._9 | |
396 | - | let currentPrice = results._10 | |
397 | - | let poolStatus = parseIntValue(results._11) | |
398 | - | if (if (isGlobalShutdown()) | |
399 | - | then true | |
400 | - | else (poolStatus == PoolShutdown)) | |
401 | - | then throw(("Get operation is blocked by admin. Status = " + toString(poolStatus))) | |
402 | - | else { | |
403 | - | let burnLPAssetOnFactory = invoke(factoryContract, "burn", [pmtAssetAmount], [AttachedPayment(pmtAssetId, pmtAssetAmount)]) | |
404 | - | if ((burnLPAssetOnFactory == burnLPAssetOnFactory)) | |
405 | - | then if ((burnLPAssetOnFactory != "success")) | |
406 | - | then throw("LP asset burn failed during emit call on factory.") | |
407 | - | else [ScriptTransfer(i.caller, outAmountAssetAmt, if ((amountAssetId == "WAVES")) | |
408 | - | then unit | |
409 | - | else fromBase58String(amountAssetId)), ScriptTransfer(i.caller, outPriceAssetAmt, fromBase58String(priceAssetId)), StringEntry(keyPoolLiquidity(amountAssetInternalId, priceAssetInternalId), dataPoolLiquidity((poolAmountAssetBalance - outAmountAssetAmt), (poolPriceAssetBalance - outPriceAssetAmt), (poolLPBalance - pmtAssetAmount))), StringEntry(keyGetActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataGetActionInfo(outAmountAssetAmt, outPriceAssetAmt, pmtAssetAmount, currentPrice, height, lastBlock.timestamp)), StringEntry(keyPriceLast(), ("%s__" + toString(currentPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(currentPrice)))] | |
410 | - | else throw("Strict value is not equal to itself.") | |
411 | - | } | |
438 | + | let res = commonGet(i) | |
439 | + | let outAmtAssetAmt = res._1 | |
440 | + | let outPriceAssetAmt = res._2 | |
441 | + | let pmtAssetAmt = res._3 | |
442 | + | let pmtAssetId = res._4 | |
443 | + | let actions = res._5 | |
444 | + | let burnLPAssetOnFactory = invoke(factoryContract, "burn", [pmtAssetAmt], [AttachedPayment(pmtAssetId, pmtAssetAmt)]) | |
445 | + | if ((burnLPAssetOnFactory == burnLPAssetOnFactory)) | |
446 | + | then actions | |
447 | + | else throw("Strict value is not equal to itself.") | |
412 | 448 | } | |
413 | 449 | ||
414 | 450 | ||
415 | 451 | ||
416 | 452 | @Callable(i) | |
417 | - | func activate (amountAssetStr,priceAssetStr) = if ((toString(i.caller) != toString(factoryContract))) | |
453 | + | func getNoLess (noLessThenAmtAsset,noLessThenPriceAsset) = { | |
454 | + | let res = commonGet(i) | |
455 | + | let outAmtAssetAmt = res._1 | |
456 | + | let outPriceAssetAmt = res._2 | |
457 | + | let pmtAssetAmt = res._3 | |
458 | + | let pmtAssetId = res._4 | |
459 | + | let actions = res._5 | |
460 | + | if ((noLessThenAmtAsset > outAmtAssetAmt)) | |
461 | + | then throw(((("noLessThenAmtAsset failed: " + toString(outAmtAssetAmt)) + " < ") + toString(noLessThenAmtAsset))) | |
462 | + | else if ((noLessThenPriceAsset > outPriceAssetAmt)) | |
463 | + | then throw(((("noLessThenPriceAsset failed: " + toString(outPriceAssetAmt)) + " < ") + toString(noLessThenPriceAsset))) | |
464 | + | else { | |
465 | + | let burnLPAssetOnFactory = invoke(factoryContract, "burn", [pmtAssetAmt], [AttachedPayment(pmtAssetId, pmtAssetAmt)]) | |
466 | + | if ((burnLPAssetOnFactory == burnLPAssetOnFactory)) | |
467 | + | then actions | |
468 | + | else throw("Strict value is not equal to itself.") | |
469 | + | } | |
470 | + | } | |
471 | + | ||
472 | + | ||
473 | + | ||
474 | + | @Callable(i) | |
475 | + | func activate (amtAssetStr,priceAssetStr) = if ((toString(i.caller) != toString(factoryContract))) | |
418 | 476 | then throw("permissions denied") | |
419 | - | else $Tuple2([StringEntry( | |
477 | + | else $Tuple2([StringEntry(keyAmtAsset(), amtAssetStr), StringEntry(keyPriceAsset(), priceAssetStr)], "success") | |
420 | 478 | ||
421 | 479 | ||
422 | 480 | ||
423 | 481 | @Callable(i) | |
424 | 482 | func statsREADONLY () = { | |
425 | - | let poolConfigDataList = getPoolConfig() | |
426 | - | let lpAssetId = fromBase58String(poolConfigDataList[idxPoolLPAssetId]) | |
427 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
428 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
429 | - | let amountAssetInternalId = poolConfigDataList[idxAmountAssetInternalId] | |
430 | - | let priceAssetInternalId = poolConfigDataList[idxPriceAssetInternalId] | |
431 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
432 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
433 | - | let poolLiquidityDataList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
434 | - | let poolLPBalance = parseIntValue(poolLiquidityDataList[idxPoolLPAssetAmt]) | |
435 | - | let accountAmountAssetBalance = privateGetAccountBalance(amountAssetId) | |
436 | - | let accountPriceAssetBalance = privateGetAccountBalance(priceAssetId) | |
437 | - | let pricesList = calculatePrices(accountAmountAssetBalance, accountPriceAssetBalance, poolLPBalance) | |
438 | - | let currentPrice = pricesList[0] | |
439 | - | let lpAmountAssetShare = pricesList[1] | |
483 | + | let cfg = getPoolConfig() | |
484 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
485 | + | let amtAssetId = cfg[idxAmtAssetId] | |
486 | + | let priceAssetId = cfg[idxPriceAssetId] | |
487 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
488 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
489 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
490 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
491 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
492 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
493 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
494 | + | let pricesList = calcPrices(accAmtAssetBalance, accPriceAssetBalance, poolLPBalance) | |
495 | + | let curPrice = pricesList[0] | |
496 | + | let lpAmtAssetShare = pricesList[1] | |
440 | 497 | let lpPriceAssetShare = pricesList[2] | |
441 | - | let poolWeight = | |
442 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString( | |
498 | + | let poolWeight = value(getInteger(factoryContract, keyPoolWeight(toString(this)))) | |
499 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString(accAmtAssetBalance), toString(accPriceAssetBalance), toString(poolLPBalance), toString(curPrice), toString(lpAmtAssetShare), toString(lpPriceAssetShare), toString(poolWeight)], SEP)) | |
443 | 500 | } | |
444 | 501 | ||
445 | 502 | ||
446 | 503 | ||
447 | 504 | @Callable(i) | |
448 | - | func | |
449 | - | let | |
450 | - | let lpAssetId = fromBase58String( | |
451 | - | let | |
452 | - | let priceAssetId = | |
453 | - | let | |
454 | - | let | |
455 | - | let | |
456 | - | let | |
457 | - | let poolStatus = | |
458 | - | let | |
459 | - | let | |
460 | - | let | |
461 | - | let | |
462 | - | let | |
463 | - | let | |
464 | - | | |
465 | - | | |
466 | - | | |
467 | - | let | |
468 | - | let inPriceAssetAmt = | |
469 | - | let estimatedPutResults = | |
470 | - | let | |
471 | - | let | |
472 | - | let | |
505 | + | func evaluatePutByAmtAssetREADONLY (slippageTolerance,inAmtAssetAmt,userAddress) = { | |
506 | + | let cfg = getPoolConfig() | |
507 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
508 | + | let amtAssetId = cfg[idxAmtAssetId] | |
509 | + | let priceAssetId = cfg[idxPriceAssetId] | |
510 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
511 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
512 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
513 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
514 | + | let poolStatus = cfg[idxPoolStatus] | |
515 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
516 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
517 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
518 | + | let amtAssetAmtX18 = toX18(accAmtAssetBalance, amtAssetDcm) | |
519 | + | let priceAssetAmtX18 = toX18(accPriceAssetBalance, priceAssetDcm) | |
520 | + | let curPriceX18 = if ((poolLPBalance == 0)) | |
521 | + | then toBigInt(0) | |
522 | + | else calcPriceBigInt(priceAssetAmtX18, amtAssetAmtX18, scale18) | |
523 | + | let inAmtAssetAmtX18 = toX18(inAmtAssetAmt, amtAssetDcm) | |
524 | + | let inPriceAssetAmtX18 = fraction(inAmtAssetAmtX18, curPriceX18, scale18) | |
525 | + | let inPriceAssetAmt = fromX18(inPriceAssetAmtX18, priceAssetDcm) | |
526 | + | let estimatedPutResults = estimatePutOperation(slippageTolerance, inAmtAssetAmt, amtAssetId, inPriceAssetAmt, priceAssetId, userAddress, true) | |
527 | + | let outLpAmtCalc = estimatedPutResults._1 | |
528 | + | let curPriceCalc = estimatedPutResults._2 | |
529 | + | let poolAmtAssetBalanceCalc = estimatedPutResults._3 | |
473 | 530 | let poolPriceAssetBalanceCAlc = estimatedPutResults._4 | |
474 | 531 | let poolLPBalanceCalc = estimatedPutResults._5 | |
475 | 532 | let poolStatusCalc = parseIntValue(estimatedPutResults._10) | |
476 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString( | |
533 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString(outLpAmtCalc), toString(fromX18(curPriceX18, scale8)), toString(poolAmtAssetBalanceCalc), toString(poolPriceAssetBalanceCAlc), toString(poolLPBalanceCalc), toString(poolStatusCalc)], SEP)) | |
477 | 534 | } | |
478 | 535 | ||
479 | 536 | ||
480 | 537 | ||
481 | 538 | @Callable(i) | |
482 | 539 | func evaluatePutByPriceAssetREADONLY (slippageTolerance,inPriceAssetAmt,userAddress) = { | |
483 | - | let poolConfigDataList = getPoolConfig() | |
484 | - | let lpAssetId = fromBase58String(poolConfigDataList[idxPoolLPAssetId]) | |
485 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
486 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
487 | - | let amountAssetInternalId = poolConfigDataList[idxAmountAssetInternalId] | |
488 | - | let priceAssetInternalId = poolConfigDataList[idxPriceAssetInternalId] | |
489 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
490 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
491 | - | let poolStatus = poolConfigDataList[idxPoolStatus] | |
492 | - | let poolLiquidityDataList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
493 | - | let poolLPBalance = parseIntValue(poolLiquidityDataList[idxPoolLPAssetAmt]) | |
494 | - | let accountAmountAssetBalance = privateGetAccountBalance(amountAssetId) | |
495 | - | let accountPriceAssetBalance = privateGetAccountBalance(priceAssetId) | |
496 | - | let accountAmountAssetBalanceCasted = privateCastToScaleOf(accountAmountAssetBalance, scale8, amoutAssetDecimals) | |
497 | - | let accountPriceAssetBalanceCasted = privateCastToScaleOf(accountPriceAssetBalance, scale8, priceAssetDecimals) | |
498 | - | let inPriceAssetAmtCalculatedTuple = privateCastToScaleOf(inPriceAssetAmt, scale8, priceAssetDecimals) | |
499 | - | let curentPrice = if ((poolLPBalance == 0)) | |
500 | - | then 0 | |
501 | - | else calculatePrice(accountPriceAssetBalanceCasted, accountAmountAssetBalanceCasted) | |
502 | - | let inAmountAssetAmtCalculated = fraction(inPriceAssetAmtCalculatedTuple, scale8, curentPrice) | |
503 | - | let inAmountAssetAmt = privateCastToScaleOf(inAmountAssetAmtCalculated, amoutAssetDecimals, scale8) | |
504 | - | let estimatedPutResults = privateEstimatePutOperation(slippageTolerance, inAmountAssetAmt, amountAssetId, inPriceAssetAmt, priceAssetId, userAddress, true) | |
505 | - | let outLpAmountCalc = estimatedPutResults._1 | |
506 | - | let curentPriceCalc = estimatedPutResults._2 | |
507 | - | let poolAmountAssetBalanceCalc = estimatedPutResults._3 | |
540 | + | let cfg = getPoolConfig() | |
541 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
542 | + | let amtAssetId = cfg[idxAmtAssetId] | |
543 | + | let priceAssetId = cfg[idxPriceAssetId] | |
544 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
545 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
546 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
547 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
548 | + | let poolStatus = cfg[idxPoolStatus] | |
549 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
550 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
551 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
552 | + | let amtAssetAmtX18 = toX18(accAmtAssetBalance, amtAssetDcm) | |
553 | + | let priceAssetAmtX18 = toX18(accPriceAssetBalance, priceAssetDcm) | |
554 | + | let curPriceX18 = if ((poolLPBalance == 0)) | |
555 | + | then toBigInt(0) | |
556 | + | else calcPriceBigInt(priceAssetAmtX18, amtAssetAmtX18, scale18) | |
557 | + | let inPriceAssetAmtX18 = toX18(inPriceAssetAmt, priceAssetDcm) | |
558 | + | let inAmtAssetAmtX18 = fraction(inPriceAssetAmtX18, scale18, curPriceX18) | |
559 | + | let inAmtAssetAmt = fromX18(inAmtAssetAmtX18, amtAssetDcm) | |
560 | + | let estimatedPutResults = estimatePutOperation(slippageTolerance, inAmtAssetAmt, amtAssetId, inPriceAssetAmt, priceAssetId, userAddress, true) | |
561 | + | let outLpAmtCalc = estimatedPutResults._1 | |
562 | + | let curPriceCalc = estimatedPutResults._2 | |
563 | + | let poolAmtAssetBalanceCalc = estimatedPutResults._3 | |
508 | 564 | let poolPriceAssetBalanceCAlc = estimatedPutResults._4 | |
509 | 565 | let poolLPBalanceCalc = estimatedPutResults._5 | |
510 | 566 | let poolStatusCalc = parseIntValue(estimatedPutResults._10) | |
511 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString( | |
567 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString(outLpAmtCalc), toString(fromX18(curPriceX18, scale8)), toString(poolAmtAssetBalanceCalc), toString(poolPriceAssetBalanceCAlc), toString(poolLPBalanceCalc), toString(poolStatusCalc)], SEP)) | |
512 | 568 | } | |
513 | 569 | ||
514 | 570 | ||
515 | 571 | ||
516 | 572 | @Callable(i) | |
517 | - | func evaluateGetREADONLY (paymentLpAssetId, | |
518 | - | let | |
519 | - | let | |
520 | - | let outPriceAssetAmt = | |
521 | - | let | |
522 | - | let poolPriceAssetBalance = | |
523 | - | let poolLPBalance = | |
524 | - | let | |
525 | - | let poolStatus = parseIntValue( | |
526 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString( | |
573 | + | func evaluateGetREADONLY (paymentLpAssetId,paymentLpAmt,userAddress) = { | |
574 | + | let res = estimateGetOperation(paymentLpAssetId, paymentLpAmt, userAddress) | |
575 | + | let outAmtAssetAmt = res._1 | |
576 | + | let outPriceAssetAmt = res._2 | |
577 | + | let poolAmtAssetBalance = res._7 | |
578 | + | let poolPriceAssetBalance = res._8 | |
579 | + | let poolLPBalance = res._9 | |
580 | + | let curPrice = res._10 | |
581 | + | let poolStatus = parseIntValue(res._11) | |
582 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString(outAmtAssetAmt), toString(outPriceAssetAmt), toString(poolAmtAssetBalance), toString(poolPriceAssetBalance), toString(poolLPBalance), toString(curPrice), toString(poolStatus)], SEP)) | |
527 | 583 | } | |
528 | - | ||
529 | - | ||
530 | - | ||
531 | - | @Callable(i) | |
532 | - | func hashREADONLY () = $Tuple2(nil, toBase64String(value(scriptHash(this)))) | |
533 | 584 | ||
534 | 585 | ||
535 | 586 | @Verifier(tx) | |
536 | 587 | func verify () = match tx { | |
537 | 588 | case order: Order => | |
538 | - | sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
589 | + | if (validateMatcherOrderAllowed(order)) | |
590 | + | then sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
591 | + | else false | |
539 | 592 | case _ => | |
540 | - | sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
593 | + | let managerPublic = valueOrElse(getString(this, keyManagerPublicKey()), EMPTY) | |
594 | + | if ((managerPublic == EMPTY)) | |
595 | + | then sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
596 | + | else sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(managerPublic)) | |
541 | 597 | } | |
542 | 598 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let lPdecimals = 8 | |
5 | 5 | ||
6 | 6 | let scale8 = 100000000 | |
7 | 7 | ||
8 | 8 | let scale8BigInt = toBigInt(100000000) | |
9 | 9 | ||
10 | 10 | let scale18 = toBigInt(1000000000000000000) | |
11 | 11 | ||
12 | 12 | let SEP = "__" | |
13 | + | ||
14 | + | let EMPTY = "" | |
13 | 15 | ||
14 | 16 | let PoolActive = 1 | |
15 | 17 | ||
16 | 18 | let PoolPutDisabled = 2 | |
17 | 19 | ||
18 | 20 | let PoolMatcherDisabled = 3 | |
19 | 21 | ||
20 | 22 | let PoolShutdown = 4 | |
21 | 23 | ||
22 | 24 | let idxPoolAddress = 1 | |
23 | 25 | ||
24 | 26 | let idxPoolStatus = 2 | |
25 | 27 | ||
26 | 28 | let idxPoolLPAssetId = 3 | |
27 | 29 | ||
28 | - | let | |
30 | + | let idxAmtAssetId = 4 | |
29 | 31 | ||
30 | 32 | let idxPriceAssetId = 5 | |
31 | 33 | ||
32 | - | let | |
34 | + | let idxAmtAssetDcm = 6 | |
33 | 35 | ||
34 | - | let | |
36 | + | let idxPriceAssetDcm = 7 | |
35 | 37 | ||
36 | - | let | |
38 | + | let idxIAmtAssetId = 8 | |
37 | 39 | ||
38 | - | let | |
40 | + | let idxIPriceAssetId = 9 | |
39 | 41 | ||
40 | - | let | |
42 | + | let idxLPAssetDcm = 10 | |
41 | 43 | ||
42 | - | let idxMatcherPublicKey = 11 | |
43 | - | ||
44 | - | let idxPoolAmountAssetAmt = 1 | |
44 | + | let idxPoolAmtAssetAmt = 1 | |
45 | 45 | ||
46 | 46 | let idxPoolPriceAssetAmt = 2 | |
47 | 47 | ||
48 | 48 | let idxPoolLPAssetAmt = 3 | |
49 | 49 | ||
50 | 50 | let idxFactoryStakingContract = 1 | |
51 | 51 | ||
52 | - | let idxFactoryBoostingConfig = 2 | |
52 | + | let idxFactorySlippageContract = 7 | |
53 | + | ||
54 | + | func toX18 (origVal,origScaleMult) = fraction(toBigInt(origVal), scale18, toBigInt(origScaleMult)) | |
55 | + | ||
56 | + | ||
57 | + | func fromX18 (val,resultScaleMult) = toInt(fraction(val, toBigInt(resultScaleMult), scale18)) | |
58 | + | ||
59 | + | ||
60 | + | func toScale (amt,resScale,curScale) = fraction(amt, resScale, curScale) | |
61 | + | ||
53 | 62 | ||
54 | 63 | func keyFactoryContract () = "%s__factoryContract" | |
64 | + | ||
65 | + | ||
66 | + | func keyManagerPublicKey () = "%s__managerPublicKey" | |
55 | 67 | ||
56 | 68 | ||
57 | 69 | func keyPriceLast () = "%s%s__price__last" | |
58 | 70 | ||
59 | 71 | ||
60 | 72 | func keyPriceHistory (h,timestamp) = makeString(["%s%s%d%d__price__history", toString(h), toString(timestamp)], SEP) | |
61 | 73 | ||
62 | 74 | ||
63 | - | func keyPoolLiquidity (internalAmountAsset,internalPriceAsset) = (((("%d%d%s__" + internalAmountAsset) + "__") + internalPriceAsset) + "__locked") | |
64 | - | ||
65 | - | ||
66 | 75 | func keyPutActionByUser (userAddress,txId) = ((("%s%s%s__P__" + userAddress) + "__") + txId) | |
67 | 76 | ||
68 | 77 | ||
69 | 78 | func keyGetActionByUser (userAddress,txId) = ((("%s%s%s__G__" + userAddress) + "__") + txId) | |
70 | 79 | ||
71 | 80 | ||
72 | - | func | |
81 | + | func keyAmtAsset () = "%s__amountAsset" | |
73 | 82 | ||
74 | 83 | ||
75 | 84 | func keyPriceAsset () = "%s__priceAsset" | |
76 | - | ||
77 | - | ||
78 | - | func keyKCurrent () = "%s%s__K_current" | |
79 | 85 | ||
80 | 86 | ||
81 | 87 | func keyKHistoric (h,timestamp) = makeString(["%s%s%d%d__K_history", toString(h), toString(timestamp)], SEP) | |
82 | 88 | ||
83 | 89 | ||
84 | 90 | func keyFactoryConfig () = "%s__factoryConfig" | |
85 | 91 | ||
86 | 92 | ||
87 | 93 | func keyMappingPoolContractAddressToPoolAssets (poolContractAddress) = (("%s%s%s__" + poolContractAddress) + "__mappings__poolContract2LpAsset") | |
88 | 94 | ||
89 | 95 | ||
90 | - | func keyPoolConfig ( | |
96 | + | func keyPoolConfig (iAmtAsset,iPriceAsset) = (((("%d%d%s__" + iAmtAsset) + "__") + iPriceAsset) + "__config") | |
91 | 97 | ||
92 | 98 | ||
93 | 99 | func keyMappingsBaseAsset2internalId (baseAssetStr) = ("%s%s%s__mappings__baseAsset2internalId__" + baseAssetStr) | |
94 | 100 | ||
95 | 101 | ||
96 | 102 | func keyAllPoolsShutdown () = "%s__shutdown" | |
97 | 103 | ||
98 | 104 | ||
99 | 105 | func keyPoolWeight (contractAddress) = ("%s%s__poolWeight__" + contractAddress) | |
100 | 106 | ||
101 | 107 | ||
102 | - | let factoryContract = addressFromStringValue(valueOrErrorMessage(getString(this, keyFactoryContract()), "No Factory Account found.")) | |
108 | + | func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined")) | |
109 | + | ||
110 | + | ||
111 | + | func getIntOrFail (address,key) = valueOrErrorMessage(getInteger(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined")) | |
112 | + | ||
113 | + | ||
114 | + | let factoryContract = addressFromStringValue(getStringOrFail(this, keyFactoryContract())) | |
103 | 115 | ||
104 | 116 | func isGlobalShutdown () = valueOrElse(getBoolean(factoryContract, keyAllPoolsShutdown()), false) | |
105 | 117 | ||
106 | 118 | ||
107 | 119 | func getPoolConfig () = { | |
108 | - | let | |
109 | - | let | |
110 | - | let | |
111 | - | let | |
112 | - | split( | |
120 | + | let amtAsset = getStringOrFail(this, keyAmtAsset()) | |
121 | + | let priceAsset = getStringOrFail(this, keyPriceAsset()) | |
122 | + | let iPriceAsset = getIntOrFail(factoryContract, keyMappingsBaseAsset2internalId(priceAsset)) | |
123 | + | let iAmtAsset = getIntOrFail(factoryContract, keyMappingsBaseAsset2internalId(amtAsset)) | |
124 | + | split(getStringOrFail(factoryContract, keyPoolConfig(toString(iAmtAsset), toString(iPriceAsset))), SEP) | |
113 | 125 | } | |
114 | 126 | ||
115 | 127 | ||
116 | - | func getFactoryConfig () = split( | |
128 | + | func getFactoryConfig () = split(getStringOrFail(factoryContract, keyFactoryConfig()), SEP) | |
117 | 129 | ||
118 | 130 | ||
119 | - | func getPoolLiquidity (amountAssetInternalId,priceAssetInternalId) = { | |
120 | - | let currentPoolLiquidityValue = getString(this, keyPoolLiquidity(amountAssetInternalId, priceAssetInternalId)) | |
121 | - | if (isDefined(currentPoolLiquidityValue)) | |
122 | - | then split(value(currentPoolLiquidityValue), SEP) | |
123 | - | else ["", "0", "0", "0"] | |
124 | - | } | |
131 | + | func dataPutActionInfo (inAmtAssetAmt,inPriceAssetAmt,outLpAmt,price,slippageTolerancePassedByUser,slippageToleranceReal,txHeight,txTimestamp,slipageAmtAssetAmt,slipagePriceAssetAmt) = makeString(["%d%d%d%d%d%d%d%d%d", toString(inAmtAssetAmt), toString(inPriceAssetAmt), toString(outLpAmt), toString(price), toString(slippageTolerancePassedByUser), toString(slippageToleranceReal), toString(txHeight), toString(txTimestamp), toString(slipageAmtAssetAmt), toString(slipagePriceAssetAmt)], SEP) | |
125 | 132 | ||
126 | 133 | ||
127 | - | func | |
134 | + | func dataGetActionInfo (outAmtAssetAmt,outPriceAssetAmt,inLpAmt,price,txHeight,txTimestamp) = makeString(["%d%d%d%d%d%d", toString(outAmtAssetAmt), toString(outPriceAssetAmt), toString(inLpAmt), toString(price), toString(txHeight), toString(txTimestamp)], SEP) | |
128 | 135 | ||
129 | 136 | ||
130 | - | func dataPutActionInfo (inAmountAssetAmt,inPriceAssetAmt,outLpAmt,price,slippageTolerancePassedByUser,slippageToleranceReal,txHeight,txTimestamp) = makeString(["%d%d%d%d%d%d%d", toString(inAmountAssetAmt), toString(inPriceAssetAmt), toString(outLpAmt), toString(price), toString(slippageTolerancePassedByUser), toString(slippageToleranceReal), toString(txHeight), toString(txTimestamp)], SEP) | |
131 | - | ||
132 | - | ||
133 | - | func dataGetActionInfo (outAmountAssetAmt,outPriceAssetAmt,inLpAmt,price,txHeight,txTimestamp) = makeString(["%d%d%d%d%d%d", toString(outAmountAssetAmt), toString(outPriceAssetAmt), toString(inLpAmt), toString(price), toString(txHeight), toString(txTimestamp)], SEP) | |
134 | - | ||
135 | - | ||
136 | - | func privateCastToScaleOf (amount,resultScale,currentScale) = fraction(amount, resultScale, currentScale) | |
137 | - | ||
138 | - | ||
139 | - | func privateBigIntCastToScaleOf (amount,resultScale,currentScale) = fraction(amount, resultScale, currentScale) | |
140 | - | ||
141 | - | ||
142 | - | func calculatePrice (priceAssetLockedAmt,amountAssetlLockedAmt) = fraction(priceAssetLockedAmt, scale8, amountAssetlLockedAmt) | |
143 | - | ||
144 | - | ||
145 | - | func calculatePriceBigInt (priceAssetLockedAmt,amountAssetlLockedAmt) = toInt(fraction(priceAssetLockedAmt, scale8BigInt, amountAssetlLockedAmt)) | |
146 | - | ||
147 | - | ||
148 | - | func privateCalculatePrice (amoutAssetDecimals,priceAssetDecimals,amountAssetAmt,priceAssetAmt) = { | |
149 | - | let amountAssetAmtCasted = privateCastToScaleOf(amountAssetAmt, scale8, amoutAssetDecimals) | |
150 | - | let priceAssetAmtCasted = privateCastToScaleOf(priceAssetAmt, scale8, priceAssetDecimals) | |
151 | - | calculatePrice(priceAssetAmtCasted, amountAssetAmtCasted) | |
152 | - | } | |
153 | - | ||
154 | - | ||
155 | - | func privateGetAccountBalance (assetId) = if ((assetId == "WAVES")) | |
137 | + | func getAccBalance (assetId) = if ((assetId == "WAVES")) | |
156 | 138 | then wavesBalance(this).available | |
157 | 139 | else assetBalance(this, fromBase58String(assetId)) | |
158 | 140 | ||
159 | 141 | ||
160 | - | func calculatePrices (amountAssetAmt,priceAssetAmt,lpAmount) = { | |
161 | - | let poolConfigDataList = getPoolConfig() | |
162 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
163 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
164 | - | let poolPrice = privateCalculatePrice(amoutAssetDecimals, priceAssetDecimals, amountAssetAmt, priceAssetAmt) | |
165 | - | let amountAssetPoolLockedAmt = privateCastToScaleOf(amountAssetAmt, scale8, amoutAssetDecimals) | |
166 | - | let priceAssetPoolLockedAmt = privateCastToScaleOf(priceAssetAmt, scale8, priceAssetDecimals) | |
167 | - | let lpPriceInAmountAsset = calculatePrice(amountAssetPoolLockedAmt, lpAmount) | |
168 | - | let lpPriceInPriceAsset = calculatePrice(priceAssetPoolLockedAmt, lpAmount) | |
169 | - | [poolPrice, lpPriceInAmountAsset, lpPriceInPriceAsset] | |
142 | + | func calcPriceBigInt (priceAssetLockedAmt,amtAssetlLockedAmt,resultScale) = fraction(priceAssetLockedAmt, resultScale, amtAssetlLockedAmt) | |
143 | + | ||
144 | + | ||
145 | + | func privateCalcPrice (amtAssetDcm,priceAssetDcm,amtAssetAmt,priceAssetAmt) = { | |
146 | + | let amtAssetAmtX18 = toX18(amtAssetAmt, amtAssetDcm) | |
147 | + | let priceAssetAmtX18 = toX18(priceAssetAmt, priceAssetDcm) | |
148 | + | toInt(calcPriceBigInt(priceAssetAmtX18, amtAssetAmtX18, scale8BigInt)) | |
170 | 149 | } | |
171 | 150 | ||
172 | 151 | ||
173 | - | func privateEstimateGetOperation (paymentLpAssetId,paymentLpAmount,userAddress) = { | |
152 | + | func calcPrices (amtAssetAmt,priceAssetAmt,lpAmt) = { | |
153 | + | let cfg = getPoolConfig() | |
154 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
155 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
156 | + | let poolPrice = privateCalcPrice(amtAssetDcm, priceAssetDcm, amtAssetAmt, priceAssetAmt) | |
157 | + | let amtAssetAmtX18 = toX18(amtAssetAmt, amtAssetDcm) | |
158 | + | let priceAssetAmtX18 = toX18(priceAssetAmt, priceAssetDcm) | |
159 | + | let lpAmtX18 = toX18(lpAmt, scale8) | |
160 | + | let lpPriceInAmtAsset = toInt(calcPriceBigInt(amtAssetAmtX18, lpAmtX18, scale8BigInt)) | |
161 | + | let lpPriceInPriceAsset = toInt(calcPriceBigInt(priceAssetAmtX18, lpAmtX18, scale8BigInt)) | |
162 | + | [poolPrice, lpPriceInAmtAsset, lpPriceInPriceAsset] | |
163 | + | } | |
164 | + | ||
165 | + | ||
166 | + | func estimateGetOperation (paymentLpAssetId,paymentLpAmt,userAddress) = { | |
174 | 167 | let poolConfigList = getPoolConfig() | |
175 | 168 | let lpAssetId = poolConfigList[idxPoolLPAssetId] | |
176 | - | let | |
169 | + | let amtAssetId = poolConfigList[idxAmtAssetId] | |
177 | 170 | let priceAssetId = poolConfigList[idxPriceAssetId] | |
178 | - | let | |
179 | - | let | |
180 | - | let | |
181 | - | let | |
171 | + | let iAmtAssetId = poolConfigList[idxIAmtAssetId] | |
172 | + | let iPriceAssetId = poolConfigList[idxIPriceAssetId] | |
173 | + | let amtAssetDcm = parseIntValue(poolConfigList[idxAmtAssetDcm]) | |
174 | + | let priceAssetDcm = parseIntValue(poolConfigList[idxPriceAssetDcm]) | |
182 | 175 | let poolStatus = poolConfigList[idxPoolStatus] | |
183 | - | let poolLiquidityList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
184 | - | let poolLPBalance = parseIntValue(poolLiquidityList[idxPoolLPAssetAmt]) | |
185 | - | let poolAmountAssetBalance = parseIntValue(poolLiquidityList[idxPoolAmountAssetAmt]) | |
186 | - | let poolPriceAssetBalance = parseIntValue(poolLiquidityList[idxPoolPriceAssetAmt]) | |
176 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(fromBase58String(lpAssetId)), (("Asset " + lpAssetId) + " doesn't exist")).quantity | |
187 | 177 | if ((lpAssetId != paymentLpAssetId)) | |
188 | 178 | then throw("Invalid asset passed.") | |
189 | 179 | else { | |
190 | - | let amountAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(privateGetAccountBalance(amountAssetId)), scale18, toBigInt(amoutAssetDecimals)) | |
191 | - | let priceAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(privateGetAccountBalance(priceAssetId)), scale18, toBigInt(priceAssetDecimals)) | |
192 | - | let currentPrice = calculatePriceBigInt(priceAssetPoolLockedAmt, amountAssetPoolLockedAmt) | |
193 | - | let paymentLpAmountBuf = privateBigIntCastToScaleOf(toBigInt(paymentLpAmount), scale18, scale8BigInt) | |
194 | - | let poolLPBalanceBuf = privateBigIntCastToScaleOf(toBigInt(poolLPBalance), scale18, scale8BigInt) | |
195 | - | let outAmountAssetAmt = fraction(amountAssetPoolLockedAmt, paymentLpAmountBuf, poolLPBalanceBuf) | |
196 | - | let outPriceAssetAmt = fraction(priceAssetPoolLockedAmt, paymentLpAmountBuf, poolLPBalanceBuf) | |
197 | - | let outAmountAssetAmtFinal = toInt(privateBigIntCastToScaleOf(outAmountAssetAmt, toBigInt(amoutAssetDecimals), scale18)) | |
198 | - | let outPriceAssetAmtFinal = toInt(privateBigIntCastToScaleOf(outPriceAssetAmt, toBigInt(priceAssetDecimals), scale18)) | |
199 | - | $Tuple11(outAmountAssetAmtFinal, outPriceAssetAmtFinal, amountAssetInternalId, priceAssetInternalId, amountAssetId, priceAssetId, poolAmountAssetBalance, poolPriceAssetBalance, poolLPBalance, currentPrice, poolStatus) | |
180 | + | let amtAssetBalance = getAccBalance(amtAssetId) | |
181 | + | let amtAssetBalanceX18 = toX18(amtAssetBalance, amtAssetDcm) | |
182 | + | let priceAssetBalance = getAccBalance(priceAssetId) | |
183 | + | let priceAssetBalanceX18 = toX18(priceAssetBalance, priceAssetDcm) | |
184 | + | let curPrice = toInt(calcPriceBigInt(priceAssetBalanceX18, amtAssetBalanceX18, scale8BigInt)) | |
185 | + | let paymentLpAmtX18 = toX18(paymentLpAmt, scale8) | |
186 | + | let poolLPBalanceX18 = toX18(poolLPBalance, scale8) | |
187 | + | let outAmtAssetAmt = fraction(amtAssetBalanceX18, paymentLpAmtX18, poolLPBalanceX18) | |
188 | + | let outPriceAssetAmt = fraction(priceAssetBalanceX18, paymentLpAmtX18, poolLPBalanceX18) | |
189 | + | let outAmtAssetAmtFinal = fromX18(outAmtAssetAmt, amtAssetDcm) | |
190 | + | let outPriceAssetAmtFinal = fromX18(outPriceAssetAmt, priceAssetDcm) | |
191 | + | $Tuple11(outAmtAssetAmtFinal, outPriceAssetAmtFinal, iAmtAssetId, iPriceAssetId, amtAssetId, priceAssetId, amtAssetBalance, priceAssetBalance, poolLPBalance, curPrice, poolStatus) | |
200 | 192 | } | |
201 | 193 | } | |
202 | 194 | ||
203 | 195 | ||
204 | - | func privateEstimatePutOperation (slippageTolerance,inAmountAssetAmt,inAmountAssetId,inPriceAssetAmt,inPriceAssetId,userAddress,isEvaluate) = { | |
205 | - | let poolConfigDataList = getPoolConfig() | |
206 | - | let lpAssetId = fromBase58String(poolConfigDataList[idxPoolLPAssetId]) | |
207 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
208 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
209 | - | let amountAssetInternalId = poolConfigDataList[idxAmountAssetInternalId] | |
210 | - | let priceAssetInternalId = poolConfigDataList[idxPriceAssetInternalId] | |
211 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
212 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
213 | - | let poolStatus = poolConfigDataList[idxPoolStatus] | |
214 | - | let poolLiquidityDataList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
215 | - | let poolAmountAssetBalance = parseIntValue(poolLiquidityDataList[idxPoolAmountAssetAmt]) | |
216 | - | let poolPriceAssetBalance = parseIntValue(poolLiquidityDataList[idxPoolPriceAssetAmt]) | |
217 | - | let poolLPBalance = parseIntValue(poolLiquidityDataList[idxPoolLPAssetAmt]) | |
218 | - | if (if ((amountAssetId != inAmountAssetId)) | |
196 | + | func estimatePutOperation (slippageTolerance,inAmtAssetAmt,inAmtAssetId,inPriceAssetAmt,inPriceAssetId,userAddress,isEvaluate) = { | |
197 | + | let cfg = getPoolConfig() | |
198 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
199 | + | let amtAssetId = cfg[idxAmtAssetId] | |
200 | + | let priceAssetId = cfg[idxPriceAssetId] | |
201 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
202 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
203 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
204 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
205 | + | let poolStatus = cfg[idxPoolStatus] | |
206 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
207 | + | if (if ((amtAssetId != inAmtAssetId)) | |
219 | 208 | then true | |
220 | 209 | else (priceAssetId != inPriceAssetId)) | |
221 | - | then throw("Invalid | |
210 | + | then throw("Invalid amt or price asset passed.") | |
222 | 211 | else { | |
223 | - | let accountAmountAssetBalance = if (isEvaluate) | |
224 | - | then privateGetAccountBalance(amountAssetId) | |
225 | - | else (privateGetAccountBalance(amountAssetId) - inAmountAssetAmt) | |
226 | - | let accountPriceAssetBalance = if (isEvaluate) | |
227 | - | then privateGetAccountBalance(priceAssetId) | |
228 | - | else (privateGetAccountBalance(priceAssetId) - inPriceAssetAmt) | |
229 | - | let amoutAssetDecimalsBigInt = toBigInt(amoutAssetDecimals) | |
230 | - | let priceAssetDecimalsBigInt = toBigInt(priceAssetDecimals) | |
231 | - | let inAmountAssetAmtCalculated = privateBigIntCastToScaleOf(toBigInt(inAmountAssetAmt), scale18, amoutAssetDecimalsBigInt) | |
232 | - | let inPriceAssetAmtCalculated = privateBigIntCastToScaleOf(toBigInt(inPriceAssetAmt), scale18, priceAssetDecimalsBigInt) | |
233 | - | let userPrice = calculatePriceBigInt(inPriceAssetAmtCalculated, inAmountAssetAmtCalculated) | |
234 | - | let amountAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(accountAmountAssetBalance), scale18, amoutAssetDecimalsBigInt) | |
235 | - | let priceAssetPoolLockedAmt = privateBigIntCastToScaleOf(toBigInt(accountPriceAssetBalance), scale18, priceAssetDecimalsBigInt) | |
236 | - | let curentPrice = if ((poolLPBalance == 0)) | |
212 | + | let accAmtAssetBalance = if (isEvaluate) | |
213 | + | then getAccBalance(amtAssetId) | |
214 | + | else (getAccBalance(amtAssetId) - inAmtAssetAmt) | |
215 | + | let accPriceAssetBalance = if (isEvaluate) | |
216 | + | then getAccBalance(priceAssetId) | |
217 | + | else (getAccBalance(priceAssetId) - inPriceAssetAmt) | |
218 | + | let inAmtAssetAmtX18 = toX18(inAmtAssetAmt, amtAssetDcm) | |
219 | + | let inPriceAssetAmtX18 = toX18(inPriceAssetAmt, priceAssetDcm) | |
220 | + | let userPrice = toInt(calcPriceBigInt(inPriceAssetAmtX18, inAmtAssetAmtX18, scale8BigInt)) | |
221 | + | let accAmtAssetBalanceX18 = toX18(accAmtAssetBalance, amtAssetDcm) | |
222 | + | let accPriceAssetBalanceX18 = toX18(accPriceAssetBalance, priceAssetDcm) | |
223 | + | let curPrice = if ((poolLPBalance == 0)) | |
237 | 224 | then 0 | |
238 | - | else | |
239 | - | let slippage = if (( | |
225 | + | else toInt(calcPriceBigInt(accPriceAssetBalanceX18, accAmtAssetBalanceX18, scale8BigInt)) | |
226 | + | let slippage = if ((curPrice == 0)) | |
240 | 227 | then 0 | |
241 | - | else if (( | |
242 | - | then fraction(( | |
243 | - | else fraction((userPrice - | |
244 | - | if (if (( | |
245 | - | then ( | |
228 | + | else if ((curPrice > userPrice)) | |
229 | + | then fraction((curPrice - userPrice), scale8, curPrice) | |
230 | + | else fraction((userPrice - curPrice), scale8, curPrice) | |
231 | + | if (if ((curPrice != 0)) | |
232 | + | then (slippage > slippageTolerance) | |
246 | 233 | else false) | |
247 | - | then throw(((("Price slippage " + toString( | |
234 | + | then throw(((("Price slippage " + toString(slippage)) + " exceeded the passed limit of ") + toString(slippageTolerance))) | |
248 | 235 | else { | |
249 | - | let finalPrice = calculatePriceBigInt((priceAssetPoolLockedAmt + inPriceAssetAmtCalculated), (amountAssetPoolLockedAmt + inAmountAssetAmtCalculated)) | |
250 | - | let outLpAmount = if ((poolLPBalance == 0)) | |
236 | + | let res = if ((poolLPBalance == 0)) | |
251 | 237 | then { | |
252 | - | let lpResultBuf = pow(( | |
253 | - | | |
238 | + | let lpResultBuf = pow((inAmtAssetAmtX18 * inPriceAssetAmtX18), 0, toBigInt(5), 1, 0, DOWN) | |
239 | + | $Tuple4(fromX18(lpResultBuf, scale8), fromX18(inAmtAssetAmtX18, amtAssetDcm), fromX18(inPriceAssetAmtX18, priceAssetDcm), toInt(calcPriceBigInt((accPriceAssetBalanceX18 + inPriceAssetAmtX18), (accAmtAssetBalanceX18 + inAmtAssetAmtX18), scale8BigInt))) | |
254 | 240 | } | |
255 | 241 | else { | |
256 | - | let poolLPBalanceBuf = privateBigIntCastToScaleOf(toBigInt(poolLPBalance), scale18, scale8BigInt) | |
257 | - | let lpAmtByAmountAsset = toInt(privateBigIntCastToScaleOf(fraction(poolLPBalanceBuf, inAmountAssetAmtCalculated, amountAssetPoolLockedAmt), scale8BigInt, scale18)) | |
258 | - | let lpAmtByPriceAsset = toInt(privateBigIntCastToScaleOf(fraction(poolLPBalanceBuf, inPriceAssetAmtCalculated, priceAssetPoolLockedAmt), scale8BigInt, scale18)) | |
259 | - | if ((lpAmtByPriceAsset > lpAmtByAmountAsset)) | |
260 | - | then lpAmtByAmountAsset | |
261 | - | else lpAmtByPriceAsset | |
242 | + | let poolLPBalanceX18 = toX18(poolLPBalance, scale8) | |
243 | + | let curPriceX18 = toX18(curPrice, scale8) | |
244 | + | let pEqByA = fraction(curPriceX18, inAmtAssetAmtX18, scale18) | |
245 | + | let resAssetsToPut = if ((pEqByA > inPriceAssetAmtX18)) | |
246 | + | then $Tuple2(fraction(inPriceAssetAmtX18, scale18, curPriceX18), inPriceAssetAmtX18) | |
247 | + | else $Tuple2(inAmtAssetAmtX18, pEqByA) | |
248 | + | let lpResult = fromX18(fraction(poolLPBalanceX18, resAssetsToPut._2, accPriceAssetBalanceX18), scale8) | |
249 | + | $Tuple4(lpResult, fromX18(resAssetsToPut._1, amtAssetDcm), fromX18(resAssetsToPut._2, priceAssetDcm), toInt(calcPriceBigInt((accPriceAssetBalanceX18 + resAssetsToPut._2), (accAmtAssetBalanceX18 + resAssetsToPut._1), scale8BigInt))) | |
262 | 250 | } | |
263 | - | if ((0 >= | |
251 | + | if ((0 >= res._1)) | |
264 | 252 | then throw("Invalid calculations. LP calculated is less than zero.") | |
265 | - | else $ | |
253 | + | else $Tuple12(res._1, res._4, accAmtAssetBalance, accPriceAssetBalance, poolLPBalance, iAmtAssetId, iPriceAssetId, lpAssetId, slippage, poolStatus, res._2, res._3) | |
266 | 254 | } | |
267 | 255 | } | |
268 | 256 | } | |
269 | 257 | ||
270 | 258 | ||
271 | 259 | func validateMatcherOrderAllowed (order) = { | |
272 | - | let poolConfigDataList = getPoolConfig() | |
273 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
274 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
275 | - | let poolStatus = parseIntValue(poolConfigDataList[idxPoolStatus]) | |
276 | - | let matcherPublicKeyStr = poolConfigDataList[idxMatcherPublicKey] | |
277 | - | let amountAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
278 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
279 | - | let accountAmountAssetBalance = privateGetAccountBalance(amountAssetId) | |
280 | - | let accountPriceAssetBalance = privateGetAccountBalance(priceAssetId) | |
281 | - | let currentPrice = if ((order.orderType == Buy)) | |
282 | - | then privateCalculatePrice(amountAssetDecimals, priceAssetDecimals, (accountAmountAssetBalance + order.amount), accountPriceAssetBalance) | |
283 | - | else privateCalculatePrice(amountAssetDecimals, priceAssetDecimals, (accountAmountAssetBalance + order.amount), accountPriceAssetBalance) | |
284 | - | let isPoolStatusValid = if (if (if (isGlobalShutdown()) | |
260 | + | let cfg = getPoolConfig() | |
261 | + | let amtAssetId = cfg[idxAmtAssetId] | |
262 | + | let priceAssetId = cfg[idxPriceAssetId] | |
263 | + | let poolStatus = parseIntValue(cfg[idxPoolStatus]) | |
264 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
265 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
266 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
267 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
268 | + | let curPrice = if ((order.orderType == Buy)) | |
269 | + | then privateCalcPrice(amtAssetDcm, priceAssetDcm, (accAmtAssetBalance + order.amount), accPriceAssetBalance) | |
270 | + | else privateCalcPrice(amtAssetDcm, priceAssetDcm, (accAmtAssetBalance - order.amount), accPriceAssetBalance) | |
271 | + | if (if (if (isGlobalShutdown()) | |
285 | 272 | then true | |
286 | 273 | else (poolStatus == PoolMatcherDisabled)) | |
287 | 274 | then true | |
288 | 275 | else (poolStatus == PoolShutdown)) | |
289 | - | then false | |
290 | - | else true | |
291 | - | let isValidMatcherSign = true | |
292 | - | let orderAmountAsset = order.assetPair.amountAsset | |
293 | - | let orderAmountAssetStr = if ((orderAmountAsset == unit)) | |
294 | - | then "WAVES" | |
295 | - | else toBase58String(value(orderAmountAsset)) | |
296 | - | let orderPriceAsset = order.assetPair.priceAsset | |
297 | - | let orderPriceAssetStr = if ((orderPriceAsset == unit)) | |
298 | - | then "WAVES" | |
299 | - | else toBase58String(value(orderPriceAsset)) | |
300 | - | let isValidAssetPair = if (if ((orderAmountAssetStr != amountAssetId)) | |
301 | - | then true | |
302 | - | else (orderPriceAssetStr != priceAssetId)) | |
303 | - | then false | |
304 | - | else true | |
305 | - | let orderPrice = order.price | |
306 | - | let priceDecimals = fraction(scale8, priceAssetDecimals, amountAssetDecimals) | |
307 | - | let castedOrderPrice = privateCastToScaleOf(orderPrice, scale8, priceDecimals) | |
308 | - | let isOrderPriceValid = if ((order.orderType == Buy)) | |
309 | - | then (currentPrice >= castedOrderPrice) | |
310 | - | else (castedOrderPrice >= currentPrice) | |
311 | - | if (if (if (isValidAssetPair) | |
312 | - | then isValidMatcherSign | |
313 | - | else false) | |
314 | - | then isPoolStatusValid | |
315 | - | else false) | |
316 | - | then isOrderPriceValid | |
317 | - | else false | |
276 | + | then throw("Exchange operations disabled") | |
277 | + | else { | |
278 | + | let orderAmtAsset = order.assetPair.amountAsset | |
279 | + | let orderAmtAssetStr = if ((orderAmtAsset == unit)) | |
280 | + | then "WAVES" | |
281 | + | else toBase58String(value(orderAmtAsset)) | |
282 | + | let orderPriceAsset = order.assetPair.priceAsset | |
283 | + | let orderPriceAssetStr = if ((orderPriceAsset == unit)) | |
284 | + | then "WAVES" | |
285 | + | else toBase58String(value(orderPriceAsset)) | |
286 | + | if (if ((orderAmtAssetStr != amtAssetId)) | |
287 | + | then true | |
288 | + | else (orderPriceAssetStr != priceAssetId)) | |
289 | + | then throw("Wrong order assets.") | |
290 | + | else { | |
291 | + | let orderPrice = order.price | |
292 | + | let priceDcm = fraction(scale8, priceAssetDcm, amtAssetDcm) | |
293 | + | let castedOrderPrice = toScale(orderPrice, scale8, priceDcm) | |
294 | + | let isOrderPriceValid = if ((order.orderType == Buy)) | |
295 | + | then (curPrice >= castedOrderPrice) | |
296 | + | else (castedOrderPrice >= curPrice) | |
297 | + | if (!(isOrderPriceValid)) | |
298 | + | then throw(((("Order price leads to K decrease. castedOrderPrice=" + toString(castedOrderPrice)) + " curPrice=") + toString(curPrice))) | |
299 | + | else true | |
300 | + | } | |
301 | + | } | |
318 | 302 | } | |
319 | 303 | ||
320 | 304 | ||
321 | - | @Callable(i) | |
322 | - | func constructor (factoryContract) = if ((i.caller != this)) | |
323 | - | then throw("permissions denied") | |
324 | - | else [StringEntry(keyFactoryContract(), factoryContract)] | |
305 | + | func commonGet (i) = { | |
306 | + | let pmtAmtAsset = value(i.payments[0]) | |
307 | + | let pmtAssetId = value(pmtAmtAsset.assetId) | |
308 | + | let pmtAssetAmt = pmtAmtAsset.amount | |
309 | + | let res = estimateGetOperation(toBase58String(pmtAssetId), pmtAssetAmt, toString(i.caller)) | |
310 | + | let outAmtAssetAmt = res._1 | |
311 | + | let outPriceAssetAmt = res._2 | |
312 | + | let iAmtAssetId = res._3 | |
313 | + | let iPriceAssetId = res._4 | |
314 | + | let amtAssetId = res._5 | |
315 | + | let priceAssetId = res._6 | |
316 | + | let poolAmtAssetBalance = res._7 | |
317 | + | let poolPriceAssetBalance = res._8 | |
318 | + | let poolLPBalance = res._9 | |
319 | + | let curPrice = res._10 | |
320 | + | let poolStatus = parseIntValue(res._11) | |
321 | + | if (if (isGlobalShutdown()) | |
322 | + | then true | |
323 | + | else (poolStatus == PoolShutdown)) | |
324 | + | then throw(("Get operation is blocked by admin. Status = " + toString(poolStatus))) | |
325 | + | else $Tuple5(outAmtAssetAmt, outPriceAssetAmt, pmtAssetAmt, pmtAssetId, [ScriptTransfer(i.caller, outAmtAssetAmt, if ((amtAssetId == "WAVES")) | |
326 | + | then unit | |
327 | + | else fromBase58String(amtAssetId)), ScriptTransfer(i.caller, outPriceAssetAmt, if ((priceAssetId == "WAVES")) | |
328 | + | then unit | |
329 | + | else fromBase58String(priceAssetId)), StringEntry(keyGetActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataGetActionInfo(outAmtAssetAmt, outPriceAssetAmt, pmtAssetAmt, curPrice, height, lastBlock.timestamp)), StringEntry(keyPriceLast(), ("%s__" + toString(curPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(curPrice)))]) | |
330 | + | } | |
325 | 331 | ||
326 | 332 | ||
327 | - | ||
328 | - | @Callable(i) | |
329 | - | func put (slippageTolerance,shouldAutoStake) = { | |
330 | - | let factoryConfigList = getFactoryConfig() | |
331 | - | let stakingContract = valueOrErrorMessage(addressFromString(factoryConfigList[idxFactoryStakingContract]), "Error. Incorrect staking address.") | |
332 | - | let pmtAmountAsset = value(i.payments[0]) | |
333 | - | let inAmountAssetAmt = pmtAmountAsset.amount | |
334 | - | let inAmountAssetId = if (!(isDefined(pmtAmountAsset.assetId))) | |
333 | + | func commonPut (i,slippageTolerance,emitLp) = { | |
334 | + | let pmtAmtAsset = value(i.payments[0]) | |
335 | + | let inAmtAssetAmt = pmtAmtAsset.amount | |
336 | + | let inAmtAssetId = if (!(isDefined(pmtAmtAsset.assetId))) | |
335 | 337 | then fromBase58String("WAVES") | |
336 | - | else value( | |
338 | + | else value(pmtAmtAsset.assetId) | |
337 | 339 | let pmtPriceAsset = value(i.payments[1]) | |
338 | 340 | let inPriceAssetAmt = pmtPriceAsset.amount | |
339 | 341 | let inPriceAssetId = if (!(isDefined(pmtPriceAsset.assetId))) | |
340 | 342 | then fromBase58String("WAVES") | |
341 | 343 | else value(pmtPriceAsset.assetId) | |
342 | - | let estimatedPutResults = privateEstimatePutOperation(slippageTolerance, inAmountAssetAmt, toBase58String(inAmountAssetId), inPriceAssetAmt, toBase58String(inPriceAssetId), toString(i.caller), false) | |
343 | - | let outLpAmount = estimatedPutResults._1 | |
344 | - | let curentPrice = estimatedPutResults._2 | |
345 | - | let poolAmountAssetBalance = estimatedPutResults._3 | |
346 | - | let poolPriceAssetBalance = estimatedPutResults._4 | |
347 | - | let poolLPBalance = estimatedPutResults._5 | |
348 | - | let amountAssetInternalId = estimatedPutResults._6 | |
349 | - | let priceAssetInternalId = estimatedPutResults._7 | |
350 | - | let lpAssetId = estimatedPutResults._8 | |
351 | - | let slippageCalculated = estimatedPutResults._9 | |
352 | - | let poolStatus = parseIntValue(estimatedPutResults._10) | |
344 | + | let estPut = estimatePutOperation(slippageTolerance, inAmtAssetAmt, toBase58String(inAmtAssetId), inPriceAssetAmt, toBase58String(inPriceAssetId), toString(i.caller), false) | |
345 | + | let outLpAmt = estPut._1 | |
346 | + | let curPrice = estPut._2 | |
347 | + | let poolAmtAssetBalance = estPut._3 | |
348 | + | let poolPriceAssetBalance = estPut._4 | |
349 | + | let poolLPBalance = estPut._5 | |
350 | + | let iAmtAssetId = estPut._6 | |
351 | + | let iPriceAssetId = estPut._7 | |
352 | + | let lpAssetId = estPut._8 | |
353 | + | let slippageCalc = estPut._9 | |
354 | + | let poolStatus = parseIntValue(estPut._10) | |
355 | + | let realAmtAssetIn = estPut._11 | |
356 | + | let realPriceAssetIn = estPut._12 | |
353 | 357 | if (if (if (isGlobalShutdown()) | |
354 | 358 | then true | |
355 | 359 | else (poolStatus == PoolPutDisabled)) | |
356 | 360 | then true | |
357 | 361 | else (poolStatus == PoolShutdown)) | |
358 | 362 | then throw(("Put operation is blocked by admin. Status = " + toString(poolStatus))) | |
359 | 363 | else { | |
360 | - | let emitLPAssetOnFactory = invoke(factoryContract, "emit", [outLpAmount], nil) | |
361 | - | if ((emitLPAssetOnFactory == emitLPAssetOnFactory)) | |
362 | - | then if ((emitLPAssetOnFactory != "success")) | |
363 | - | then throw("LP asset reissue failed during emit call on factory.") | |
364 | - | else { | |
365 | - | let results = [StringEntry(keyPriceLast(), ("%s__" + toString(curentPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(curentPrice))), StringEntry(keyPutActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataPutActionInfo(inAmountAssetAmt, inPriceAssetAmt, outLpAmount, curentPrice, slippageTolerance, slippageCalculated, height, lastBlock.timestamp)), StringEntry(keyPoolLiquidity(amountAssetInternalId, priceAssetInternalId), dataPoolLiquidity((poolAmountAssetBalance + inAmountAssetAmt), (poolPriceAssetBalance + inPriceAssetAmt), (poolLPBalance + outLpAmount)))] | |
366 | - | if (shouldAutoStake) | |
367 | - | then { | |
368 | - | let stake = invoke(stakingContract, "stake", nil, [AttachedPayment(lpAssetId, outLpAmount)]) | |
369 | - | if ((stake == stake)) | |
370 | - | then results | |
371 | - | else throw("Strict value is not equal to itself.") | |
372 | - | } | |
373 | - | else (results :+ ScriptTransfer(i.caller, outLpAmount, lpAssetId)) | |
374 | - | } | |
364 | + | let lpAmtToEmit = if (!(emitLp)) | |
365 | + | then 0 | |
366 | + | else outLpAmt | |
367 | + | let diffInAmtAsset = (inAmtAssetAmt - realAmtAssetIn) | |
368 | + | let diffInPriceAsset = (inPriceAssetAmt - realPriceAssetIn) | |
369 | + | let dataStateWrite = [StringEntry(keyPriceLast(), ("%s__" + toString(curPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(curPrice))), StringEntry(keyPutActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataPutActionInfo(realAmtAssetIn, realPriceAssetIn, lpAmtToEmit, curPrice, slippageTolerance, slippageCalc, height, lastBlock.timestamp, diffInAmtAsset, diffInPriceAsset))] | |
370 | + | $Tuple9(lpAmtToEmit, lpAssetId, dataStateWrite, inAmtAssetAmt, realAmtAssetIn, inPriceAssetAmt, realPriceAssetIn, pmtAmtAsset.assetId, pmtPriceAsset.assetId) | |
371 | + | } | |
372 | + | } | |
373 | + | ||
374 | + | ||
375 | + | @Callable(i) | |
376 | + | func constructor (factoryContract,managerPublicKey) = if ((i.caller != this)) | |
377 | + | then throw("permissions denied") | |
378 | + | else [StringEntry(keyFactoryContract(), factoryContract), StringEntry(keyManagerPublicKey(), managerPublicKey)] | |
379 | + | ||
380 | + | ||
381 | + | ||
382 | + | @Callable(i) | |
383 | + | func put (slippageTolerance,shouldAutoStake) = { | |
384 | + | let factoryCfg = getFactoryConfig() | |
385 | + | let stakingContract = valueOrErrorMessage(addressFromString(factoryCfg[idxFactoryStakingContract]), "Error. Incorrect staking address.") | |
386 | + | let slippageContract = valueOrErrorMessage(addressFromString(factoryCfg[idxFactorySlippageContract]), "Error. Incorrect slippage contract address.") | |
387 | + | let actions = commonPut(i, slippageTolerance, true) | |
388 | + | let amtToEmit = actions._1 | |
389 | + | let lpAssetId = actions._2 | |
390 | + | let state = actions._3 | |
391 | + | let inAmt = actions._4 | |
392 | + | let realInAmt = actions._5 | |
393 | + | let inPrice = actions._6 | |
394 | + | let realInPrice = actions._7 | |
395 | + | let amId = actions._8 | |
396 | + | let prId = actions._9 | |
397 | + | let diffInAmtAsset = (inAmt - realInAmt) | |
398 | + | let diffInPriceAsset = (inPrice - realInPrice) | |
399 | + | let emitInv = invoke(factoryContract, "emit", [amtToEmit], nil) | |
400 | + | if ((emitInv == emitInv)) | |
401 | + | then { | |
402 | + | let slippageAInv = if ((inAmt > realInAmt)) | |
403 | + | then invoke(slippageContract, "put", nil, [AttachedPayment(amId, diffInAmtAsset)]) | |
404 | + | else nil | |
405 | + | if ((slippageAInv == slippageAInv)) | |
406 | + | then { | |
407 | + | let slippagePInv = if ((inPrice > realInPrice)) | |
408 | + | then invoke(slippageContract, "put", nil, [AttachedPayment(prId, diffInPriceAsset)]) | |
409 | + | else nil | |
410 | + | if ((slippagePInv == slippagePInv)) | |
411 | + | then { | |
412 | + | let lpStakeInv = if (shouldAutoStake) | |
413 | + | then invoke(stakingContract, "stake", nil, [AttachedPayment(lpAssetId, amtToEmit)]) | |
414 | + | else nil | |
415 | + | if ((lpStakeInv == lpStakeInv)) | |
416 | + | then (state :+ ScriptTransfer(i.caller, amtToEmit, lpAssetId)) | |
417 | + | else throw("Strict value is not equal to itself.") | |
418 | + | } | |
419 | + | else throw("Strict value is not equal to itself.") | |
420 | + | } | |
375 | 421 | else throw("Strict value is not equal to itself.") | |
376 | 422 | } | |
423 | + | else throw("Strict value is not equal to itself.") | |
424 | + | } | |
425 | + | ||
426 | + | ||
427 | + | ||
428 | + | @Callable(i) | |
429 | + | func putForFree () = { | |
430 | + | let actions = commonPut(i, 0, false) | |
431 | + | actions._3 | |
377 | 432 | } | |
378 | 433 | ||
379 | 434 | ||
380 | 435 | ||
381 | 436 | @Callable(i) | |
382 | 437 | func get () = { | |
383 | - | let pmtAmountAsset = value(i.payments[0]) | |
384 | - | let pmtAssetId = value(pmtAmountAsset.assetId) | |
385 | - | let pmtAssetAmount = pmtAmountAsset.amount | |
386 | - | let results = privateEstimateGetOperation(toBase58String(pmtAssetId), pmtAssetAmount, toString(i.caller)) | |
387 | - | let outAmountAssetAmt = results._1 | |
388 | - | let outPriceAssetAmt = results._2 | |
389 | - | let amountAssetInternalId = results._3 | |
390 | - | let priceAssetInternalId = results._4 | |
391 | - | let amountAssetId = results._5 | |
392 | - | let priceAssetId = results._6 | |
393 | - | let poolAmountAssetBalance = results._7 | |
394 | - | let poolPriceAssetBalance = results._8 | |
395 | - | let poolLPBalance = results._9 | |
396 | - | let currentPrice = results._10 | |
397 | - | let poolStatus = parseIntValue(results._11) | |
398 | - | if (if (isGlobalShutdown()) | |
399 | - | then true | |
400 | - | else (poolStatus == PoolShutdown)) | |
401 | - | then throw(("Get operation is blocked by admin. Status = " + toString(poolStatus))) | |
402 | - | else { | |
403 | - | let burnLPAssetOnFactory = invoke(factoryContract, "burn", [pmtAssetAmount], [AttachedPayment(pmtAssetId, pmtAssetAmount)]) | |
404 | - | if ((burnLPAssetOnFactory == burnLPAssetOnFactory)) | |
405 | - | then if ((burnLPAssetOnFactory != "success")) | |
406 | - | then throw("LP asset burn failed during emit call on factory.") | |
407 | - | else [ScriptTransfer(i.caller, outAmountAssetAmt, if ((amountAssetId == "WAVES")) | |
408 | - | then unit | |
409 | - | else fromBase58String(amountAssetId)), ScriptTransfer(i.caller, outPriceAssetAmt, fromBase58String(priceAssetId)), StringEntry(keyPoolLiquidity(amountAssetInternalId, priceAssetInternalId), dataPoolLiquidity((poolAmountAssetBalance - outAmountAssetAmt), (poolPriceAssetBalance - outPriceAssetAmt), (poolLPBalance - pmtAssetAmount))), StringEntry(keyGetActionByUser(toString(i.caller), toBase58String(i.transactionId)), dataGetActionInfo(outAmountAssetAmt, outPriceAssetAmt, pmtAssetAmount, currentPrice, height, lastBlock.timestamp)), StringEntry(keyPriceLast(), ("%s__" + toString(currentPrice))), StringEntry(keyPriceHistory(height, lastBlock.timestamp), ("%s__" + toString(currentPrice)))] | |
410 | - | else throw("Strict value is not equal to itself.") | |
411 | - | } | |
438 | + | let res = commonGet(i) | |
439 | + | let outAmtAssetAmt = res._1 | |
440 | + | let outPriceAssetAmt = res._2 | |
441 | + | let pmtAssetAmt = res._3 | |
442 | + | let pmtAssetId = res._4 | |
443 | + | let actions = res._5 | |
444 | + | let burnLPAssetOnFactory = invoke(factoryContract, "burn", [pmtAssetAmt], [AttachedPayment(pmtAssetId, pmtAssetAmt)]) | |
445 | + | if ((burnLPAssetOnFactory == burnLPAssetOnFactory)) | |
446 | + | then actions | |
447 | + | else throw("Strict value is not equal to itself.") | |
412 | 448 | } | |
413 | 449 | ||
414 | 450 | ||
415 | 451 | ||
416 | 452 | @Callable(i) | |
417 | - | func activate (amountAssetStr,priceAssetStr) = if ((toString(i.caller) != toString(factoryContract))) | |
453 | + | func getNoLess (noLessThenAmtAsset,noLessThenPriceAsset) = { | |
454 | + | let res = commonGet(i) | |
455 | + | let outAmtAssetAmt = res._1 | |
456 | + | let outPriceAssetAmt = res._2 | |
457 | + | let pmtAssetAmt = res._3 | |
458 | + | let pmtAssetId = res._4 | |
459 | + | let actions = res._5 | |
460 | + | if ((noLessThenAmtAsset > outAmtAssetAmt)) | |
461 | + | then throw(((("noLessThenAmtAsset failed: " + toString(outAmtAssetAmt)) + " < ") + toString(noLessThenAmtAsset))) | |
462 | + | else if ((noLessThenPriceAsset > outPriceAssetAmt)) | |
463 | + | then throw(((("noLessThenPriceAsset failed: " + toString(outPriceAssetAmt)) + " < ") + toString(noLessThenPriceAsset))) | |
464 | + | else { | |
465 | + | let burnLPAssetOnFactory = invoke(factoryContract, "burn", [pmtAssetAmt], [AttachedPayment(pmtAssetId, pmtAssetAmt)]) | |
466 | + | if ((burnLPAssetOnFactory == burnLPAssetOnFactory)) | |
467 | + | then actions | |
468 | + | else throw("Strict value is not equal to itself.") | |
469 | + | } | |
470 | + | } | |
471 | + | ||
472 | + | ||
473 | + | ||
474 | + | @Callable(i) | |
475 | + | func activate (amtAssetStr,priceAssetStr) = if ((toString(i.caller) != toString(factoryContract))) | |
418 | 476 | then throw("permissions denied") | |
419 | - | else $Tuple2([StringEntry( | |
477 | + | else $Tuple2([StringEntry(keyAmtAsset(), amtAssetStr), StringEntry(keyPriceAsset(), priceAssetStr)], "success") | |
420 | 478 | ||
421 | 479 | ||
422 | 480 | ||
423 | 481 | @Callable(i) | |
424 | 482 | func statsREADONLY () = { | |
425 | - | let poolConfigDataList = getPoolConfig() | |
426 | - | let lpAssetId = fromBase58String(poolConfigDataList[idxPoolLPAssetId]) | |
427 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
428 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
429 | - | let amountAssetInternalId = poolConfigDataList[idxAmountAssetInternalId] | |
430 | - | let priceAssetInternalId = poolConfigDataList[idxPriceAssetInternalId] | |
431 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
432 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
433 | - | let poolLiquidityDataList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
434 | - | let poolLPBalance = parseIntValue(poolLiquidityDataList[idxPoolLPAssetAmt]) | |
435 | - | let accountAmountAssetBalance = privateGetAccountBalance(amountAssetId) | |
436 | - | let accountPriceAssetBalance = privateGetAccountBalance(priceAssetId) | |
437 | - | let pricesList = calculatePrices(accountAmountAssetBalance, accountPriceAssetBalance, poolLPBalance) | |
438 | - | let currentPrice = pricesList[0] | |
439 | - | let lpAmountAssetShare = pricesList[1] | |
483 | + | let cfg = getPoolConfig() | |
484 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
485 | + | let amtAssetId = cfg[idxAmtAssetId] | |
486 | + | let priceAssetId = cfg[idxPriceAssetId] | |
487 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
488 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
489 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
490 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
491 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
492 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
493 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
494 | + | let pricesList = calcPrices(accAmtAssetBalance, accPriceAssetBalance, poolLPBalance) | |
495 | + | let curPrice = pricesList[0] | |
496 | + | let lpAmtAssetShare = pricesList[1] | |
440 | 497 | let lpPriceAssetShare = pricesList[2] | |
441 | - | let poolWeight = | |
442 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString( | |
498 | + | let poolWeight = value(getInteger(factoryContract, keyPoolWeight(toString(this)))) | |
499 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString(accAmtAssetBalance), toString(accPriceAssetBalance), toString(poolLPBalance), toString(curPrice), toString(lpAmtAssetShare), toString(lpPriceAssetShare), toString(poolWeight)], SEP)) | |
443 | 500 | } | |
444 | 501 | ||
445 | 502 | ||
446 | 503 | ||
447 | 504 | @Callable(i) | |
448 | - | func | |
449 | - | let | |
450 | - | let lpAssetId = fromBase58String( | |
451 | - | let | |
452 | - | let priceAssetId = | |
453 | - | let | |
454 | - | let | |
455 | - | let | |
456 | - | let | |
457 | - | let poolStatus = | |
458 | - | let | |
459 | - | let | |
460 | - | let | |
461 | - | let | |
462 | - | let | |
463 | - | let | |
464 | - | | |
465 | - | | |
466 | - | | |
467 | - | let | |
468 | - | let inPriceAssetAmt = | |
469 | - | let estimatedPutResults = | |
470 | - | let | |
471 | - | let | |
472 | - | let | |
505 | + | func evaluatePutByAmtAssetREADONLY (slippageTolerance,inAmtAssetAmt,userAddress) = { | |
506 | + | let cfg = getPoolConfig() | |
507 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
508 | + | let amtAssetId = cfg[idxAmtAssetId] | |
509 | + | let priceAssetId = cfg[idxPriceAssetId] | |
510 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
511 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
512 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
513 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
514 | + | let poolStatus = cfg[idxPoolStatus] | |
515 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
516 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
517 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
518 | + | let amtAssetAmtX18 = toX18(accAmtAssetBalance, amtAssetDcm) | |
519 | + | let priceAssetAmtX18 = toX18(accPriceAssetBalance, priceAssetDcm) | |
520 | + | let curPriceX18 = if ((poolLPBalance == 0)) | |
521 | + | then toBigInt(0) | |
522 | + | else calcPriceBigInt(priceAssetAmtX18, amtAssetAmtX18, scale18) | |
523 | + | let inAmtAssetAmtX18 = toX18(inAmtAssetAmt, amtAssetDcm) | |
524 | + | let inPriceAssetAmtX18 = fraction(inAmtAssetAmtX18, curPriceX18, scale18) | |
525 | + | let inPriceAssetAmt = fromX18(inPriceAssetAmtX18, priceAssetDcm) | |
526 | + | let estimatedPutResults = estimatePutOperation(slippageTolerance, inAmtAssetAmt, amtAssetId, inPriceAssetAmt, priceAssetId, userAddress, true) | |
527 | + | let outLpAmtCalc = estimatedPutResults._1 | |
528 | + | let curPriceCalc = estimatedPutResults._2 | |
529 | + | let poolAmtAssetBalanceCalc = estimatedPutResults._3 | |
473 | 530 | let poolPriceAssetBalanceCAlc = estimatedPutResults._4 | |
474 | 531 | let poolLPBalanceCalc = estimatedPutResults._5 | |
475 | 532 | let poolStatusCalc = parseIntValue(estimatedPutResults._10) | |
476 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString( | |
533 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString(outLpAmtCalc), toString(fromX18(curPriceX18, scale8)), toString(poolAmtAssetBalanceCalc), toString(poolPriceAssetBalanceCAlc), toString(poolLPBalanceCalc), toString(poolStatusCalc)], SEP)) | |
477 | 534 | } | |
478 | 535 | ||
479 | 536 | ||
480 | 537 | ||
481 | 538 | @Callable(i) | |
482 | 539 | func evaluatePutByPriceAssetREADONLY (slippageTolerance,inPriceAssetAmt,userAddress) = { | |
483 | - | let poolConfigDataList = getPoolConfig() | |
484 | - | let lpAssetId = fromBase58String(poolConfigDataList[idxPoolLPAssetId]) | |
485 | - | let amountAssetId = poolConfigDataList[idxAmountAssetId] | |
486 | - | let priceAssetId = poolConfigDataList[idxPriceAssetId] | |
487 | - | let amountAssetInternalId = poolConfigDataList[idxAmountAssetInternalId] | |
488 | - | let priceAssetInternalId = poolConfigDataList[idxPriceAssetInternalId] | |
489 | - | let amoutAssetDecimals = parseIntValue(poolConfigDataList[idxAmountAssetDecimals]) | |
490 | - | let priceAssetDecimals = parseIntValue(poolConfigDataList[idxPriceAssetDecimals]) | |
491 | - | let poolStatus = poolConfigDataList[idxPoolStatus] | |
492 | - | let poolLiquidityDataList = getPoolLiquidity(amountAssetInternalId, priceAssetInternalId) | |
493 | - | let poolLPBalance = parseIntValue(poolLiquidityDataList[idxPoolLPAssetAmt]) | |
494 | - | let accountAmountAssetBalance = privateGetAccountBalance(amountAssetId) | |
495 | - | let accountPriceAssetBalance = privateGetAccountBalance(priceAssetId) | |
496 | - | let accountAmountAssetBalanceCasted = privateCastToScaleOf(accountAmountAssetBalance, scale8, amoutAssetDecimals) | |
497 | - | let accountPriceAssetBalanceCasted = privateCastToScaleOf(accountPriceAssetBalance, scale8, priceAssetDecimals) | |
498 | - | let inPriceAssetAmtCalculatedTuple = privateCastToScaleOf(inPriceAssetAmt, scale8, priceAssetDecimals) | |
499 | - | let curentPrice = if ((poolLPBalance == 0)) | |
500 | - | then 0 | |
501 | - | else calculatePrice(accountPriceAssetBalanceCasted, accountAmountAssetBalanceCasted) | |
502 | - | let inAmountAssetAmtCalculated = fraction(inPriceAssetAmtCalculatedTuple, scale8, curentPrice) | |
503 | - | let inAmountAssetAmt = privateCastToScaleOf(inAmountAssetAmtCalculated, amoutAssetDecimals, scale8) | |
504 | - | let estimatedPutResults = privateEstimatePutOperation(slippageTolerance, inAmountAssetAmt, amountAssetId, inPriceAssetAmt, priceAssetId, userAddress, true) | |
505 | - | let outLpAmountCalc = estimatedPutResults._1 | |
506 | - | let curentPriceCalc = estimatedPutResults._2 | |
507 | - | let poolAmountAssetBalanceCalc = estimatedPutResults._3 | |
540 | + | let cfg = getPoolConfig() | |
541 | + | let lpAssetId = fromBase58String(cfg[idxPoolLPAssetId]) | |
542 | + | let amtAssetId = cfg[idxAmtAssetId] | |
543 | + | let priceAssetId = cfg[idxPriceAssetId] | |
544 | + | let iAmtAssetId = cfg[idxIAmtAssetId] | |
545 | + | let iPriceAssetId = cfg[idxIPriceAssetId] | |
546 | + | let amtAssetDcm = parseIntValue(cfg[idxAmtAssetDcm]) | |
547 | + | let priceAssetDcm = parseIntValue(cfg[idxPriceAssetDcm]) | |
548 | + | let poolStatus = cfg[idxPoolStatus] | |
549 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
550 | + | let accAmtAssetBalance = getAccBalance(amtAssetId) | |
551 | + | let accPriceAssetBalance = getAccBalance(priceAssetId) | |
552 | + | let amtAssetAmtX18 = toX18(accAmtAssetBalance, amtAssetDcm) | |
553 | + | let priceAssetAmtX18 = toX18(accPriceAssetBalance, priceAssetDcm) | |
554 | + | let curPriceX18 = if ((poolLPBalance == 0)) | |
555 | + | then toBigInt(0) | |
556 | + | else calcPriceBigInt(priceAssetAmtX18, amtAssetAmtX18, scale18) | |
557 | + | let inPriceAssetAmtX18 = toX18(inPriceAssetAmt, priceAssetDcm) | |
558 | + | let inAmtAssetAmtX18 = fraction(inPriceAssetAmtX18, scale18, curPriceX18) | |
559 | + | let inAmtAssetAmt = fromX18(inAmtAssetAmtX18, amtAssetDcm) | |
560 | + | let estimatedPutResults = estimatePutOperation(slippageTolerance, inAmtAssetAmt, amtAssetId, inPriceAssetAmt, priceAssetId, userAddress, true) | |
561 | + | let outLpAmtCalc = estimatedPutResults._1 | |
562 | + | let curPriceCalc = estimatedPutResults._2 | |
563 | + | let poolAmtAssetBalanceCalc = estimatedPutResults._3 | |
508 | 564 | let poolPriceAssetBalanceCAlc = estimatedPutResults._4 | |
509 | 565 | let poolLPBalanceCalc = estimatedPutResults._5 | |
510 | 566 | let poolStatusCalc = parseIntValue(estimatedPutResults._10) | |
511 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString( | |
567 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d", toString(outLpAmtCalc), toString(fromX18(curPriceX18, scale8)), toString(poolAmtAssetBalanceCalc), toString(poolPriceAssetBalanceCAlc), toString(poolLPBalanceCalc), toString(poolStatusCalc)], SEP)) | |
512 | 568 | } | |
513 | 569 | ||
514 | 570 | ||
515 | 571 | ||
516 | 572 | @Callable(i) | |
517 | - | func evaluateGetREADONLY (paymentLpAssetId, | |
518 | - | let | |
519 | - | let | |
520 | - | let outPriceAssetAmt = | |
521 | - | let | |
522 | - | let poolPriceAssetBalance = | |
523 | - | let poolLPBalance = | |
524 | - | let | |
525 | - | let poolStatus = parseIntValue( | |
526 | - | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString( | |
573 | + | func evaluateGetREADONLY (paymentLpAssetId,paymentLpAmt,userAddress) = { | |
574 | + | let res = estimateGetOperation(paymentLpAssetId, paymentLpAmt, userAddress) | |
575 | + | let outAmtAssetAmt = res._1 | |
576 | + | let outPriceAssetAmt = res._2 | |
577 | + | let poolAmtAssetBalance = res._7 | |
578 | + | let poolPriceAssetBalance = res._8 | |
579 | + | let poolLPBalance = res._9 | |
580 | + | let curPrice = res._10 | |
581 | + | let poolStatus = parseIntValue(res._11) | |
582 | + | $Tuple2(nil, makeString(["%d%d%d%d%d%d%d", toString(outAmtAssetAmt), toString(outPriceAssetAmt), toString(poolAmtAssetBalance), toString(poolPriceAssetBalance), toString(poolLPBalance), toString(curPrice), toString(poolStatus)], SEP)) | |
527 | 583 | } | |
528 | - | ||
529 | - | ||
530 | - | ||
531 | - | @Callable(i) | |
532 | - | func hashREADONLY () = $Tuple2(nil, toBase64String(value(scriptHash(this)))) | |
533 | 584 | ||
534 | 585 | ||
535 | 586 | @Verifier(tx) | |
536 | 587 | func verify () = match tx { | |
537 | 588 | case order: Order => | |
538 | - | sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
589 | + | if (validateMatcherOrderAllowed(order)) | |
590 | + | then sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
591 | + | else false | |
539 | 592 | case _ => | |
540 | - | sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
593 | + | let managerPublic = valueOrElse(getString(this, keyManagerPublicKey()), EMPTY) | |
594 | + | if ((managerPublic == EMPTY)) | |
595 | + | then sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
596 | + | else sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(managerPublic)) | |
541 | 597 | } | |
542 | 598 |
github/deemru/w8io/169f3d6 129.38 ms ◑