tx · 2d6bLXky5ew8shWypfaoKRMWAdjLP6iYm4VAi58pCxiw

3Mx3zmXrMcLFCafMuPtXAzR4ZPVeZYb6qLz:  -0.03500000 Waves

2022.06.29 14:05 [2117642] smart account 3Mx3zmXrMcLFCafMuPtXAzR4ZPVeZYb6qLz > SELF 0.00000000 Waves

{ "type": 13, "id": "2d6bLXky5ew8shWypfaoKRMWAdjLP6iYm4VAi58pCxiw", "fee": 3500000, "feeAssetId": null, "timestamp": 1656500733711, "version": 2, "chainId": 84, "sender": "3Mx3zmXrMcLFCafMuPtXAzR4ZPVeZYb6qLz", "senderPublicKey": "D28XoueZWsMfm8Y5pa6C5ZFuYoWgre2Wm8tzJANJgMnq", "proofs": [ "2xAD8gG289RvjAseK661n7Vyw2qbnHZKpCbKFiVui5zDKazMP9cq1YZCkfsX43WLWyuPrQfKxsbkaJzHevjHgnez" ], "script": "base64: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a2luZ0RhcHAAAAAAAAAAAAEAAAAAGUlkeEZhY3RvcnlDZmdCb29zdGluZ0RhcHAAAAAAAAAAAAIAAAAAFElkeEZhY3RvcnlDZmdJZG9EYXBwAAAAAAAAAAADAAAAABVJZHhGYWN0b3J5Q2ZnVGVhbURhcHAAAAAAAAAAAAQAAAAAGUlkeEZhY3RvcnlDZmdFbWlzc2lvbkRhcHAAAAAAAAAAAAUAAAAAFUlkeEZhY3RvcnlDZmdSZXN0RGFwcAAAAAAAAAAABgAAAAAZSWR4RmFjdG9yeUNmZ1NsaXBwYWdlRGFwcAAAAAAAAAAABwEAAAANa2V5RmFjdG9yeUNmZwAAAAACAAAAESVzX19mYWN0b3J5Q29uZmlnAQAAABNrZXlNYW5hZ2VyUHVibGljS2V5AAAAAAIAAAAUJXNfX21hbmFnZXJQdWJsaWNLZXkBAAAAFGtleU1pZ3JhdG9yUHVibGljS2V5AAAAAAIAAAAVJXNfX21pZ3JhdG9yUHVibGljS2V5AQAAABprZXlQZW5kaW5nTWFuYWdlclB1YmxpY0tleQAAAAACAAAAGyVzX19wZW5kaW5nTWFuYWdlclB1YmxpY0tleQEAAAAWa2V5U3RhYmxlUG9vbEFkZG9uQWRkcgAAAAACAAAAFyVzX19zdGFibGVQb29sQWRkb25BZGRyAQAAABprZXlGYWN0b3J5THAyQXNzZXRzTWFwcGluZwAAAAEAAAAKbHBBc3NldFN0cgkABLkAAAACCQAETAAAAAICAAAABiVzJXMlcwkABEwAAAACBQAAAApscEFzc2V0U3RyCQAETAAAAAICAAAAHm1hcHBpbmdzX19scEFzc2V0MlBvb2xDb250cmFjdAUAAAADbmlsBQAAAANTRVABAAAAEGtleUZhY3RvcnlMcExpc3QAAAAAAgAAABAlc19fbHBUb2tlbnNMaXN0AQAAACZrZXlGYWN0b3J5THBBc3NldFRvUG9vbENvbnRyYWN0QWRkcmVzcwAAAAEAAAAKbHBBc3NldFN0cgkABLkAAAACCQAETAAAAAICAAAABiVzJXMlcwkABEwAAAACBQAAAApscEFzc2V0U3RyCQAETAAAAAICAAAAHm1hcHBpbmdzX19scEFzc2V0MlBvb2xDb250cmFjdAUAAAADbmlsBQAAAANTRVABAAAAFGtleUZhY3RvcnlQb29sV2VpZ2h0AAAAAQAAAA9jb250cmFjdEFkZHJlc3MJAAS5AAAAAgkABEwAAAACAgAAAAQlcyVzCQAETAAAAAICAAAACnBvb2xXZWlnaHQJAARMAAAAAgUAAAAPY29udHJhY3RBZGRyZXNzBQAAAANuaWwFAAAAA1NFUAEAAAAKcmVhZExwTGlzdAAAAAEAAAAHZmFjdG9yeQkABLUAAAACCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHQAAAAIFAAAAB2ZhY3RvcnkJAQAAABBrZXlGYWN0b3J5THBMaXN0AAAAAAIAAAAABQAAAANTRVABAAAAFHJlYWRGYWN0b3J5Q2ZnT3JGYWlsAAAAAQAAAAdmYWN0b3J5CQAEtQAAAAIJAQAAABhnZXRTdHJpbmdCeUFkZHJlc3NPckZhaWwAAAACBQAAAAdmYWN0b3J5CQEAAAANa2V5RmFjdG9yeUNmZwAAAAAFAAAAA1NFUAEAAAAYZ2V0Qm9vc3RpbmdBZGRyZXNzT3JGYWlsAAAAAQAAAApmYWN0b3J5Q2ZnCQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABCQABkQAAAAIFAAAACmZhY3RvcnlDZmcFAAAAGUlkeEZhY3RvcnlDZmdCb29zdGluZ0RhcHABAAAAGGdldEVtaXNzaW9uQWRkcmVzc09yRmFpbAAAAAEAAAAKZmFjdG9yeUNmZwkBAAAAEUBleHRyTmF0aXZlKDEwNjIpAAAAAQkAAZEAAAACBQAAAApmYWN0b3J5Q2ZnBQAAABlJZHhGYWN0b3J5Q2ZnRW1pc3Npb25EYXBwAQAAABdnZXRTdGFraW5nQWRkcmVzc09yRmFpbAAAAAEAAAAKZmFjdG9yeUNmZwkBAAAAEUBleHRyTmF0aXZlKDEwNjIpAAAAAQkAAZEAAAACBQAAAApmYWN0b3J5Q2ZnBQAAABhJZHhGYWN0b3J5Q2ZnU3Rha2luZ0RhcHABAAAAHmtleUVtaXNzaW9uUmF0ZVBlckJsb2NrQ3VycmVudAAAAAACAAAAGyVzJXNfX3JhdGVQZXJCbG9ja19fY3VycmVudAEAAAAha2V5RW1pc3Npb25SYXRlUGVyQmxvY2tNYXhDdXJyZW50AAAAAAIAAAAeJXMlc19fcmF0ZVBlckJsb2NrTWF4X19jdXJyZW50AQAAABVrZXlFbWlzc2lvblN0YXJ0QmxvY2sAAAAAAgAAABolcyVzX19lbWlzc2lvbl9fc3RhcnRCbG9jawEAAAAba2V5RW1pc3Npb25EdXJhdGlvbkluQmxvY2tzAAAAAAIAAAAYJXMlc19fZW1pc3Npb25fX2R1cmF0aW9uAQAAABNrZXlFbWlzc2lvbkVuZEJsb2NrAAAAAAIAAAAYJXMlc19fZW1pc3Npb25fX2VuZEJsb2NrAQAAAA9rZXlTdGFrZWRCeVVzZXIAAAACAAAADnVzZXJBZGRyZXNzU3RyAAAADGxwQXNzZXRJZFN0cgkABLkAAAACCQAETAAAAAICAAAADiVzJXMlc19fc3Rha2VkCQAETAAAAAIFAAAADnVzZXJBZGRyZXNzU3RyCQAETAAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAADbmlsBQAAAANTRVABAAAADmtleVN0YWtlZFRvdGFsAAAAAQAAAAxscEFzc2V0SWRTdHIJAAEsAAAAAgIAAAAXJXMlcyVzX19zdGFrZWRfX3RvdGFsX18FAAAADGxwQXNzZXRJZFN0cgEAAAAQa2V5Q2xhaW1lZEJ5VXNlcgAAAAIAAAAMbHBBc3NldElkU3RyAAAADnVzZXJBZGRyZXNzU3RyCQAEuQAAAAIJAARMAAAAAgIAAAAPJXMlcyVzX19jbGFpbWVkCQAETAAAAAIFAAAADnVzZXJBZGRyZXNzU3RyCQAETAAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAADbmlsBQAAAANTRVABAAAAGWtleUNsYWltZWRCeVVzZXJNaW5SZXdhcmQAAAACAAAADGxwQXNzZXRJZFN0cgAAAA51c2VyQWRkcmVzc1N0cgkABLkAAAACCQAETAAAAAICAAAAGCVzJXMlc19fY2xhaW1lZE1pblJld2FyZAkABEwAAAACBQAAAA51c2VyQWRkcmVzc1N0cgkABEwAAAACBQAAAAxscEFzc2V0SWRTdHIFAAAAA25pbAUAAAADU0VQAQAAABtrZXlDbGFpbWVkQnlVc2VyQm9vc3RSZXdhcmQAAAACAAAADGxwQXNzZXRJZFN0cgAAAA51c2VyQWRkcmVzc1N0cgkABLkAAAACCQAETAAAAAICAAAAGiVzJXMlc19fY2xhaW1lZEJvb3N0UmV3YXJkCQAE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YXN0QmxvY2sAAAAJdGltZXN0YW1wCQAETAAAAAIJAAGkAAAAAQUAAAAGYW1vdW50BQAAAANuaWwFAAAAA1NFUAEAAAAVT3BlcmF0aW9uSGlzdG9yeUVudHJ5AAAABQAAAAR0eXBlAAAAC3VzZXJBZGRyZXNzAAAACWxwQXNzZXRJZAAAAAZhbW91bnQAAAAEdHhJZAkBAAAAC1N0cmluZ0VudHJ5AAAAAgkBAAAAGWtleU9wZXJhdGlvbkhpc3RvcnlSZWNvcmQAAAADBQAAAAR0eXBlBQAAAAt1c2VyQWRkcmVzcwkAAlgAAAABBQAAAAR0eElkCQEAAAATZm9ybWF0SGlzdG9yeVJlY29yZAAAAAQFAAAAC3VzZXJBZGRyZXNzBQAAAAlscEFzc2V0SWQFAAAABHR5cGUFAAAABmFtb3VudAAAAAAOZmFjdG9yeUFkZHJlc3MJAQAAAA9nZXRTdHJpbmdPckZhaWwAAAABCQEAAAARa2V5RmFjdG9yeUFkZHJlc3MAAAAAAAAAAA9mYWN0b3J5Q29udHJhY3QJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEFAAAADmZhY3RvcnlBZGRyZXNzAAAAAApmYWN0b3J5Q2ZnCQEAAAAUcmVhZEZhY3RvcnlDZmdPckZhaWwAAAABBQAAAA9mYWN0b3J5Q29udHJhY3QAAAAAEGVtaXNzaW9uQ29udHJhY3QJAQAAABhnZXRFbWlzc2lvbkFkZHJlc3NPckZhaWwAAAABBQAAAApmYWN0b3J5Q2ZnAAAAABBib29zdGluZ0NvbnRyYWN0CQEAAAAYZ2V0Qm9vc3RpbmdBZGRyZXNzT3JGYWlsAAAAAQUAAAAKZmFjdG9yeUNmZwEAAAAbY2FsY1d4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0AAAABAAAAAxzdGFrZWRCeVVzZXIAAAAmd3hQZXJMcEludGVncmFsVXNlckxhc3RVcGRIZWlnaHRPclplcm8AAAASd3hQZXJMcEludGVncmFsTmV3AAAAGnd4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0S0VZAwMJAAAAAAAAAgUAAAAmd3hQZXJMcEludGVncmFsVXNlckxhc3RVcGRIZWlnaHRPclplcm8AAAAAAAAAAAAJAABmAAAAAgUAAAAMc3Rha2VkQnlVc2VyAAAAAAAAAAAABwAAAAAAAAAAAAMJAAAAAAAAAgUAAAAMc3Rha2VkQnlVc2VyAAAAAAAAAAAABQAAABJ3eFBlckxwSW50ZWdyYWxOZXcDAwkAAGYAAAACBQAAACZ3eFBlckxwSW50ZWdyYWxVc2VyTGFzdFVwZEhlaWdodE9yWmVybwAAAAAAAAAAAAkAAGYAAAACBQAAAAxzdGFrZWRCeVVzZXIAAAAAAAAAAAAHCQEAAAAMZ2V0SW50T3JGYWlsAAAAAgUAAAAEdGhpcwUAAAAad3hQZXJMcEludGVncmFsVXNlckxhc3RLRVkJAAACAAAAAQIAAAAtY2FsY1d4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0OiB1bmV4cGVjdGVkIHN0YXRlAQAAABRyZWZyZXNoUG9vbElOVEVHUkFMUwAAAAMAAAAMbHBBc3NldElkU3RyAAAADnBvb2xBZGRyZXNzU3RyAAAADWxwRGVsdGFBbW91bnQEAAAADnN0YWtlZFRvdGFsS0VZCQEAAAAOa2V5U3Rha2VkVG90YWwAAAABBQAAAAxscEFzc2V0SWRTdHIEAAAAC3N0YWtlZFRvdGFsCQEAAAAKcmVhZFN0YWtlZAAAAAEFAAAADnN0YWtlZFRvdGFsS0VZBAAAAApwb29sV2VpZ2h0CQEAAAARQGV4dHJOYXRpdmUoMTA1MCkAAAACBQAAAA9mYWN0b3J5Q29udHJhY3QJAQAAABRrZXlGYWN0b3J5UG9vbFdlaWdodAAAAAEFAAAADnBvb2xBZGRyZXNzU3RyBAAAABJlbWlzc2lvblN0YXJ0QmxvY2sJAQAAAAxnZXRJbnRPckZhaWwAAAACBQAAABBlbWlzc2lvbkNvbnRyYWN0CQEAAAAVa2V5RW1pc3Npb25TdGFydEJsb2NrAAAAAAQAAAAFTVVMVDMAAAAAAAAAA+gEAAAAFHd4RW1pc3Npb25QZXJCbG9ja1gzCQAAaAAAAAIJAQAAAAxnZXRJbnRPckZhaWwAAAACBQAAABBlbWlzc2lvbkNvbnRyYWN0CQEAAAAea2V5RW1pc3Npb25SYXRlUGVyQmxvY2tDdXJyZW50AAAAAAUAAAAFTVVMVDMEAAAAGHBvb2xXeEVtaXNzaW9uUGVyQmxvY2tYMwkAAGsAAAADBQAAABR3eEVtaXNzaW9uUGVyQmxvY2tYMwUAAAAKcG9vbFdlaWdodAkAAGgAAAACBQAAAA5QT09MV0VJR0hUTVVMVAAAAAAAAAAAAwQAAAASd3hQZXJMcEludGVncmFsS0VZCQEAAAASa2V5V3hQZXJMcEludGVncmFsAAAAAQUAAAAMbHBBc3NldElkU3RyBAAAAB93eFBlckxwSW50ZWdyYWxMYXN0VXBkSGVpZ2h0S0VZCQEAAAAfa2V5V3hQZXJMcEludGVncmFsTGFzdFVwZEhlaWdodAAAAAEFAAAADGxwQXNzZXRJZFN0cgQAAAAKd3hQZXJMcEtFWQkBAAAACmtleVd4UGVyTHAAAAABBQAAAAxscEFzc2V0SWRTdHIEAAAAHHd4UGVyTHBJbnRlZ3JhbExhc3RVcGRIZWlnaHQJAQAAAA9nZXRJbnRPckRlZmF1bHQAAAADBQAAAAR0aGlzBQAAAB93eFBlckxwSW50ZWdyYWxMYXN0VXBkSGVpZ2h0S0VZBQAAABJlbWlzc2lvblN0YXJ0QmxvY2sEAAAAD3d4UGVyTHBJbnRlZ3JhbAkBAAAADGdldEludE9yWmVybwAAAAIFAAAABHRoaXMFAAAAEnd4UGVyTHBJbnRlZ3JhbEtFWQQAAAAPd3hQZXJMcE9yWmVyb1gzAAAAAAAAAAAABAAAAAJkaAkAAZYAAAABCQAETAAAAAIJAABlAAAAAgUAAAAGaGVpZ2h0BQAAABx3eFBlckxwSW50ZWdyYWxMYXN0VXBkSGVpZ2h0CQAETAAAAAIAAAAAAAAAAAAFAAAAA25pbAQAAAAJd3hQZXJMcFgzAwkBAAAAAiE9AAAAAgUAAAAPd3hQZXJMcE9yWmVyb1gzAAAAAAAAAAAABQAAAA93eFBlckxwT3JaZXJvWDMJAABrAAAAAwUAAAAYcG9vbFd4RW1pc3Npb25QZXJCbG9ja1gzBQAAAAVNVUxUOAUAAAALc3Rha2VkVG90YWwEAAAADnN0YWtlZFRvdGFsTmV3CQAAZAAAAAIFAAAAC3N0YWtlZFRvdGFsBQAAAA1scERlbHRhQW1vdW50BAAAABJ3eFBlckxwSW50ZWdyYWxOZXcJAABkAAAAAgUAAAAPd3hQZXJMcEludGVncmFsCQAAaAAAAAIFAAAACXd4UGVyTHBYMwUAAAACZGgEAAAADHd4UGVyTHBYM05ldwkAAGkAAAACBQAAABhwb29sV3hFbWlzc2lvblBlckJsb2NrWDMFAAAADnN0YWtlZFRvdGFsTmV3BAAAAB93eFBlckxwSW50ZWdyYWxMYXN0VXBkSGVpZ2h0TmV3BQAAAAZoZWlnaHQEAAAABWRlYnVnCQAEuQAAAAIJAARMAAAAAgkAAaQAAAABBQAAABJ3eFBlckxwSW50ZWdyYWxOZXcJAARMAAAAAgIAAAADZGg9CQAETAAAAAIJAAGkAAAAAQUAAAACZGgJAARMAAAAAgIAAAAKd3hQZXJMcFgzPQkABEwAAAACCQABpAAAAAEFAAAACXd4UGVyTHBYMwkABEwAAAACAgAAAAxzdGFrZWRUb3RhbD0JAARMAAAAAgkAAaQAAAABBQAAAAtzdGFrZWRUb3RhbAkABEwAAAACCQABpAAAAAEFAAAAGHBvb2xXeEVtaXNzaW9uUGVyQmxvY2tYMwkABEwAAAACCQABpAAAAAEFAAAAFHd4RW1pc3Npb25QZXJCbG9ja1gzCQAETAAAAAICAAAAC3Bvb2xXZWlnaHQ9CQAETAAAAAIJAAGkAAAAAQUAAAAKcG9vbFdlaWdodAUAAAADbmlsAgAAAAI6OgkABRUAAAADBQAAABJ3eFBlckxwSW50ZWdyYWxOZXcJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAEnd4UGVyTHBJbnRlZ3JhbEtFWQUAAAASd3hQZXJMcEludGVncmFsTmV3CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAB93eFBlckxwSW50ZWdyYWxMYXN0VXBkSGVpZ2h0S0VZBQAAAB93eFBlckxwSW50ZWdyYWxMYXN0VXBkSGVpZ2h0TmV3CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAp3eFBlckxwS0VZBQAAAAx3eFBlckxwWDNOZXcFAAAAA25pbAUAAAAFZGVidWcBAAAAEHJlZnJlc2hJTlRFR1JBTFMAAAAEAAAADGxwQXNzZXRJZFN0cgAAAA51c2VyQWRkcmVzc1N0cgAAAA5wb29sQWRkcmVzc1N0cgAAAA1scERlbHRhQW1vdW50BAAAAA0kdDAxMDkxMTExMDMzCQEAAAAUcmVmcmVzaFBvb2xJTlRFR1JBTFMAAAADBQAAAAxscEFzc2V0SWRTdHIFAAAADnBvb2xBZGRyZXNzU3RyBQAAAA1scERlbHRhQW1vdW50BAAAABJ3eFBlckxwSW50ZWdyYWxOZXcIBQAAAA0kdDAxMDkxMTExMDMzAAAAAl8xBAAAABFwb29sSW50ZWdyYWxTVEFURQgFAAAADSR0MDEwOTExMTEwMzMAAAACXzIEAAAACXBvb2xERUJVRwgFAAAADSR0MDEwOTExMTEwMzMAAAACXzMEAAAABU1VTFQzAAAAAAAAAAPoBAAAAA9zdGFrZWRCeVVzZXJLRVkJAQAAAA9rZXlTdGFrZWRCeVVzZXIAAAACBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAMbHBBc3NldElkU3RyBAAAAAxzdGFrZWRCeVVzZXIJAQAAAApyZWFkU3Rha2VkAAAAAQUAAAAPc3Rha2VkQnlVc2VyS0VZBAAAABB3eFRvQ2xhaW1Vc2VyS0VZCQEAAAAQa2V5V3hUb0NsYWltVXNlcgAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAAOdXNlckFkZHJlc3NTdHIEAAAAI3d4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0VXBkSGVpZ2h0S0VZCQEAAAAja2V5V3hQZXJMcEludGVncmFsVXNlckxhc3RVcGRIZWlnaHQAAAACBQAAAAxscEFzc2V0SWRTdHIFAAAADnVzZXJBZGRyZXNzU3RyBAAAABp3eFBlckxwSW50ZWdyYWxVc2VyTGFzdEtFWQkBAAAAGmtleVd4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0AAAAAgUAAAAMbHBBc3NldElkU3RyBQAAAA51c2VyQWRkcmVzc1N0cgQAAAANd3hUb0NsYWltVXNlcgkBAAAADGdldEludE9yWmVybwAAAAIFAAAABHRoaXMFAAAAEHd4VG9DbGFpbVVzZXJLRVkEAAAAJnd4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0VXBkSGVpZ2h0T3JaZXJvCQEAAAAMZ2V0SW50T3JaZXJvAAAAAgUAAAAEdGhpcwUAAAAjd3hQZXJMcEludGVncmFsVXNlckxhc3RVcGRIZWlnaHRLRVkEAAAAF3d4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0CQEAAAAbY2FsY1d4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0AAAABAUAAAAMc3Rha2VkQnlVc2VyBQAAACZ3eFBlckxwSW50ZWdyYWxVc2VyTGFzdFVwZEhlaWdodE9yWmVybwUAAAASd3hQZXJMcEludGVncmFsTmV3BQAAABp3eFBlckxwSW50ZWdyYWxVc2VyTGFzdEtFWQQAAAAQd3hUb0NsYWltVXNlck5ldwkAAGQAAAACBQAAAA13eFRvQ2xhaW1Vc2VyCQAAawAAAAMJAABlAAAAAgUAAAASd3hQZXJMcEludGVncmFsTmV3BQAAABd3eFBlckxwSW50ZWdyYWxVc2VyTGFzdAUAAAAMc3Rha2VkQnlVc2VyCQAAaAAAAAIFAAAABU1VTFQ4BQAAAAVNVUxUMwQAAAAad3hQZXJMcEludGVncmFsVXNlckxhc3ROZXcFAAAAEnd4UGVyTHBJbnRlZ3JhbE5ldwQAAAAjd3hQZXJMcEludGVncmFsVXNlckxhc3RVcGRIZWlnaHROZXcFAAAABmhlaWdodAQAAAAFZGVidWcJAAS5AAAAAgkABEwAAAACAgAAAA13eFRvQ2xhaW1Vc2VyCQAETAAAAAIJAAGkAAAAAQUAAAANd3hUb0NsYWltVXNlcgkABEwAAAACAgAAAAx3eFBlckxwSW50ZWcJAARMAAAAAgkAAaQAAAABBQAAACZ3eFBlckxwSW50ZWdyYWxVc2VyTGFzdFVwZEhlaWdodE9yWmVybwkABEwAAAACAgAAABJ3eFBlckxwSW50ZWdyYWxOZXcJAARMAAAAAgkAAaQAAAABBQAAABJ3eFBlckxwSW50ZWdyYWxOZXcJAARMAAAAAgIAAAAQd3hUb0NsYWltVXNlck5ldwkABEwAAAACCQABpAAAAAEFAAAAEHd4VG9DbGFpbVVzZXJOZXcJAARMAAAAAgIAAAAXd3hQZXJMcEludGVncmFsVXNlckxhc3QJAARMAAAAAgkAAaQAAAABBQAAABd3eFBlckxwSW50ZWdyYWxVc2VyTGFzdAkABEwAAAACCQABpAAAAAEFAAAADHN0YWtlZEJ5VXNlcgkABEwAAAACAgAAAApwb29sREVCVUc9CQAETAAAAAIFAAAACXBvb2xERUJVRwkABEwAAAACAgAAAAdoZWlnaHQ9CQAETAAAAAIJAAGkAAAAAQUAAAAGaGVpZ2h0BQAAAANuaWwCAAAAAjo6CQAFFQAAAAMFAAAAEHd4VG9DbGFpbVVzZXJOZXcJAAROAAAAAgUAAAARcG9vbEludGVncmFsU1RBVEUJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAEHd4VG9DbGFpbVVzZXJLRVkFAAAAEHd4VG9DbGFpbVVzZXJOZXcJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAI3d4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0VXBkSGVpZ2h0S0VZBQAAACN3eFBlckxwSW50ZWdyYWxVc2VyTGFzdFVwZEhlaWdodE5ldwkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAad3hQZXJMcEludGVncmFsVXNlckxhc3RLRVkFAAAAGnd4UGVyTHBJbnRlZ3JhbFVzZXJMYXN0TmV3BQAAAANuaWwFAAAABWRlYnVnAQAAABZtYW5hZ2VyUHVibGljS2V5T3JVbml0AAAAAAQAAAAHJG1hdGNoMAkABCIAAAABCQEAAAATa2V5TWFuYWdlclB1YmxpY0tleQAAAAADCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABlN0cmluZwQAAAABcwUAAAAHJG1hdGNoMAkAAlkAAAABBQAAAAFzAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAARVbml0BQAAAAR1bml0CQAAAgAAAAECAAAAC01hdGNoIGVycm9yAQAAABdtaWdyYXRvclB1YmxpY0tleU9yVW5pdAAAAAAEAAAAByRtYXRjaDAJAAQiAAAAAQkBAAAAFGtleU1pZ3JhdG9yUHVibGljS2V5AAAAAAMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAGU3RyaW5nBAAAAAFzBQAAAAckbWF0Y2gwCQACWQAAAAEFAAAAAXMDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABFVuaXQFAAAABHVuaXQJAAACAAAAAQIAAAALTWF0Y2ggZXJyb3IBAAAAHXBlbmRpbmdNYW5hZ2VyUHVibGljS2V5T3JVbml0AAAAAAQAAAAHJG1hdGNoMAkABCIAAAABCQEAAAAaa2V5UGVuZGluZ01hbmFnZXJQdWJsaWNLZXkAAAAAAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAAZTdHJpbmcEAAAAAXMFAAAAByRtYXRjaDAJ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IG11c3QgYmUgYXR0YWNoZWQEAAAAA3BtdAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAQAAAAJbHBBc3NldElkCQEAAAAFdmFsdWUAAAABCAUAAAADcG10AAAAB2Fzc2V0SWQEAAAADGxwQXNzZXRJZFN0cgkAAlgAAAABBQAAAAlscEFzc2V0SWQEAAAABmFtb3VudAgFAAAAA3BtdAAAAAZhbW91bnQEAAAADnBvb2xBZGRyZXNzU3RyCQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQdAAAAAgUAAAAPZmFjdG9yeUNvbnRyYWN0CQEAAAAaa2V5RmFjdG9yeUxwMkFzc2V0c01hcHBpbmcAAAABBQAAAAxscEFzc2V0SWRTdHIJAAEsAAAAAgIAAAAVdW5zdXBwb3J0ZWQgbHAgYXNzZXQgBQAAAAxscEFzc2V0SWRTdHIEAAAACWNhbGxlclN0cgkABCUAAAABCAUAAAABaQAAAAZjYWxsZXIEAAAADnVzZXJBZGRyZXNzU3RyAwkAAAAAAAACBQAAAAljYWxsZXJTdHIFAAAADnBvb2xBZGRyZXNzU3RyCQAEJQAAAAEIBQAAAAFpAAAADG9yaWdpbkNhbGxlcgUAAAAJY2FsbGVyU3RyBAAAAA9zdGFrZWRCeVVzZXJLRVkJAQAAAA9rZXlTdGFrZWRCeVVzZXIAAAACBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAMbHBBc3NldElkU3RyBAAAAA5zdGFrZWRUb3RhbEtFWQkBAAAADmtleVN0YWtlZFRvdGFsAAAAAQUAAAAMbHBBc3NldElkU3RyBAAAAAxzdGFrZWRCeVVzZXIJAQAAAApyZWFkU3Rha2VkAAAAAQUAAAAPc3Rha2VkQnlVc2VyS0VZBAAAAAtzdGFrZWRUb3RhbAkBAAAACnJlYWRTdGFrZWQAAAABBQAAAA5zdGFrZWRUb3RhbEtFWQQAAAANJHQwMTUzMjQxNTQ0MQkBAAAAEHJlZnJlc2hJTlRFR1JBTFMAAAAEBQAAAAxscEFzc2V0SWRTdHIFAAAADnVzZXJBZGRyZXNzU3RyBQAAAA5wb29sQWRkcmVzc1N0cgUAAAAGYW1vdW50BAAAABB3eFRvQ2xhaW1Vc2VyTmV3CAUAAAANJHQwMTUzMjQxNTQ0MQAAAAJfMQQAAAANaW50ZWdyYWxTVEFURQgFAAAADSR0MDE1MzI0MTU0NDEAAAACXzIEAAAABWRlYnVnCAUAAAANJHQwMTUzMjQxNTQ0MQAAAAJfMwkABE4AAAACCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAA9zdGFrZWRCeVVzZXJLRVkJAABkAAAAAgUAAAAMc3Rha2VkQnlVc2VyBQAAAAZhbW91bnQJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAADnN0YWtlZFRvdGFsS0VZCQAAZAAAAAIFAAAAC3N0YWtlZFRvdGFsBQAAAAZhbW91bnQJAARMAAAAAgkBAAAAFU9wZXJhdGlvbkhpc3RvcnlFbnRyeQAAAAUCAAAABXN0YWtlBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAMbHBBc3NldElkU3RyBQAAAAZhbW91bnQIBQAAAAFpAAAADXRyYW5zYWN0aW9uSWQFAAAAA25pbAUAAAANaW50ZWdyYWxTVEFURQAAAAFpAQAAAAd1bnN0YWtlAAAAAgAAAAxscEFzc2V0SWRTdHIAAAAGYW1vdW50BAAAAAlscEFzc2V0SWQJAAJZAAAAAQUAAAAMbHBBc3NldElkU3RyBAAAAA5wb29sQWRkcmVzc1N0cgkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEHQAAAAIFAAAAD2ZhY3RvcnlDb250cmFjdAkBAAAAGmtleUZhY3RvcnlMcDJBc3NldHNNYXBwaW5nAAAAAQUAAAAMbHBBc3NldElkU3RyCQABLAAAAAICAAAAFXVuc3VwcG9ydGVkIGxwIGFzc2V0IAUAAAAMbHBBc3NldElkU3RyBAAAAAlwb29sQWRkb24JAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQdAAAAAgUAAAAEdGhpcwkBAAAAFmtleVN0YWJsZVBvb2xBZGRvbkFkZHIAAAAABQAAAA5wb29sQWRkcmVzc1N0cgQAAAAJY2FsbGVyU3RyCQAEJQAAAAEIBQAAAAFpAAAABmNhbGxlcgQAAAAOdXNlckFkZHJlc3NTdHIDAwkAAAAAAAACBQAAAAljYWxsZXJTdHIFAAAADnBvb2xBZGRyZXNzU3RyBgkAAAAAAAACBQAAAAljYWxsZXJTdHIFAAAACXBvb2xBZGRvbgkABCUAAAABCAUAAAABaQAAAAxvcmlnaW5DYWxsZXIFAAAACWNhbGxlclN0cgQAAAAPc3Rha2VkQnlVc2VyS0VZCQEAAAAPa2V5U3Rha2VkQnlVc2VyAAAAAgUAAAAOdXNlckFkZHJlc3NTdHIFAAAADGxwQXNzZXRJZFN0cgQAAAAOc3Rha2VkVG90YWxLRVkJAQAAAA5rZXlTdGFrZWRUb3RhbAAAAAEFAAAADGxwQXNzZXRJZFN0cgQAAAAMc3Rha2VkQnlVc2VyCQEAAAAKcmVhZFN0YWtlZAAAAAEFAAAAD3N0YWtlZEJ5VXNlcktFWQQAAAALc3Rha2VkVG90YWwJAQAAAApyZWFkU3Rha2VkAAAAAQUAAAAOc3Rha2VkVG90YWxLRVkEAAAADSR0MDE2NTc4MTY2OTYJAQAAABByZWZyZXNoSU5URUdSQUxTAAAABAUAAAAMbHBBc3NldElkU3RyBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAOcG9vbEFkZHJlc3NTdHIJAQAAAAEtAAAAAQUAAAAGYW1vdW50BAAAABB3eFRvQ2xhaW1Vc2VyTmV3CAUAAAANJHQwMTY1NzgxNjY5NgAAAAJfMQQAAAANaW50ZWdyYWxTVEFURQgFAAAADSR0MDE2NTc4MTY2OTYAAAACXzIEAAAABWRlYnVnCAUAAAANJHQwMTY1NzgxNjY5NgAAAAJfMwMJAABmAAAAAgUAAAAGYW1vdW50BQAAAAxzdGFrZWRCeVVzZXIJAAACAAAAAQIAAAAkcGFzc2VkIGFtb3VudCBpcyBsZXNzIHRoZW4gYXZhaWxhYmxlCQAETgAAAAIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAD3N0YWtlZEJ5VXNlcktFWQkAAGUAAAACBQAAAAxzdGFrZWRCeVVzZXIFAAAABmFtb3VudAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAOc3Rha2VkVG90YWxLRVkJAABlAAAAAgUAAAALc3Rha2VkVG90YWwFAAAABmFtb3VudAkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAABmFtb3VudAUAAAAJbHBBc3NldElkCQAETAAAAAIJAQAAABVPcGVyYXRpb25IaXN0b3J5RW50cnkAAAAFAgAAAAd1bnN0YWtlBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAMbHBBc3NldElkU3RyBQAAAAZhbW91bnQIBQAAAAFpAAAADXRyYW5zYWN0aW9uSWQFAAAAA25pbAUAAAANaW50ZWdyYWxTVEFURQAAAAFpAQAAAAdjbGFpbVd4AAAAAQAAAAxscEFzc2V0SWRTdHIEAAAAC3VzZXJBZGRyZXNzCAUAAAABaQAAAAZjYWxsZXIEAAAADnVzZXJBZGRyZXNzU3RyCQAEJQAAAAEIBQAAAAFpAAAABmNhbGxlcgQAAAAOcG9vbEFkZHJlc3NTdHIJAQAAABhnZXRTdHJpbmdCeUFkZHJlc3NPckZhaWwAAAACBQAAAA9mYWN0b3J5Q29udHJhY3QJAQAAACZrZXlGYWN0b3J5THBBc3NldFRvUG9vbENvbnRyYWN0QWRkcmVzcwAAAAEFAAAADGxwQXNzZXRJZFN0cgQAAAAQY2xhaW1lZEJ5VXNlcktFWQkBAAAAEGtleUNsYWltZWRCeVVzZXIAAAACBQAAAAxscEFzc2V0SWRTdHIFAAAADnVzZXJBZGRyZXNzU3RyBAAAAA9jbGFpbWVkVG90YWxLRVkJAQAAAA9rZXlDbGFpbWVkVG90YWwAAAABBQAAAAxscEFzc2V0SWRTdHIEAAAAGWNsYWltZWRCeVVzZXJNaW5SZXdhcmRLRVkJAQAAABlrZXlDbGFpbWVkQnlVc2VyTWluUmV3YXJkAAAAAgUAAAAMbHBBc3NldElkU3RyBQAAAA51c2VyQWRkcmVzc1N0cgQAAAAbY2xhaW1lZEJ5VXNlckJvb3N0UmV3YXJkS0VZCQEAAAAba2V5Q2xhaW1lZEJ5VXNlckJvb3N0UmV3YXJkAAAAAgUAAAAMbHBBc3NldElkU3RyBQAAAA51c2VyQWRkcmVzc1N0cgQAAAANY2xhaW1lZEJ5VXNlcgkBAAAADGdldEludE9yWmVybwAAAAIFAAAABHRoaXMFAAAAEGNsYWltZWRCeVVzZXJLRVkEAAAAFmNsYWltZWRCeVVzZXJNaW5SZXdhcmQJAQAAAAxnZXRJbnRPclplcm8AAAACBQAAAAR0aGlzBQAAABljbGFpbWVkQnlVc2VyTWluUmV3YXJkS0VZBAAAABhjbGFpbWVkQnlVc2VyQm9vc3RSZXdhcmQJAQAAAAxnZXRJbnRPclplcm8AAAACBQAAAAR0aGlzBQAAABtjbGFpbWVkQnlVc2VyQm9vc3RSZXdhcmRLRVkEAAAADGNsYWltZWRUb3RhbAkBAAAADGdldEludE9yWmVybwAAAAIFAAAABHRoaXMFAAAAD2NsYWltZWRUb3RhbEtFWQQAAAANJHQwMTc4ODgxODAwMAkBAAAAEHJlZnJlc2hJTlRFR1JBTFMAAAAEBQAAAAxscEFzc2V0SWRTdHIFAAAADnVzZXJBZGRyZXNzU3RyBQAAAA5wb29sQWRkcmVzc1N0cgAAAAAAAAAAAAQAAAAQd3hUb0NsYWltVXNlck5ldwgFAAAADSR0MDE3ODg4MTgwMDAAAAACXzEEAAAADWludGVncmFsU1RBVEUIBQAAAA0kdDAxNzg4ODE4MDAwAAAAAl8yBAAAAAVkZWJ1ZwgFAAAADSR0MDE3ODg4MTgwMDAAAAACXzMEAAAAEGF2YWlsYWJsZVRvQ2xhaW0JAABlAAAAAgUAAAAQd3hUb0NsYWltVXNlck5ldwUAAAANY2xhaW1lZEJ5VXNlcgMJAABnAAAAAgAAAAAAAAAAAAUAAAAQYXZhaWxhYmxlVG9DbGFpbQkAAAIAAAABAgAAABBub3RoaW5nIHRvIGNsYWltBAAAABJ3eEFtb3VudEJvb3N0VG90YWwJAQAAAAVhc0ludAAAAAEJAAGRAAAAAgkBAAAACWFzQW55TGlzdAAAAAEJAAP8AAAABAUAAAAQYm9vc3RpbmdDb250cmFjdAIAAAAMY2xhaW1XeEJvb3N0CQAETAAAAAIFAAAADGxwQXNzZXRJZFN0cgkABEwAAAACBQAAAA51c2VyQWRkcmVzc1N0cgUAAAADbmlsBQAAAANuaWwAAAAAAAAAAAAEAAAADW1pblJld2FyZFBhcnQFAAAAEGF2YWlsYWJsZVRvQ2xhaW0EAAAAD2Jvb3N0UmV3YXJkUGFydAkAAZcAAAABCQAETAAAAAIJAABoAAAAAgUAAAANbWluUmV3YXJkUGFydAAAAAAAAAAAAgkABEwAAAACBQAAABJ3eEFtb3VudEJvb3N0VG90YWwFAAAAA25pbAQAAAAJd3hBc3NldElkCQEAAAAMYXNCeXRlVmVjdG9yAAAAAQkAAZEAAAACCQEAAAAJYXNBbnlMaXN0AAAAAQkAA/wAAAAEBQAAABBlbWlzc2lvbkNvbnRyYWN0AgAAAARlbWl0CQAETAAAAAIFAAAADW1pblJld2FyZFBhcnQFAAAAA25pbAUAAAADbmlsAAAAAAAAAAAABAAAAAllbWl0Qm9vc3QJAQAAAAlhc0FueUxpc3QAAAABCQAD/AAAAAQFAAAAEGVtaXNzaW9uQ29udHJhY3QCAAAABGVtaXQJAARMAAAAAgUAAAAPYm9vc3RSZXdhcmRQYXJ0BQAAAANuaWwFAAAAA25pbAMJAAAAAAAAAgUAAAAJZW1pdEJvb3N0BQAAAAllbWl0Qm9vc3QJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAEGNsYWltZWRCeVVzZXJLRVkJAABkAAAAAgUAAAANY2xhaW1lZEJ5VXNlcgUAAAAQYXZhaWxhYmxlVG9DbGFpbQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAZY2xhaW1lZEJ5VXNlck1pblJld2FyZEtFWQkAAGQAAAACBQAAABZjbGFpbWVkQnlVc2VyTWluUmV3YXJkBQAAAA1taW5SZXdhcmRQYXJ0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABtjbGFpbWVkQnlVc2VyQm9vc3RSZXdhcmRLRVkJAABkAAAAAgUAAAAYY2xhaW1lZEJ5VXNlckJvb3N0UmV3YXJkBQAAAA9ib29zdFJld2FyZFBhcnQJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAD2NsYWltZWRUb3RhbEtFWQkAAGQAAAACBQAAAAxjbGFpbWVkVG90YWwFAAAAEGF2YWlsYWJsZVRvQ2xhaW0JAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwUAAAALdXNlckFkZHJlc3MFAAAADW1pblJld2FyZFBhcnQFAAAACXd4QXNzZXRJZAkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADBQAAAAt1c2VyQWRkcmVzcwUAAAAPYm9vc3RSZXdhcmRQYXJ0BQAAAAl3eEFzc2V0SWQJAARMAAAAAgkBAAAAFU9wZXJhdGlvbkhpc3RvcnlFbnRyeQAAAAUCAAAABWNsYWltBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAMbHBBc3NldElkU3RyBQAAABBhdmFpbGFibGVUb0NsYWltCAUAAAABaQAAAA10cmFuc2FjdGlvbklkBQAAAANuaWwJAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAAAAAWkBAAAAD2NsYWltV3hSRUFET05MWQAAAAIAAAAMbHBBc3NldElkU3RyAAAADnVzZXJBZGRyZXNzU3RyBAAAAA9zdGFrZWRCeVVzZXJLRVkJAQAAAA9rZXlTdGFrZWRCeVVzZXIAAAACBQAAAA51c2VyQWRkcmVzc1N0cgUAAAAMbHBBc3NldElkU3RyBAAAAA5zdGFrZWRUb3RhbEtFWQkBAAAADmtleVN0YWtlZFRvdGFsAAAAAQUAAAAMbHBBc3NldElkU3RyBAAAABBjbGFpbWVkQnlVc2VyS0VZCQEAAAAQa2V5Q2xhaW1lZEJ5VXNlcgAAAAIFAAAADGxwQXNzZXRJZFN0cgUAAAAOdXNlckFkZHJlc3NTdHIEAAAADHN0YWtlZEJ5VXNlcgkBAAAACnJlYWRTdGFrZWQAAAABBQAAAA9zdGFrZWRCeVVzZXJLRVkEAAAAC3N0YWtlZFRvdGFsCQEAAAAKcmVhZFN0YWtlZAAAAAEFAAAADnN0YWtlZFRvdGFsS0VZBAAAAA1jbGFpbWVkQnlVc2VyCQEAAAAMZ2V0SW50T3JaZXJvAAAAAgUAAAAEdGhpcwUAAAAQY2xhaW1lZEJ5VXNlcktFWQQAAAAOcG9vbEFkZHJlc3NTdHIJAQAAABhnZXRTdHJpbmdCeUFkZHJlc3NPckZhaWwAAAACBQAAAA9mYWN0b3J5Q29udHJhY3QJAQAAACZrZXlGYWN0b3J5THBBc3NldFRvUG9vbENvbnRyYWN0QWRkcmVzcwAAAAEFAAAADGxwQXNzZXRJZFN0cgQAAAAKcG9vbFdlaWdodAkBAAAAEUBleHRyTmF0aXZlKDEwNTApAAAAAgUAAAAPZmFjdG9yeUNvbnRyYWN0CQEAAAAUa2V5RmFjdG9yeVBvb2xXZWlnaHQAAAABBQAAAA5wb29sQWRkcmVzc1N0cgQAAAASd3hFbWlzc2lvblBlckJsb2NrCQEAAAAMZ2V0SW50T3JGYWlsAAAAAgUAAAAQZW1pc3Npb25Db250cmFjdAkBAAAAHmtleUVtaXNzaW9uUmF0ZVBlckJsb2NrQ3VycmVudAAAAAAEAAAAEmVtaXNzaW9uU3RhcnRCbG9jawkBAAAADGdldEludE9yRmFpbAAAAAIFAAAAEGVtaXNzaW9uQ29udHJhY3QJAQAAABVrZXlFbWlzc2lvblN0YXJ0QmxvY2sAAAAABAAAAAxwYXNzZWRCbG9ja3MDCQAAZgAAAAIFAAAAEmVtaXNzaW9uU3RhcnRCbG9jawUAAAAGaGVpZ2h0AAAAAAAAAAAACQAAZQAAAAIFAAAABmhlaWdodAUAAAASZW1pc3Npb25TdGFydEJsb2NrBAAAAA5wb29sV3hFbWlzc2lvbgkAAGsAAAADCQAAaAAAAAIFAAAAEnd4RW1pc3Npb25QZXJCbG9jawUAAAAMcGFzc2VkQmxvY2tzBQAAAApwb29sV2VpZ2h0BQAAAA5QT09MV0VJR0hUTVVMVAQAAAAMdXNlcld4UmV3YXJkCQAAawAAAAMFAAAADnBvb2xXeEVtaXNzaW9uBQAAAAxzdGFrZWRCeVVzZXIFAAAAC3N0YWtlZFRvdGFsBAAAAA0kdDAyMDIwMTIwMzEzCQEAAAAQcmVmcmVzaElOVEVHUkFMUwAAAAQFAAAADGxwQXNzZXRJZFN0cgUAAAAOdXNlckFkZHJlc3NTdHIFAAAADnBvb2xBZGRyZXNzU3RyAAAAAAAAAAAABAAAABB3eFRvQ2xhaW1Vc2VyTmV3CAUAAAANJHQwMjAyMDEyMDMxMwAAAAJfMQQAAAANaW50ZWdyYWxTVEFURQgFAAAADSR0MDIwMjAxMjAzMTMAAAACXzIEAAAABWRlYnVnCAUAAAANJHQwMjAyMDEyMDMxMwAAAAJfMwQAAAAQYXZhaWxhYmxlVG9DbGFpbQkAAGUAAAACBQAAABB3eFRvQ2xhaW1Vc2VyTmV3BQAAAA1jbGFpbWVkQnlVc2VyBAAAAA5ib29zdEludlJlc3VsdAkBAAAACWFzQW55TGlzdAAAAAEJAAP8AAAABAUAAAAQYm9vc3RpbmdDb250cmFjdAIAAAAUY2xhaW1XeEJvb3N0UkVBRE9OTFkJAARMAAAAAgUAAAAMbHBBc3NldElkU3RyCQAETAAAAAIF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", "height": 2117642, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: BwbieajXhagmtA8tuBL4GTidZSsNdBzZqd7siH6gHXZe Next: EbzDBzCUkdbZn2mDaY84vdeDZ3rt6tNknu8bv1GXu6ud Diff:
OldNewDifferences
1-{-# STDLIB_VERSION 6 #-}
1+{-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let SCALE8 = 8
1212 let SEP = "__"
1313
1414 let POOLWEIGHTMULT = MULT8
15-
16-let zeroBigInt = toBigInt(0)
17-
18-let oneBigInt = toBigInt(1)
1915
2016 func asAnyList (val) = match val {
2117 case valAnyLyst: List[Any] =>
4945 }
5046
5147
52-func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (("mandatory this." + key) + " is not defined"))
48+func getStringOrFail (key) = valueOrErrorMessage(getString(key), (("mandatory this." + key) + " is not defined"))
5349
5450
5551 func getStringByAddressOrFail (address,key) = valueOrErrorMessage(getString(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined"))
6460 func getIntOrFail (address,key) = valueOrErrorMessage(getInteger(address, key), (("mandatory this." + key) + " is not defined"))
6561
6662
67-func getBigIntFromStringOrZero (address,key) = value(parseBigInt(valueOrElse(getString(address, key), "0")))
68-
69-
70-func getBigIntFromStringOrDefault (address,key,defaultVal) = match getString(address, key) {
71- case s: String =>
72- value(parseBigInt(s))
73- case _: Unit =>
74- defaultVal
75- case _ =>
76- throw("Match error")
77-}
78-
79-
8063 func toX18 (origVal,origScaleMult) = fraction(toBigInt(origVal), MULT18, toBigInt(origScaleMult))
8164
8265
8467
8568
8669 func keyFactoryAddress () = "%s%s__config__factoryAddress"
87-
88-
89-func keyVotingEmissionContract () = "%s__votingEmissionContract"
9070
9171
9272 let IdxFactoryCfgStakingDapp = 1
175155 func keyClaimedByUserBoostReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedBoostReward", userAddressStr, lpAssetIdStr], SEP)
176156
177157
178-func keyClaimedTotal (lpAssetIdStr) = makeString(["%s%s%s__claimed__total", lpAssetIdStr], SEP)
158+func keyClaimedTotal (lpAssetIdStr) = makeString(["%s%s%s__claimed", "total", lpAssetIdStr], SEP)
179159
180160
181161 func readStaked (key) = valueOrElse(getInteger(this, key), 0)
229209 func OperationHistoryEntry (type,userAddress,lpAssetId,amount,txId) = StringEntry(keyOperationHistoryRecord(type, userAddress, toBase58String(txId)), formatHistoryRecord(userAddress, lpAssetId, type, amount))
230210
231211
232-let factoryAddress = getStringOrFail(this, keyFactoryAddress())
212+let factoryAddress = getStringOrFail(keyFactoryAddress())
233213
234214 let factoryContract = addressFromStringValue(factoryAddress)
235215
239219
240220 let boostingContract = getBoostingAddressOrFail(factoryCfg)
241221
242-func keyNextUser (lpAssetId) = makeString(["%s%s", lpAssetId, "nextUser"], SEP)
243-
244-
245-func getUsersListName (lpAssetId) = makeString(["users", lpAssetId], SEP)
246-
247-
248-func keyListHead (listName) = makeString(["%s%s%s", listName, "head"], SEP)
249-
250-
251-func keyListSize (listName) = makeString(["%s%s%s", listName, "size"], SEP)
252-
253-
254-func keyListPrev (listName,id) = makeString(["%s%s%s%s", listName, id, "prev"], SEP)
255-
256-
257-func keyListNext (listName,id) = makeString(["%s%s%s%s", listName, id, "next"], SEP)
258-
259-
260-func containsNode (listName,id) = {
261- let headOrUnit = getString(this, keyListHead(listName))
262- let prevOrUnit = getString(this, keyListPrev(listName, id))
263- let nextOrUnit = getString(this, keyListNext(listName, id))
264- if (if ((id == valueOrElse(headOrUnit, "")))
265- then true
266- else (prevOrUnit != unit))
267- then true
268- else (nextOrUnit != unit)
269- }
270-
271-
272-func insertNodeActions (listName,id) = {
273- let headOrUnit = getString(this, keyListHead(listName))
274- let listSize = valueOrElse(getInteger(this, keyListSize(listName)), 0)
275- let checkNode = if (!(containsNode(listName, id)))
276- then true
277- else throw("Node exists")
278- if ((checkNode == checkNode))
279- then (([IntegerEntry(keyListSize(listName), (listSize + 1))] ++ (if ((headOrUnit != unit))
280- then [StringEntry(keyListNext(listName, id), value(headOrUnit)), StringEntry(keyListPrev(listName, value(headOrUnit)), id)]
281- else nil)) ++ [StringEntry(keyListHead(listName), id)])
282- else throw("Strict value is not equal to itself.")
283- }
284-
285-
286-func deleteNodeActions (listName,id) = {
287- let headOrUnit = getString(this, keyListHead(listName))
288- let listSize = valueOrElse(getInteger(this, keyListSize(listName)), 0)
289- let prevOrUnit = getString(this, keyListPrev(listName, id))
290- let nextOrUnit = getString(this, keyListNext(listName, id))
291- ([IntegerEntry(keyListSize(listName), (listSize - 1))] ++ (if (if ((prevOrUnit != unit))
292- then (nextOrUnit != unit)
293- else false)
294- then [StringEntry(keyListNext(listName, value(prevOrUnit)), value(nextOrUnit)), StringEntry(keyListPrev(listName, value(nextOrUnit)), value(prevOrUnit)), DeleteEntry(keyListPrev(listName, id)), DeleteEntry(keyListNext(listName, id))]
295- else if ((nextOrUnit != unit))
296- then [StringEntry(keyListHead(listName), value(nextOrUnit)), DeleteEntry(keyListNext(listName, id)), DeleteEntry(keyListPrev(listName, value(nextOrUnit)))]
297- else if ((prevOrUnit != unit))
298- then [DeleteEntry(keyListPrev(listName, id)), DeleteEntry(keyListNext(listName, value(prevOrUnit)))]
299- else if ((id == valueOrElse(headOrUnit, "")))
300- then [DeleteEntry(keyListHead(listName))]
301- else throw(((("invalid node: " + listName) + ".") + id))))
302- }
303-
304-
305-func calcWxPerLpIntegralUserLast (stakedByUser,wxPerLpIntegralUserLastUpdHeightOrZero,wxPerLpIntegralNew,wxPerLpIntegralUserLastKEY) = if (if ((wxPerLpIntegralUserLastUpdHeightOrZero == zeroBigInt))
306- then (stakedByUser > zeroBigInt)
222+func calcWxPerLpIntegralUserLast (stakedByUser,wxPerLpIntegralUserLastUpdHeightOrZero,wxPerLpIntegralNew,wxPerLpIntegralUserLastKEY) = if (if ((wxPerLpIntegralUserLastUpdHeightOrZero == 0))
223+ then (stakedByUser > 0)
307224 else false)
308- then zeroBigInt
309- else if ((stakedByUser == zeroBigInt))
225+ then 0
226+ else if ((stakedByUser == 0))
310227 then wxPerLpIntegralNew
311- else if (if ((wxPerLpIntegralUserLastUpdHeightOrZero > zeroBigInt))
312- then (stakedByUser > zeroBigInt)
228+ else if (if ((wxPerLpIntegralUserLastUpdHeightOrZero > 0))
229+ then (stakedByUser > 0)
313230 else false)
314- then value(parseBigInt(getStringOrFail(this, wxPerLpIntegralUserLastKEY)))
231+ then getIntOrFail(this, wxPerLpIntegralUserLastKEY)
315232 else throw("calcWxPerLpIntegralUserLast: unexpected state")
316233
317234
318235 func refreshPoolINTEGRALS (lpAssetIdStr,poolAddressStr,lpDeltaAmount) = {
319236 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
320- let stakedTotal = toBigInt(readStaked(stakedTotalKEY))
321- let nonZeroStakedTotal = if ((stakedTotal == zeroBigInt))
322- then oneBigInt
323- else stakedTotal
237+ let stakedTotal = readStaked(stakedTotalKEY)
324238 let poolWeight = getIntegerValue(factoryContract, keyFactoryPoolWeight(poolAddressStr))
325239 let emissionStartBlock = getIntOrFail(emissionContract, keyEmissionStartBlock())
326240 let MULT3 = 1000
330244 let wxPerLpIntegralLastUpdHeightKEY = keyWxPerLpIntegralLastUpdHeight(lpAssetIdStr)
331245 let wxPerLpKEY = keyWxPerLp(lpAssetIdStr)
332246 let wxPerLpIntegralLastUpdHeight = getIntOrDefault(this, wxPerLpIntegralLastUpdHeightKEY, emissionStartBlock)
333- let wxPerLpIntegral = getBigIntFromStringOrZero(this, wxPerLpIntegralKEY)
247+ let wxPerLpIntegral = getIntOrZero(this, wxPerLpIntegralKEY)
334248 let wxPerLpOrZeroX3 = 0
335249 let dh = max([(height - wxPerLpIntegralLastUpdHeight), 0])
336250 let wxPerLpX3 = if ((wxPerLpOrZeroX3 != 0))
337- then toBigInt(wxPerLpOrZeroX3)
338- else fraction(toBigInt(poolWxEmissionPerBlockX3), toBigInt(MULT8), nonZeroStakedTotal)
339- let stakedTotalNew = (stakedTotal + toBigInt(lpDeltaAmount))
340- let nonZeroStakedTotalNew = if ((stakedTotalNew == zeroBigInt))
341- then oneBigInt
342- else stakedTotalNew
343- let wxPerLpIntegralNew = (wxPerLpIntegral + (wxPerLpX3 * toBigInt(dh)))
344- let wxPerLpX3New = (toBigInt(poolWxEmissionPerBlockX3) / nonZeroStakedTotalNew)
251+ then wxPerLpOrZeroX3
252+ else fraction(poolWxEmissionPerBlockX3, MULT8, stakedTotal)
253+ let stakedTotalNew = (stakedTotal + lpDeltaAmount)
254+ let wxPerLpIntegralNew = (wxPerLpIntegral + (wxPerLpX3 * dh))
255+ let wxPerLpX3New = (poolWxEmissionPerBlockX3 / stakedTotalNew)
345256 let wxPerLpIntegralLastUpdHeightNew = height
346- let debug = makeString([toString(wxPerLpIntegralNew), toString(dh), toString(wxPerLpX3), toString(stakedTotal), toString(poolWxEmissionPerBlockX3), toString(wxEmissionPerBlockX3), toString(poolWeight)], "::")
347- $Tuple3(wxPerLpIntegralNew, [StringEntry(wxPerLpIntegralKEY, toString(wxPerLpIntegralNew)), IntegerEntry(wxPerLpIntegralLastUpdHeightKEY, wxPerLpIntegralLastUpdHeightNew), StringEntry(wxPerLpKEY, toString(wxPerLpX3New))], debug)
257+ let debug = makeString([toString(wxPerLpIntegralNew), "dh=", toString(dh), "wxPerLpX3=", toString(wxPerLpX3), "stakedTotal=", toString(stakedTotal), toString(poolWxEmissionPerBlockX3), toString(wxEmissionPerBlockX3), "poolWeight=", toString(poolWeight)], "::")
258+ $Tuple3(wxPerLpIntegralNew, [IntegerEntry(wxPerLpIntegralKEY, wxPerLpIntegralNew), IntegerEntry(wxPerLpIntegralLastUpdHeightKEY, wxPerLpIntegralLastUpdHeightNew), IntegerEntry(wxPerLpKEY, wxPerLpX3New)], debug)
348259 }
349260
350261
351262 func refreshINTEGRALS (lpAssetIdStr,userAddressStr,poolAddressStr,lpDeltaAmount) = {
352- let $t01430214424 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, lpDeltaAmount)
353- let wxPerLpIntegralNew = $t01430214424._1
354- let poolIntegralSTATE = $t01430214424._2
355- let poolDEBUG = $t01430214424._3
263+ let $t01091111033 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, lpDeltaAmount)
264+ let wxPerLpIntegralNew = $t01091111033._1
265+ let poolIntegralSTATE = $t01091111033._2
266+ let poolDEBUG = $t01091111033._3
356267 let MULT3 = 1000
357268 let stakedByUserKEY = keyStakedByUser(userAddressStr, lpAssetIdStr)
358269 let stakedByUser = readStaked(stakedByUserKEY)
359270 let wxToClaimUserKEY = keyWxToClaimUser(lpAssetIdStr, userAddressStr)
360271 let wxPerLpIntegralUserLastUpdHeightKEY = keyWxPerLpIntegralUserLastUpdHeight(lpAssetIdStr, userAddressStr)
361272 let wxPerLpIntegralUserLastKEY = keyWxPerLpIntegralUserLast(lpAssetIdStr, userAddressStr)
362- let wxToClaimUser = getBigIntFromStringOrZero(this, wxToClaimUserKEY)
273+ let wxToClaimUser = getIntOrZero(this, wxToClaimUserKEY)
363274 let wxPerLpIntegralUserLastUpdHeightOrZero = getIntOrZero(this, wxPerLpIntegralUserLastUpdHeightKEY)
364- let wxPerLpIntegralUserLast = calcWxPerLpIntegralUserLast(toBigInt(stakedByUser), toBigInt(wxPerLpIntegralUserLastUpdHeightOrZero), wxPerLpIntegralNew, wxPerLpIntegralUserLastKEY)
365- let MULT11 = (MULT8 * MULT3)
366- let wxToClaimUserNew = max([(wxToClaimUser + fraction((wxPerLpIntegralNew - wxPerLpIntegralUserLast), toBigInt(stakedByUser), toBigInt(MULT11))), zeroBigInt])
275+ let wxPerLpIntegralUserLast = calcWxPerLpIntegralUserLast(stakedByUser, wxPerLpIntegralUserLastUpdHeightOrZero, wxPerLpIntegralNew, wxPerLpIntegralUserLastKEY)
276+ let wxToClaimUserNew = (wxToClaimUser + fraction((wxPerLpIntegralNew - wxPerLpIntegralUserLast), stakedByUser, (MULT8 * MULT3)))
367277 let wxPerLpIntegralUserLastNew = wxPerLpIntegralNew
368278 let wxPerLpIntegralUserLastUpdHeightNew = height
369- let debug = makeString([toString(wxToClaimUser), toString(wxPerLpIntegralUserLast), toString(stakedByUser), poolDEBUG, toString(height)], "::")
370- $Tuple3(wxToClaimUserNew, (poolIntegralSTATE ++ [StringEntry(wxToClaimUserKEY, toString(wxToClaimUserNew)), IntegerEntry(wxPerLpIntegralUserLastUpdHeightKEY, wxPerLpIntegralUserLastUpdHeightNew), StringEntry(wxPerLpIntegralUserLastKEY, toString(wxPerLpIntegralUserLastNew))]), debug)
279+ let debug = makeString(["wxToClaimUser", toString(wxToClaimUser), "wxPerLpInteg", toString(wxPerLpIntegralUserLastUpdHeightOrZero), "wxPerLpIntegralNew", toString(wxPerLpIntegralNew), "wxToClaimUserNew", toString(wxToClaimUserNew), "wxPerLpIntegralUserLast", toString(wxPerLpIntegralUserLast), toString(stakedByUser), "poolDEBUG=", poolDEBUG, "height=", toString(height)], "::")
280+ $Tuple3(wxToClaimUserNew, (poolIntegralSTATE ++ [IntegerEntry(wxToClaimUserKEY, wxToClaimUserNew), IntegerEntry(wxPerLpIntegralUserLastUpdHeightKEY, wxPerLpIntegralUserLastUpdHeightNew), IntegerEntry(wxPerLpIntegralUserLastKEY, wxPerLpIntegralUserLastNew)]), debug)
371281 }
372282
373283
401311 }
402312
403313
404-let permissionDeniedError = throw("Permission denied")
405-
406-func mustThis (i) = if ((i.caller == this))
407- then true
408- else permissionDeniedError
409-
410-
411-func mustManager (i) = match managerPublicKeyOrUnit() {
412- case pk: ByteVector =>
413- if ((i.callerPublicKey == pk))
414- then true
415- else permissionDeniedError
416- case _: Unit =>
417- if ((i.caller == this))
418- then true
419- else permissionDeniedError
420- case _ =>
421- throw("Match error")
422-}
314+func mustManager (i) = {
315+ let pd = throw("Permission denied")
316+ match managerPublicKeyOrUnit() {
317+ case pk: ByteVector =>
318+ if ((i.callerPublicKey == pk))
319+ then true
320+ else pd
321+ case _: Unit =>
322+ if ((i.caller == this))
323+ then true
324+ else pd
325+ case _ =>
326+ throw("Match error")
327+ }
328+ }
423329
424330
425331 @Callable(i)
427333 let checkCaller = mustManager(i)
428334 if ((checkCaller == checkCaller))
429335 then [StringEntry(keyFactoryAddress(), factoryAddressStr)]
430- else throw("Strict value is not equal to itself.")
431- }
432-
433-
434-
435-@Callable(i)
436-func constructorV2 (votingEmissionContract) = {
437- let cheks = [mustManager(i), if ((addressFromString(votingEmissionContract) != unit))
438- then true
439- else "invalid voting emission contract address"]
440- if ((cheks == cheks))
441- then [StringEntry(keyVotingEmissionContract(), votingEmissionContract)]
442336 else throw("Strict value is not equal to itself.")
443337 }
444338
496390 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
497391 let stakedByUser = readStaked(stakedByUserKEY)
498392 let stakedTotal = readStaked(stakedTotalKEY)
499- let $t01903419151 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, amount)
500- let wxToClaimUserNew = $t01903419151._1
501- let integralSTATE = $t01903419151._2
502- let debug = $t01903419151._3
503- let listName = getUsersListName(lpAssetIdStr)
504- let listActions = if (containsNode(listName, userAddressStr))
505- then nil
506- else insertNodeActions(listName, userAddressStr)
507- (([IntegerEntry(stakedByUserKEY, (stakedByUser + amount)), IntegerEntry(stakedTotalKEY, (stakedTotal + amount)), OperationHistoryEntry("stake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE) ++ listActions)
393+ let $t01532415441 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, amount)
394+ let wxToClaimUserNew = $t01532415441._1
395+ let integralSTATE = $t01532415441._2
396+ let debug = $t01532415441._3
397+ ([IntegerEntry(stakedByUserKEY, (stakedByUser + amount)), IntegerEntry(stakedTotalKEY, (stakedTotal + amount)), OperationHistoryEntry("stake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE)
508398 }
509399
510400
524414 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
525415 let stakedByUser = readStaked(stakedByUserKEY)
526416 let stakedTotal = readStaked(stakedTotalKEY)
527- let $t02048620604 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, -(amount))
528- let wxToClaimUserNew = $t02048620604._1
529- let integralSTATE = $t02048620604._2
530- let debug = $t02048620604._3
531- let listName = getUsersListName(lpAssetIdStr)
532- let listActions = if (containsNode(listName, userAddressStr))
533- then deleteNodeActions(listName, userAddressStr)
534- else nil
417+ let $t01657816696 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, -(amount))
418+ let wxToClaimUserNew = $t01657816696._1
419+ let integralSTATE = $t01657816696._2
420+ let debug = $t01657816696._3
535421 if ((amount > stakedByUser))
536422 then throw("passed amount is less then available")
537- else (([IntegerEntry(stakedByUserKEY, (stakedByUser - amount)), IntegerEntry(stakedTotalKEY, (stakedTotal - amount)), ScriptTransfer(i.caller, amount, lpAssetId), OperationHistoryEntry("unstake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE) ++ listActions)
423+ else ([IntegerEntry(stakedByUserKEY, (stakedByUser - amount)), IntegerEntry(stakedTotalKEY, (stakedTotal - amount)), ScriptTransfer(i.caller, amount, lpAssetId), OperationHistoryEntry("unstake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE)
538424 }
539425
540426
541427
542428 @Callable(i)
543429 func claimWx (lpAssetIdStr) = {
430+ let userAddress = i.caller
544431 let userAddressStr = toString(i.caller)
545- let result = invoke(this, "claimWxINTERNAL", [lpAssetIdStr, userAddressStr], nil)
546- $Tuple2(nil, result)
547- }
548-
549-
550-
551-@Callable(i)
552-func claimWxINTERNAL (lpAssetIdStr,userAddressStr) = {
553- let checkCaller = mustThis(i)
554- if ((checkCaller == checkCaller))
555- then {
556- let userAddress = valueOrErrorMessage(addressFromString(userAddressStr), "claimWxINTERNAL: invalid user address")
557- let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetIdStr))
558- let claimedByUserKEY = keyClaimedByUser(lpAssetIdStr, userAddressStr)
559- let claimedTotalKEY = keyClaimedTotal(lpAssetIdStr)
560- let claimedByUserMinRewardKEY = keyClaimedByUserMinReward(lpAssetIdStr, userAddressStr)
561- let claimedByUserBoostRewardKEY = keyClaimedByUserBoostReward(lpAssetIdStr, userAddressStr)
562- let claimedByUser = getBigIntFromStringOrZero(this, claimedByUserKEY)
563- let claimedByUserMinReward = getBigIntFromStringOrZero(this, claimedByUserMinRewardKEY)
564- let claimedByUserBoostReward = getBigIntFromStringOrZero(this, claimedByUserBoostRewardKEY)
565- let claimedTotal = getBigIntFromStringOrZero(this, claimedTotalKEY)
566- let $t02235222464 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
567- let wxToClaimUserNew = $t02235222464._1
568- let integralSTATE = $t02235222464._2
569- let debug = $t02235222464._3
570- let availableToClaim = max([(wxToClaimUserNew - claimedByUser), zeroBigInt])
571- if ((zeroBigInt >= availableToClaim))
572- then throw("nothing to claim")
573- else {
574- let wxAmountBoostTotal = max([asInt(asAnyList(invoke(boostingContract, "claimWxBoost", [lpAssetIdStr, userAddressStr], nil))[0]), 0])
575- let minRewardPart = availableToClaim
576- let boostRewardPart = min([(minRewardPart * toBigInt(2)), toBigInt(wxAmountBoostTotal)])
577- let wxAssetId = asByteVector(asAnyList(invoke(emissionContract, "emit", [toInt(minRewardPart)], nil))[0])
578- let emitBoost = asAnyList(invoke(emissionContract, "emit", [toInt(boostRewardPart)], nil))
579- if ((emitBoost == emitBoost))
580- then {
581- let claimedByUserValue = (claimedByUser + availableToClaim)
582- let claimedByUserMinRewardPlusPart = (claimedByUserMinReward + minRewardPart)
583- let claimedByUserBoostRewardPlusBoostRewardPart = (claimedByUserMinReward + minRewardPart)
584- let claimedTotalPlusAvailableToClaim = (claimedByUserMinReward + minRewardPart)
585-[StringEntry(claimedByUserKEY, toString(claimedByUserValue)), StringEntry(claimedByUserMinRewardKEY, toString(claimedByUserMinRewardPlusPart)), StringEntry(claimedByUserBoostRewardKEY, toString(claimedByUserBoostRewardPlusBoostRewardPart)), StringEntry(claimedTotalKEY, toString(claimedTotalPlusAvailableToClaim)), ScriptTransfer(userAddress, toInt(minRewardPart), wxAssetId), ScriptTransfer(userAddress, toInt(boostRewardPart), wxAssetId), OperationHistoryEntry("claim", userAddressStr, lpAssetIdStr, toInt(availableToClaim), i.transactionId)]
586- }
587- else throw("Strict value is not equal to itself.")
588- }
432+ let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetIdStr))
433+ let claimedByUserKEY = keyClaimedByUser(lpAssetIdStr, userAddressStr)
434+ let claimedTotalKEY = keyClaimedTotal(lpAssetIdStr)
435+ let claimedByUserMinRewardKEY = keyClaimedByUserMinReward(lpAssetIdStr, userAddressStr)
436+ let claimedByUserBoostRewardKEY = keyClaimedByUserBoostReward(lpAssetIdStr, userAddressStr)
437+ let claimedByUser = getIntOrZero(this, claimedByUserKEY)
438+ let claimedByUserMinReward = getIntOrZero(this, claimedByUserMinRewardKEY)
439+ let claimedByUserBoostReward = getIntOrZero(this, claimedByUserBoostRewardKEY)
440+ let claimedTotal = getIntOrZero(this, claimedTotalKEY)
441+ let $t01788818000 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
442+ let wxToClaimUserNew = $t01788818000._1
443+ let integralSTATE = $t01788818000._2
444+ let debug = $t01788818000._3
445+ let availableToClaim = (wxToClaimUserNew - claimedByUser)
446+ if ((0 >= availableToClaim))
447+ then throw("nothing to claim")
448+ else {
449+ let wxAmountBoostTotal = asInt(asAnyList(invoke(boostingContract, "claimWxBoost", [lpAssetIdStr, userAddressStr], nil))[0])
450+ let minRewardPart = availableToClaim
451+ let boostRewardPart = min([(minRewardPart * 2), wxAmountBoostTotal])
452+ let wxAssetId = asByteVector(asAnyList(invoke(emissionContract, "emit", [minRewardPart], nil))[0])
453+ let emitBoost = asAnyList(invoke(emissionContract, "emit", [boostRewardPart], nil))
454+ if ((emitBoost == emitBoost))
455+ then [IntegerEntry(claimedByUserKEY, (claimedByUser + availableToClaim)), IntegerEntry(claimedByUserMinRewardKEY, (claimedByUserMinReward + minRewardPart)), IntegerEntry(claimedByUserBoostRewardKEY, (claimedByUserBoostReward + boostRewardPart)), IntegerEntry(claimedTotalKEY, (claimedTotal + availableToClaim)), ScriptTransfer(userAddress, minRewardPart, wxAssetId), ScriptTransfer(userAddress, boostRewardPart, wxAssetId), OperationHistoryEntry("claim", userAddressStr, lpAssetIdStr, availableToClaim, i.transactionId)]
456+ else throw("Strict value is not equal to itself.")
589457 }
590- else throw("Strict value is not equal to itself.")
591458 }
592459
593460
599466 let claimedByUserKEY = keyClaimedByUser(lpAssetIdStr, userAddressStr)
600467 let stakedByUser = readStaked(stakedByUserKEY)
601468 let stakedTotal = readStaked(stakedTotalKEY)
602- let claimedByUser = getBigIntFromStringOrZero(this, claimedByUserKEY)
469+ let claimedByUser = getIntOrZero(this, claimedByUserKEY)
603470 let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetIdStr))
604471 let poolWeight = getIntegerValue(factoryContract, keyFactoryPoolWeight(poolAddressStr))
605472 let wxEmissionPerBlock = getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent())
609476 else (height - emissionStartBlock)
610477 let poolWxEmission = fraction((wxEmissionPerBlock * passedBlocks), poolWeight, POOLWEIGHTMULT)
611478 let userWxReward = fraction(poolWxEmission, stakedByUser, stakedTotal)
612- let $t02510625218 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
613- let wxToClaimUserNew = $t02510625218._1
614- let integralSTATE = $t02510625218._2
615- let debug = $t02510625218._3
616- let availableToClaim = max([(wxToClaimUserNew - claimedByUser), zeroBigInt])
479+ let $t02020120313 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
480+ let wxToClaimUserNew = $t02020120313._1
481+ let integralSTATE = $t02020120313._2
482+ let debug = $t02020120313._3
483+ let availableToClaim = (wxToClaimUserNew - claimedByUser)
617484 let boostInvResult = asAnyList(invoke(boostingContract, "claimWxBoostREADONLY", [lpAssetIdStr, userAddressStr], nil))
618- let wxAmountBoostTotal = max([asInt(boostInvResult[0]), 0])
485+ let wxAmountBoostTotal = asInt(boostInvResult[0])
619486 let boostDebug = asString(boostInvResult[1])
620487 let minRewardPart = availableToClaim
621- let boostRewardPart = min([(minRewardPart * toBigInt(2)), toBigInt(wxAmountBoostTotal)])
488+ let boostRewardPart = min([(minRewardPart * 2), wxAmountBoostTotal])
622489 let totalReward = (minRewardPart + boostRewardPart)
623- $Tuple2(nil, makeString(["%s%s%d%d%d%d%s", lpAssetIdStr, userAddressStr, toString(totalReward), toString(claimedByUser), toString(minRewardPart), toString(boostRewardPart), ((((debug + "::") + toString(userWxReward)) + "::BOOST::") + boostDebug)], SEP))
624- }
625-
626-
627-
628-@Callable(i)
629-func usersListTraversal (lpAssetId) = {
630- let checkCaller = if ((toBase58String(i.caller.bytes) == valueOrElse(getString(this, keyVotingEmissionContract()), "")))
631- then true
632- else mustManager(i)
633- if ((checkCaller == checkCaller))
634- then {
635- let listName = getUsersListName(lpAssetId)
636- let userOrUnit = getString(keyNextUser(lpAssetId))
637- let headOrUnit = getString(keyListHead(listName))
638- match userOrUnit {
639- case _: Unit =>
640- match headOrUnit {
641- case _: Unit =>
642- $Tuple2(nil, false)
643- case head: String =>
644- $Tuple2([StringEntry(keyNextUser(lpAssetId), head)], true)
645- case _ =>
646- throw("Match error")
647- }
648- case userAddress: String =>
649- let claimedByUser = getBigIntFromStringOrZero(this, keyClaimedByUser(lpAssetId, userAddress))
650- let poolAddress = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetId))
651- let $t02681326916 = refreshINTEGRALS(lpAssetId, userAddress, poolAddress, 0)
652- let wxToClaimUserNew = $t02681326916._1
653- let integralSTATE = $t02681326916._2
654- let debug = $t02681326916._3
655- let availableToClaim = (wxToClaimUserNew - claimedByUser)
656- let r = if ((availableToClaim > zeroBigInt))
657- then invoke(this, "claimWxINTERNAL", [lpAssetId, userAddress], nil)
658- else unit
659- if ((r == r))
660- then {
661- let nextUserOrUnit = getString(keyListNext(listName, userAddress))
662- match nextUserOrUnit {
663- case _: Unit =>
664- $Tuple2([DeleteEntry(keyNextUser(lpAssetId))], false)
665- case nextUser: String =>
666- $Tuple2([StringEntry(keyNextUser(lpAssetId), nextUser)], true)
667- case _ =>
668- throw("Match error")
669- }
670- }
671- else throw("Strict value is not equal to itself.")
672- case _ =>
673- throw("Match error")
674- }
675- }
676- else throw("Strict value is not equal to itself.")
490+ $Tuple2(nil, makeString(["%s%s%d%d%d%d%s", lpAssetIdStr, userAddressStr, toString(totalReward), toString(claimedByUser), toString(minRewardPart), toString(boostRewardPart), ((((debug + "::") + toString(userWxReward)) + "::BOOSTDEBUG::") + boostDebug)], SEP))
677491 }
678492
679493
682496 func onModifyWeight (lpAssetIdStr,poolAddressStr) = if ((i.caller != factoryContract))
683497 then throw("permissions denied")
684498 else {
685- let $t02759027700 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, 0)
686- let wxPerLpIntegralNew = $t02759027700._1
687- let poolIntegralSTATE = $t02759027700._2
688- let poolDEBUG = $t02759027700._3
499+ let $t02119321303 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, 0)
500+ let wxPerLpIntegralNew = $t02119321303._1
501+ let poolIntegralSTATE = $t02119321303._2
502+ let poolDEBUG = $t02119321303._3
689503 poolIntegralSTATE
690504 }
691505
Full:
OldNewDifferences
1-{-# STDLIB_VERSION 6 #-}
1+{-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let SCALE8 = 8
55
66 let MULT8 = 100000000
77
88 let SCALE18 = 18
99
1010 let MULT18 = toBigInt(1000000000000000000)
1111
1212 let SEP = "__"
1313
1414 let POOLWEIGHTMULT = MULT8
15-
16-let zeroBigInt = toBigInt(0)
17-
18-let oneBigInt = toBigInt(1)
1915
2016 func asAnyList (val) = match val {
2117 case valAnyLyst: List[Any] =>
2218 valAnyLyst
2319 case _ =>
2420 throw("fail to cast into List[Any]")
2521 }
2622
2723
2824 func asInt (val) = match val {
2925 case valInt: Int =>
3026 valInt
3127 case _ =>
3228 throw("fail to cast into Int")
3329 }
3430
3531
3632 func asString (val) = match val {
3733 case valStr: String =>
3834 valStr
3935 case _ =>
4036 throw("fail to cast into Int")
4137 }
4238
4339
4440 func asByteVector (val) = match val {
4541 case valBin: ByteVector =>
4642 valBin
4743 case _ =>
4844 throw("fail to cast into Int")
4945 }
5046
5147
52-func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (("mandatory this." + key) + " is not defined"))
48+func getStringOrFail (key) = valueOrErrorMessage(getString(key), (("mandatory this." + key) + " is not defined"))
5349
5450
5551 func getStringByAddressOrFail (address,key) = valueOrErrorMessage(getString(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined"))
5652
5753
5854 func getIntOrZero (address,key) = valueOrElse(getInteger(address, key), 0)
5955
6056
6157 func getIntOrDefault (address,key,defaultVal) = valueOrElse(getInteger(address, key), defaultVal)
6258
6359
6460 func getIntOrFail (address,key) = valueOrErrorMessage(getInteger(address, key), (("mandatory this." + key) + " is not defined"))
6561
6662
67-func getBigIntFromStringOrZero (address,key) = value(parseBigInt(valueOrElse(getString(address, key), "0")))
68-
69-
70-func getBigIntFromStringOrDefault (address,key,defaultVal) = match getString(address, key) {
71- case s: String =>
72- value(parseBigInt(s))
73- case _: Unit =>
74- defaultVal
75- case _ =>
76- throw("Match error")
77-}
78-
79-
8063 func toX18 (origVal,origScaleMult) = fraction(toBigInt(origVal), MULT18, toBigInt(origScaleMult))
8164
8265
8366 func fromX18 (val,resultScaleMult) = toInt(fraction(val, toBigInt(resultScaleMult), MULT18))
8467
8568
8669 func keyFactoryAddress () = "%s%s__config__factoryAddress"
87-
88-
89-func keyVotingEmissionContract () = "%s__votingEmissionContract"
9070
9171
9272 let IdxFactoryCfgStakingDapp = 1
9373
9474 let IdxFactoryCfgBoostingDapp = 2
9575
9676 let IdxFactoryCfgIdoDapp = 3
9777
9878 let IdxFactoryCfgTeamDapp = 4
9979
10080 let IdxFactoryCfgEmissionDapp = 5
10181
10282 let IdxFactoryCfgRestDapp = 6
10383
10484 let IdxFactoryCfgSlippageDapp = 7
10585
10686 func keyFactoryCfg () = "%s__factoryConfig"
10787
10888
10989 func keyManagerPublicKey () = "%s__managerPublicKey"
11090
11191
11292 func keyMigratorPublicKey () = "%s__migratorPublicKey"
11393
11494
11595 func keyPendingManagerPublicKey () = "%s__pendingManagerPublicKey"
11696
11797
11898 func keyStablePoolAddonAddr () = "%s__stablePoolAddonAddr"
11999
120100
121101 func keyFactoryLp2AssetsMapping (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP)
122102
123103
124104 func keyFactoryLpList () = "%s__lpTokensList"
125105
126106
127107 func keyFactoryLpAssetToPoolContractAddress (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP)
128108
129109
130110 func keyFactoryPoolWeight (contractAddress) = makeString(["%s%s", "poolWeight", contractAddress], SEP)
131111
132112
133113 func readLpList (factory) = split(valueOrElse(getString(factory, keyFactoryLpList()), ""), SEP)
134114
135115
136116 func readFactoryCfgOrFail (factory) = split(getStringByAddressOrFail(factory, keyFactoryCfg()), SEP)
137117
138118
139119 func getBoostingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgBoostingDapp])
140120
141121
142122 func getEmissionAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgEmissionDapp])
143123
144124
145125 func getStakingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgStakingDapp])
146126
147127
148128 func keyEmissionRatePerBlockCurrent () = "%s%s__ratePerBlock__current"
149129
150130
151131 func keyEmissionRatePerBlockMaxCurrent () = "%s%s__ratePerBlockMax__current"
152132
153133
154134 func keyEmissionStartBlock () = "%s%s__emission__startBlock"
155135
156136
157137 func keyEmissionDurationInBlocks () = "%s%s__emission__duration"
158138
159139
160140 func keyEmissionEndBlock () = "%s%s__emission__endBlock"
161141
162142
163143 func keyStakedByUser (userAddressStr,lpAssetIdStr) = makeString(["%s%s%s__staked", userAddressStr, lpAssetIdStr], SEP)
164144
165145
166146 func keyStakedTotal (lpAssetIdStr) = ("%s%s%s__staked__total__" + lpAssetIdStr)
167147
168148
169149 func keyClaimedByUser (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimed", userAddressStr, lpAssetIdStr], SEP)
170150
171151
172152 func keyClaimedByUserMinReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedMinReward", userAddressStr, lpAssetIdStr], SEP)
173153
174154
175155 func keyClaimedByUserBoostReward (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s__claimedBoostReward", userAddressStr, lpAssetIdStr], SEP)
176156
177157
178-func keyClaimedTotal (lpAssetIdStr) = makeString(["%s%s%s__claimed__total", lpAssetIdStr], SEP)
158+func keyClaimedTotal (lpAssetIdStr) = makeString(["%s%s%s__claimed", "total", lpAssetIdStr], SEP)
179159
180160
181161 func readStaked (key) = valueOrElse(getInteger(this, key), 0)
182162
183163
184164 func keyLastTotalLpBalance (lpAssetId) = makeString(["%s%s%s", lpAssetId, "total", "bal"], SEP)
185165
186166
187167 func keyLastUserLpBalance (lpAssetId,userAddress) = makeString(["%s%s%s", lpAssetId, userAddress, "bal"], SEP)
188168
189169
190170 func keyTotalLpBalanceIntegral (lpAssetId) = makeString(["%s%s%s", lpAssetId, "total", "balINT"], SEP)
191171
192172
193173 func keyUserLpBalanceIntegral (lpAssetId,userAddress) = makeString(["%s%s%s", lpAssetId, userAddress, "balINT"], SEP)
194174
195175
196176 func keyTotalLpBalanceIntegralLastUpdHeight (lpAssetId) = makeString(["%s%s%s", lpAssetId, "total", "lastUpd"], SEP)
197177
198178
199179 func keyUserLpBalanceIntegralLastUpdHeight (lpAssetId,userAddress) = makeString(["%s%s%s", lpAssetId, userAddress, "lastUpd"], SEP)
200180
201181
202182 func keyWxPerLpIntegral (lpAssetId) = makeString(["%s%s%s%s", lpAssetId, "common", "lpInt"], SEP)
203183
204184
205185 func keyWxPerLpIntegralLastUpdHeight (lpAssetId) = makeString(["%s%s%s%s", lpAssetId, "common", "lpIntH"], SEP)
206186
207187
208188 func keyWxToClaimUser (lpAssetId,userAddress) = makeString(["%s%s%s%s", lpAssetId, userAddress, "lpInt"], SEP)
209189
210190
211191 func keyWxPerLpIntegralUserLastUpdHeight (lpAssetId,userAddress) = makeString(["%s%s%s%s", lpAssetId, userAddress, "lpIntH"], SEP)
212192
213193
214194 func keyWxPerLp (lpAssetId) = makeString(["%s", lpAssetId, "wxPerLp"], SEP)
215195
216196
217197 func keyWxPerLpX18 (lpAssetId) = makeString(["%s", lpAssetId, "wxPerLpX18"], SEP)
218198
219199
220200 func keyWxPerLpIntegralUserLast (lpAssetId,userAddress) = makeString(["%s%s%s%s", lpAssetId, userAddress, "uIntL"], SEP)
221201
222202
223203 func keyOperationHistoryRecord (type,userAddress,txId58) = makeString(["%s%s%s%s__history", type, userAddress, txId58], SEP)
224204
225205
226206 func formatHistoryRecord (userAddress,lpAssetId,type,amount) = makeString(["%s%s%s%d%d%d", userAddress, lpAssetId, type, toString(height), toString(lastBlock.timestamp), toString(amount)], SEP)
227207
228208
229209 func OperationHistoryEntry (type,userAddress,lpAssetId,amount,txId) = StringEntry(keyOperationHistoryRecord(type, userAddress, toBase58String(txId)), formatHistoryRecord(userAddress, lpAssetId, type, amount))
230210
231211
232-let factoryAddress = getStringOrFail(this, keyFactoryAddress())
212+let factoryAddress = getStringOrFail(keyFactoryAddress())
233213
234214 let factoryContract = addressFromStringValue(factoryAddress)
235215
236216 let factoryCfg = readFactoryCfgOrFail(factoryContract)
237217
238218 let emissionContract = getEmissionAddressOrFail(factoryCfg)
239219
240220 let boostingContract = getBoostingAddressOrFail(factoryCfg)
241221
242-func keyNextUser (lpAssetId) = makeString(["%s%s", lpAssetId, "nextUser"], SEP)
243-
244-
245-func getUsersListName (lpAssetId) = makeString(["users", lpAssetId], SEP)
246-
247-
248-func keyListHead (listName) = makeString(["%s%s%s", listName, "head"], SEP)
249-
250-
251-func keyListSize (listName) = makeString(["%s%s%s", listName, "size"], SEP)
252-
253-
254-func keyListPrev (listName,id) = makeString(["%s%s%s%s", listName, id, "prev"], SEP)
255-
256-
257-func keyListNext (listName,id) = makeString(["%s%s%s%s", listName, id, "next"], SEP)
258-
259-
260-func containsNode (listName,id) = {
261- let headOrUnit = getString(this, keyListHead(listName))
262- let prevOrUnit = getString(this, keyListPrev(listName, id))
263- let nextOrUnit = getString(this, keyListNext(listName, id))
264- if (if ((id == valueOrElse(headOrUnit, "")))
265- then true
266- else (prevOrUnit != unit))
267- then true
268- else (nextOrUnit != unit)
269- }
270-
271-
272-func insertNodeActions (listName,id) = {
273- let headOrUnit = getString(this, keyListHead(listName))
274- let listSize = valueOrElse(getInteger(this, keyListSize(listName)), 0)
275- let checkNode = if (!(containsNode(listName, id)))
276- then true
277- else throw("Node exists")
278- if ((checkNode == checkNode))
279- then (([IntegerEntry(keyListSize(listName), (listSize + 1))] ++ (if ((headOrUnit != unit))
280- then [StringEntry(keyListNext(listName, id), value(headOrUnit)), StringEntry(keyListPrev(listName, value(headOrUnit)), id)]
281- else nil)) ++ [StringEntry(keyListHead(listName), id)])
282- else throw("Strict value is not equal to itself.")
283- }
284-
285-
286-func deleteNodeActions (listName,id) = {
287- let headOrUnit = getString(this, keyListHead(listName))
288- let listSize = valueOrElse(getInteger(this, keyListSize(listName)), 0)
289- let prevOrUnit = getString(this, keyListPrev(listName, id))
290- let nextOrUnit = getString(this, keyListNext(listName, id))
291- ([IntegerEntry(keyListSize(listName), (listSize - 1))] ++ (if (if ((prevOrUnit != unit))
292- then (nextOrUnit != unit)
293- else false)
294- then [StringEntry(keyListNext(listName, value(prevOrUnit)), value(nextOrUnit)), StringEntry(keyListPrev(listName, value(nextOrUnit)), value(prevOrUnit)), DeleteEntry(keyListPrev(listName, id)), DeleteEntry(keyListNext(listName, id))]
295- else if ((nextOrUnit != unit))
296- then [StringEntry(keyListHead(listName), value(nextOrUnit)), DeleteEntry(keyListNext(listName, id)), DeleteEntry(keyListPrev(listName, value(nextOrUnit)))]
297- else if ((prevOrUnit != unit))
298- then [DeleteEntry(keyListPrev(listName, id)), DeleteEntry(keyListNext(listName, value(prevOrUnit)))]
299- else if ((id == valueOrElse(headOrUnit, "")))
300- then [DeleteEntry(keyListHead(listName))]
301- else throw(((("invalid node: " + listName) + ".") + id))))
302- }
303-
304-
305-func calcWxPerLpIntegralUserLast (stakedByUser,wxPerLpIntegralUserLastUpdHeightOrZero,wxPerLpIntegralNew,wxPerLpIntegralUserLastKEY) = if (if ((wxPerLpIntegralUserLastUpdHeightOrZero == zeroBigInt))
306- then (stakedByUser > zeroBigInt)
222+func calcWxPerLpIntegralUserLast (stakedByUser,wxPerLpIntegralUserLastUpdHeightOrZero,wxPerLpIntegralNew,wxPerLpIntegralUserLastKEY) = if (if ((wxPerLpIntegralUserLastUpdHeightOrZero == 0))
223+ then (stakedByUser > 0)
307224 else false)
308- then zeroBigInt
309- else if ((stakedByUser == zeroBigInt))
225+ then 0
226+ else if ((stakedByUser == 0))
310227 then wxPerLpIntegralNew
311- else if (if ((wxPerLpIntegralUserLastUpdHeightOrZero > zeroBigInt))
312- then (stakedByUser > zeroBigInt)
228+ else if (if ((wxPerLpIntegralUserLastUpdHeightOrZero > 0))
229+ then (stakedByUser > 0)
313230 else false)
314- then value(parseBigInt(getStringOrFail(this, wxPerLpIntegralUserLastKEY)))
231+ then getIntOrFail(this, wxPerLpIntegralUserLastKEY)
315232 else throw("calcWxPerLpIntegralUserLast: unexpected state")
316233
317234
318235 func refreshPoolINTEGRALS (lpAssetIdStr,poolAddressStr,lpDeltaAmount) = {
319236 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
320- let stakedTotal = toBigInt(readStaked(stakedTotalKEY))
321- let nonZeroStakedTotal = if ((stakedTotal == zeroBigInt))
322- then oneBigInt
323- else stakedTotal
237+ let stakedTotal = readStaked(stakedTotalKEY)
324238 let poolWeight = getIntegerValue(factoryContract, keyFactoryPoolWeight(poolAddressStr))
325239 let emissionStartBlock = getIntOrFail(emissionContract, keyEmissionStartBlock())
326240 let MULT3 = 1000
327241 let wxEmissionPerBlockX3 = (getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent()) * MULT3)
328242 let poolWxEmissionPerBlockX3 = fraction(wxEmissionPerBlockX3, poolWeight, (POOLWEIGHTMULT * 3))
329243 let wxPerLpIntegralKEY = keyWxPerLpIntegral(lpAssetIdStr)
330244 let wxPerLpIntegralLastUpdHeightKEY = keyWxPerLpIntegralLastUpdHeight(lpAssetIdStr)
331245 let wxPerLpKEY = keyWxPerLp(lpAssetIdStr)
332246 let wxPerLpIntegralLastUpdHeight = getIntOrDefault(this, wxPerLpIntegralLastUpdHeightKEY, emissionStartBlock)
333- let wxPerLpIntegral = getBigIntFromStringOrZero(this, wxPerLpIntegralKEY)
247+ let wxPerLpIntegral = getIntOrZero(this, wxPerLpIntegralKEY)
334248 let wxPerLpOrZeroX3 = 0
335249 let dh = max([(height - wxPerLpIntegralLastUpdHeight), 0])
336250 let wxPerLpX3 = if ((wxPerLpOrZeroX3 != 0))
337- then toBigInt(wxPerLpOrZeroX3)
338- else fraction(toBigInt(poolWxEmissionPerBlockX3), toBigInt(MULT8), nonZeroStakedTotal)
339- let stakedTotalNew = (stakedTotal + toBigInt(lpDeltaAmount))
340- let nonZeroStakedTotalNew = if ((stakedTotalNew == zeroBigInt))
341- then oneBigInt
342- else stakedTotalNew
343- let wxPerLpIntegralNew = (wxPerLpIntegral + (wxPerLpX3 * toBigInt(dh)))
344- let wxPerLpX3New = (toBigInt(poolWxEmissionPerBlockX3) / nonZeroStakedTotalNew)
251+ then wxPerLpOrZeroX3
252+ else fraction(poolWxEmissionPerBlockX3, MULT8, stakedTotal)
253+ let stakedTotalNew = (stakedTotal + lpDeltaAmount)
254+ let wxPerLpIntegralNew = (wxPerLpIntegral + (wxPerLpX3 * dh))
255+ let wxPerLpX3New = (poolWxEmissionPerBlockX3 / stakedTotalNew)
345256 let wxPerLpIntegralLastUpdHeightNew = height
346- let debug = makeString([toString(wxPerLpIntegralNew), toString(dh), toString(wxPerLpX3), toString(stakedTotal), toString(poolWxEmissionPerBlockX3), toString(wxEmissionPerBlockX3), toString(poolWeight)], "::")
347- $Tuple3(wxPerLpIntegralNew, [StringEntry(wxPerLpIntegralKEY, toString(wxPerLpIntegralNew)), IntegerEntry(wxPerLpIntegralLastUpdHeightKEY, wxPerLpIntegralLastUpdHeightNew), StringEntry(wxPerLpKEY, toString(wxPerLpX3New))], debug)
257+ let debug = makeString([toString(wxPerLpIntegralNew), "dh=", toString(dh), "wxPerLpX3=", toString(wxPerLpX3), "stakedTotal=", toString(stakedTotal), toString(poolWxEmissionPerBlockX3), toString(wxEmissionPerBlockX3), "poolWeight=", toString(poolWeight)], "::")
258+ $Tuple3(wxPerLpIntegralNew, [IntegerEntry(wxPerLpIntegralKEY, wxPerLpIntegralNew), IntegerEntry(wxPerLpIntegralLastUpdHeightKEY, wxPerLpIntegralLastUpdHeightNew), IntegerEntry(wxPerLpKEY, wxPerLpX3New)], debug)
348259 }
349260
350261
351262 func refreshINTEGRALS (lpAssetIdStr,userAddressStr,poolAddressStr,lpDeltaAmount) = {
352- let $t01430214424 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, lpDeltaAmount)
353- let wxPerLpIntegralNew = $t01430214424._1
354- let poolIntegralSTATE = $t01430214424._2
355- let poolDEBUG = $t01430214424._3
263+ let $t01091111033 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, lpDeltaAmount)
264+ let wxPerLpIntegralNew = $t01091111033._1
265+ let poolIntegralSTATE = $t01091111033._2
266+ let poolDEBUG = $t01091111033._3
356267 let MULT3 = 1000
357268 let stakedByUserKEY = keyStakedByUser(userAddressStr, lpAssetIdStr)
358269 let stakedByUser = readStaked(stakedByUserKEY)
359270 let wxToClaimUserKEY = keyWxToClaimUser(lpAssetIdStr, userAddressStr)
360271 let wxPerLpIntegralUserLastUpdHeightKEY = keyWxPerLpIntegralUserLastUpdHeight(lpAssetIdStr, userAddressStr)
361272 let wxPerLpIntegralUserLastKEY = keyWxPerLpIntegralUserLast(lpAssetIdStr, userAddressStr)
362- let wxToClaimUser = getBigIntFromStringOrZero(this, wxToClaimUserKEY)
273+ let wxToClaimUser = getIntOrZero(this, wxToClaimUserKEY)
363274 let wxPerLpIntegralUserLastUpdHeightOrZero = getIntOrZero(this, wxPerLpIntegralUserLastUpdHeightKEY)
364- let wxPerLpIntegralUserLast = calcWxPerLpIntegralUserLast(toBigInt(stakedByUser), toBigInt(wxPerLpIntegralUserLastUpdHeightOrZero), wxPerLpIntegralNew, wxPerLpIntegralUserLastKEY)
365- let MULT11 = (MULT8 * MULT3)
366- let wxToClaimUserNew = max([(wxToClaimUser + fraction((wxPerLpIntegralNew - wxPerLpIntegralUserLast), toBigInt(stakedByUser), toBigInt(MULT11))), zeroBigInt])
275+ let wxPerLpIntegralUserLast = calcWxPerLpIntegralUserLast(stakedByUser, wxPerLpIntegralUserLastUpdHeightOrZero, wxPerLpIntegralNew, wxPerLpIntegralUserLastKEY)
276+ let wxToClaimUserNew = (wxToClaimUser + fraction((wxPerLpIntegralNew - wxPerLpIntegralUserLast), stakedByUser, (MULT8 * MULT3)))
367277 let wxPerLpIntegralUserLastNew = wxPerLpIntegralNew
368278 let wxPerLpIntegralUserLastUpdHeightNew = height
369- let debug = makeString([toString(wxToClaimUser), toString(wxPerLpIntegralUserLast), toString(stakedByUser), poolDEBUG, toString(height)], "::")
370- $Tuple3(wxToClaimUserNew, (poolIntegralSTATE ++ [StringEntry(wxToClaimUserKEY, toString(wxToClaimUserNew)), IntegerEntry(wxPerLpIntegralUserLastUpdHeightKEY, wxPerLpIntegralUserLastUpdHeightNew), StringEntry(wxPerLpIntegralUserLastKEY, toString(wxPerLpIntegralUserLastNew))]), debug)
279+ let debug = makeString(["wxToClaimUser", toString(wxToClaimUser), "wxPerLpInteg", toString(wxPerLpIntegralUserLastUpdHeightOrZero), "wxPerLpIntegralNew", toString(wxPerLpIntegralNew), "wxToClaimUserNew", toString(wxToClaimUserNew), "wxPerLpIntegralUserLast", toString(wxPerLpIntegralUserLast), toString(stakedByUser), "poolDEBUG=", poolDEBUG, "height=", toString(height)], "::")
280+ $Tuple3(wxToClaimUserNew, (poolIntegralSTATE ++ [IntegerEntry(wxToClaimUserKEY, wxToClaimUserNew), IntegerEntry(wxPerLpIntegralUserLastUpdHeightKEY, wxPerLpIntegralUserLastUpdHeightNew), IntegerEntry(wxPerLpIntegralUserLastKEY, wxPerLpIntegralUserLastNew)]), debug)
371281 }
372282
373283
374284 func managerPublicKeyOrUnit () = match getString(keyManagerPublicKey()) {
375285 case s: String =>
376286 fromBase58String(s)
377287 case _: Unit =>
378288 unit
379289 case _ =>
380290 throw("Match error")
381291 }
382292
383293
384294 func migratorPublicKeyOrUnit () = match getString(keyMigratorPublicKey()) {
385295 case s: String =>
386296 fromBase58String(s)
387297 case _: Unit =>
388298 unit
389299 case _ =>
390300 throw("Match error")
391301 }
392302
393303
394304 func pendingManagerPublicKeyOrUnit () = match getString(keyPendingManagerPublicKey()) {
395305 case s: String =>
396306 fromBase58String(s)
397307 case _: Unit =>
398308 unit
399309 case _ =>
400310 throw("Match error")
401311 }
402312
403313
404-let permissionDeniedError = throw("Permission denied")
405-
406-func mustThis (i) = if ((i.caller == this))
407- then true
408- else permissionDeniedError
409-
410-
411-func mustManager (i) = match managerPublicKeyOrUnit() {
412- case pk: ByteVector =>
413- if ((i.callerPublicKey == pk))
414- then true
415- else permissionDeniedError
416- case _: Unit =>
417- if ((i.caller == this))
418- then true
419- else permissionDeniedError
420- case _ =>
421- throw("Match error")
422-}
314+func mustManager (i) = {
315+ let pd = throw("Permission denied")
316+ match managerPublicKeyOrUnit() {
317+ case pk: ByteVector =>
318+ if ((i.callerPublicKey == pk))
319+ then true
320+ else pd
321+ case _: Unit =>
322+ if ((i.caller == this))
323+ then true
324+ else pd
325+ case _ =>
326+ throw("Match error")
327+ }
328+ }
423329
424330
425331 @Callable(i)
426332 func constructor (factoryAddressStr) = {
427333 let checkCaller = mustManager(i)
428334 if ((checkCaller == checkCaller))
429335 then [StringEntry(keyFactoryAddress(), factoryAddressStr)]
430- else throw("Strict value is not equal to itself.")
431- }
432-
433-
434-
435-@Callable(i)
436-func constructorV2 (votingEmissionContract) = {
437- let cheks = [mustManager(i), if ((addressFromString(votingEmissionContract) != unit))
438- then true
439- else "invalid voting emission contract address"]
440- if ((cheks == cheks))
441- then [StringEntry(keyVotingEmissionContract(), votingEmissionContract)]
442336 else throw("Strict value is not equal to itself.")
443337 }
444338
445339
446340
447341 @Callable(i)
448342 func setManager (pendingManagerPublicKey) = {
449343 let checkCaller = mustManager(i)
450344 if ((checkCaller == checkCaller))
451345 then {
452346 let checkManagerPublicKey = fromBase58String(pendingManagerPublicKey)
453347 if ((checkManagerPublicKey == checkManagerPublicKey))
454348 then [StringEntry(keyPendingManagerPublicKey(), pendingManagerPublicKey)]
455349 else throw("Strict value is not equal to itself.")
456350 }
457351 else throw("Strict value is not equal to itself.")
458352 }
459353
460354
461355
462356 @Callable(i)
463357 func confirmManager () = {
464358 let pm = pendingManagerPublicKeyOrUnit()
465359 let hasPM = if (isDefined(pm))
466360 then true
467361 else throw("No pending manager")
468362 if ((hasPM == hasPM))
469363 then {
470364 let checkPM = if ((i.callerPublicKey == value(pm)))
471365 then true
472366 else throw("You are not pending manager")
473367 if ((checkPM == checkPM))
474368 then [StringEntry(keyManagerPublicKey(), toBase58String(value(pm))), DeleteEntry(keyPendingManagerPublicKey())]
475369 else throw("Strict value is not equal to itself.")
476370 }
477371 else throw("Strict value is not equal to itself.")
478372 }
479373
480374
481375
482376 @Callable(i)
483377 func stake () = if ((size(i.payments) != 1))
484378 then throw("invalid payment - exact one payment must be attached")
485379 else {
486380 let pmt = i.payments[0]
487381 let lpAssetId = value(pmt.assetId)
488382 let lpAssetIdStr = toBase58String(lpAssetId)
489383 let amount = pmt.amount
490384 let poolAddressStr = valueOrErrorMessage(getString(factoryContract, keyFactoryLp2AssetsMapping(lpAssetIdStr)), ("unsupported lp asset " + lpAssetIdStr))
491385 let callerStr = toString(i.caller)
492386 let userAddressStr = if ((callerStr == poolAddressStr))
493387 then toString(i.originCaller)
494388 else callerStr
495389 let stakedByUserKEY = keyStakedByUser(userAddressStr, lpAssetIdStr)
496390 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
497391 let stakedByUser = readStaked(stakedByUserKEY)
498392 let stakedTotal = readStaked(stakedTotalKEY)
499- let $t01903419151 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, amount)
500- let wxToClaimUserNew = $t01903419151._1
501- let integralSTATE = $t01903419151._2
502- let debug = $t01903419151._3
503- let listName = getUsersListName(lpAssetIdStr)
504- let listActions = if (containsNode(listName, userAddressStr))
505- then nil
506- else insertNodeActions(listName, userAddressStr)
507- (([IntegerEntry(stakedByUserKEY, (stakedByUser + amount)), IntegerEntry(stakedTotalKEY, (stakedTotal + amount)), OperationHistoryEntry("stake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE) ++ listActions)
393+ let $t01532415441 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, amount)
394+ let wxToClaimUserNew = $t01532415441._1
395+ let integralSTATE = $t01532415441._2
396+ let debug = $t01532415441._3
397+ ([IntegerEntry(stakedByUserKEY, (stakedByUser + amount)), IntegerEntry(stakedTotalKEY, (stakedTotal + amount)), OperationHistoryEntry("stake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE)
508398 }
509399
510400
511401
512402 @Callable(i)
513403 func unstake (lpAssetIdStr,amount) = {
514404 let lpAssetId = fromBase58String(lpAssetIdStr)
515405 let poolAddressStr = valueOrErrorMessage(getString(factoryContract, keyFactoryLp2AssetsMapping(lpAssetIdStr)), ("unsupported lp asset " + lpAssetIdStr))
516406 let poolAddon = valueOrElse(getString(this, keyStablePoolAddonAddr()), poolAddressStr)
517407 let callerStr = toString(i.caller)
518408 let userAddressStr = if (if ((callerStr == poolAddressStr))
519409 then true
520410 else (callerStr == poolAddon))
521411 then toString(i.originCaller)
522412 else callerStr
523413 let stakedByUserKEY = keyStakedByUser(userAddressStr, lpAssetIdStr)
524414 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
525415 let stakedByUser = readStaked(stakedByUserKEY)
526416 let stakedTotal = readStaked(stakedTotalKEY)
527- let $t02048620604 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, -(amount))
528- let wxToClaimUserNew = $t02048620604._1
529- let integralSTATE = $t02048620604._2
530- let debug = $t02048620604._3
531- let listName = getUsersListName(lpAssetIdStr)
532- let listActions = if (containsNode(listName, userAddressStr))
533- then deleteNodeActions(listName, userAddressStr)
534- else nil
417+ let $t01657816696 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, -(amount))
418+ let wxToClaimUserNew = $t01657816696._1
419+ let integralSTATE = $t01657816696._2
420+ let debug = $t01657816696._3
535421 if ((amount > stakedByUser))
536422 then throw("passed amount is less then available")
537- else (([IntegerEntry(stakedByUserKEY, (stakedByUser - amount)), IntegerEntry(stakedTotalKEY, (stakedTotal - amount)), ScriptTransfer(i.caller, amount, lpAssetId), OperationHistoryEntry("unstake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE) ++ listActions)
423+ else ([IntegerEntry(stakedByUserKEY, (stakedByUser - amount)), IntegerEntry(stakedTotalKEY, (stakedTotal - amount)), ScriptTransfer(i.caller, amount, lpAssetId), OperationHistoryEntry("unstake", userAddressStr, lpAssetIdStr, amount, i.transactionId)] ++ integralSTATE)
538424 }
539425
540426
541427
542428 @Callable(i)
543429 func claimWx (lpAssetIdStr) = {
430+ let userAddress = i.caller
544431 let userAddressStr = toString(i.caller)
545- let result = invoke(this, "claimWxINTERNAL", [lpAssetIdStr, userAddressStr], nil)
546- $Tuple2(nil, result)
547- }
548-
549-
550-
551-@Callable(i)
552-func claimWxINTERNAL (lpAssetIdStr,userAddressStr) = {
553- let checkCaller = mustThis(i)
554- if ((checkCaller == checkCaller))
555- then {
556- let userAddress = valueOrErrorMessage(addressFromString(userAddressStr), "claimWxINTERNAL: invalid user address")
557- let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetIdStr))
558- let claimedByUserKEY = keyClaimedByUser(lpAssetIdStr, userAddressStr)
559- let claimedTotalKEY = keyClaimedTotal(lpAssetIdStr)
560- let claimedByUserMinRewardKEY = keyClaimedByUserMinReward(lpAssetIdStr, userAddressStr)
561- let claimedByUserBoostRewardKEY = keyClaimedByUserBoostReward(lpAssetIdStr, userAddressStr)
562- let claimedByUser = getBigIntFromStringOrZero(this, claimedByUserKEY)
563- let claimedByUserMinReward = getBigIntFromStringOrZero(this, claimedByUserMinRewardKEY)
564- let claimedByUserBoostReward = getBigIntFromStringOrZero(this, claimedByUserBoostRewardKEY)
565- let claimedTotal = getBigIntFromStringOrZero(this, claimedTotalKEY)
566- let $t02235222464 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
567- let wxToClaimUserNew = $t02235222464._1
568- let integralSTATE = $t02235222464._2
569- let debug = $t02235222464._3
570- let availableToClaim = max([(wxToClaimUserNew - claimedByUser), zeroBigInt])
571- if ((zeroBigInt >= availableToClaim))
572- then throw("nothing to claim")
573- else {
574- let wxAmountBoostTotal = max([asInt(asAnyList(invoke(boostingContract, "claimWxBoost", [lpAssetIdStr, userAddressStr], nil))[0]), 0])
575- let minRewardPart = availableToClaim
576- let boostRewardPart = min([(minRewardPart * toBigInt(2)), toBigInt(wxAmountBoostTotal)])
577- let wxAssetId = asByteVector(asAnyList(invoke(emissionContract, "emit", [toInt(minRewardPart)], nil))[0])
578- let emitBoost = asAnyList(invoke(emissionContract, "emit", [toInt(boostRewardPart)], nil))
579- if ((emitBoost == emitBoost))
580- then {
581- let claimedByUserValue = (claimedByUser + availableToClaim)
582- let claimedByUserMinRewardPlusPart = (claimedByUserMinReward + minRewardPart)
583- let claimedByUserBoostRewardPlusBoostRewardPart = (claimedByUserMinReward + minRewardPart)
584- let claimedTotalPlusAvailableToClaim = (claimedByUserMinReward + minRewardPart)
585-[StringEntry(claimedByUserKEY, toString(claimedByUserValue)), StringEntry(claimedByUserMinRewardKEY, toString(claimedByUserMinRewardPlusPart)), StringEntry(claimedByUserBoostRewardKEY, toString(claimedByUserBoostRewardPlusBoostRewardPart)), StringEntry(claimedTotalKEY, toString(claimedTotalPlusAvailableToClaim)), ScriptTransfer(userAddress, toInt(minRewardPart), wxAssetId), ScriptTransfer(userAddress, toInt(boostRewardPart), wxAssetId), OperationHistoryEntry("claim", userAddressStr, lpAssetIdStr, toInt(availableToClaim), i.transactionId)]
586- }
587- else throw("Strict value is not equal to itself.")
588- }
432+ let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetIdStr))
433+ let claimedByUserKEY = keyClaimedByUser(lpAssetIdStr, userAddressStr)
434+ let claimedTotalKEY = keyClaimedTotal(lpAssetIdStr)
435+ let claimedByUserMinRewardKEY = keyClaimedByUserMinReward(lpAssetIdStr, userAddressStr)
436+ let claimedByUserBoostRewardKEY = keyClaimedByUserBoostReward(lpAssetIdStr, userAddressStr)
437+ let claimedByUser = getIntOrZero(this, claimedByUserKEY)
438+ let claimedByUserMinReward = getIntOrZero(this, claimedByUserMinRewardKEY)
439+ let claimedByUserBoostReward = getIntOrZero(this, claimedByUserBoostRewardKEY)
440+ let claimedTotal = getIntOrZero(this, claimedTotalKEY)
441+ let $t01788818000 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
442+ let wxToClaimUserNew = $t01788818000._1
443+ let integralSTATE = $t01788818000._2
444+ let debug = $t01788818000._3
445+ let availableToClaim = (wxToClaimUserNew - claimedByUser)
446+ if ((0 >= availableToClaim))
447+ then throw("nothing to claim")
448+ else {
449+ let wxAmountBoostTotal = asInt(asAnyList(invoke(boostingContract, "claimWxBoost", [lpAssetIdStr, userAddressStr], nil))[0])
450+ let minRewardPart = availableToClaim
451+ let boostRewardPart = min([(minRewardPart * 2), wxAmountBoostTotal])
452+ let wxAssetId = asByteVector(asAnyList(invoke(emissionContract, "emit", [minRewardPart], nil))[0])
453+ let emitBoost = asAnyList(invoke(emissionContract, "emit", [boostRewardPart], nil))
454+ if ((emitBoost == emitBoost))
455+ then [IntegerEntry(claimedByUserKEY, (claimedByUser + availableToClaim)), IntegerEntry(claimedByUserMinRewardKEY, (claimedByUserMinReward + minRewardPart)), IntegerEntry(claimedByUserBoostRewardKEY, (claimedByUserBoostReward + boostRewardPart)), IntegerEntry(claimedTotalKEY, (claimedTotal + availableToClaim)), ScriptTransfer(userAddress, minRewardPart, wxAssetId), ScriptTransfer(userAddress, boostRewardPart, wxAssetId), OperationHistoryEntry("claim", userAddressStr, lpAssetIdStr, availableToClaim, i.transactionId)]
456+ else throw("Strict value is not equal to itself.")
589457 }
590- else throw("Strict value is not equal to itself.")
591458 }
592459
593460
594461
595462 @Callable(i)
596463 func claimWxREADONLY (lpAssetIdStr,userAddressStr) = {
597464 let stakedByUserKEY = keyStakedByUser(userAddressStr, lpAssetIdStr)
598465 let stakedTotalKEY = keyStakedTotal(lpAssetIdStr)
599466 let claimedByUserKEY = keyClaimedByUser(lpAssetIdStr, userAddressStr)
600467 let stakedByUser = readStaked(stakedByUserKEY)
601468 let stakedTotal = readStaked(stakedTotalKEY)
602- let claimedByUser = getBigIntFromStringOrZero(this, claimedByUserKEY)
469+ let claimedByUser = getIntOrZero(this, claimedByUserKEY)
603470 let poolAddressStr = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetIdStr))
604471 let poolWeight = getIntegerValue(factoryContract, keyFactoryPoolWeight(poolAddressStr))
605472 let wxEmissionPerBlock = getIntOrFail(emissionContract, keyEmissionRatePerBlockCurrent())
606473 let emissionStartBlock = getIntOrFail(emissionContract, keyEmissionStartBlock())
607474 let passedBlocks = if ((emissionStartBlock > height))
608475 then 0
609476 else (height - emissionStartBlock)
610477 let poolWxEmission = fraction((wxEmissionPerBlock * passedBlocks), poolWeight, POOLWEIGHTMULT)
611478 let userWxReward = fraction(poolWxEmission, stakedByUser, stakedTotal)
612- let $t02510625218 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
613- let wxToClaimUserNew = $t02510625218._1
614- let integralSTATE = $t02510625218._2
615- let debug = $t02510625218._3
616- let availableToClaim = max([(wxToClaimUserNew - claimedByUser), zeroBigInt])
479+ let $t02020120313 = refreshINTEGRALS(lpAssetIdStr, userAddressStr, poolAddressStr, 0)
480+ let wxToClaimUserNew = $t02020120313._1
481+ let integralSTATE = $t02020120313._2
482+ let debug = $t02020120313._3
483+ let availableToClaim = (wxToClaimUserNew - claimedByUser)
617484 let boostInvResult = asAnyList(invoke(boostingContract, "claimWxBoostREADONLY", [lpAssetIdStr, userAddressStr], nil))
618- let wxAmountBoostTotal = max([asInt(boostInvResult[0]), 0])
485+ let wxAmountBoostTotal = asInt(boostInvResult[0])
619486 let boostDebug = asString(boostInvResult[1])
620487 let minRewardPart = availableToClaim
621- let boostRewardPart = min([(minRewardPart * toBigInt(2)), toBigInt(wxAmountBoostTotal)])
488+ let boostRewardPart = min([(minRewardPart * 2), wxAmountBoostTotal])
622489 let totalReward = (minRewardPart + boostRewardPart)
623- $Tuple2(nil, makeString(["%s%s%d%d%d%d%s", lpAssetIdStr, userAddressStr, toString(totalReward), toString(claimedByUser), toString(minRewardPart), toString(boostRewardPart), ((((debug + "::") + toString(userWxReward)) + "::BOOST::") + boostDebug)], SEP))
624- }
625-
626-
627-
628-@Callable(i)
629-func usersListTraversal (lpAssetId) = {
630- let checkCaller = if ((toBase58String(i.caller.bytes) == valueOrElse(getString(this, keyVotingEmissionContract()), "")))
631- then true
632- else mustManager(i)
633- if ((checkCaller == checkCaller))
634- then {
635- let listName = getUsersListName(lpAssetId)
636- let userOrUnit = getString(keyNextUser(lpAssetId))
637- let headOrUnit = getString(keyListHead(listName))
638- match userOrUnit {
639- case _: Unit =>
640- match headOrUnit {
641- case _: Unit =>
642- $Tuple2(nil, false)
643- case head: String =>
644- $Tuple2([StringEntry(keyNextUser(lpAssetId), head)], true)
645- case _ =>
646- throw("Match error")
647- }
648- case userAddress: String =>
649- let claimedByUser = getBigIntFromStringOrZero(this, keyClaimedByUser(lpAssetId, userAddress))
650- let poolAddress = getStringByAddressOrFail(factoryContract, keyFactoryLpAssetToPoolContractAddress(lpAssetId))
651- let $t02681326916 = refreshINTEGRALS(lpAssetId, userAddress, poolAddress, 0)
652- let wxToClaimUserNew = $t02681326916._1
653- let integralSTATE = $t02681326916._2
654- let debug = $t02681326916._3
655- let availableToClaim = (wxToClaimUserNew - claimedByUser)
656- let r = if ((availableToClaim > zeroBigInt))
657- then invoke(this, "claimWxINTERNAL", [lpAssetId, userAddress], nil)
658- else unit
659- if ((r == r))
660- then {
661- let nextUserOrUnit = getString(keyListNext(listName, userAddress))
662- match nextUserOrUnit {
663- case _: Unit =>
664- $Tuple2([DeleteEntry(keyNextUser(lpAssetId))], false)
665- case nextUser: String =>
666- $Tuple2([StringEntry(keyNextUser(lpAssetId), nextUser)], true)
667- case _ =>
668- throw("Match error")
669- }
670- }
671- else throw("Strict value is not equal to itself.")
672- case _ =>
673- throw("Match error")
674- }
675- }
676- else throw("Strict value is not equal to itself.")
490+ $Tuple2(nil, makeString(["%s%s%d%d%d%d%s", lpAssetIdStr, userAddressStr, toString(totalReward), toString(claimedByUser), toString(minRewardPart), toString(boostRewardPart), ((((debug + "::") + toString(userWxReward)) + "::BOOSTDEBUG::") + boostDebug)], SEP))
677491 }
678492
679493
680494
681495 @Callable(i)
682496 func onModifyWeight (lpAssetIdStr,poolAddressStr) = if ((i.caller != factoryContract))
683497 then throw("permissions denied")
684498 else {
685- let $t02759027700 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, 0)
686- let wxPerLpIntegralNew = $t02759027700._1
687- let poolIntegralSTATE = $t02759027700._2
688- let poolDEBUG = $t02759027700._3
499+ let $t02119321303 = refreshPoolINTEGRALS(lpAssetIdStr, poolAddressStr, 0)
500+ let wxPerLpIntegralNew = $t02119321303._1
501+ let poolIntegralSTATE = $t02119321303._2
502+ let poolDEBUG = $t02119321303._3
689503 poolIntegralSTATE
690504 }
691505
692506
693507 @Verifier(tx)
694508 func verify () = {
695509 let targetPublicKey = match managerPublicKeyOrUnit() {
696510 case pk: ByteVector =>
697511 pk
698512 case _: Unit =>
699513 tx.senderPublicKey
700514 case _ =>
701515 throw("Match error")
702516 }
703517 let migratorPublicKey = match migratorPublicKeyOrUnit() {
704518 case pk: ByteVector =>
705519 pk
706520 case _: Unit =>
707521 tx.senderPublicKey
708522 case _ =>
709523 throw("Match error")
710524 }
711525 if (sigVerify(tx.bodyBytes, tx.proofs[0], targetPublicKey))
712526 then true
713527 else sigVerify(tx.bodyBytes, tx.proofs[0], migratorPublicKey)
714528 }
715529

github/deemru/w8io/026f985 
99.39 ms