tx · 79JHwhhAP413ryz4rGAd9AqvdRoFdAK7oM9kG8AwNPAp 3P8MoPnsaurofk1VyhsdAFkeQ6ijpJYXCpW: -0.01000000 Waves 2021.11.25 21:17 [2871836] smart account 3P8MoPnsaurofk1VyhsdAFkeQ6ijpJYXCpW > SELF 0.00000000 Waves
{ "type": 13, "id": "79JHwhhAP413ryz4rGAd9AqvdRoFdAK7oM9kG8AwNPAp", "fee": 1000000, "feeAssetId": null, "timestamp": 1637864276800, "version": 1, "sender": "3P8MoPnsaurofk1VyhsdAFkeQ6ijpJYXCpW", "senderPublicKey": "6tusy8LfPEh2eoAsxHwZZn6cw8DBMGTHAce3gqLXwQxC", "proofs": [ "Q8Vio3kaqR8hC8ExDpXDvhCuWpLNuYaQnQxSr1iZ5KrPYyXUCmwwa5VwMnSzFiShk17Bz8dYQa7rh2Ctgs9MX1j" ], "script": "base64:AAIFAAAAAAAAACwIAhIDCgEIEgMKAQgSBAoCCAgSAwoBCBIHCgUBAQEICBIAEgMKAQgSAwoBCAAAAD8AAAAABlNDQUxFOAAAAAAAAAAACAAAAAAFTVVMVDgAAAAAAAX14QAAAAAAB1NDQUxFMTgAAAAAAAAAABIAAAAABk1VTFQxOAkAATYAAAABAA3gtrOnZAAAAAAAAANTRVACAAAAAl9fAAAAAA5pZHhQb29sQWRkcmVzcwAAAAAAAAAAAQAAAAANaWR4UG9vbFN0YXR1cwAAAAAAAAAAAgAAAAAQaWR4UG9vbExQQXNzZXRJZAAAAAAAAAAAAwAAAAANaWR4QW10QXNzZXRJZAAAAAAAAAAABAAAAAAPaWR4UHJpY2VBc3NldElkAAAAAAAAAAAFAAAAAA5pZHhBbXRBc3NldERjbQAAAAAAAAAABgAAAAAQaWR4UHJpY2VBc3NldERjbQAAAAAAAAAABwAAAAAOaWR4SUFtdEFzc2V0SWQAAAAAAAAAAAgAAAAAEGlkeElQcmljZUFzc2V0SWQAAAAAAAAAAAkAAAAADWlkeExQQXNzZXREY20AAAAAAAAAAAoBAAAAD2dldFN0cmluZ09yRmFpbAAAAAEAAAADa2V5CQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQiAAAAAQUAAAADa2V5CQABLAAAAAIJAAEsAAAAAgIAAAAPbWFuZGF0b3J5IHRoaXMuBQAAAANrZXkCAAAADyBpcyBub3QgZGVmaW5lZAEAAAAYZ2V0U3RyaW5nQnlBZGRyZXNzT3JGYWlsAAAAAgAAAAdhZGRyZXNzAAAAA2tleQkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEHQAAAAIFAAAAB2FkZHJlc3MFAAAAA2tleQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACAgAAAAptYW5kYXRvcnkgCQAEJQAAAAEFAAAAB2FkZHJlc3MCAAAAAS4FAAAAA2tleQIAAAAPIGlzIG5vdCBkZWZpbmVkAQAAAAxnZXRJbnRPclplcm8AAAACAAAAB2FkZHJlc3MAAAADa2V5CQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAAB2FkZHJlc3MFAAAAA2tleQAAAAAAAAAAAAEAAAAMZ2V0SW50T3JGYWlsAAAAAgAAAAdhZGRyZXNzAAAAA2tleQkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEGgAAAAIFAAAAB2FkZHJlc3MFAAAAA2tleQkAASwAAAACCQABLAAAAAICAAAAD21hbmRhdG9yeSB0aGlzLgUAAAADa2V5AgAAAA8gaXMgbm90IGRlZmluZWQBAAAAEWtleUZhY3RvcnlBZGRyZXNzAAAAAAIAAAAcJXMlc19fY29uZmlnX19mYWN0b3J5QWRkcmVzcwAAAAAYSWR4RmFjdG9yeUNmZ1N0YWtpbmdEYXBwAAAAAAAAAAABAAAAABlJZHhGYWN0b3J5Q2ZnQm9vc3RpbmdEYXBwAAAAAAAAAAACAAAAABRJZHhGYWN0b3J5Q2ZnSWRvRGFwcAAAAAAAAAAAAwAAAAAVSWR4RmFjdG9yeUNmZ1RlYW1EYXBwAAAAAAAAAAAEAAAAABlJZHhGYWN0b3J5Q2ZnRW1pc3Npb25EYXBwAAAAAAAAAAAFAAAAABVJZHhGYWN0b3J5Q2ZnUmVzdERhcHAAAAAAAAAAAAYAAAAAGUlkeEZhY3RvcnlDZmdTbGlwcGFnZURhcHAAAAAAAAAAAAcBAAAADWtleUZhY3RvcnlDZmcAAAAAAgAAABElc19fZmFjdG9yeUNvbmZpZwEAAAAaa2V5RmFjdG9yeUxwMkFzc2V0c01hcHBpbmcAAAABAAAACmxwQXNzZXRTdHIJAAS5AAAAAgkABEwAAAACAgAAAAYlcyVzJXMJAARMAAAAAgUAAAAKbHBBc3NldFN0cgkABEwAAAACAgAAAB5tYXBwaW5nc19fbHBBc3NldDJQb29sQ29udHJhY3QFAAAAA25pbAUAAAADU0VQAQAAABBrZXlGYWN0b3J5THBMaXN0AAAAAAIAAAAQJXNfX2xwVG9rZW5zTGlzdAEAAAAma2V5RmFjdG9yeUxwQXNzZXRUb1Bvb2xDb250cmFjdEFkZHJlc3MAAAABAAAACmxwQXNzZXRTdHIJAAS5AAAAAgkABEwAAAACAgAAAAYlcyVzJXMJAARMAAAAAgUAAAAKbHBBc3NldFN0cgkABEwAAAACAgAAAB5tYXBwaW5nc19fbHBBc3NldDJQb29sQ29udHJhY3QFAAAAA25pbAUAAAADU0VQAQAAABRrZXlGYWN0b3J5UG9vbFdlaWdodAAAAAEAAAAPY29udHJhY3RBZGRyZXNzCQAEuQAAAAIJAARMAAAAAgIAAAAEJXMlcwkABEwAAAACAgAAAApwb29sV2VpZ2h0CQAETAAAAAIFAAAAD2NvbnRyYWN0QWRkcmVzcwUAAAADbmlsBQAAAANTRVABAAAAGHJlYWRGYWN0b3J5QWRkcmVzc09yRmFpbAAAAAAJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEJAQAAAA9nZXRTdHJpbmdPckZhaWwAAAABCQEAAAARa2V5RmFjdG9yeUFkZHJlc3MAAAAAAQAAAApyZWFkTHBMaXN0AAAAAAkABLUAAAACCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHQAAAAIJAQAAABhyZWFkRmFjdG9yeUFkZHJlc3NPckZhaWwAAAAACQEAAAAQa2V5RmFjdG9yeUxwTGlzdAAAAAACAAAAAAUAAAADU0VQAQAAABRyZWFkRmFjdG9yeUNmZ09yRmFpbAAAAAEAAAAHZmFjdG9yeQkABLUAAAACCQEAAAAYZ2V0U3RyaW5nQnlBZGRyZXNzT3JGYWlsAAAAAgUAAAAHZmFjdG9yeQkBAAAADWtleUZhY3RvcnlDZmcAAAAABQAAAANTRVABAAAAGGdldEJvb3N0aW5nQWRkcmVzc09yRmFpbAAAAAEAAAAKZmFjdG9yeUNmZwkBAAAAEUBleHRyTmF0aXZlKDEwNjIpAAAAAQkAAZEAAAACBQAAAApmYWN0b3J5Q2ZnBQAAABlJZHhGYWN0b3J5Q2ZnQm9vc3RpbmdEYXBwAQAAABhnZXRFbWlzc2lvbkFkZHJlc3NPckZhaWwAAAABAAAACmZhY3RvcnlDZmcJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEJAAGRAAAAAgUAAAAKZmFjdG9yeUNmZwUAAAAZSWR4RmFjdG9yeUNmZ0VtaXNzaW9uRGFwcAEAAAAXZ2V0U3Rha2luZ0FkZHJlc3NPckZhaWwAAAABAAAACmZhY3RvcnlDZmcJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEJAAGRAAAAAgUAAAAKZmFjdG9yeUNmZwUAAAAYSWR4RmFjdG9yeUNmZ1N0YWtpbmdEYXBwAQAAAAtrZXlCb29zdENmZwAAAAACAAAACiVzX19jb25maWcBAAAAH2tleUJvb3N0aW5nTG9ja1BhcmFtVG90YWxBbW91bnQAAAAAAgAAAB4lcyVzX19zdGF0c19fYWN0aXZlVG90YWxMb2NrZWQBAAAAKGtleUJvb3N0aW5nU3RhdHNMb2Nrc0R1cmF0aW9uU3VtSW5CbG9ja3MAAAAAAgAAACUlcyVzX19zdGF0c19fbG9ja3NEdXJhdGlvblN1bUluQmxvY2tzAQAAABprZXlCb29zdGluZ1N0YXRzTG9ja3NDb3VudAAAAAACAAAAFyVzJXNfX3N0YXRzX19sb2Nrc0NvdW50AQAAABprZXlCb29zdGluZ1N0YXRzVXNlcnNDb3VudAAAAAACAAAAHSVzJXNfX3N0YXRzX19hY3RpdmVVc2Vyc0NvdW50AQAAABJrZXlVc2VyMk51bU1hcHBpbmcAAAABAAAAC3VzZXJBZGRyZXNzCQAEuQAAAAIJAARMAAAAAgIAAAAZJXMlcyVzX19tYXBwaW5nX191c2VyMm51bQkABEwAAAACBQAAAAt1c2VyQWRkcmVzcwUAAAADbmlsBQAAAANTRVABAAAAEmtleU51bTJVc2VyTWFwcGluZwAAAAEAAAADbnVtCQAEuQAAAAIJAARMAAAAAgIAAAAZJXMlcyVzX19tYXBwaW5nX19udW0ydXNlcgkABEwAAAACBQAAAANudW0FAAAAA25pbAUAAAADU0VQAQAAABZrZXlMb2NrUGFyYW1Vc2VyQW1vdW50AAAAAQAAAAd1c2VyTnVtCQAEuQAAAAIJAARMAAAAAgIAAAAWJXMlZCVzX19wYXJhbUJ5VXNlck51bQkABEwAAAACBQAAAAd1c2VyTnVtCQAETAAAAAICAAAABmFtb3VudAUAAAADbmlsBQAAAANTRVABAAAAFmtleUxvY2tQYXJhbVN0YXJ0QmxvY2sAAAABAAAAB3VzZXJOdW0JAAS5AAAAAgkABEwAAAACAgAAABYlcyVkJXNfX3BhcmFtQnlVc2VyTnVtCQAETAAAAAIFAAAAB3VzZXJOdW0JAARMAAAAAgIAAAAFc3RhcnQFAAAAA25pbAUAAAADU0VQAQAAABRrZXlMb2NrUGFyYW1EdXJhdGlvbgAAAAEAAAAHdXNlck51bQkABLkAAAACCQAETAAAAAICAAAAFiVzJWQlc19fcGFyYW1CeVVzZXJOdW0JAARMAAAAAgUAAAAHdXNlck51bQkABEwAAAACAgAAAAhkdXJhdGlvbgUAAAADbmlsBQAAAANTRVABAAAADWtleUxvY2tQYXJhbUsAAAABAAAAB3VzZXJOdW0JAAS5AAAAAgkABEwAAAACAgAAABYlcyVkJXNfX3BhcmFtQnlVc2VyTnVtCQAETAAAAAIFAAAAB3VzZXJOdW0JAARMAAAAAgIAAAABawUAAAADbmlsBQAAAANTRVABAAAADWtleUxvY2tQYXJhbUIAAAABAAAAB3VzZXJOdW0JAAS5AAAAAgkABEwAAAACAgAAABYlcyVkJXNfX3BhcmFtQnlVc2VyTnVtCQAETAAAAAIFAAAAB3VzZXJOdW0JAARMAAAAAgIAAAABYgUAAAADbmlsBQAAAANTRVABAAAAFWtleUxvY2tQYXJhbUJ5UGVyaW9kSwAAAAIAAAAHdXNlck51bQAAAAZwZXJpb2QJAAS5AAAAAgkABEwAAAACAgAAABclcyVkJXMlZF9fcGFyYW1CeVBlcmlvZAkABEwAAAACBQAAAAd1c2VyTnVtCQAETAAAAAICAAAAAWsJAARMAAAAAgUAAAAGcGVyaW9kBQAAAANuaWwFAAAAA1NFUAEAAAAVa2V5TG9ja1BhcmFtQnlQZXJpb2RCAAAAAgAAAAd1c2VyTnVtAAAABnBlcmlvZAkABLkAAAACCQAETAAAAAICAAAAFyVzJWQlcyVkX19wYXJhbUJ5UGVyaW9kCQAETAAAAAIFAAAAB3VzZXJOdW0JAARMAAAAAgIAAAABYgkABEwAAAACBQAAAAZwZXJpb2QFAAAAA25pbAUAAAADU0VQAQAAAA9rZXlTdGFrZWRCeVVzZXIAAAACAAAADnVzZXJBZGRyZXNzU3RyAAAADGxwQXNzZXRJZFN0cgkABLkAAAACCQAETAAAAAICAAAADiVzJXMlc19fc3Rha2VkCQAETAAAAAIFAAAADnVzZXJBZGRyZXNzU3RyCQAETAAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAADbmlsBQAAAANTRVABAAAADmtleVN0YWtlZFRvdGFsAAAAAQAAAAxscEFzc2V0SWRTdHIJAAEsAAAAAgIAAAAXJXMlcyVzX19zdGFrZWRfX3RvdGFsX18FAAAADGxwQXNzZXRJZFN0cgEAAAAQa2V5Q2xhaW1lZEJ5VXNlcgAAAAIAAAAMbHBBc3NldElkU3RyAAAADnVzZXJBZGRyZXNzU3RyCQAEuQAAAAIJAARMAAAAAgIAAAAPJXMlcyVzX19jbGFpbWVkCQAETAAAAAIFAAAADnVzZXJBZGRyZXNzU3RyCQAETAAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAADbmlsBQAAAANTRVABAAAAGWtleUNsYWltZWRCeVVzZXJNaW5SZXdhcmQAAAACAAAADGxwQXNzZXRJZFN0cgAAAA51c2VyQWRkcmVzc1N0cgkABLkAAAACCQAETAAAAAICAAAAGCVzJXMlc19fY2xhaW1lZE1pblJld2FyZAkABEwAAAACBQAAAA51c2VyQWRkcmVzc1N0cgkABEwAAAACBQAAAAxscEFzc2V0SWRTdHIFAAAAA25pbAUAAAADU0VQAQAAABtrZXlDbGFpbWVkQnlVc2VyQm9vc3RSZXdhcmQAAAACAAAADGxwQXNzZXRJZFN0cgAAAA51c2VyQWRkcmVzc1N0cgkABLkAAAACCQAETAAAAAICAAAAGiVzJXMlc19fY2xhaW1lZEJvb3N0UmV3YXJkCQAETAAAAAIFAAAADnVzZXJBZGRyZXNzU3RyCQAETAAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAADbmlsBQAAAANTRVABAAAAHmtleUVtaXNzaW9uUmF0ZVBlckJsb2NrQ3VycmVudAAAAAACAAAAGyVzJXNfX3JhdGVQZXJCbG9ja19fY3VycmVudAEAAAAha2V5RW1pc3Npb25SYXRlUGVyQmxvY2tNYXhDdXJyZW50AAAAAAIAAAAeJXMlc19fcmF0ZVBlckJsb2NrTWF4X19jdXJyZW50AQAAABVrZXlFbWlzc2lvblN0YXJ0QmxvY2sAAAAAAgAAABolcyVzX19lbWlzc2lvbl9fc3RhcnRCbG9jawEAAAAba2V5RW1pc3Npb25EdXJhdGlvbkluQmxvY2tzAAAAAAIAAAAYJXMlc19fZW1pc3Npb25fX2R1cmF0aW9uAQAAABNrZXlFbWlzc2lvbkVuZEJsb2NrAAAAAAIAAAAYJXMlc19fZW1pc3Npb25fX2VuZEJsb2NrAQAAABlpbnRlcm5hbEN1cnJlbnRSZXdhcmRSYXRlAAAABAAAAA9mYWN0b3J5Q29udHJhY3QAAAAPc3Rha2luZ0NvbnRyYWN0AAAAEGVtaXNzaW9uQ29udHJhY3QAAAAJbHBBc3NldElkBAAAAA5wb29sQWRkcmVzc1N0cgkBAAAAGGdldFN0cmluZ0J5QWRkcmVzc09yRmFpbAAAAAIFAAAAD2ZhY3RvcnlDb250cmFjdAkBAAAAJmtleUZhY3RvcnlMcEFzc2V0VG9Qb29sQ29udHJhY3RBZGRyZXNzAAAAAQUAAAAJbHBBc3NldElkBAAAAA5wb29sV2VpZ2h0TXVsdAUAAAAFTVVMVDgEAAAACnBvb2xXZWlnaHQJAQAAABFAZXh0ck5hdGl2ZSgxMDUwKQAAAAIFAAAAD2ZhY3RvcnlDb250cmFjdAkBAAAAFGtleUZhY3RvcnlQb29sV2VpZ2h0AAAAAQUAAAAOcG9vbEFkZHJlc3NTdHIEAAAAEnd4RW1pc3Npb25QZXJCbG9jawkBAAAADGdldEludE9yRmFpbAAAAAIFAAAAEGVtaXNzaW9uQ29udHJhY3QJAQAAAB5rZXlFbWlzc2lvblJhdGVQZXJCbG9ja0N1cnJlbnQAAAAABAAAABV3eEVtaXNzaW9uUGVyQmxvY2tNYXgJAQAAAAxnZXRJbnRPckZhaWwAAAACBQAAABBlbWlzc2lvbkNvbnRyYWN0CQEAAAAha2V5RW1pc3Npb25SYXRlUGVyQmxvY2tNYXhDdXJyZW50AAAAAAQAAAAWcG9vbFd4RW1pc3Npb25QZXJCbG9jawkAAGsAAAADBQAAABJ3eEVtaXNzaW9uUGVyQmxvY2sFAAAACnBvb2xXZWlnaHQFAAAADnBvb2xXZWlnaHRNdWx0BAAAABlwb29sV3hFbWlzc2lvblBlckJsb2NrTWF4CQAAawAAAAMFAAAAFXd4RW1pc3Npb25QZXJCbG9ja01heAUAAAAKcG9vbFdlaWdodAUAAAAOcG9vbFdlaWdodE11bHQEAAAACW1heEZhY3RvcgkAAGgAAAACAAAAAAAAAAADBQAAAAVNVUxUOAQAAAANdG90YWxMcFN0YWtlZAkBAAAADGdldEludE9yWmVybwAAAAIFAAAAD3N0YWtpbmdDb250cmFjdAkBAAAADmtleVN0YWtlZFRvdGFsAAAAAQUAAAAJbHBBc3NldElkCQAETAAAAAIFAAAAFnBvb2xXeEVtaXNzaW9uUGVyQmxvY2sJAARMAAAAAgUAAAAJbWF4RmFjdG9yCQAETAAAAAIFAAAADXRvdGFsTHBTdGFrZWQFAAAAA25pbAAAAAgAAAABaQEAAAALY29uc3RydWN0b3IAAAABAAAADmZhY3RvcnlBZGRyZXNzAwkBAAAAAiE9AAAAAggFAAAAAWkAAAAGY2FsbGVyBQAAAAR0aGlzCQAAAgAAAAECAAAADm5vdCBhdXRob3JpemVkCQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABFrZXlGYWN0b3J5QWRkcmVzcwAAAAAFAAAADmZhY3RvcnlBZGRyZXNzBQAAAANuaWwAAAABaQEAAAAZY3VycmVudFJld2FyZFJhdGVSRUFET05MWQAAAAEAAAAJbHBBc3NldElkBAAAAA9mYWN0b3J5Q29udHJhY3QJAQAAABhyZWFkRmFjdG9yeUFkZHJlc3NPckZhaWwAAAAABAAAAApmYWN0b3J5Q2ZnCQEAAAAUcmVhZEZhY3RvcnlDZmdPckZhaWwAAAABBQAAAA9mYWN0b3J5Q29udHJhY3QEAAAAD3N0YWtpbmdDb250cmFjdAkBAAAAF2dldFN0YWtpbmdBZGRyZXNzT3JGYWlsAAAAAQUAAAAKZmFjdG9yeUNmZwQAAAAQZW1pc3Npb25Db250cmFjdAkBAAAAGGdldEVtaXNzaW9uQWRkcmVzc09yRmFpbAAAAAEFAAAACmZhY3RvcnlDZmcEAAAACnJld2FyZERhdGEJAQAAABlpbnRlcm5hbEN1cnJlbnRSZXdhcmRSYXRlAAAABAUAAAAPZmFjdG9yeUNvbnRyYWN0BQAAAA9zdGFraW5nQ29udHJhY3QFAAAAEGVtaXNzaW9uQ29udHJhY3QFAAAACWxwQXNzZXRJZAQAAAASd3hFbWlzc2lvblBlckJsb2NrCQABkQAAAAIFAAAACnJld2FyZERhdGEAAAAAAAAAAAAEAAAACW1heEZhY3RvcgkAAZEAAAACBQAAAApyZXdhcmREYXRhAAAAAAAAAAABBAAAAA10b3RhbExwU3Rha2VkCQABkQAAAAIFAAAACnJld2FyZERhdGEAAAAAAAAAAAIJAAUUAAAAAgUAAAADbmlsCQAEuQAAAAIJAARMAAAAAgIAAAAGJWQlZCVkCQAETAAAAAIJAAGkAAAAAQUAAAASd3hFbWlzc2lvblBlckJsb2NrCQAETAAAAAIJAAGkAAAAAQUAAAAJbWF4RmFjdG9yCQAETAAAAAIJAAGkAAAAAQUAAAANdG90YWxMcFN0YWtlZAUAAAADbmlsBQAAAANTRVAAAAABaQEAAAAdY3VycmVudFVzZXJSZXdhcmRSYXRlUkVBRE9OTFkAAAACAAAACWxwQXNzZXRJZAAAAAt1c2VyQWRkcmVzcwQAAAAPZmFjdG9yeUNvbnRyYWN0CQEAAAAYcmVhZEZhY3RvcnlBZGRyZXNzT3JGYWlsAAAAAAQAAAAKZmFjdG9yeUNmZwkBAAAAFHJlYWRGYWN0b3J5Q2ZnT3JGYWlsAAAAAQUAAAAPZmFjdG9yeUNvbnRyYWN0BAAAAA9zdGFraW5nQ29udHJhY3QJAQAAABdnZXRTdGFraW5nQWRkcmVzc09yRmFpbAAAAAEFAAAACmZhY3RvcnlDZmcEAAAAEGVtaXNzaW9uQ29udHJhY3QJAQAAABhnZXRFbWlzc2lvbkFkZHJlc3NPckZhaWwAAAABBQAAAApmYWN0b3J5Q2ZnBAAAAApyZXdhcmREYXRhCQEAAAAZaW50ZXJuYWxDdXJyZW50UmV3YXJkUmF0ZQAAAAQFAAAAD2ZhY3RvcnlDb250cmFjdAUAAAAPc3Rha2luZ0NvbnRyYWN0BQAAABBlbWlzc2lvbkNvbnRyYWN0BQAAAAlscEFzc2V0SWQEAAAAEnd4RW1pc3Npb25QZXJCbG9jawkAAZEAAAACBQAAAApyZXdhcmREYXRhAAAAAAAAAAAABAAAAAltYXhGYWN0b3IJAAGRAAAAAgUAAAAKcmV3YXJkRGF0YQAAAAAAAAAAAQQAAAANdG90YWxMcFN0YWtlZAkAAZEAAAACBQAAAApyZXdhcmREYXRhAAAAAAAAAAACBAAAAA5scFN0YWtlZEJ5VXNlcgkBAAAADGdldEludE9yWmVybwAAAAIFAAAAD3N0YWtpbmdDb250cmFjdAkBAAAAD2tleVN0YWtlZEJ5VXNlcgAAAAIFAAAAC3VzZXJBZGRyZXNzBQAAAAlscEFzc2V0SWQEAAAADWJvb3N0aW5nUG93ZXIJAABoAAAAAgAAAAAAAAAAAQUAAAAFTVVMVDgJAAUUAAAAAgUAAAADbmlsCQAEuQAAAAIJAARMAAAAAgIAAAAKJWQlZCVkJWQlZAkABEwAAAACCQABpAAAAAEFAAAAEnd4RW1pc3Npb25QZXJCbG9jawkABEwAAAACCQABpAAAAAEFAAAACW1heEZhY3RvcgkABEwAAAACCQABpAAAAAEFAAAADXRvdGFsTHBTdGFrZWQJAARMAAAAAgkAAaQAAAABBQAAAA5scFN0YWtlZEJ5VXNlcgkABEwAAAACCQABpAAAAAEFAAAADWJvb3N0aW5nUG93ZXIFAAAAA25pbAUAAAADU0VQAAAAAWkBAAAAFWNsYWltZWRSZXdhcmRSRUFET05MWQAAAAEAAAALdXNlckFkZHJlc3MEAAAAD2ZhY3RvcnlDb250cmFjdAkBAAAAGHJlYWRGYWN0b3J5QWRkcmVzc09yRmFpbAAAAAAEAAAACmZhY3RvcnlDZmcJAQAAABRyZWFkRmFjdG9yeUNmZ09yRmFpbAAAAAEFAAAAD2ZhY3RvcnlDb250cmFjdAQAAAAPc3Rha2luZ0NvbnRyYWN0CQEAAAAXZ2V0U3Rha2luZ0FkZHJlc3NPckZhaWwAAAABBQAAAApmYWN0b3J5Q2ZnBAAAAAZscExpc3QJAQAAAApyZWFkTHBMaXN0AAAAAAQAAAAGcHJlZml4AgAAAAolcyVkJWQlZCVzCgEAAAAbY2xhaW1lZFJld2FyZEJ5THBBZ2dyZWdhdG9yAAAAAgAAAAlyZXN1bHRTdHIAAAAGbmV4dExwBAAAABljbGFpbWVkQnlVc2VyTWluUmV3YXJkS0VZCQEAAAAZa2V5Q2xhaW1lZEJ5VXNlck1pblJld2FyZAAAAAIFAAAABm5leHRMcAUAAAALdXNlckFkZHJlc3MEAAAAG2NsYWltZWRCeVVzZXJCb29zdFJld2FyZEtFWQkBAAAAG2tleUNsYWltZWRCeVVzZXJCb29zdFJld2FyZAAAAAIFAAAABm5leHRMcAUAAAALdXNlckFkZHJlc3MEAAAAEG1pblJld2FyZENsYWltZWQJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAPc3Rha2luZ0NvbnRyYWN0BQAAABljbGFpbWVkQnlVc2VyTWluUmV3YXJkS0VZAAAAAAAAAAAABAAAABJib29zdFJld2FyZENsYWltZWQJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAPc3Rha2luZ0NvbnRyYWN0BQAAABtjbGFpbWVkQnlVc2VyQm9vc3RSZXdhcmRLRVkAAAAAAAAAAAAEAAAAC2dGZWVDbGFpbWVkAAAAAAAAAAAACQAEuQAAAAIJAARMAAAAAgkAASwAAAACBQAAAAZwcmVmaXgFAAAACXJlc3VsdFN0cgkABEwAAAACBQAAAAZuZXh0THAJAARMAAAAAgkAAaQAAAABBQAAABBtaW5SZXdhcmRDbGFpbWVkCQAETAAAAAIJAAGkAAAAAQUAAAASYm9vc3RSZXdhcmRDbGFpbWVkCQAETAAAAAIJAAGkAAAAAQUAAAALZ0ZlZUNsYWltZWQJAARMAAAAAgIAAAADZW5kBQAAAANuaWwFAAAAA1NFUAQAAAAGcmVzdWx0CgAAAAACJGwFAAAABmxwTGlzdAoAAAAAAiRzCQABkAAAAAEFAAAAAiRsCgAAAAAFJGFjYzACAAAAAiVzCgEAAAABMQAAAAIAAAACJGEAAAACJGkDCQAAZwAAAAIFAAAAAiRpBQAAAAIkcwUAAAACJGEJAQAAABtjbGFpbWVkUmV3YXJkQnlMcEFnZ3JlZ2F0b3IAAAACBQAAAAIkYQkAAZEAAAACBQAAAAIkbAUAAAACJGkKAQAAAAEyAAAAAgAAAAIkYQAAAAIkaQMJAABnAAAAAgUAAAACJGkFAAAAAiRzBQAAAAIkYQkAAAIAAAABAgAAABRMaXN0IHNpemUgZXhjZWVkcyAxMAkBAAAAATIAAAACCQEAAAABMQAAAAIJAQAAAAExAAAAAgkBAAAAATEAAAACCQEAAAABMQAAAAIJAQAAAAExAAAAAgkBAAAAATEAAAACCQEAAAABMQAAAAIJAQAAAAExAAAAAgkBAAAAATEAAAACCQEAAAABMQAAAAIFAAAABSRhY2MwAAAAAAAAAAAAAAAAAAAAAAABAAAAAAAAAAACAAAAAAAAAAADAAAAAAAAAAAEAAAAAAAAAAAFAAAAAAAAAAAGAAAAAAAAAAAHAAAAAAAAAAAIAAAAAAAAAAAJAAAAAAAAAAAKCQAFFAAAAAIFAAAAA25pbAkAASwAAAACCQABLAAAAAIFAAAABnJlc3VsdAUAAAADU0VQBQAAAAt1c2VyQWRkcmVzcwAAAAFpAQAAABFjYWxjQm9vc3RSRUFET05MWQAAAAUAAAANZGVsdGFXeEFtb3VudAAAABdkZWx0YUxvY2tQZXJpb2RJbkJsb2NrcwAAAA1kZWx0YUxwQW1vdW50AAAADGxwQXNzZXRJZE9wdAAAAA51c2VyQWRkcmVzc09wdAQAAAAPZmFjdG9yeUNvbnRyYWN0CQEAAAAYcmVhZEZhY3RvcnlBZGRyZXNzT3JGYWlsAAAAAAQAAAAKZmFjdG9yeUNmZwkBAAAAFHJlYWRGYWN0b3J5Q2ZnT3JGYWlsAAAAAQUAAAAPZmFjdG9yeUNvbnRyYWN0BAAAABBib29zdGluZ0NvbnRyYWN0CQEAAAAYZ2V0Qm9vc3RpbmdBZGRyZXNzT3JGYWlsAAAAAQUAAAAKZmFjdG9yeUNmZwQAAAAIRU1QVFlTVFICAAAABWVtcHR5BAAAABdtYXhMb2NrRHVyYXRpb25JbkJsb2NrcwkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIJAAS1AAAAAgkBAAAAGGdldFN0cmluZ0J5QWRkcmVzc09yRmFpbAAAAAIFAAAAEGJvb3N0aW5nQ29udHJhY3QJAQAAAAtrZXlCb29zdENmZwAAAAAFAAAAA1NFUAAAAAAAAAAABAQAAAAHdXNlck51bQkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABB0AAAACBQAAABBib29zdGluZ0NvbnRyYWN0CQEAAAASa2V5VXNlcjJOdW1NYXBwaW5nAAAAAQUAAAAOdXNlckFkZHJlc3NPcHQFAAAACEVNUFRZU1RSBAAAAAp1c2VyQW1vdW50CQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAAEGJvb3N0aW5nQ29udHJhY3QJAQAAABZrZXlMb2NrUGFyYW1Vc2VyQW1vdW50AAAAAQUAAAAHdXNlck51bQAAAAAAAAAAAAQAAAAJbG9ja1N0YXJ0CQEAAAALdmFsdWVPckVsc2UAAAACCQAEGgAAAAIFAAAAEGJvb3N0aW5nQ29udHJhY3QJAQAAABZrZXlMb2NrUGFyYW1TdGFydEJsb2NrAAAAAQUAAAAHdXNlck51bQUAAAAGaGVpZ2h0BAAAAAxsb2NrRHVyYXRpb24JAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQaAAAAAgUAAAAQYm9vc3RpbmdDb250cmFjdAkBAAAAFGtleUxvY2tQYXJhbUR1cmF0aW9uAAAAAQUAAAAHdXNlck51bQAAAAAAAAAAAAQAAAAHbG9ja0VuZAkAAGQAAAACBQAAAAlsb2NrU3RhcnQFAAAADGxvY2tEdXJhdGlvbgQAAAARcmVtYWluaW5nRHVyYXRpb24JAAGWAAAAAQkABEwAAAACCQAAZQAAAAIFAAAAB2xvY2tFbmQFAAAABmhlaWdodAkABEwAAAACAAAAAAAAAAAABQAAAANuaWwEAAAABWJvb3N0AwkBAAAAAiE9AAAAAgUAAAAOdXNlckFkZHJlc3NPcHQCAAAAAAkAAGgAAAACAAAAAAAAAAADBQAAAAVNVUxUOAkAAGgAAAACAAAAAAAAAAABBQAAAAVNVUxUOAQAAAAFU0NBTEUAAAAAAAAAA+gEAAAADXVzZXJBbW91bnROZXcJAABkAAAAAgUAAAAKdXNlckFtb3VudAUAAAANZGVsdGFXeEFtb3VudAQAAAAPbG9ja0R1cmF0aW9uTmV3CQABlwAAAAEJAARMAAAAAgkAAGQAAAACBQAAABFyZW1haW5pbmdEdXJhdGlvbgUAAAAXZGVsdGFMb2NrUGVyaW9kSW5CbG9ja3MJAARMAAAAAgUAAAAXbWF4TG9ja0R1cmF0aW9uSW5CbG9ja3MFAAAAA25pbAQAAAAHY29lZmZYOAkAAGsAAAADBQAAAA9sb2NrRHVyYXRpb25OZXcFAAAABU1VTFQ4BQAAABdtYXhMb2NrRHVyYXRpb25JbkJsb2NrcwQAAAANd3hBbW91bnRTdGFydAkAAGsAAAADBQAAAA11c2VyQW1vdW50TmV3BQAAAAdjb2VmZlg4BQAAAAVNVUxUOAQAAAANbG9ja0VuZEhlaWdodAkAAGQAAAACBQAAAAZoZWlnaHQFAAAAD2xvY2tEdXJhdGlvbk5ldwQAAAAMc2NhbGU4UGFyYW1LCQEAAAABLQAAAAEJAABrAAAAAwUAAAANd3hBbW91bnRTdGFydAUAAAAFU0NBTEUFAAAAD2xvY2tEdXJhdGlvbk5ldwQAAAAMc2NhbGU4UGFyYW1CCQAAaAAAAAIJAABrAAAAAwUAAAANd3hBbW91bnRTdGFydAUAAAAFU0NBTEUFAAAAD2xvY2tEdXJhdGlvbk5ldwUAAAANbG9ja0VuZEhlaWdodAQAAAAJZ1d4QW1vdW50CQAAaQAAAAIJAABkAAAAAgkAAGgAAAACBQAAAAxzY2FsZThQYXJhbUsFAAAABmhlaWdodAUAAAAMc2NhbGU4UGFyYW1CBQAAAAVTQ0FMRQkABRQAAAACBQAAAANuaWwJAAS5AAAAAgkABEwAAAACAgAAAAQlZCVkCQAETAAAAAIJAAGkAAAAAQUAAAAJZ1d4QW1vdW50CQAETAAAAAIJAAGkAAAAAQUAAAAFYm9vc3QFAAAAA25pbAUAAAADU0VQAAAAAWkBAAAAF3d4RW1pc3Npb25TdGF0c1JFQURPTkxZAAAAAAQAAAAHT05FTVVMVAkAAaQAAAABBQAAAAVNVUxUOAQAAAADT05FAgAAAAExBAAAAA9mYWN0b3J5Q29udHJhY3QJAQAAABhyZWFkRmFjdG9yeUFkZHJlc3NPckZhaWwAAAAABAAAAApmYWN0b3J5Q2ZnCQEAAAAUcmVhZEZhY3RvcnlDZmdPckZhaWwAAAABBQAAAA9mYWN0b3J5Q29udHJhY3QEAAAAEGJvb3N0aW5nQ29udHJhY3QJAQAAABhnZXRCb29zdGluZ0FkZHJlc3NPckZhaWwAAAABBQAAAApmYWN0b3J5Q2ZnBAAAABBlbWlzc2lvbkNvbnRyYWN0CQEAAAAYZ2V0RW1pc3Npb25BZGRyZXNzT3JGYWlsAAAAAQUAAAAKZmFjdG9yeUNmZwQAAAASd3hFbWlzc2lvblBlckJsb2NrCQEAAAAMZ2V0SW50T3JGYWlsAAAAAgUAAAAQZW1pc3Npb25Db250cmFjdAkBAAAAHmtleUVtaXNzaW9uUmF0ZVBlckJsb2NrQ3VycmVudAAAAAAEAAAAEmVtaXNzaW9uU3RhcnRCbG9jawkBAAAADGdldEludE9yRmFpbAAAAAIFAAAAEGVtaXNzaW9uQ29udHJhY3QJAQAAABVrZXlFbWlzc2lvblN0YXJ0QmxvY2sAAAAABAAAAAxwYXNzZWRCbG9ja3MDCQAAZgAAAAIFAAAAEmVtaXNzaW9uU3RhcnRCbG9jawUAAAAGaGVpZ2h0AAAAAAAAAAAACQAAZQAAAAIFAAAABmhlaWdodAUAAAASZW1pc3Npb25TdGFydEJsb2NrBAAAAA50ZWFtRW1EdXJhdGlvbgkAAGgAAAACAAAAAAAAAAWgAAAAAAAAAAFtBAAAAAl0ZWFtRW1NYXgJAABoAAAAAgAAAAAAC/sEQAUAAAAFTVVMVDgEAAAABnRlYW1FbQMJAABmAAAAAgUAAAAMcGFzc2VkQmxvY2tzBQAAAA50ZWFtRW1EdXJhdGlvbgUAAAAJdGVhbUVtTWF4CQAAawAAAAMFAAAACXRlYW1FbU1heAUAAAAMcGFzc2VkQmxvY2tzBQAAAA50ZWFtRW1EdXJhdGlvbgQAAAAPdG90YWxXeFJlbGVhc2VkCQAAZAAAAAIJAABoAAAAAgUAAAASd3hFbWlzc2lvblBlckJsb2NrBQAAAAxwYXNzZWRCbG9ja3MFAAAABnRlYW1FbQQAAAANdG90YWxXeExvY2tlZAkBAAAADGdldEludE9yWmVybwAAAAIFAAAAEGJvb3N0aW5nQ29udHJhY3QJAQAAAB9rZXlCb29zdGluZ0xvY2tQYXJhbVRvdGFsQW1vdW50AAAAAAQAAAAYbG9ja3NEdXJhdGlvblN1bUluQmxvY2tzCQEAAAAMZ2V0SW50T3JaZXJvAAAAAgUAAAAQYm9vc3RpbmdDb250cmFjdAkBAAAAKGtleUJvb3N0aW5nU3RhdHNMb2Nrc0R1cmF0aW9uU3VtSW5CbG9ja3MAAAAABAAAAApsb2Nrc0NvdW50CQEAAAAMZ2V0SW50T3JaZXJvAAAAAgUAAAAQYm9vc3RpbmdDb250cmFjdAkBAAAAGmtleUJvb3N0aW5nU3RhdHNMb2Nrc0NvdW50AAAAAAkABRQAAAACBQAAAANuaWwJAAS5AAAAAgkABEwAAAACAgAAAAglZCVkJWQlZAkABEwAAAACCQABpAAAAAEFAAAAD3RvdGFsV3hSZWxlYXNlZAkABEwAAAACCQABpAAAAAEFAAAADXRvdGFsV3hMb2NrZWQJAARMAAAAAgkAAaQAAAABBQAAABhsb2Nrc0R1cmF0aW9uU3VtSW5CbG9ja3MJAARMAAAAAgkAAaQAAAABBQAAAApsb2Nrc0NvdW50BQAAAANuaWwFAAAAA1NFUAAAAAFpAQAAAA9scFN0YXRzUkVBRE9OTFkAAAABAAAAB2xwQXNzZXQEAAAADmZhY3RvcnlBZGRyZXNzCQEAAAAYcmVhZEZhY3RvcnlBZGRyZXNzT3JGYWlsAAAAAAQAAAALcG9vbEFk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", "chainId": 87, "height": 2871836, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 2to8QunTWPFUcJZ6pcbjmh5bmy2ibMofGyDfHUPE8Fu7 Next: 79jzKR5twVF2ZZ8Q2EgzfrGef1HvB5iog4dCHo5vugyv Diff:
Old | New | Differences | |
---|---|---|---|
10 | 10 | let MULT18 = toBigInt(1000000000000000000) | |
11 | 11 | ||
12 | 12 | let SEP = "__" | |
13 | + | ||
14 | + | let idxPoolAddress = 1 | |
15 | + | ||
16 | + | let idxPoolStatus = 2 | |
17 | + | ||
18 | + | let idxPoolLPAssetId = 3 | |
19 | + | ||
20 | + | let idxAmtAssetId = 4 | |
21 | + | ||
22 | + | let idxPriceAssetId = 5 | |
23 | + | ||
24 | + | let idxAmtAssetDcm = 6 | |
25 | + | ||
26 | + | let idxPriceAssetDcm = 7 | |
27 | + | ||
28 | + | let idxIAmtAssetId = 8 | |
29 | + | ||
30 | + | let idxIPriceAssetId = 9 | |
31 | + | ||
32 | + | let idxLPAssetDcm = 10 | |
13 | 33 | ||
14 | 34 | func getStringOrFail (key) = valueOrErrorMessage(getString(key), (("mandatory this." + key) + " is not defined")) | |
15 | 35 | ||
26 | 46 | func keyFactoryAddress () = "%s%s__config__factoryAddress" | |
27 | 47 | ||
28 | 48 | ||
49 | + | let IdxFactoryCfgStakingDapp = 1 | |
50 | + | ||
51 | + | let IdxFactoryCfgBoostingDapp = 2 | |
52 | + | ||
53 | + | let IdxFactoryCfgIdoDapp = 3 | |
54 | + | ||
55 | + | let IdxFactoryCfgTeamDapp = 4 | |
56 | + | ||
57 | + | let IdxFactoryCfgEmissionDapp = 5 | |
58 | + | ||
59 | + | let IdxFactoryCfgRestDapp = 6 | |
60 | + | ||
61 | + | let IdxFactoryCfgSlippageDapp = 7 | |
62 | + | ||
63 | + | func keyFactoryCfg () = "%s__factoryConfig" | |
64 | + | ||
65 | + | ||
29 | 66 | func keyFactoryLp2AssetsMapping (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
30 | 67 | ||
31 | 68 | ||
32 | 69 | func keyFactoryLpList () = "%s__lpTokensList" | |
33 | 70 | ||
34 | 71 | ||
35 | - | func keyFactoryCfg () = "%s__factoryConfig" | |
72 | + | func keyFactoryLpAssetToPoolContractAddress (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
73 | + | ||
74 | + | ||
75 | + | func keyFactoryPoolWeight (contractAddress) = makeString(["%s%s", "poolWeight", contractAddress], SEP) | |
36 | 76 | ||
37 | 77 | ||
38 | 78 | func readFactoryAddressOrFail () = addressFromStringValue(getStringOrFail(keyFactoryAddress())) | |
41 | 81 | func readLpList () = split(valueOrElse(getString(readFactoryAddressOrFail(), keyFactoryLpList()), ""), SEP) | |
42 | 82 | ||
43 | 83 | ||
44 | - | func readFactoryCfgOrFail () = split(getStringByAddressOrFail(readFactoryAddressOrFail(), keyFactoryCfg()), SEP) | |
84 | + | func readFactoryCfgOrFail (factory) = split(getStringByAddressOrFail(factory, keyFactoryCfg()), SEP) | |
85 | + | ||
86 | + | ||
87 | + | func getBoostingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgBoostingDapp]) | |
88 | + | ||
89 | + | ||
90 | + | func getEmissionAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgEmissionDapp]) | |
91 | + | ||
92 | + | ||
93 | + | func getStakingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgStakingDapp]) | |
94 | + | ||
95 | + | ||
96 | + | func keyBoostCfg () = "%s__config" | |
97 | + | ||
98 | + | ||
99 | + | func keyBoostingLockParamTotalAmount () = "%s%s__stats__activeTotalLocked" | |
100 | + | ||
101 | + | ||
102 | + | func keyBoostingStatsLocksDurationSumInBlocks () = "%s%s__stats__locksDurationSumInBlocks" | |
103 | + | ||
104 | + | ||
105 | + | func keyBoostingStatsLocksCount () = "%s%s__stats__locksCount" | |
106 | + | ||
107 | + | ||
108 | + | func keyBoostingStatsUsersCount () = "%s%s__stats__activeUsersCount" | |
109 | + | ||
110 | + | ||
111 | + | func keyUser2NumMapping (userAddress) = makeString(["%s%s%s__mapping__user2num", userAddress], SEP) | |
112 | + | ||
113 | + | ||
114 | + | func keyNum2UserMapping (num) = makeString(["%s%s%s__mapping__num2user", num], SEP) | |
115 | + | ||
116 | + | ||
117 | + | func keyLockParamUserAmount (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "amount"], SEP) | |
118 | + | ||
119 | + | ||
120 | + | func keyLockParamStartBlock (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "start"], SEP) | |
121 | + | ||
122 | + | ||
123 | + | func keyLockParamDuration (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "duration"], SEP) | |
124 | + | ||
125 | + | ||
126 | + | func keyLockParamK (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "k"], SEP) | |
127 | + | ||
128 | + | ||
129 | + | func keyLockParamB (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "b"], SEP) | |
130 | + | ||
131 | + | ||
132 | + | func keyLockParamByPeriodK (userNum,period) = makeString(["%s%d%s%d__paramByPeriod", userNum, "k", period], SEP) | |
133 | + | ||
134 | + | ||
135 | + | func keyLockParamByPeriodB (userNum,period) = makeString(["%s%d%s%d__paramByPeriod", userNum, "b", period], SEP) | |
45 | 136 | ||
46 | 137 | ||
47 | 138 | func keyStakedByUser (userAddressStr,lpAssetIdStr) = makeString(["%s%s%s__staked", userAddressStr, lpAssetIdStr], SEP) | |
50 | 141 | func keyStakedTotal (lpAssetIdStr) = ("%s%s%s__staked__total__" + lpAssetIdStr) | |
51 | 142 | ||
52 | 143 | ||
53 | - | func | |
144 | + | func keyClaimedByUser (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimed", userAddressStr, lpAssetIdStr], SEP) | |
54 | 145 | ||
55 | 146 | ||
56 | - | func internalCurrentRewardRate (stakingContract,lpAssetId) = { | |
57 | - | let wxEmissionPerBlock = 0 | |
58 | - | let maxFactor = (1 * MULT8) | |
147 | + | func keyClaimedByUserMinReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedMinReward", userAddressStr, lpAssetIdStr], SEP) | |
148 | + | ||
149 | + | ||
150 | + | func keyClaimedByUserBoostReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedBoostReward", userAddressStr, lpAssetIdStr], SEP) | |
151 | + | ||
152 | + | ||
153 | + | func keyEmissionRatePerBlockCurrent () = "%s%s__ratePerBlock__current" | |
154 | + | ||
155 | + | ||
156 | + | func keyEmissionRatePerBlockMaxCurrent () = "%s%s__ratePerBlockMax__current" | |
157 | + | ||
158 | + | ||
159 | + | func keyEmissionStartBlock () = "%s%s__emission__startBlock" | |
160 | + | ||
161 | + | ||
162 | + | func keyEmissionDurationInBlocks () = "%s%s__emission__duration" | |
163 | + | ||
164 | + | ||
165 | + | func keyEmissionEndBlock () = "%s%s__emission__endBlock" | |
166 | + | ||
167 | + | ||
168 | + | func internalCurrentRewardRate (factoryContract,stakingContract,emissionContract,lpAssetId) = { | |
169 | + | let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetId)) | |
170 | + | let poolWeightMult = MULT8 | |
171 | + | let poolWeight = getIntegerValue(factoryContract, keyFactoryPoolWeight(poolAddressStr)) | |
172 | + | let wxEmissionPerBlock = getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent()) | |
173 | + | let wxEmissionPerBlockMax = getIntOrFail(emissionContract, keyEmissionRatePerBlockMaxCurrent()) | |
174 | + | let poolWxEmissionPerBlock = fraction(wxEmissionPerBlock, poolWeight, poolWeightMult) | |
175 | + | let poolWxEmissionPerBlockMax = fraction(wxEmissionPerBlockMax, poolWeight, poolWeightMult) | |
176 | + | let maxFactor = (3 * MULT8) | |
59 | 177 | let totalLpStaked = getIntOrZero(stakingContract, keyStakedTotal(lpAssetId)) | |
60 | - | [ | |
178 | + | [poolWxEmissionPerBlock, maxFactor, totalLpStaked] | |
61 | 179 | } | |
62 | 180 | ||
63 | 181 | ||
70 | 188 | ||
71 | 189 | @Callable(i) | |
72 | 190 | func currentRewardRateREADONLY (lpAssetId) = { | |
73 | - | let factoryCfgArray = readFactoryCfgOrFail() | |
74 | - | let stakingContract = getStakingAddressOrFail(factoryCfgArray) | |
75 | - | let rewardData = internalCurrentRewardRate(stakingContract, lpAssetId) | |
191 | + | let factoryContract = readFactoryAddressOrFail() | |
192 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
193 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
194 | + | let emissionContract = getEmissionAddressOrFail(factoryCfg) | |
195 | + | let rewardData = internalCurrentRewardRate(factoryContract, stakingContract, emissionContract, lpAssetId) | |
76 | 196 | let wxEmissionPerBlock = rewardData[0] | |
77 | 197 | let maxFactor = rewardData[1] | |
78 | 198 | let totalLpStaked = rewardData[2] | |
83 | 203 | ||
84 | 204 | @Callable(i) | |
85 | 205 | func currentUserRewardRateREADONLY (lpAssetId,userAddress) = { | |
86 | - | let factoryCfgArray = readFactoryCfgOrFail() | |
87 | - | let stakingContract = getStakingAddressOrFail(factoryCfgArray) | |
88 | - | let rewardData = internalCurrentRewardRate(stakingContract, lpAssetId) | |
206 | + | let factoryContract = readFactoryAddressOrFail() | |
207 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
208 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
209 | + | let emissionContract = getEmissionAddressOrFail(factoryCfg) | |
210 | + | let rewardData = internalCurrentRewardRate(factoryContract, stakingContract, emissionContract, lpAssetId) | |
89 | 211 | let wxEmissionPerBlock = rewardData[0] | |
90 | 212 | let maxFactor = rewardData[1] | |
91 | 213 | let totalLpStaked = rewardData[2] | |
92 | 214 | let lpStakedByUser = getIntOrZero(stakingContract, keyStakedByUser(userAddress, lpAssetId)) | |
93 | 215 | let boostingPower = (1 * MULT8) | |
94 | - | $Tuple2(nil, makeString(["%d%d%d", toString(wxEmissionPerBlock), toString(maxFactor), toString(totalLpStaked), toString(lpStakedByUser), toString(boostingPower)], SEP)) | |
216 | + | $Tuple2(nil, makeString(["%d%d%d%d%d", toString(wxEmissionPerBlock), toString(maxFactor), toString(totalLpStaked), toString(lpStakedByUser), toString(boostingPower)], SEP)) | |
95 | 217 | } | |
96 | 218 | ||
97 | 219 | ||
98 | 220 | ||
99 | 221 | @Callable(i) | |
100 | 222 | func claimedRewardREADONLY (userAddress) = { | |
223 | + | let factoryContract = readFactoryAddressOrFail() | |
224 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
225 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
101 | 226 | let lpList = readLpList() | |
102 | 227 | let prefix = "%s%d%d%d%s" | |
103 | 228 | func claimedRewardByLpAggregator (resultStr,nextLp) = { | |
104 | - | let rand = ((toInt(fromBase58String(nextLp)) % 100000) + (toInt(toBytes(userAddress)) % 100000)) | |
105 | - | let absRand = if ((0 > rand)) | |
106 | - | then -(rand) | |
107 | - | else rand | |
108 | - | let val = (((absRand % 100) * MULT8) + (absRand * (MULT8 / 100000))) | |
109 | - | let minRewardClaimed = 0 | |
110 | - | let boostRewardClaimed = 0 | |
229 | + | let claimedByUserMinRewardKEY = keyClaimedByUserMinReward(nextLp, userAddress) | |
230 | + | let claimedByUserBoostRewardKEY = keyClaimedByUserBoostReward(nextLp, userAddress) | |
231 | + | let minRewardClaimed = valueOrElse(getInteger(stakingContract, claimedByUserMinRewardKEY), 0) | |
232 | + | let boostRewardClaimed = valueOrElse(getInteger(stakingContract, claimedByUserBoostRewardKEY), 0) | |
111 | 233 | let gFeeClaimed = 0 | |
112 | 234 | makeString([(prefix + resultStr), nextLp, toString(minRewardClaimed), toString(boostRewardClaimed), toString(gFeeClaimed), "end"], SEP) | |
113 | 235 | } | |
133 | 255 | ||
134 | 256 | @Callable(i) | |
135 | 257 | func calcBoostREADONLY (deltaWxAmount,deltaLockPeriodInBlocks,deltaLpAmount,lpAssetIdOpt,userAddressOpt) = { | |
258 | + | let factoryContract = readFactoryAddressOrFail() | |
259 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
260 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
261 | + | let EMPTYSTR = "empty" | |
262 | + | let maxLockDurationInBlocks = parseIntValue(split(getStringByAddressOrFail(boostingContract, keyBoostCfg()), SEP)[4]) | |
263 | + | let userNum = valueOrElse(getString(boostingContract, keyUser2NumMapping(userAddressOpt)), EMPTYSTR) | |
264 | + | let userAmount = valueOrElse(getInteger(boostingContract, keyLockParamUserAmount(userNum)), 0) | |
265 | + | let lockStart = valueOrElse(getInteger(boostingContract, keyLockParamStartBlock(userNum)), height) | |
266 | + | let lockDuration = valueOrElse(getInteger(boostingContract, keyLockParamDuration(userNum)), 0) | |
267 | + | let lockEnd = (lockStart + lockDuration) | |
268 | + | let remainingDuration = max([(lockEnd - height), 0]) | |
136 | 269 | let boost = if ((userAddressOpt != "")) | |
137 | 270 | then (3 * MULT8) | |
138 | 271 | else (1 * MULT8) | |
139 | - | let monthes18 = fraction(toBigInt(deltaLockPeriodInBlocks), MULT18, toBigInt((30 * 1440))) | |
140 | - | let coeff18 = (pow(monthes18, SCALE18, toBigInt(2), 0, SCALE18, HALFUP) / toBigInt(1000)) | |
141 | - | let gWxAmount = fraction(toBigInt(deltaWxAmount), coeff18, MULT18) | |
142 | - | $Tuple2(nil, makeString(["%d%d", toString(toInt(gWxAmount)), toString(boost)], SEP)) | |
272 | + | let SCALE = 1000 | |
273 | + | let userAmountNew = (userAmount + deltaWxAmount) | |
274 | + | let lockDurationNew = min([(remainingDuration + deltaLockPeriodInBlocks), maxLockDurationInBlocks]) | |
275 | + | let coeffX8 = fraction(lockDurationNew, MULT8, maxLockDurationInBlocks) | |
276 | + | let wxAmountStart = fraction(userAmountNew, coeffX8, MULT8) | |
277 | + | let lockEndHeight = (height + lockDurationNew) | |
278 | + | let scale8ParamK = -(fraction(wxAmountStart, SCALE, lockDurationNew)) | |
279 | + | let scale8ParamB = (fraction(wxAmountStart, SCALE, lockDurationNew) * lockEndHeight) | |
280 | + | let gWxAmount = (((scale8ParamK * height) + scale8ParamB) / SCALE) | |
281 | + | $Tuple2(nil, makeString(["%d%d", toString(gWxAmount), toString(boost)], SEP)) | |
143 | 282 | } | |
144 | 283 | ||
145 | 284 | ||
148 | 287 | func wxEmissionStatsREADONLY () = { | |
149 | 288 | let ONEMULT = toString(MULT8) | |
150 | 289 | let ONE = "1" | |
151 | - | let totalWxReleased = 0 | |
152 | - | let totalWxLocked = 0 | |
153 | - | let locksDurationSumInBlocks = 0 | |
154 | - | let locksCount = 0 | |
290 | + | let factoryContract = readFactoryAddressOrFail() | |
291 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
292 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
293 | + | let emissionContract = getEmissionAddressOrFail(factoryCfg) | |
294 | + | let wxEmissionPerBlock = getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent()) | |
295 | + | let emissionStartBlock = getIntOrFail(emissionContract, keyEmissionStartBlock()) | |
296 | + | let passedBlocks = if ((emissionStartBlock > height)) | |
297 | + | then 0 | |
298 | + | else (height - emissionStartBlock) | |
299 | + | let teamEmDuration = (1440 * 365) | |
300 | + | let teamEmMax = (201000000 * MULT8) | |
301 | + | let teamEm = if ((passedBlocks > teamEmDuration)) | |
302 | + | then teamEmMax | |
303 | + | else fraction(teamEmMax, passedBlocks, teamEmDuration) | |
304 | + | let totalWxReleased = ((wxEmissionPerBlock * passedBlocks) + teamEm) | |
305 | + | let totalWxLocked = getIntOrZero(boostingContract, keyBoostingLockParamTotalAmount()) | |
306 | + | let locksDurationSumInBlocks = getIntOrZero(boostingContract, keyBoostingStatsLocksDurationSumInBlocks()) | |
307 | + | let locksCount = getIntOrZero(boostingContract, keyBoostingStatsLocksCount()) | |
155 | 308 | $Tuple2(nil, makeString(["%d%d%d%d", toString(totalWxReleased), toString(totalWxLocked), toString(locksDurationSumInBlocks), toString(locksCount)], SEP)) | |
309 | + | } | |
310 | + | ||
311 | + | ||
312 | + | ||
313 | + | @Callable(i) | |
314 | + | func lpStatsREADONLY (lpAsset) = { | |
315 | + | let factoryAddress = readFactoryAddressOrFail() | |
316 | + | let poolAddress = addressFromStringValue(getStringByAddressOrFail(factoryAddress, keyFactoryLpAssetToPoolContractAddress(lpAsset))) | |
317 | + | let cfg = if ($isInstanceOf(invoke(poolAddress, "getPoolConfigWrapperREADONLY", nil, nil), "List[Any]")) | |
318 | + | then invoke(poolAddress, "getPoolConfigWrapperREADONLY", nil, nil) | |
319 | + | else throw("Couldn't cast Any to List[Any]") | |
320 | + | let lpAssetId = fromBase58String(if ($isInstanceOf(cfg[idxPoolLPAssetId], "String")) | |
321 | + | then cfg[idxPoolLPAssetId] | |
322 | + | else throw("Couldn't cast Any to String")) | |
323 | + | let amtAssetId = if ($isInstanceOf(cfg[idxAmtAssetId], "String")) | |
324 | + | then cfg[idxAmtAssetId] | |
325 | + | else throw("Couldn't cast Any to String") | |
326 | + | let priceAssetId = if ($isInstanceOf(cfg[idxPriceAssetId], "String")) | |
327 | + | then cfg[idxPriceAssetId] | |
328 | + | else throw("Couldn't cast Any to String") | |
329 | + | let iAmtAssetId = if ($isInstanceOf(cfg[idxIAmtAssetId], "String")) | |
330 | + | then cfg[idxIAmtAssetId] | |
331 | + | else throw("Couldn't cast Any to String") | |
332 | + | let iPriceAssetId = if ($isInstanceOf(cfg[idxIPriceAssetId], "String")) | |
333 | + | then cfg[idxIPriceAssetId] | |
334 | + | else throw("Couldn't cast Any to String") | |
335 | + | let amtAssetDcm = parseIntValue(if ($isInstanceOf(cfg[idxAmtAssetDcm], "String")) | |
336 | + | then cfg[idxAmtAssetDcm] | |
337 | + | else throw("Couldn't cast Any to String")) | |
338 | + | let priceAssetDcm = parseIntValue(if ($isInstanceOf(cfg[idxPriceAssetDcm], "String")) | |
339 | + | then cfg[idxPriceAssetDcm] | |
340 | + | else throw("Couldn't cast Any to String")) | |
341 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
342 | + | let accAmtAssetBalance = if ($isInstanceOf(invoke(poolAddress, "getAccBalanceWrapperREADONLY", [amtAssetId], nil), "Int")) | |
343 | + | then invoke(poolAddress, "getAccBalanceWrapperREADONLY", [amtAssetId], nil) | |
344 | + | else throw("Couldn't cast Any to Int") | |
345 | + | let accPriceAssetBalance = if ($isInstanceOf(invoke(poolAddress, "getAccBalanceWrapperREADONLY", [priceAssetId], nil), "Int")) | |
346 | + | then invoke(poolAddress, "getAccBalanceWrapperREADONLY", [priceAssetId], nil) | |
347 | + | else throw("Couldn't cast Any to Int") | |
348 | + | let pricesList = if ($isInstanceOf(invoke(poolAddress, "calcPricesWrapperREADONLY", [accAmtAssetBalance, accPriceAssetBalance, poolLPBalance], nil), "List[Any]")) | |
349 | + | then invoke(poolAddress, "calcPricesWrapperREADONLY", [accAmtAssetBalance, accPriceAssetBalance, poolLPBalance], nil) | |
350 | + | else throw("Couldn't cast Any to List[Any]") | |
351 | + | let curPrice = 0 | |
352 | + | let lpAmtAssetShare = if ($isInstanceOf(invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[1], MULT8], nil), "Int")) | |
353 | + | then invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[1], MULT8], nil) | |
354 | + | else throw("Couldn't cast Any to Int") | |
355 | + | let lpPriceAssetShare = if ($isInstanceOf(invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[2], MULT8], nil), "Int")) | |
356 | + | then invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[2], MULT8], nil) | |
357 | + | else throw("Couldn't cast Any to Int") | |
358 | + | let poolWeight = getIntegerValue(factoryAddress, keyFactoryPoolWeight(toString(poolAddress))) | |
359 | + | $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)) | |
360 | + | } | |
361 | + | ||
362 | + | ||
363 | + | ||
364 | + | @Callable(i) | |
365 | + | func gwxUserInfoREADONLY (userAddress) = { | |
366 | + | let factoryContract = readFactoryAddressOrFail() | |
367 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
368 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
369 | + | let gwxUserInfoLIST = if ($isInstanceOf(invoke(boostingContract, "gwxUserInfoREADONLY", [userAddress], nil), "List[Any]")) | |
370 | + | then invoke(boostingContract, "gwxUserInfoREADONLY", [userAddress], nil) | |
371 | + | else throw("Couldn't cast Any to List[Any]") | |
372 | + | let gwxAmount = if ($isInstanceOf(gwxUserInfoLIST[0], "Int")) | |
373 | + | then gwxUserInfoLIST[0] | |
374 | + | else throw("Couldn't cast Any to Int") | |
375 | + | $Tuple2(nil, makeString(["%d", toString(gwxAmount)], SEP)) | |
156 | 376 | } | |
157 | 377 | ||
158 | 378 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let SCALE8 = 8 | |
5 | 5 | ||
6 | 6 | let MULT8 = 100000000 | |
7 | 7 | ||
8 | 8 | let SCALE18 = 18 | |
9 | 9 | ||
10 | 10 | let MULT18 = toBigInt(1000000000000000000) | |
11 | 11 | ||
12 | 12 | let SEP = "__" | |
13 | + | ||
14 | + | let idxPoolAddress = 1 | |
15 | + | ||
16 | + | let idxPoolStatus = 2 | |
17 | + | ||
18 | + | let idxPoolLPAssetId = 3 | |
19 | + | ||
20 | + | let idxAmtAssetId = 4 | |
21 | + | ||
22 | + | let idxPriceAssetId = 5 | |
23 | + | ||
24 | + | let idxAmtAssetDcm = 6 | |
25 | + | ||
26 | + | let idxPriceAssetDcm = 7 | |
27 | + | ||
28 | + | let idxIAmtAssetId = 8 | |
29 | + | ||
30 | + | let idxIPriceAssetId = 9 | |
31 | + | ||
32 | + | let idxLPAssetDcm = 10 | |
13 | 33 | ||
14 | 34 | func getStringOrFail (key) = valueOrErrorMessage(getString(key), (("mandatory this." + key) + " is not defined")) | |
15 | 35 | ||
16 | 36 | ||
17 | 37 | func getStringByAddressOrFail (address,key) = valueOrErrorMessage(getString(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined")) | |
18 | 38 | ||
19 | 39 | ||
20 | 40 | func getIntOrZero (address,key) = valueOrElse(getInteger(address, key), 0) | |
21 | 41 | ||
22 | 42 | ||
23 | 43 | func getIntOrFail (address,key) = valueOrErrorMessage(getInteger(address, key), (("mandatory this." + key) + " is not defined")) | |
24 | 44 | ||
25 | 45 | ||
26 | 46 | func keyFactoryAddress () = "%s%s__config__factoryAddress" | |
27 | 47 | ||
28 | 48 | ||
49 | + | let IdxFactoryCfgStakingDapp = 1 | |
50 | + | ||
51 | + | let IdxFactoryCfgBoostingDapp = 2 | |
52 | + | ||
53 | + | let IdxFactoryCfgIdoDapp = 3 | |
54 | + | ||
55 | + | let IdxFactoryCfgTeamDapp = 4 | |
56 | + | ||
57 | + | let IdxFactoryCfgEmissionDapp = 5 | |
58 | + | ||
59 | + | let IdxFactoryCfgRestDapp = 6 | |
60 | + | ||
61 | + | let IdxFactoryCfgSlippageDapp = 7 | |
62 | + | ||
63 | + | func keyFactoryCfg () = "%s__factoryConfig" | |
64 | + | ||
65 | + | ||
29 | 66 | func keyFactoryLp2AssetsMapping (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
30 | 67 | ||
31 | 68 | ||
32 | 69 | func keyFactoryLpList () = "%s__lpTokensList" | |
33 | 70 | ||
34 | 71 | ||
35 | - | func keyFactoryCfg () = "%s__factoryConfig" | |
72 | + | func keyFactoryLpAssetToPoolContractAddress (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
73 | + | ||
74 | + | ||
75 | + | func keyFactoryPoolWeight (contractAddress) = makeString(["%s%s", "poolWeight", contractAddress], SEP) | |
36 | 76 | ||
37 | 77 | ||
38 | 78 | func readFactoryAddressOrFail () = addressFromStringValue(getStringOrFail(keyFactoryAddress())) | |
39 | 79 | ||
40 | 80 | ||
41 | 81 | func readLpList () = split(valueOrElse(getString(readFactoryAddressOrFail(), keyFactoryLpList()), ""), SEP) | |
42 | 82 | ||
43 | 83 | ||
44 | - | func readFactoryCfgOrFail () = split(getStringByAddressOrFail(readFactoryAddressOrFail(), keyFactoryCfg()), SEP) | |
84 | + | func readFactoryCfgOrFail (factory) = split(getStringByAddressOrFail(factory, keyFactoryCfg()), SEP) | |
85 | + | ||
86 | + | ||
87 | + | func getBoostingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgBoostingDapp]) | |
88 | + | ||
89 | + | ||
90 | + | func getEmissionAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgEmissionDapp]) | |
91 | + | ||
92 | + | ||
93 | + | func getStakingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgStakingDapp]) | |
94 | + | ||
95 | + | ||
96 | + | func keyBoostCfg () = "%s__config" | |
97 | + | ||
98 | + | ||
99 | + | func keyBoostingLockParamTotalAmount () = "%s%s__stats__activeTotalLocked" | |
100 | + | ||
101 | + | ||
102 | + | func keyBoostingStatsLocksDurationSumInBlocks () = "%s%s__stats__locksDurationSumInBlocks" | |
103 | + | ||
104 | + | ||
105 | + | func keyBoostingStatsLocksCount () = "%s%s__stats__locksCount" | |
106 | + | ||
107 | + | ||
108 | + | func keyBoostingStatsUsersCount () = "%s%s__stats__activeUsersCount" | |
109 | + | ||
110 | + | ||
111 | + | func keyUser2NumMapping (userAddress) = makeString(["%s%s%s__mapping__user2num", userAddress], SEP) | |
112 | + | ||
113 | + | ||
114 | + | func keyNum2UserMapping (num) = makeString(["%s%s%s__mapping__num2user", num], SEP) | |
115 | + | ||
116 | + | ||
117 | + | func keyLockParamUserAmount (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "amount"], SEP) | |
118 | + | ||
119 | + | ||
120 | + | func keyLockParamStartBlock (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "start"], SEP) | |
121 | + | ||
122 | + | ||
123 | + | func keyLockParamDuration (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "duration"], SEP) | |
124 | + | ||
125 | + | ||
126 | + | func keyLockParamK (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "k"], SEP) | |
127 | + | ||
128 | + | ||
129 | + | func keyLockParamB (userNum) = makeString(["%s%d%s__paramByUserNum", userNum, "b"], SEP) | |
130 | + | ||
131 | + | ||
132 | + | func keyLockParamByPeriodK (userNum,period) = makeString(["%s%d%s%d__paramByPeriod", userNum, "k", period], SEP) | |
133 | + | ||
134 | + | ||
135 | + | func keyLockParamByPeriodB (userNum,period) = makeString(["%s%d%s%d__paramByPeriod", userNum, "b", period], SEP) | |
45 | 136 | ||
46 | 137 | ||
47 | 138 | func keyStakedByUser (userAddressStr,lpAssetIdStr) = makeString(["%s%s%s__staked", userAddressStr, lpAssetIdStr], SEP) | |
48 | 139 | ||
49 | 140 | ||
50 | 141 | func keyStakedTotal (lpAssetIdStr) = ("%s%s%s__staked__total__" + lpAssetIdStr) | |
51 | 142 | ||
52 | 143 | ||
53 | - | func | |
144 | + | func keyClaimedByUser (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimed", userAddressStr, lpAssetIdStr], SEP) | |
54 | 145 | ||
55 | 146 | ||
56 | - | func internalCurrentRewardRate (stakingContract,lpAssetId) = { | |
57 | - | let wxEmissionPerBlock = 0 | |
58 | - | let maxFactor = (1 * MULT8) | |
147 | + | func keyClaimedByUserMinReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedMinReward", userAddressStr, lpAssetIdStr], SEP) | |
148 | + | ||
149 | + | ||
150 | + | func keyClaimedByUserBoostReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedBoostReward", userAddressStr, lpAssetIdStr], SEP) | |
151 | + | ||
152 | + | ||
153 | + | func keyEmissionRatePerBlockCurrent () = "%s%s__ratePerBlock__current" | |
154 | + | ||
155 | + | ||
156 | + | func keyEmissionRatePerBlockMaxCurrent () = "%s%s__ratePerBlockMax__current" | |
157 | + | ||
158 | + | ||
159 | + | func keyEmissionStartBlock () = "%s%s__emission__startBlock" | |
160 | + | ||
161 | + | ||
162 | + | func keyEmissionDurationInBlocks () = "%s%s__emission__duration" | |
163 | + | ||
164 | + | ||
165 | + | func keyEmissionEndBlock () = "%s%s__emission__endBlock" | |
166 | + | ||
167 | + | ||
168 | + | func internalCurrentRewardRate (factoryContract,stakingContract,emissionContract,lpAssetId) = { | |
169 | + | let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetId)) | |
170 | + | let poolWeightMult = MULT8 | |
171 | + | let poolWeight = getIntegerValue(factoryContract, keyFactoryPoolWeight(poolAddressStr)) | |
172 | + | let wxEmissionPerBlock = getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent()) | |
173 | + | let wxEmissionPerBlockMax = getIntOrFail(emissionContract, keyEmissionRatePerBlockMaxCurrent()) | |
174 | + | let poolWxEmissionPerBlock = fraction(wxEmissionPerBlock, poolWeight, poolWeightMult) | |
175 | + | let poolWxEmissionPerBlockMax = fraction(wxEmissionPerBlockMax, poolWeight, poolWeightMult) | |
176 | + | let maxFactor = (3 * MULT8) | |
59 | 177 | let totalLpStaked = getIntOrZero(stakingContract, keyStakedTotal(lpAssetId)) | |
60 | - | [ | |
178 | + | [poolWxEmissionPerBlock, maxFactor, totalLpStaked] | |
61 | 179 | } | |
62 | 180 | ||
63 | 181 | ||
64 | 182 | @Callable(i) | |
65 | 183 | func constructor (factoryAddress) = if ((i.caller != this)) | |
66 | 184 | then throw("not authorized") | |
67 | 185 | else [StringEntry(keyFactoryAddress(), factoryAddress)] | |
68 | 186 | ||
69 | 187 | ||
70 | 188 | ||
71 | 189 | @Callable(i) | |
72 | 190 | func currentRewardRateREADONLY (lpAssetId) = { | |
73 | - | let factoryCfgArray = readFactoryCfgOrFail() | |
74 | - | let stakingContract = getStakingAddressOrFail(factoryCfgArray) | |
75 | - | let rewardData = internalCurrentRewardRate(stakingContract, lpAssetId) | |
191 | + | let factoryContract = readFactoryAddressOrFail() | |
192 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
193 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
194 | + | let emissionContract = getEmissionAddressOrFail(factoryCfg) | |
195 | + | let rewardData = internalCurrentRewardRate(factoryContract, stakingContract, emissionContract, lpAssetId) | |
76 | 196 | let wxEmissionPerBlock = rewardData[0] | |
77 | 197 | let maxFactor = rewardData[1] | |
78 | 198 | let totalLpStaked = rewardData[2] | |
79 | 199 | $Tuple2(nil, makeString(["%d%d%d", toString(wxEmissionPerBlock), toString(maxFactor), toString(totalLpStaked)], SEP)) | |
80 | 200 | } | |
81 | 201 | ||
82 | 202 | ||
83 | 203 | ||
84 | 204 | @Callable(i) | |
85 | 205 | func currentUserRewardRateREADONLY (lpAssetId,userAddress) = { | |
86 | - | let factoryCfgArray = readFactoryCfgOrFail() | |
87 | - | let stakingContract = getStakingAddressOrFail(factoryCfgArray) | |
88 | - | let rewardData = internalCurrentRewardRate(stakingContract, lpAssetId) | |
206 | + | let factoryContract = readFactoryAddressOrFail() | |
207 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
208 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
209 | + | let emissionContract = getEmissionAddressOrFail(factoryCfg) | |
210 | + | let rewardData = internalCurrentRewardRate(factoryContract, stakingContract, emissionContract, lpAssetId) | |
89 | 211 | let wxEmissionPerBlock = rewardData[0] | |
90 | 212 | let maxFactor = rewardData[1] | |
91 | 213 | let totalLpStaked = rewardData[2] | |
92 | 214 | let lpStakedByUser = getIntOrZero(stakingContract, keyStakedByUser(userAddress, lpAssetId)) | |
93 | 215 | let boostingPower = (1 * MULT8) | |
94 | - | $Tuple2(nil, makeString(["%d%d%d", toString(wxEmissionPerBlock), toString(maxFactor), toString(totalLpStaked), toString(lpStakedByUser), toString(boostingPower)], SEP)) | |
216 | + | $Tuple2(nil, makeString(["%d%d%d%d%d", toString(wxEmissionPerBlock), toString(maxFactor), toString(totalLpStaked), toString(lpStakedByUser), toString(boostingPower)], SEP)) | |
95 | 217 | } | |
96 | 218 | ||
97 | 219 | ||
98 | 220 | ||
99 | 221 | @Callable(i) | |
100 | 222 | func claimedRewardREADONLY (userAddress) = { | |
223 | + | let factoryContract = readFactoryAddressOrFail() | |
224 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
225 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
101 | 226 | let lpList = readLpList() | |
102 | 227 | let prefix = "%s%d%d%d%s" | |
103 | 228 | func claimedRewardByLpAggregator (resultStr,nextLp) = { | |
104 | - | let rand = ((toInt(fromBase58String(nextLp)) % 100000) + (toInt(toBytes(userAddress)) % 100000)) | |
105 | - | let absRand = if ((0 > rand)) | |
106 | - | then -(rand) | |
107 | - | else rand | |
108 | - | let val = (((absRand % 100) * MULT8) + (absRand * (MULT8 / 100000))) | |
109 | - | let minRewardClaimed = 0 | |
110 | - | let boostRewardClaimed = 0 | |
229 | + | let claimedByUserMinRewardKEY = keyClaimedByUserMinReward(nextLp, userAddress) | |
230 | + | let claimedByUserBoostRewardKEY = keyClaimedByUserBoostReward(nextLp, userAddress) | |
231 | + | let minRewardClaimed = valueOrElse(getInteger(stakingContract, claimedByUserMinRewardKEY), 0) | |
232 | + | let boostRewardClaimed = valueOrElse(getInteger(stakingContract, claimedByUserBoostRewardKEY), 0) | |
111 | 233 | let gFeeClaimed = 0 | |
112 | 234 | makeString([(prefix + resultStr), nextLp, toString(minRewardClaimed), toString(boostRewardClaimed), toString(gFeeClaimed), "end"], SEP) | |
113 | 235 | } | |
114 | 236 | ||
115 | 237 | let result = { | |
116 | 238 | let $l = lpList | |
117 | 239 | let $s = size($l) | |
118 | 240 | let $acc0 = "%s" | |
119 | 241 | func 1 ($a,$i) = if (($i >= $s)) | |
120 | 242 | then $a | |
121 | 243 | else claimedRewardByLpAggregator($a, $l[$i]) | |
122 | 244 | ||
123 | 245 | func 2 ($a,$i) = if (($i >= $s)) | |
124 | 246 | then $a | |
125 | 247 | else throw("List size exceeds 10") | |
126 | 248 | ||
127 | 249 | 2(1(1(1(1(1(1(1(1(1(1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10) | |
128 | 250 | } | |
129 | 251 | $Tuple2(nil, ((result + SEP) + userAddress)) | |
130 | 252 | } | |
131 | 253 | ||
132 | 254 | ||
133 | 255 | ||
134 | 256 | @Callable(i) | |
135 | 257 | func calcBoostREADONLY (deltaWxAmount,deltaLockPeriodInBlocks,deltaLpAmount,lpAssetIdOpt,userAddressOpt) = { | |
258 | + | let factoryContract = readFactoryAddressOrFail() | |
259 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
260 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
261 | + | let EMPTYSTR = "empty" | |
262 | + | let maxLockDurationInBlocks = parseIntValue(split(getStringByAddressOrFail(boostingContract, keyBoostCfg()), SEP)[4]) | |
263 | + | let userNum = valueOrElse(getString(boostingContract, keyUser2NumMapping(userAddressOpt)), EMPTYSTR) | |
264 | + | let userAmount = valueOrElse(getInteger(boostingContract, keyLockParamUserAmount(userNum)), 0) | |
265 | + | let lockStart = valueOrElse(getInteger(boostingContract, keyLockParamStartBlock(userNum)), height) | |
266 | + | let lockDuration = valueOrElse(getInteger(boostingContract, keyLockParamDuration(userNum)), 0) | |
267 | + | let lockEnd = (lockStart + lockDuration) | |
268 | + | let remainingDuration = max([(lockEnd - height), 0]) | |
136 | 269 | let boost = if ((userAddressOpt != "")) | |
137 | 270 | then (3 * MULT8) | |
138 | 271 | else (1 * MULT8) | |
139 | - | let monthes18 = fraction(toBigInt(deltaLockPeriodInBlocks), MULT18, toBigInt((30 * 1440))) | |
140 | - | let coeff18 = (pow(monthes18, SCALE18, toBigInt(2), 0, SCALE18, HALFUP) / toBigInt(1000)) | |
141 | - | let gWxAmount = fraction(toBigInt(deltaWxAmount), coeff18, MULT18) | |
142 | - | $Tuple2(nil, makeString(["%d%d", toString(toInt(gWxAmount)), toString(boost)], SEP)) | |
272 | + | let SCALE = 1000 | |
273 | + | let userAmountNew = (userAmount + deltaWxAmount) | |
274 | + | let lockDurationNew = min([(remainingDuration + deltaLockPeriodInBlocks), maxLockDurationInBlocks]) | |
275 | + | let coeffX8 = fraction(lockDurationNew, MULT8, maxLockDurationInBlocks) | |
276 | + | let wxAmountStart = fraction(userAmountNew, coeffX8, MULT8) | |
277 | + | let lockEndHeight = (height + lockDurationNew) | |
278 | + | let scale8ParamK = -(fraction(wxAmountStart, SCALE, lockDurationNew)) | |
279 | + | let scale8ParamB = (fraction(wxAmountStart, SCALE, lockDurationNew) * lockEndHeight) | |
280 | + | let gWxAmount = (((scale8ParamK * height) + scale8ParamB) / SCALE) | |
281 | + | $Tuple2(nil, makeString(["%d%d", toString(gWxAmount), toString(boost)], SEP)) | |
143 | 282 | } | |
144 | 283 | ||
145 | 284 | ||
146 | 285 | ||
147 | 286 | @Callable(i) | |
148 | 287 | func wxEmissionStatsREADONLY () = { | |
149 | 288 | let ONEMULT = toString(MULT8) | |
150 | 289 | let ONE = "1" | |
151 | - | let totalWxReleased = 0 | |
152 | - | let totalWxLocked = 0 | |
153 | - | let locksDurationSumInBlocks = 0 | |
154 | - | let locksCount = 0 | |
290 | + | let factoryContract = readFactoryAddressOrFail() | |
291 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
292 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
293 | + | let emissionContract = getEmissionAddressOrFail(factoryCfg) | |
294 | + | let wxEmissionPerBlock = getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent()) | |
295 | + | let emissionStartBlock = getIntOrFail(emissionContract, keyEmissionStartBlock()) | |
296 | + | let passedBlocks = if ((emissionStartBlock > height)) | |
297 | + | then 0 | |
298 | + | else (height - emissionStartBlock) | |
299 | + | let teamEmDuration = (1440 * 365) | |
300 | + | let teamEmMax = (201000000 * MULT8) | |
301 | + | let teamEm = if ((passedBlocks > teamEmDuration)) | |
302 | + | then teamEmMax | |
303 | + | else fraction(teamEmMax, passedBlocks, teamEmDuration) | |
304 | + | let totalWxReleased = ((wxEmissionPerBlock * passedBlocks) + teamEm) | |
305 | + | let totalWxLocked = getIntOrZero(boostingContract, keyBoostingLockParamTotalAmount()) | |
306 | + | let locksDurationSumInBlocks = getIntOrZero(boostingContract, keyBoostingStatsLocksDurationSumInBlocks()) | |
307 | + | let locksCount = getIntOrZero(boostingContract, keyBoostingStatsLocksCount()) | |
155 | 308 | $Tuple2(nil, makeString(["%d%d%d%d", toString(totalWxReleased), toString(totalWxLocked), toString(locksDurationSumInBlocks), toString(locksCount)], SEP)) | |
309 | + | } | |
310 | + | ||
311 | + | ||
312 | + | ||
313 | + | @Callable(i) | |
314 | + | func lpStatsREADONLY (lpAsset) = { | |
315 | + | let factoryAddress = readFactoryAddressOrFail() | |
316 | + | let poolAddress = addressFromStringValue(getStringByAddressOrFail(factoryAddress, keyFactoryLpAssetToPoolContractAddress(lpAsset))) | |
317 | + | let cfg = if ($isInstanceOf(invoke(poolAddress, "getPoolConfigWrapperREADONLY", nil, nil), "List[Any]")) | |
318 | + | then invoke(poolAddress, "getPoolConfigWrapperREADONLY", nil, nil) | |
319 | + | else throw("Couldn't cast Any to List[Any]") | |
320 | + | let lpAssetId = fromBase58String(if ($isInstanceOf(cfg[idxPoolLPAssetId], "String")) | |
321 | + | then cfg[idxPoolLPAssetId] | |
322 | + | else throw("Couldn't cast Any to String")) | |
323 | + | let amtAssetId = if ($isInstanceOf(cfg[idxAmtAssetId], "String")) | |
324 | + | then cfg[idxAmtAssetId] | |
325 | + | else throw("Couldn't cast Any to String") | |
326 | + | let priceAssetId = if ($isInstanceOf(cfg[idxPriceAssetId], "String")) | |
327 | + | then cfg[idxPriceAssetId] | |
328 | + | else throw("Couldn't cast Any to String") | |
329 | + | let iAmtAssetId = if ($isInstanceOf(cfg[idxIAmtAssetId], "String")) | |
330 | + | then cfg[idxIAmtAssetId] | |
331 | + | else throw("Couldn't cast Any to String") | |
332 | + | let iPriceAssetId = if ($isInstanceOf(cfg[idxIPriceAssetId], "String")) | |
333 | + | then cfg[idxIPriceAssetId] | |
334 | + | else throw("Couldn't cast Any to String") | |
335 | + | let amtAssetDcm = parseIntValue(if ($isInstanceOf(cfg[idxAmtAssetDcm], "String")) | |
336 | + | then cfg[idxAmtAssetDcm] | |
337 | + | else throw("Couldn't cast Any to String")) | |
338 | + | let priceAssetDcm = parseIntValue(if ($isInstanceOf(cfg[idxPriceAssetDcm], "String")) | |
339 | + | then cfg[idxPriceAssetDcm] | |
340 | + | else throw("Couldn't cast Any to String")) | |
341 | + | let poolLPBalance = valueOrErrorMessage(assetInfo(lpAssetId), (("Asset " + toBase58String(lpAssetId)) + " doesn't exist")).quantity | |
342 | + | let accAmtAssetBalance = if ($isInstanceOf(invoke(poolAddress, "getAccBalanceWrapperREADONLY", [amtAssetId], nil), "Int")) | |
343 | + | then invoke(poolAddress, "getAccBalanceWrapperREADONLY", [amtAssetId], nil) | |
344 | + | else throw("Couldn't cast Any to Int") | |
345 | + | let accPriceAssetBalance = if ($isInstanceOf(invoke(poolAddress, "getAccBalanceWrapperREADONLY", [priceAssetId], nil), "Int")) | |
346 | + | then invoke(poolAddress, "getAccBalanceWrapperREADONLY", [priceAssetId], nil) | |
347 | + | else throw("Couldn't cast Any to Int") | |
348 | + | let pricesList = if ($isInstanceOf(invoke(poolAddress, "calcPricesWrapperREADONLY", [accAmtAssetBalance, accPriceAssetBalance, poolLPBalance], nil), "List[Any]")) | |
349 | + | then invoke(poolAddress, "calcPricesWrapperREADONLY", [accAmtAssetBalance, accPriceAssetBalance, poolLPBalance], nil) | |
350 | + | else throw("Couldn't cast Any to List[Any]") | |
351 | + | let curPrice = 0 | |
352 | + | let lpAmtAssetShare = if ($isInstanceOf(invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[1], MULT8], nil), "Int")) | |
353 | + | then invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[1], MULT8], nil) | |
354 | + | else throw("Couldn't cast Any to Int") | |
355 | + | let lpPriceAssetShare = if ($isInstanceOf(invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[2], MULT8], nil), "Int")) | |
356 | + | then invoke(poolAddress, "fromX18WrapperREADONLY", [pricesList[2], MULT8], nil) | |
357 | + | else throw("Couldn't cast Any to Int") | |
358 | + | let poolWeight = getIntegerValue(factoryAddress, keyFactoryPoolWeight(toString(poolAddress))) | |
359 | + | $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)) | |
360 | + | } | |
361 | + | ||
362 | + | ||
363 | + | ||
364 | + | @Callable(i) | |
365 | + | func gwxUserInfoREADONLY (userAddress) = { | |
366 | + | let factoryContract = readFactoryAddressOrFail() | |
367 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
368 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
369 | + | let gwxUserInfoLIST = if ($isInstanceOf(invoke(boostingContract, "gwxUserInfoREADONLY", [userAddress], nil), "List[Any]")) | |
370 | + | then invoke(boostingContract, "gwxUserInfoREADONLY", [userAddress], nil) | |
371 | + | else throw("Couldn't cast Any to List[Any]") | |
372 | + | let gwxAmount = if ($isInstanceOf(gwxUserInfoLIST[0], "Int")) | |
373 | + | then gwxUserInfoLIST[0] | |
374 | + | else throw("Couldn't cast Any to Int") | |
375 | + | $Tuple2(nil, makeString(["%d", toString(gwxAmount)], SEP)) | |
156 | 376 | } | |
157 | 377 | ||
158 | 378 |
github/deemru/w8io/3ef1775 64.03 ms ◑