tx · C6SXkLNGTKA1vfzzZVy9qK9W7MrDUyE3LgE6DEFxQS3P

3Mp3nG9T5fKGziHW8mcCqx3jMFx3towh3WA:  -0.02300000 Waves

2023.03.06 16:58 [2478322] smart account 3Mp3nG9T5fKGziHW8mcCqx3jMFx3towh3WA > SELF 0.00000000 Waves

{ "type": 13, "id": "C6SXkLNGTKA1vfzzZVy9qK9W7MrDUyE3LgE6DEFxQS3P", "fee": 2300000, "feeAssetId": null, "timestamp": 1678111132033, "version": 2, "chainId": 84, "sender": "3Mp3nG9T5fKGziHW8mcCqx3jMFx3towh3WA", "senderPublicKey": "D46gQxzmdzdCTYRg6dMcwFoUrc6ZmxF6GKd5ZoTkvzp1", "proofs": [ "4smgc5yGeNTwwJdZchnKZTgv6Vxs3sBjpD3jXkjuNeJV7Yyxrnf8dhP5wQgyLXRmgJSY3TzPDMsfj6DCyF5tcdwU" ], "script": "base64:BgLEFQgCEgMKAQgSABIECgIIARIHCgUICAgIARIECgIICBIFCgMICAESBAoCCAgSAwoBARIDCgEBEgQKAggIEgQKAggEEgASAwoBARIAEgQKAggIEgQKAggIEgQKAggIIglzZXBhcmF0b3IiDnBvb2xXZWlnaHRNdWx0Ig9tYXhEZXB0aERlZmF1bHQiFmZpbmFsaXphdGlvblN0YWdlVG90YWwiF2ZpbmFsaXphdGlvblN0YWdlU2hhcmVzIg5rZXlFcG9jaExlbmd0aCIRa2V5RXBvY2hMZW5ndGhOZXciFWtleUVwb2NoTGVuZ3RoQnlFcG9jaCIFZXBvY2giD2tleUN1cnJlbnRFcG9jaCILa2V5TWF4RGVwdGgiImtleVZvdGluZ0VtaXNzaW9uQ2FuZGlkYXRlQ29udHJhY3QiEmtleUZhY3RvcnlDb250cmFjdCITa2V5Qm9vc3RpbmdDb250cmFjdCISa2V5U3Rha2luZ0NvbnRyYWN0IhRrZXlGaW5hbGl6YXRpb25TdGFnZSILa2V5TmV4dFBvb2wiC2tleU5leHRVc2VyIg5rZXlTdGFydEhlaWdodCIRa2V5Q3VycmVudEVwb2NoVWkiEGtleVN0YXJ0SGVpZ2h0VWkiHWtleUZpbmFsaXphdGlvblNob3VsZEJlRm9yY2VkIhVrZXlTdGFydEhlaWdodEJ5RXBvY2giDGtleUZpbmFsaXplZCIJa2V5SW5MaXN0IgRwb29sIgskdDAxNjkzMTczMyINYW1vdW50QXNzZXRJZCIMcHJpY2VBc3NldElkIgdrZXlVc2VkIgdhZGRyZXNzIgdrZXlWb3RlIgskdDAyMDA3MjA0NyIPa2V5Vm90aW5nUmVzdWx0IgskdDAyMjI5MjI2OSIVa2V5Vm90aW5nUmVzdWx0U3Rha2VkIgxscEFzc2V0SWRTdHIiDGtleVBvb2xTaGFyZSILJHQwMjU5MDI2MzAiDWtleVRvdGFsVm90ZXMiD2tleVN0YWtlZEJ5VXNlciIOdXNlckFkZHJlc3NTdHIiB3dyYXBFcnIiA21zZyIIdGhyb3dFcnIiDmdldFZhbHVlT3JGYWlsIgNrZXkiBHR5cGUiBWVycm9yIgckbWF0Y2gwIgNzdHIiA2ludCIMZ2V0U3RyT3JGYWlsIgFAIgxnZXRJbnRPckZhaWwiDHBvb2xUb1N0cmluZyIMc3RyaW5nVG9Qb29sIgVwYXJ0cyIWZ2V0THBBc3NldEJ5UG9vbEFzc2V0cyIfa2V5TWFwcGluZ3NCYXNlQXNzZXQyaW50ZXJuYWxJZCIMYmFzZUFzc2V0U3RyIilrZXlNYXBwaW5nUG9vbEFzc2V0c1RvUG9vbENvbnRyYWN0QWRkcmVzcyIYaW50ZXJuYWxBbW91bnRBc3NldElkU3RyIhdpbnRlcm5hbFByaWNlQXNzZXRJZFN0ciIfa2V5TWFwcGluZ1Bvb2xDb250cmFjdFRvTFBBc3NldCITcG9vbENvbnRyYWN0QWRkcmVzcyIPZmFjdG9yeUNvbnRyYWN0IhVhbW91bnRBc3NldEludGVybmFsSWQiFHByaWNlQXNzZXRJbnRlcm5hbElkIglscEFzc2V0SWQiGGNoZWNrV3hFbWlzc2lvblBvb2xMYWJlbCILJHQwNTE1MjUxOTIiDXBvb2xzTGlzdE5hbWUiEGdldFZvdGVzTGlzdE5hbWUiCyR0MDU0OTM1NTMzIgtrZXlMaXN0SGVhZCIIbGlzdE5hbWUiBG1ldGEiC2tleUxpc3RTaXplIgtrZXlMaXN0UHJldiICaWQiC2tleUxpc3ROZXh0Igxjb250YWluc05vZGUiCmhlYWRPclVuaXQiCnByZXZPclVuaXQiCm5leHRPclVuaXQiEWluc2VydE5vZGVBY3Rpb25zIghsaXN0U2l6ZSIJY2hlY2tOb2RlIhFkZWxldGVOb2RlQWN0aW9ucyITa2V5TWFuYWdlclB1YmxpY0tleSIaa2V5UGVuZGluZ01hbmFnZXJQdWJsaWNLZXkiFm1hbmFnZXJQdWJsaWNLZXlPclVuaXQiAXMiHXBlbmRpbmdNYW5hZ2VyUHVibGljS2V5T3JVbml0IhVwZXJtaXNzaW9uRGVuaWVkRXJyb3IiCG11c3RUaGlzIgFpIgttdXN0TWFuYWdlciICcGsiF3BlbmRpbmdNYW5hZ2VyUHVibGljS2V5IgtjaGVja0NhbGxlciIVY2hlY2tNYW5hZ2VyUHVibGljS2V5IgJwbSIFaGFzUE0iB2NoZWNrUE0iC3VzZXJBZGRyZXNzIgx0YXJnZXRIZWlnaHQiF2Jvb3N0aW5nQ29udHJhY3RBZGRyZXNzIh92b3RpbmdFbWlzc2lvbkNhbmRpZGF0ZUNvbnRyYWN0IhBib29zdGluZ0NvbnRyYWN0Ig9zdGFraW5nQ29udHJhY3QiC2Vwb2NoTGVuZ3RoIgZjaGVja3MiDWluTGlzdEFjdGlvbnMiGGN1cnJlbnRFcG9jaElzTm90RGVmaW5lZCISc3RhcnRIZWlnaHRBY3Rpb25zIgZhbW91bnQiC3N0YXJ0SGVpZ2h0IgllbmRIZWlnaHQiF2ZpbmFsaXphdGlvblN0YWdlT3JVbml0IgR1c2VkIgR2b3RlIgpwb29sUmVzdWx0Igp0b3RhbFZvdGVzIhNnd3hBbW91bnRBdEVuZFRvdGFsIglhdmFpbGFibGUiB25ld1ZvdGUiCnd4RW1pc3Npb24iDXZvdGVzTGlzdE5hbWUiEHZvdGVzTGlzdEFjdGlvbnMiDm5ld0Vwb2NoTGVuZ3RoIgtuZXdNYXhEZXB0aCIHcG9vbFN0ciINZXBvY2hQcmV2aW91cyITc3RhcnRIZWlnaHRQcmV2aW91cyITZXBvY2hMZW5ndGhQcmV2aW91cyIRZW5kSGVpZ2h0UHJldmlvdXMiEGNoZWNrVGFyZ2V0RXBvY2giDSR0MDE3OTg0MTgwMjQiG2d3eEFtb3VudEF0RW5kVG90YWxQcmV2aW91cyIMdm90aW5nUmVzdWx0Igx2b3RlUHJldmlvdXMiGnZvdGluZ1Jlc3VsdFN0YWtlZFByZXZpb3VzIgxzdGFrZWRCeVVzZXIiGXZvdGluZ1Jlc3VsdFN0YWtlZEFjdGlvbnMiB2FjdGlvbnMiBWZvcmNlIgt0YXJnZXRFcG9jaCIMY3VycmVudEVwb2NoIg0kdDAyMDM2NjIwNDA2IgFyIgVzaGFyZSIPbW9kaWZ5V2VpZ2h0SW52IhBwb29sc0xpc3RBY3Rpb25zIg1wcmV2aW91c0Vwb2NoIghuZXdFcG9jaCIUbmV3RXBvY2hMZW5ndGhPcHRpb24iFW5ld0Vwb2NoTGVuZ3RoQWN0aW9ucyIKcG9vbE9yVW5pdCIKdXNlck9yVW5pdCIHJG1hdGNoMSIMcG9vbHNIZWFkU3RyIg5uZXh0VXNlck9yVW5pdCIEdXNlciIEbmV4dCIOcHJvY2Vzc1ZvdGVJbnYiDm5leHRQb29sT3JVbml0IgckbWF0Y2gyIghuZXh0VXNlciILbmV4dFBvb2xTdHIiB2NvdW50ZXIiBnJlc3VsdCIIbWF4RGVwdGgiA2ludiICdHgiBnZlcmlmeSIPdGFyZ2V0UHVibGljS2V5OAABYQICX18AAWIAgMLXLwABYwAKAAFkAAAAAWUAAQABZgkAuQkCCQDMCAICAiVzCQDMCAICC2Vwb2NoTGVuZ3RoBQNuaWwFAWEAAWcJALkJAgkAzAgCAgQlcyVzCQDMCAICEGVwb2NoTGVuZ3RoX19uZXcFA25pbAUBYQEBaAEBaQkAuQkCCQDMCAICBCVzJWQJAMwIAgILZXBvY2hMZW5ndGgJAMwIAgkApAMBBQFpBQNuaWwFAWEAAWoJALkJAgkAzAgCAgIlcwkAzAgCAgxjdXJyZW50RXBvY2gFA25pbAUBYQABawkAuQkCCQDMCAICAiVzCQDMCAICCG1heERlcHRoBQNuaWwFAWEAAWwJALkJAgkAzAgCAgIlcwkAzAgCAh92b3RpbmdFbWlzc2lvbkNhbmRpZGF0ZUNvbnRyYWN0BQNuaWwFAWEAAW0JALkJAgkAzAgCAgIlcwkAzAgCAg9mYWN0b3J5Q29udHJhY3QFA25pbAUBYQABbgkAuQkCCQDMCAICAiVzCQDMCAICEGJvb3N0aW5nQ29udHJhY3QFA25pbAUBYQABbwkAuQkCCQDMCAICAiVzCQDMCAICD3N0YWtpbmdDb250cmFjdAUDbmlsBQFhAAFwCQC5CQIJAMwIAgICJXMJAMwIAgIRZmluYWxpemF0aW9uU3RhZ2UFA25pbAUBYQABcQkAuQkCCQDMCAICAiVzCQDMCAICCG5leHRQb29sBQNuaWwFAWEAAXIJALkJAgkAzAgCAgIlcwkAzAgCAghuZXh0VXNlcgUDbmlsBQFhAAFzCQC5CQIJAMwIAgICJXMJAMwIAgILc3RhcnRIZWlnaHQFA25pbAUBYQABdAkAuQkCCQDMCAICAiVzCQDMCAICDmN1cnJlbnRFcG9jaFVpBQNuaWwFAWEAAXUJALkJAgkAzAgCAgIlcwkAzAgCAg1zdGFydEhlaWdodFVpBQNuaWwFAWEAAXYJALkJAgkAzAgCAgIlcwkAzAgCAgVmb3JjZQUDbmlsBQFhAQF3AQFpCQC5CQIJAMwIAgIEJXMlZAkAzAgCAgtzdGFydEhlaWdodAkAzAgCCQCkAwEFAWkFA25pbAUBYQEBeAEBaQkAuQkCCQDMCAICBCVzJWQJAMwIAgIJZmluYWxpemVkCQDMCAIJAKQDAQUBaQUDbmlsBQFhAQF5AQF6BAFBBQF6BAFCCAUBQQJfMQQBQwgFAUECXzIJALkJAgkAzAgCAgYlcyVzJXMJAMwIAgIGaW5MaXN0CQDMCAIFAUIJAMwIAgUBQwUDbmlsBQFhAQFEAgFFAWkJALkJAgkAzAgCAgYlcyVzJWQJAMwIAgIEdXNlZAkAzAgCCQClCAEFAUUJAMwIAgkApAMBBQFpBQNuaWwFAWEBAUYDAXoBRQFpBAFHBQF6BAFCCAUBRwJfMQQBQwgFAUcCXzIJALkJAgkAzAgCAgolcyVzJXMlcyVkCQDMCAICBHZvdGUJAMwIAgUBQgkAzAgCBQFDCQDMCAIJAKUIAQUBRQkAzAgCCQCkAwEFAWkFA25pbAUBYQEBSAIBegFpBAFJBQF6BAFCCAUBSQJfMQQBQwgFAUkCXzIJALkJAgkAzAgCAgglcyVzJXMlZAkAzAgCAgx2b3RpbmdSZXN1bHQJAMwIAgUBQgkAzAgCBQFDCQDMCAIJAKQDAQUBaQUDbmlsBQFhAQFKAgFLAWkJALkJAgkAzAgCAgYlcyVzJWQJAMwIAgISdm90aW5nUmVzdWx0U3Rha2VkCQDMCAIFAUsJAMwIAgkApAMBBQFpBQNuaWwFAWEBAUwCAXoBaQQBTQUBegQBQggFAU0CXzEEAUMIBQFNAl8yCQC5CQIJAMwIAgIIJXMlcyVzJWQJAMwIAgIJcG9vbFNoYXJlCQDMCAIFAUIJAMwIAgUBQwkAzAgCCQCkAwEFAWkFA25pbAUBYQEBTgEBaQkAuQkCCQDMCAICBCVzJWQJAMwIAgIKdG90YWxWb3RlcwkAzAgCCQCkAwEFAWkFA25pbAUBYQEBTwIBSwFQCQC5CQIJAMwIAgIGJXMlcyVzCQDMCAICBnN0YWtlZAkAzAgCBQFQCQDMCAIFAUsFA25pbAUBYQEBUQEBUgkAuQkCCQDMCAICFXZvdGluZ19lbWlzc2lvbi5yaWRlOgkAzAgCBQFSBQNuaWwCASABAVMBAVIJAAIBCQEBUQEFAVIBAVQDAUUBVQFWBAFXCQC5CQIJAMwIAgIKbWFuZGF0b3J5IAkAzAgCCQClCAEFAUUJAMwIAgIBLgkAzAgCBQFVCQDMCAICDyBpcyBub3QgZGVmaW5lZAUDbmlsAgAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgQBWAUBVgMJAAECBQFYAgZTdHJpbmcEAVkFAVgJAJ0IAgUBRQUBVQMJAAECBQFYAgNJbnQEAVoFAVgJAJoIAgUBRQUBVQkAAgECEmludmFsaWQgZW50cnkgdHlwZQUBVwECYWECAUUBVQoAAmFiCQEBVAMFAUUFAVUCAAMJAAECBQJhYgIGU3RyaW5nBQJhYgkAAgEJAKwCAgkAAwEFAmFiAhsgY291bGRuJ3QgYmUgY2FzdCB0byBTdHJpbmcBAmFjAgFFAVUKAAJhYgkBAVQDBQFFBQFVAAADCQABAgUCYWICA0ludAUCYWIJAAIBCQCsAgIJAAMBBQJhYgIYIGNvdWxkbid0IGJlIGNhc3QgdG8gSW50AQJhZAEBegkArAICCQCsAgIIBQF6Al8xBQFhCAUBegJfMgECYWUBAVkEAmFmCQC1CQIFAVkFAWEDCQAAAgkAkAMBBQJhZgACCQCUCgIJAJEDAgUCYWYAAAkAkQMCBQJhZgABCQACAQITaW52YWxpZCBwb29sIHN0cmluZwECYWcCAUIBQwoBAmFoAQJhaQkArAICAiglcyVzJXNfX21hcHBpbmdzX19iYXNlQXNzZXQyaW50ZXJuYWxJZF9fBQJhaQoBAmFqAgJhawJhbAkArAICCQCsAgIJAKwCAgkArAICAgolZCVkJXMlc19fCQCkAwEFAmFrAgJfXwkApAMBBQJhbAIjX19tYXBwaW5nc19fcG9vbEFzc2V0czJQb29sQ29udHJhY3QKAQJhbQECYW4JAKwCAgkArAICAgglcyVzJXNfXwUCYW4CIF9fbWFwcGluZ3NfX3Bvb2xDb250cmFjdDJMcEFzc2V0BAJhbwkBEUBleHRyTmF0aXZlKDEwNjIpAQkBAmFhAgUEdGhpcwUBbQQCYXAJAQJhYwIFAmFvCQECYWgBBQFCBAJhcQkBAmFjAgUCYW8JAQJhaAEFAUMEAmFuCQECYWECBQJhbwkBAmFqAgUCYXAFAmFxBAJhcgkBAmFhAgUCYW8JAQJhbQEFAmFuBQJhcgECYXMBAXoEAmF0BQF6BAFCCAUCYXQCXzEEAUMIBQJhdAJfMgQCYW8JARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQJhYQIFBHRoaXMFAW0KAAJhYgkA/AcEBQJhbwIYY2hlY2tXeEVtaXNzaW9uUG9vbExhYmVsCQDMCAIFAUIJAMwIAgUBQwUDbmlsBQNuaWwDCQABAgUCYWICB0Jvb2xlYW4FAmFiCQACAQkArAICCQADAQUCYWICHCBjb3VsZG4ndCBiZSBjYXN0IHRvIEJvb2xlYW4AAmF1AgVwb29scwECYXYBAXoEAmF3BQF6BAFCCAUCYXcCXzEEAUMIBQJhdwJfMgkAuQkCCQDMCAICBXZvdGVzCQDMCAIFAUIJAMwIAgUBQwUDbmlsBQFhAQJheAECYXkEAmF6AwkAAAIFAmF5BQJhdQIEJXMlcwIIJXMlcyVzJXMJALkJAgkAzAgCBQJhegkAzAgCBQJheQkAzAgCAgRoZWFkBQNuaWwFAWEBAmFBAQJheQQCYXoDCQAAAgUCYXkFAmF1AgQlcyVzAgglcyVzJXMlcwkAuQkCCQDMCAIFAmF6CQDMCAIFAmF5CQDMCAICBHNpemUFA25pbAUBYQECYUICAmF5AmFDBAJhegMJAAACBQJheQUCYXUCCCVzJXMlcyVzAgolcyVzJXMlcyVzCQC5CQIJAMwIAgUCYXoJAMwIAgUCYXkJAMwIAgUCYUMJAMwIAgIEcHJldgUDbmlsBQFhAQJhRAICYXkCYUMEAmF6AwkAAAIFAmF5BQJhdQIIJXMlcyVzJXMCCiVzJXMlcyVzJXMJALkJAgkAzAgCBQJhegkAzAgCBQJheQkAzAgCBQJhQwkAzAgCAgRuZXh0BQNuaWwFAWEBAmFFAgJheQJhQwQCYUYJAJ0IAgUEdGhpcwkBAmF4AQUCYXkEAmFHCQCdCAIFBHRoaXMJAQJhQgIFAmF5BQJhQwQCYUgJAJ0IAgUEdGhpcwkBAmFEAgUCYXkFAmFDAwMJAAACBQJhQwkBC3ZhbHVlT3JFbHNlAgUCYUYCAAYJAQIhPQIFAmFHBQR1bml0BgkBAiE9AgUCYUgFBHVuaXQBAmFJAgJheQJhQwQCYUYJAJ0IAgUEdGhpcwkBAmF4AQUCYXkEAmFKCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQJhQQEFAmF5AAAEAmFLAwkBASEBCQECYUUCBQJheQUCYUMGCQACAQILTm9kZSBleGlzdHMDCQAAAgUCYUsFAmFLCQDOCAIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBAmFBAQUCYXkJAGQCBQJhSgABBQNuaWwDCQECIT0CBQJhRgUEdW5pdAkAzAgCCQELU3RyaW5nRW50cnkCCQECYUQCBQJheQUCYUMJAQV2YWx1ZQEFAmFGCQDMCAIJAQtTdHJpbmdFbnRyeQIJAQJhQgIFAmF5CQEFdmFsdWUBBQJhRgUCYUMFA25pbAUDbmlsCQDMCAIJAQtTdHJpbmdFbnRyeQIJAQJheAEFAmF5BQJhQwUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQJhTAICYXkCYUMEAmFGCQCdCAIFBHRoaXMJAQJheAEFAmF5BAJhSgkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQECYUEBBQJheQAABAJhRwkAnQgCBQR0aGlzCQECYUICBQJheQUCYUMEAmFICQCdCAIFBHRoaXMJAQJhRAIFAmF5BQJhQwkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQECYUEBBQJheQkAZQIFAmFKAAEFA25pbAMDCQECIT0CBQJhRwUEdW5pdAkBAiE9AgUCYUgFBHVuaXQHCQDMCAIJAQtTdHJpbmdFbnRyeQIJAQJhRAIFAmF5CQEFdmFsdWUBBQJhRwkBBXZhbHVlAQUCYUgJAMwIAgkBC1N0cmluZ0VudHJ5AgkBAmFCAgUCYXkJAQV2YWx1ZQEFAmFICQEFdmFsdWUBBQJhRwkAzAgCCQELRGVsZXRlRW50cnkBCQECYUICBQJheQUCYUMJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBAmFEAgUCYXkFAmFDBQNuaWwDCQECIT0CBQJhSAUEdW5pdAkAzAgCCQELU3RyaW5nRW50cnkCCQECYXgBBQJheQkBBXZhbHVlAQUCYUgJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBAmFEAgUCYXkFAmFDCQDMCAIJAQtEZWxldGVFbnRyeQEJAQJhQgIFAmF5CQEFdmFsdWUBBQJhSAUDbmlsAwkBAiE9AgUCYUcFBHVuaXQJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBAmFCAgUCYXkFAmFDCQDMCAIJAQtEZWxldGVFbnRyeQEJAQJhRAIFAmF5CQEFdmFsdWUBBQJhRwUDbmlsAwkAAAIFAmFDCQELdmFsdWVPckVsc2UCBQJhRgIACQDMCAIJAQtEZWxldGVFbnRyeQEJAQJheAEFAmF5BQNuaWwJAAIBCQCsAgIJAKwCAgkArAICAg5pbnZhbGlkIG5vZGU6IAUCYXkCAS4FAmFDAQJhTQACFCVzX19tYW5hZ2VyUHVibGljS2V5AQJhTgACGyVzX19wZW5kaW5nTWFuYWdlclB1YmxpY0tleQECYU8ABAFYCQCiCAEJAQJhTQADCQABAgUBWAIGU3RyaW5nBAJhUAUBWAkA2QQBBQJhUAMJAAECBQFYAgRVbml0BQR1bml0CQACAQILTWF0Y2ggZXJyb3IBAmFRAAQBWAkAoggBCQECYU4AAwkAAQIFAVgCBlN0cmluZwQCYVAFAVgJANkEAQUCYVADCQABAgUBWAIEVW5pdAUEdW5pdAkAAgECC01hdGNoIGVycm9yAAJhUgkAAgECEVBlcm1pc3Npb24gZGVuaWVkAQJhUwECYVQDCQAAAggFAmFUBmNhbGxlcgUEdGhpcwYFAmFSAQJhVQECYVQEAVgJAQJhTwADCQABAgUBWAIKQnl0ZVZlY3RvcgQCYVYFAVgDCQAAAggFAmFUD2NhbGxlclB1YmxpY0tleQUCYVYGBQJhUgMJAAECBQFYAgRVbml0CQECYVMBBQJhVAkAAgECC01hdGNoIGVycm9yEQJhVAEKc2V0TWFuYWdlcgECYVcEAmFYCQECYVUBBQJhVAMJAAACBQJhWAUCYVgEAmFZCQDZBAEFAmFXAwkAAAIFAmFZBQJhWQkAzAgCCQELU3RyaW5nRW50cnkCCQECYU4ABQJhVwUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUAQ5jb25maXJtTWFuYWdlcgAEAmFaCQECYVEABAJiYQMJAQlpc0RlZmluZWQBBQJhWgYJAAIBAhJObyBwZW5kaW5nIG1hbmFnZXIDCQAAAgUCYmEFAmJhBAJiYgMJAAACCAUCYVQPY2FsbGVyUHVibGljS2V5CQEFdmFsdWUBBQJhWgYJAAIBAhtZb3UgYXJlIG5vdCBwZW5kaW5nIG1hbmFnZXIDCQAAAgUCYmIFAmJiCQDMCAIJAQtTdHJpbmdFbnRyeQIJAQJhTQAJANgEAQkBBXZhbHVlAQUCYVoJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBAmFOAAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUARhnZXRVc2VyR3d4QW1vdW50QXRIZWlnaHQCAmJjAmJkBAJiZQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJAQJhYQIFBHRoaXMFAW4CIWludmFsaWQgYm9vc3RpbmcgY29udHJhY3QgYWRkcmVzcwkAlAoCBQNuaWwKAAJhYgkA/AcEBQJiZQIgZ2V0VXNlckd3eEFtb3VudEF0SGVpZ2h0UkVBRE9OTFkJAMwIAgUCYmMJAMwIAgUCYmQFA25pbAUDbmlsAwkAAQIFAmFiAgNJbnQFAmFiCQACAQkArAICCQADAQUCYWICGCBjb3VsZG4ndCBiZSBjYXN0IHRvIEludAJhVAELY29uc3RydWN0b3IFAmFvAmJmAmJnAmJoAmJpBAJiagkAzAgCCQECYVUBBQJhVAkAzAgCAwkBAiE9AgkApggBBQJhbwUEdW5pdAYCIGludmFsaWQgZmFjdG9yeSBjb250cmFjdCBhZGRyZXNzCQDMCAIDCQECIT0CCQCmCAEFAmJmBQR1bml0BgIyaW52YWxpZCB2b3RpbmcgZW1pc3Npb24gY2FuZGlkYXRlIGNvbnRyYWN0IGFkZHJlc3MJAMwIAgMJAQIhPQIJAKYIAQUCYmcFBHVuaXQGAiFpbnZhbGlkIGJvb3N0aW5nIGNvbnRyYWN0IGFkZHJlc3MJAMwIAgMJAQIhPQIJAKYIAQUCYmgFBHVuaXQGAiBpbnZhbGlkIHN0YWtpbmcgY29udHJhY3QgYWRkcmVzcwkAzAgCAwkAZgIFAmJpAAAGCQACAQIUaW52YWxpZCBlcG9jaCBsZW5ndGgFA25pbAMJAAACBQJiagUCYmoJAJQKAgkAzAgCCQELU3RyaW5nRW50cnkCBQFtBQJhbwkAzAgCCQELU3RyaW5nRW50cnkCBQFsBQJiZgkAzAgCCQELU3RyaW5nRW50cnkCBQFuBQJiZwkAzAgCCQELU3RyaW5nRW50cnkCBQFvBQJiaAkAzAgCCQEMSW50ZWdlckVudHJ5AgUBZgUCYmkFA25pbAUEdW5pdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgJhVAEGY3JlYXRlAgFCAUMEAmJqCQDMCAIDCQAAAgkA2AQBCAgFAmFUBmNhbGxlcgVieXRlcwkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzBQFsAgAGCQECYVUBBQJhVAUDbmlsAwkAAAIFAmJqBQJiagQBegkAlAoCBQFCBQFDBAJiawkAzggCCQDMCAIJAQxCb29sZWFuRW50cnkCCQEBeQEFAXoGBQNuaWwJAQJhSQIFAmF1CQECYWQBBQF6BAJibAkAAAIJAJoIAgUEdGhpcwUBagUEdW5pdAQCYm0DBQJibAQBaQAACQDMCAIJAQxJbnRlZ2VyRW50cnkCBQFqBQFpCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEBdwEFAWkFBmhlaWdodAkAzAgCCQEMSW50ZWdlckVudHJ5AgUBcwUGaGVpZ2h0CQDMCAIJAQxJbnRlZ2VyRW50cnkCBQF0BQFpCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQF1BQZoZWlnaHQFA25pbAUDbmlsCQCUCgIJAM4IAgUCYmsFAmJtBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUAQR2b3RlAwFCAUMCYm4EAXoJAJQKAgUBQgUBQwQBaQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBQFqAAAEAmJvCQECYWMCBQR0aGlzCQEBdwEFAWkEAmJpCQECYWMCBQR0aGlzBQFmBAJicAkAZAIFAmJvBQJiaQQCYnEJAJoIAgUEdGhpcwUBcAQCYnIJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBAUQCCAUCYVQGY2FsbGVyBQFpAAAEAmJzCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQFGAwUBeggFAmFUBmNhbGxlcgUBaQAABAJidAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEBSAIFAXoFAWkAAAQCYnUJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBAU4BBQFpAAAEAmJ2CgACYWIJAPwHBAUEdGhpcwIYZ2V0VXNlckd3eEFtb3VudEF0SGVpZ2h0CQDMCAIJANgEAQgIBQJhVAZjYWxsZXIFYnl0ZXMJAMwIAgUCYnAFA25pbAUDbmlsAwkAAQIFAmFiAgNJbnQFAmFiCQACAQkArAICCQADAQUCYWICGCBjb3VsZG4ndCBiZSBjYXN0IHRvIEludAQCYncJAGUCBQJidgUCYnIEAmJ4CQBkAgUCYnMFAmJuBAJieQkBAmFzAQUBegQCYmoJAMwIAgMJAQIhPQIJAKAIAQkBAXkBBQF6BQR1bml0BgkAAgECDmludmFsaWQgYXNzZXRzCQDMCAIDCQBmAgUCYnAFBmhlaWdodAYJAAIBAg5pbnZhbGlkIGhlaWdodAkAzAgCAwkAAAIFAmJxBQR1bml0BgkAAgECGGZpbmFsaXphdGlvbiBpbiBwcm9ncmVzcwkAzAgCAwkAZgIFAmJ2AAAGCQACAQITeW91IGRvIG5vdCBoYXZlIGdXWAkAzAgCAwMJAGYCBQJibgAACQBnAgUCYncFAmJuBwYJAAIBAg5pbnZhbGlkIGFtb3VudAkAzAgCAwUCYnkGCQACAQIdcG9vbCBoYXNuJ3QgV1hfRU1JU1NJT04gbGFiZWwFA25pbAMJAAACBQJiagUCYmoEAmJ6CQECYXYBBQF6BAFQCQClCAEIBQJhVAZjYWxsZXIEAmJBAwkBAmFFAgUCYnoFAVAFA25pbAkBAmFJAgUCYnoFAVAJAJQKAgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEBRAIIBQJhVAZjYWxsZXIFAWkJAGQCBQJicgUCYm4JAMwIAgkBDEludGVnZXJFbnRyeQIJAQFGAwUBeggFAmFUBmNhbGxlcgUBaQUCYngJAMwIAgkBDEludGVnZXJFbnRyeQIJAQFIAgUBegUBaQkAZAIFAmJ0BQJibgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBAU4BBQFpCQBkAgUCYnUFAmJuBQNuaWwFAmJBBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUAQpjYW5jZWxWb3RlAgFCAUMEAXoJAJQKAgUBQgUBQwQBaQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBQFqAAAEAmJvCQECYWMCBQR0aGlzCQEBdwEFAWkEAmJpCQECYWMCBQR0aGlzBQFmBAJicAkAZAIFAmJvBQJiaQQCYnEJAJoIAgUEdGhpcwUBcAQCYnIJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBAUQCCAUCYVQGY2FsbGVyBQFpAAAEAmJzCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQFGAwUBeggFAmFUBmNhbGxlcgUBaQAABAJidAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEBSAIFAXoFAWkAAAQCYnUJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBAU4BBQFpAAAEAmJqCQDMCAIDCQECIT0CCQCgCAEJAQF5AQUBegUEdW5pdAYJAAIBAg5pbnZhbGlkIGFzc2V0cwkAzAgCAwkAZgIFAmJwBQZoZWlnaHQGCQACAQIOaW52YWxpZCBoZWlnaHQJAMwIAgMJAAACBQJicQUEdW5pdAYJAAIBAhhmaW5hbGl6YXRpb24gaW4gcHJvZ3Jlc3MJAMwIAgMJAGYCBQJicwAABgkAAgECB25vIHZvdGUFA25pbAMJAAACBQJiagUCYmoEAmJ6CQECYXYBBQF6BAFQCQClCAEIBQJhVAZjYWxsZXIJAJQKAgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEBRAIIBQJhVAZjYWxsZXIFAWkJAJYDAQkAzAgCCQBlAgUCYnIFAmJzCQDMCAIAAAUDbmlsCQDMCAIJAQtEZWxldGVFbnRyeQEJAQFGAwUBeggFAmFUBmNhbGxlcgUBaQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBAUgCBQF6BQFpCQBlAgUCYnQFAmJzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEBTgEFAWkJAGUCBQJidQUCYnMFA25pbAkBAmFMAgUCYnoFAVAFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4CYVQBDnNldEVwb2NoTGVuZ3RoAQJiQgQCYmoJAMwIAgkBAmFVAQUCYVQJAMwIAgMJAGYCBQJiQgAABgkAAgECFGludmFsaWQgZXBvY2ggbGVuZ3RoBQNuaWwDCQAAAgUCYmoFAmJqCQCUCgIJAMwIAgkBDEludGVnZXJFbnRyeQIFAWcFAmJCBQNuaWwFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4CYVQBC3NldE1heERlcHRoAQJiQwQCYmoJAMwIAgkBAmFVAQUCYVQJAMwIAgMJAGYCBQJiQwAABgkAAgECEWludmFsaWQgbWF4IGRlcHRoBQNuaWwDCQAAAgUCYmoFAmJqCQCUCgIJAMwIAgkBDEludGVnZXJFbnRyeQIFAWsFAmJDBQNuaWwFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4CYVQBE3Byb2Nlc3NWb3RlSU5URVJOQUwCAmJEAVAEAmFYCQECYVMBBQJhVAMJAAACBQJhWAUCYVgEAmJjCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQUBUAkArAICAipwcm9jZXNzVm90ZUlOVEVSTkFMOiBpbnZhbGlkIHVzZXIgYWRkcmVzcyAFAVAEAWkJAQJhYwIFBHRoaXMFAWoEAmJFCQBlAgUBaQABBAJiaQkBAmFjAgUEdGhpcwUBZgQCYm8JAQJhYwIFBHRoaXMJAQF3AQUBaQQCYnAJAGQCBQJibwUCYmkEAmJGCQECYWMCBQR0aGlzCQEBdwEFAmJFBAJiRwkBAmFjAgUEdGhpcwkBAWgBBQJiRQQCYkgJAGQCBQJiRgUCYkcEAmJJAwkAZwIFAmJFAAAGCQACAQIrcHJvY2Vzc1ZvdGVJTlRFUk5BTDogaW52YWxpZCBwcmV2aW91cyBlcG9jaAMJAAACBQJiSQUCYkkEAXoJAQJhZQEFAmJEBAJiSgUBegQBQggFAmJKAl8xBAFDCAUCYkoCXzIEAmJ5CQECYXMBBQF6BAJidgoAAmFiCQD8BwQFBHRoaXMCGGdldFVzZXJHd3hBbW91bnRBdEhlaWdodAkAzAgCBQFQCQDMCAIFAmJwBQNuaWwFA25pbAMJAAECBQJhYgIDSW50BQJhYgkAAgEJAKwCAgkAAwEFAmFiAhggY291bGRuJ3QgYmUgY2FzdCB0byBJbnQEAmJLCgACYWIJAPwHBAUEdGhpcwIYZ2V0VXNlckd3eEFtb3VudEF0SGVpZ2h0CQDMCAIFAVAJAMwIAgUCYkgFA25pbAUDbmlsAwkAAQIFAmFiAgNJbnQFAmFiCQACAQkArAICCQADAQUCYWICGCBjb3VsZG4ndCBiZSBjYXN0IHRvIEludAQCYnUJAQt2YWx1ZU9yRWxzZQIJAJ8IAQkBAU4BBQFpAAAEAmJMCQELdmFsdWVPckVsc2UCCQCfCAEJAQFIAgUBegUBaQAABAJiTQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEJAQFGAwUBegUCYmMFAmJFCQCsAgIJAKwCAgkArAICCQCsAgICFHByb2Nlc3NWb3RlSU5URVJOQUwgBQJiRAIBIAUBUAISOiBubyBwcmV2aW91cyB2b3RlBAJicgkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEBRAIFAmJjBQFpAAAEAUsJAQJhZwIFAUIFAUMEAmJOCQELdmFsdWVPckVsc2UCCQCfCAEJAQFKAgUBSwUCYkUAAAQCYmgJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQJhYQIFBHRoaXMFAW8EAmJPCQELdmFsdWVPckVsc2UCCQCaCAIFAmJoCQEBTwIFAUsFAVAAAAQCYlADCQAAAgUCYk8AAAUDbmlsCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEBSgIFAUsFAmJFCQBkAgUCYk4FAmJNBQNuaWwEAmJ4AwkAZgIFAmJLAAAJAGsDBQJiTQUCYnYFAmJLAAAEAmJRAwMJAGYCBQJieAAABQJieQcJAMwIAgkBDEludGVnZXJFbnRyeQIJAQFGAwUBegUCYmMFAWkFAmJ4CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEBTgEFAWkJAGQCBQJidQUCYngJAMwIAgkBDEludGVnZXJFbnRyeQIJAQFIAgUBegUBaQkAZAIFAmJMBQJieAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBAUQCBQJiYwUBaQkAZAIFAmJyBQJieAUDbmlsCQECYUwCCQECYXYBBQF6BQFQCQCUCgIJAM4IAgUCYlEFAmJQBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUARNwcm9jZXNzUG9vbElOVEVSTkFMAgJiRAJiUgQCYVgJAQJhUwEFAmFUAwkAAAIFAmFYBQJhWAQCYlMEAmJUCQECYWMCBQR0aGlzBQFqAwUCYlIFAmJUCQBlAgUCYlQAAQQCYkkDCQBnAgUCYlMAAAYJAAIBAilwcm9jZXNzUG9vbElOVEVSTkFMOiBpbnZhbGlkIHRhcmdldCBlcG9jaAMJAAACBQJiSQUCYkkEAXoJAQJhZQEFAmJEBAJiVQUBegQBQggFAmJVAl8xBAFDCAUCYlUCXzIEAmJoCQERQGV4dHJOYXRpdmUoMTA2MikBCQECYWECBQR0aGlzBQFvBAJhbwkBEUBleHRyTmF0aXZlKDEwNjIpAQkBAmFhAgUEdGhpcwUBbQQCYXIJAQJhZwIFAUIFAUMEAmJWCgACYWIJAPwHBAUCYmgCEnVzZXJzTGlzdFRyYXZlcnNhbAkAzAgCBQJhcgUDbmlsBQNuaWwDCQABAgUCYWICB0Jvb2xlYW4FAmFiCQACAQkArAICCQADAQUCYWICHCBjb3VsZG4ndCBiZSBjYXN0IHRvIEJvb2xlYW4DCQAAAgUCYlYFAmJWAwUCYlYJAJQKAgUDbmlsBgQCYnkJAQJhcwEFAXoEAmJ1CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQFOAQUCYlMAAAQCYkwJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBAUgCBQF6BQJiUwAABAJiVwMDCQAAAgUCYnUAAAYJAQEhAQUCYnkAAAkAawMFAmJMBQFiBQJidQQCYlgJAPwHBAUCYW8CDG1vZGlmeVdlaWdodAkAzAgCBQJhcgkAzAgCBQJiVwUDbmlsBQNuaWwDCQAAAgUCYlgFAmJYBAJiWQMDBQJieQYFAmJSBQNuaWwJAM4IAgkAzAgCCQELRGVsZXRlRW50cnkBCQEBeQEFAXoFA25pbAkBAmFMAgUCYXUFAmJECQCUCgIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBAUwCBQF6BQJiUwUCYlcFA25pbAUCYlkHCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUAQ5maW5hbGl6ZUhlbHBlcgAEAmJSCQELdmFsdWVPckVsc2UCCQCgCAEFAXYHBAFpCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMFAWoAAAQCYloJAGUCBQFpAAEEAmJvCQECYWMCBQR0aGlzCQEBdwEFAWkEAmJpCQECYWMCBQR0aGlzBQFmBAJicAkAZAIFAmJvBQJiaQQCYnEJAJoIAgUEdGhpcwUBcAMDAwkAZwIFBmhlaWdodAUCYnAJAAACBQJicQUEdW5pdAcJAQEhAQUCYlIHBAJjYQkAZAIFAWkAAQQCY2IJAJoIAgUEdGhpcwUBZwQCY2MEAVgFAmNiAwkAAQIFAVgCA0ludAQCYkIFAVgJAMwIAgkBDEludGVnZXJFbnRyeQIFAWYFAmJCCQDMCAIJAQtEZWxldGVFbnRyeQEFAWcFA25pbAMJAAECBQFYAgRVbml0BQNuaWwJAAIBAgtNYXRjaCBlcnJvcgkAlAoCCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAQF3AQUCY2EFBmhlaWdodAkAzAgCCQEMSW50ZWdlckVudHJ5AgUBcwUGaGVpZ2h0CQDMCAIJAQxJbnRlZ2VyRW50cnkCBQFqBQJjYQkAzAgCCQEMSW50ZWdlckVudHJ5AgUBcAUBZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBAWgBBQFpBQJiaQUDbmlsBQJjYwYDAwUCYlIJAAACBQJicQUEdW5pdAcJAJQKAgkAzAgCCQEMSW50ZWdlckVudHJ5AgUBcAUBZQUDbmlsBgMJAAACBQJicQUEdW5pdAkAlAoCBQNuaWwHAwkAAAIFAmJxBQFkBAJjZAkAoggBBQFxBAJjZQkAoggBBQFyBAFYBQJjZAMJAAECBQFYAgRVbml0BAJjZgkAoggBCQECYXgBBQJhdQMJAAECBQJjZgIEVW5pdAkAlAoCCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQFwBQFlCQDMCAIJAQtEZWxldGVFbnRyeQEFAXEJAMwIAgkBC0RlbGV0ZUVudHJ5AQUBcgUDbmlsBgMJAAECBQJjZgIGU3RyaW5nBAJjZwUCY2YJAJQKAgkAzAgCCQELU3RyaW5nRW50cnkCBQFxBQJjZwUDbmlsBgkAAgECC01hdGNoIGVycm9yAwkAAQIFAVgCBlN0cmluZwQCYkQFAVgEAXoJAQJhZQEFAmJEBAJjaAQCY2YFAmNlAwkAAQIFAmNmAgRVbml0CQCiCAEJAQJheAEJAQJhdgEFAXoDCQABAgUCY2YCBlN0cmluZwQCY2kFAmNmBAJjagkAoggBCQECYUQCCQECYXYBBQF6BQJjaQMJAAACBQJjagUCY2oEAmNrCQD8BwQFBHRoaXMCE3Byb2Nlc3NWb3RlSU5URVJOQUwJAMwIAgUCYkQJAMwIAgUCY2kFA25pbAUDbmlsAwkAAAIFAmNrBQJjawUCY2oJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAgtNYXRjaCBlcnJvcgQCY2YFAmNoAwkAAQIFAmNmAgRVbml0BAJjbAkAoggBCQECYUQCBQJhdQUCYkQEAmNtBQJjbAMJAAECBQJjbQIEVW5pdAkAlAoCCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQFwBQFlCQDMCAIJAQtEZWxldGVFbnRyeQEFAXEJAMwIAgkBC0RlbGV0ZUVudHJ5AQUBcgUDbmlsBgMJAAECBQJjbQIGU3RyaW5nBAJhUAUCY20JAJQKAgkAzAgCCQELU3RyaW5nRW50cnkCBQFxBQJhUAkAzAgCCQELRGVsZXRlRW50cnkBBQFyBQNuaWwGCQACAQILTWF0Y2ggZXJyb3IDCQABAgUCY2YCBlN0cmluZwQCY24FAmNmCQCUCgIJAMwIAgkBC1N0cmluZ0VudHJ5AgUBcgUCY24FA25pbAYJAAIBAgtNYXRjaCBlcnJvcgkAAgECC01hdGNoIGVycm9yAwkAAAIFAmJxBQFlBAJjZAkAoggBBQFxBAFYBQJjZAMJAAECBQFYAgRVbml0BAJjZgkAoggBCQECYXgBBQJhdQMJAAECBQJjZgIEVW5pdAQCYlEDBQJiUgkAzAgCCQELRGVsZXRlRW50cnkBBQFwCQDMCAIJAQtEZWxldGVFbnRyeQEFAXYFA25pbAkAzAgCCQELRGVsZXRlRW50cnkBBQFwCQDMCAIJAQxCb29sZWFuRW50cnkCCQEBeAEFAmJaBgkAzAgCCQEMSW50ZWdlckVudHJ5AgUBdAUBaQkAzAgCCQEMSW50ZWdlckVudHJ5AgUBdQUCYm8FA25pbAkAlAoCBQJiUQYDCQABAgUCY2YCBlN0cmluZwQCY28FAmNmCQCUCgIJAMwIAgkBC1N0cmluZ0VudHJ5AgUBcQUCY28FA25pbAYJAAIBAgtNYXRjaCBlcnJvcgMJAAECBQFYAgZTdHJpbmcEAmJEBQFYBAJjbAkAoggBCQECYUQCBQJhdQUCYkQDCQAAAgUCY2wFAmNsBAJiVgoAAmFiCQD8BwQFBHRoaXMCE3Byb2Nlc3NQb29sSU5URVJOQUwJAMwIAgUCYkQJAMwIAgUCYlIFA25pbAUDbmlsAwkAAQIFAmFiAgdCb29sZWFuBQJhYgkAAgEJAKwCAgkAAwEFAmFiAhwgY291bGRuJ3QgYmUgY2FzdCB0byBCb29sZWFuAwkAAAIFAmJWBQJiVgMFAmJWCQCUCgIFA25pbAYEAmNmBQJjbAMJAAECBQJjZgIEVW5pdAQCYlEDBQJiUgkAzAgCCQELRGVsZXRlRW50cnkBBQFwCQDMCAIJAQtEZWxldGVFbnRyeQEFAXYJAMwIAgkBC0RlbGV0ZUVudHJ5AQUBcQUDbmlsCQDMCAIJAQtEZWxldGVFbnRyeQEFAXAJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAQF4AQUCYloGCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQF0BQFpCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQF1BQJibwkAzAgCCQELRGVsZXRlRW50cnkBBQFxBQNuaWwJAJQKAgUCYlEGAwkAAQIFAmNmAgZTdHJpbmcEAmNvBQJjZgkAlAoCCQDMCAIJAQtTdHJpbmdFbnRyeQIFAXEFAmNvBQNuaWwGCQACAQILTWF0Y2ggZXJyb3IJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAgtNYXRjaCBlcnJvcgkAAgECFmZpbmFsaXphdGlvbiBpcyBicm9rZW4CYVQBD2ZpbmFsaXplV3JhcHBlcgECY3AEAmNxCgACYWIJAPwHBAUEdGhpcwIOZmluYWxpemVIZWxwZXIFA25pbAUDbmlsAwkAAQIFAmFiAgdCb29sZWFuBQJhYgkAAgEJAKwCAgkAAwEFAmFiAhwgY291bGRuJ3QgYmUgY2FzdCB0byBCb29sZWFuAwkAAAIFAmNxBQJjcQMJAQEhAQUCY3EDCQAAAgUCY3AAAAkAAgECHkN1cnJlbnQgdm90aW5nIGlzIG5vdCBvdmVyIHlldAkAlAoCBQNuaWwFBHVuaXQEAmNyCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMFAWsFAWMDCQBmAgUCY3IFAmNwBAJjcwkA/AcEBQR0aGlzAg9maW5hbGl6ZVdyYXBwZXIJAMwIAgkAZAIFAmNwAAEFA25pbAUDbmlsAwkAAAIFAmNzBQJjcwkAlAoCBQNuaWwFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAJQKAgUDbmlsBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUAQhmaW5hbGl6ZQAEAmNzCQD8BwQFBHRoaXMCD2ZpbmFsaXplV3JhcHBlcgkAzAgCAAAFA25pbAUDbmlsAwkAAAIFAmNzBQJjcwkAlAoCBQNuaWwFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4CYVQBFGNvbnRhaW5zTm9kZVJFQURPTkxZAgJheQJhQwkAlAoCBQNuaWwJAQJhRQIFAmF5BQJhQwJhVAEKaW5zZXJ0Tm9kZQICYXkCYUMEAmFYCQECYVUBBQJhVAMJAAACBQJhWAUCYVgJAJQKAgkBAmFJAgUCYXkFAmFDBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAmFUAQpkZWxldGVOb2RlAgJheQJhQwQCYVgJAQJhVQEFAmFUAwkAAAIFAmFYBQJhWAkAlAoCCQECYUwCBQJheQUCYUMFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BAmN0AQJjdQAEAmN2BAFYCQECYU8AAwkAAQIFAVgCCkJ5dGVWZWN0b3IEAmFWBQFYBQJhVgMJAAECBQFYAgRVbml0CAUCY3QPc2VuZGVyUHVibGljS2V5CQACAQILTWF0Y2ggZXJyb3IJAPQDAwgFAmN0CWJvZHlCeXRlcwkAkQMCCAUCY3QGcHJvb2ZzAAAFAmN2kCt4wA==", "height": 2478322, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: E7CQSQMXjfFp8F6XpfFC3ntKsHHUSUUM79JVTsDc1ptZ Next: A4Y8NgoKZAi6bn8LtwQQ2u2MaLzM4gE5GEyU9V4s5CAm Diff:
OldNewDifferences
517517 if ((checkTargetEpoch == checkTargetEpoch))
518518 then {
519519 let pool = stringToPool(poolStr)
520- let $t01801718057 = pool
521- let amountAssetId = $t01801718057._1
522- let priceAssetId = $t01801718057._2
520+ let $t01798418024 = pool
521+ let amountAssetId = $t01798418024._1
522+ let priceAssetId = $t01798418024._2
523523 let wxEmission = checkWxEmissionPoolLabel(pool)
524524 let gwxAmountAtEndTotal = {
525525 let @ = invoke(this, "getUserGwxAmountAtHeight", [userAddressStr, endHeight], nil)
578578 if ((checkTargetEpoch == checkTargetEpoch))
579579 then {
580580 let pool = stringToPool(poolStr)
581- let $t02039920439 = pool
582- let amountAssetId = $t02039920439._1
583- let priceAssetId = $t02039920439._2
581+ let $t02036620406 = pool
582+ let amountAssetId = $t02036620406._1
583+ let priceAssetId = $t02036620406._2
584584 let stakingContract = addressFromStringValue(getStrOrFail(this, keyStakingContract))
585585 let factoryContract = addressFromStringValue(getStrOrFail(this, keyFactoryContract))
586586 let lpAssetId = getLpAssetByPoolAssets(amountAssetId, priceAssetId)
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let separator = "__"
55
66 let poolWeightMult = 100000000
77
88 let maxDepthDefault = 10
99
1010 let finalizationStageTotal = 0
1111
1212 let finalizationStageShares = 1
1313
1414 let keyEpochLength = makeString(["%s", "epochLength"], separator)
1515
1616 let keyEpochLengthNew = makeString(["%s%s", "epochLength__new"], separator)
1717
1818 func keyEpochLengthByEpoch (epoch) = makeString(["%s%d", "epochLength", toString(epoch)], separator)
1919
2020
2121 let keyCurrentEpoch = makeString(["%s", "currentEpoch"], separator)
2222
2323 let keyMaxDepth = makeString(["%s", "maxDepth"], separator)
2424
2525 let keyVotingEmissionCandidateContract = makeString(["%s", "votingEmissionCandidateContract"], separator)
2626
2727 let keyFactoryContract = makeString(["%s", "factoryContract"], separator)
2828
2929 let keyBoostingContract = makeString(["%s", "boostingContract"], separator)
3030
3131 let keyStakingContract = makeString(["%s", "stakingContract"], separator)
3232
3333 let keyFinalizationStage = makeString(["%s", "finalizationStage"], separator)
3434
3535 let keyNextPool = makeString(["%s", "nextPool"], separator)
3636
3737 let keyNextUser = makeString(["%s", "nextUser"], separator)
3838
3939 let keyStartHeight = makeString(["%s", "startHeight"], separator)
4040
4141 let keyCurrentEpochUi = makeString(["%s", "currentEpochUi"], separator)
4242
4343 let keyStartHeightUi = makeString(["%s", "startHeightUi"], separator)
4444
4545 let keyFinalizationShouldBeForced = makeString(["%s", "force"], separator)
4646
4747 func keyStartHeightByEpoch (epoch) = makeString(["%s%d", "startHeight", toString(epoch)], separator)
4848
4949
5050 func keyFinalized (epoch) = makeString(["%s%d", "finalized", toString(epoch)], separator)
5151
5252
5353 func keyInList (pool) = {
5454 let $t016931733 = pool
5555 let amountAssetId = $t016931733._1
5656 let priceAssetId = $t016931733._2
5757 makeString(["%s%s%s", "inList", amountAssetId, priceAssetId], separator)
5858 }
5959
6060
6161 func keyUsed (address,epoch) = makeString(["%s%s%d", "used", toString(address), toString(epoch)], separator)
6262
6363
6464 func keyVote (pool,address,epoch) = {
6565 let $t020072047 = pool
6666 let amountAssetId = $t020072047._1
6767 let priceAssetId = $t020072047._2
6868 makeString(["%s%s%s%s%d", "vote", amountAssetId, priceAssetId, toString(address), toString(epoch)], separator)
6969 }
7070
7171
7272 func keyVotingResult (pool,epoch) = {
7373 let $t022292269 = pool
7474 let amountAssetId = $t022292269._1
7575 let priceAssetId = $t022292269._2
7676 makeString(["%s%s%s%d", "votingResult", amountAssetId, priceAssetId, toString(epoch)], separator)
7777 }
7878
7979
8080 func keyVotingResultStaked (lpAssetIdStr,epoch) = makeString(["%s%s%d", "votingResultStaked", lpAssetIdStr, toString(epoch)], separator)
8181
8282
8383 func keyPoolShare (pool,epoch) = {
8484 let $t025902630 = pool
8585 let amountAssetId = $t025902630._1
8686 let priceAssetId = $t025902630._2
8787 makeString(["%s%s%s%d", "poolShare", amountAssetId, priceAssetId, toString(epoch)], separator)
8888 }
8989
9090
9191 func keyTotalVotes (epoch) = makeString(["%s%d", "totalVotes", toString(epoch)], separator)
9292
9393
9494 func keyStakedByUser (lpAssetIdStr,userAddressStr) = makeString(["%s%s%s", "staked", userAddressStr, lpAssetIdStr], separator)
9595
9696
9797 func wrapErr (msg) = makeString(["voting_emission.ride:", msg], " ")
9898
9999
100100 func throwErr (msg) = throw(wrapErr(msg))
101101
102102
103103 func getValueOrFail (address,key,type) = {
104104 let error = makeString(["mandatory ", toString(address), ".", key, " is not defined"], "")
105105 valueOrErrorMessage( match type {
106106 case str: String =>
107107 getString(address, key)
108108 case int: Int =>
109109 getInteger(address, key)
110110 case _ =>
111111 throw("invalid entry type")
112112 }, error)
113113 }
114114
115115
116116 func getStrOrFail (address,key) = {
117117 let @ = getValueOrFail(address, key, "")
118118 if ($isInstanceOf(@, "String"))
119119 then @
120120 else throw(($getType(@) + " couldn't be cast to String"))
121121 }
122122
123123
124124 func getIntOrFail (address,key) = {
125125 let @ = getValueOrFail(address, key, 0)
126126 if ($isInstanceOf(@, "Int"))
127127 then @
128128 else throw(($getType(@) + " couldn't be cast to Int"))
129129 }
130130
131131
132132 func poolToString (pool) = ((pool._1 + separator) + pool._2)
133133
134134
135135 func stringToPool (str) = {
136136 let parts = split(str, separator)
137137 if ((size(parts) == 2))
138138 then $Tuple2(parts[0], parts[1])
139139 else throw("invalid pool string")
140140 }
141141
142142
143143 func getLpAssetByPoolAssets (amountAssetId,priceAssetId) = {
144144 func keyMappingsBaseAsset2internalId (baseAssetStr) = ("%s%s%s__mappings__baseAsset2internalId__" + baseAssetStr)
145145
146146 func keyMappingPoolAssetsToPoolContractAddress (internalAmountAssetIdStr,internalPriceAssetIdStr) = (((("%d%d%s%s__" + toString(internalAmountAssetIdStr)) + "__") + toString(internalPriceAssetIdStr)) + "__mappings__poolAssets2PoolContract")
147147
148148 func keyMappingPoolContractToLPAsset (poolContractAddress) = (("%s%s%s__" + poolContractAddress) + "__mappings__poolContract2LpAsset")
149149
150150 let factoryContract = addressFromStringValue(getStrOrFail(this, keyFactoryContract))
151151 let amountAssetInternalId = getIntOrFail(factoryContract, keyMappingsBaseAsset2internalId(amountAssetId))
152152 let priceAssetInternalId = getIntOrFail(factoryContract, keyMappingsBaseAsset2internalId(priceAssetId))
153153 let poolContractAddress = getStrOrFail(factoryContract, keyMappingPoolAssetsToPoolContractAddress(amountAssetInternalId, priceAssetInternalId))
154154 let lpAssetId = getStrOrFail(factoryContract, keyMappingPoolContractToLPAsset(poolContractAddress))
155155 lpAssetId
156156 }
157157
158158
159159 func checkWxEmissionPoolLabel (pool) = {
160160 let $t051525192 = pool
161161 let amountAssetId = $t051525192._1
162162 let priceAssetId = $t051525192._2
163163 let factoryContract = addressFromStringValue(getStrOrFail(this, keyFactoryContract))
164164 let @ = invoke(factoryContract, "checkWxEmissionPoolLabel", [amountAssetId, priceAssetId], nil)
165165 if ($isInstanceOf(@, "Boolean"))
166166 then @
167167 else throw(($getType(@) + " couldn't be cast to Boolean"))
168168 }
169169
170170
171171 let poolsListName = "pools"
172172
173173 func getVotesListName (pool) = {
174174 let $t054935533 = pool
175175 let amountAssetId = $t054935533._1
176176 let priceAssetId = $t054935533._2
177177 makeString(["votes", amountAssetId, priceAssetId], separator)
178178 }
179179
180180
181181 func keyListHead (listName) = {
182182 let meta = if ((listName == poolsListName))
183183 then "%s%s"
184184 else "%s%s%s%s"
185185 makeString([meta, listName, "head"], separator)
186186 }
187187
188188
189189 func keyListSize (listName) = {
190190 let meta = if ((listName == poolsListName))
191191 then "%s%s"
192192 else "%s%s%s%s"
193193 makeString([meta, listName, "size"], separator)
194194 }
195195
196196
197197 func keyListPrev (listName,id) = {
198198 let meta = if ((listName == poolsListName))
199199 then "%s%s%s%s"
200200 else "%s%s%s%s%s"
201201 makeString([meta, listName, id, "prev"], separator)
202202 }
203203
204204
205205 func keyListNext (listName,id) = {
206206 let meta = if ((listName == poolsListName))
207207 then "%s%s%s%s"
208208 else "%s%s%s%s%s"
209209 makeString([meta, listName, id, "next"], separator)
210210 }
211211
212212
213213 func containsNode (listName,id) = {
214214 let headOrUnit = getString(this, keyListHead(listName))
215215 let prevOrUnit = getString(this, keyListPrev(listName, id))
216216 let nextOrUnit = getString(this, keyListNext(listName, id))
217217 if (if ((id == valueOrElse(headOrUnit, "")))
218218 then true
219219 else (prevOrUnit != unit))
220220 then true
221221 else (nextOrUnit != unit)
222222 }
223223
224224
225225 func insertNodeActions (listName,id) = {
226226 let headOrUnit = getString(this, keyListHead(listName))
227227 let listSize = valueOrElse(getInteger(this, keyListSize(listName)), 0)
228228 let checkNode = if (!(containsNode(listName, id)))
229229 then true
230230 else throw("Node exists")
231231 if ((checkNode == checkNode))
232232 then (([IntegerEntry(keyListSize(listName), (listSize + 1))] ++ (if ((headOrUnit != unit))
233233 then [StringEntry(keyListNext(listName, id), value(headOrUnit)), StringEntry(keyListPrev(listName, value(headOrUnit)), id)]
234234 else nil)) ++ [StringEntry(keyListHead(listName), id)])
235235 else throw("Strict value is not equal to itself.")
236236 }
237237
238238
239239 func deleteNodeActions (listName,id) = {
240240 let headOrUnit = getString(this, keyListHead(listName))
241241 let listSize = valueOrElse(getInteger(this, keyListSize(listName)), 0)
242242 let prevOrUnit = getString(this, keyListPrev(listName, id))
243243 let nextOrUnit = getString(this, keyListNext(listName, id))
244244 ([IntegerEntry(keyListSize(listName), (listSize - 1))] ++ (if (if ((prevOrUnit != unit))
245245 then (nextOrUnit != unit)
246246 else false)
247247 then [StringEntry(keyListNext(listName, value(prevOrUnit)), value(nextOrUnit)), StringEntry(keyListPrev(listName, value(nextOrUnit)), value(prevOrUnit)), DeleteEntry(keyListPrev(listName, id)), DeleteEntry(keyListNext(listName, id))]
248248 else if ((nextOrUnit != unit))
249249 then [StringEntry(keyListHead(listName), value(nextOrUnit)), DeleteEntry(keyListNext(listName, id)), DeleteEntry(keyListPrev(listName, value(nextOrUnit)))]
250250 else if ((prevOrUnit != unit))
251251 then [DeleteEntry(keyListPrev(listName, id)), DeleteEntry(keyListNext(listName, value(prevOrUnit)))]
252252 else if ((id == valueOrElse(headOrUnit, "")))
253253 then [DeleteEntry(keyListHead(listName))]
254254 else throw(((("invalid node: " + listName) + ".") + id))))
255255 }
256256
257257
258258 func keyManagerPublicKey () = "%s__managerPublicKey"
259259
260260
261261 func keyPendingManagerPublicKey () = "%s__pendingManagerPublicKey"
262262
263263
264264 func managerPublicKeyOrUnit () = match getString(keyManagerPublicKey()) {
265265 case s: String =>
266266 fromBase58String(s)
267267 case _: Unit =>
268268 unit
269269 case _ =>
270270 throw("Match error")
271271 }
272272
273273
274274 func pendingManagerPublicKeyOrUnit () = match getString(keyPendingManagerPublicKey()) {
275275 case s: String =>
276276 fromBase58String(s)
277277 case _: Unit =>
278278 unit
279279 case _ =>
280280 throw("Match error")
281281 }
282282
283283
284284 let permissionDeniedError = throw("Permission denied")
285285
286286 func mustThis (i) = if ((i.caller == this))
287287 then true
288288 else permissionDeniedError
289289
290290
291291 func mustManager (i) = match managerPublicKeyOrUnit() {
292292 case pk: ByteVector =>
293293 if ((i.callerPublicKey == pk))
294294 then true
295295 else permissionDeniedError
296296 case _: Unit =>
297297 mustThis(i)
298298 case _ =>
299299 throw("Match error")
300300 }
301301
302302
303303 @Callable(i)
304304 func setManager (pendingManagerPublicKey) = {
305305 let checkCaller = mustManager(i)
306306 if ((checkCaller == checkCaller))
307307 then {
308308 let checkManagerPublicKey = fromBase58String(pendingManagerPublicKey)
309309 if ((checkManagerPublicKey == checkManagerPublicKey))
310310 then [StringEntry(keyPendingManagerPublicKey(), pendingManagerPublicKey)]
311311 else throw("Strict value is not equal to itself.")
312312 }
313313 else throw("Strict value is not equal to itself.")
314314 }
315315
316316
317317
318318 @Callable(i)
319319 func confirmManager () = {
320320 let pm = pendingManagerPublicKeyOrUnit()
321321 let hasPM = if (isDefined(pm))
322322 then true
323323 else throw("No pending manager")
324324 if ((hasPM == hasPM))
325325 then {
326326 let checkPM = if ((i.callerPublicKey == value(pm)))
327327 then true
328328 else throw("You are not pending manager")
329329 if ((checkPM == checkPM))
330330 then [StringEntry(keyManagerPublicKey(), toBase58String(value(pm))), DeleteEntry(keyPendingManagerPublicKey())]
331331 else throw("Strict value is not equal to itself.")
332332 }
333333 else throw("Strict value is not equal to itself.")
334334 }
335335
336336
337337
338338 @Callable(i)
339339 func getUserGwxAmountAtHeight (userAddress,targetHeight) = {
340340 let boostingContractAddress = valueOrErrorMessage(addressFromString(getStrOrFail(this, keyBoostingContract)), "invalid boosting contract address")
341341 $Tuple2(nil, {
342342 let @ = invoke(boostingContractAddress, "getUserGwxAmountAtHeightREADONLY", [userAddress, targetHeight], nil)
343343 if ($isInstanceOf(@, "Int"))
344344 then @
345345 else throw(($getType(@) + " couldn't be cast to Int"))
346346 })
347347 }
348348
349349
350350
351351 @Callable(i)
352352 func constructor (factoryContract,votingEmissionCandidateContract,boostingContract,stakingContract,epochLength) = {
353353 let checks = [mustManager(i), if ((addressFromString(factoryContract) != unit))
354354 then true
355355 else "invalid factory contract address", if ((addressFromString(votingEmissionCandidateContract) != unit))
356356 then true
357357 else "invalid voting emission candidate contract address", if ((addressFromString(boostingContract) != unit))
358358 then true
359359 else "invalid boosting contract address", if ((addressFromString(stakingContract) != unit))
360360 then true
361361 else "invalid staking contract address", if ((epochLength > 0))
362362 then true
363363 else throw("invalid epoch length")]
364364 if ((checks == checks))
365365 then $Tuple2([StringEntry(keyFactoryContract, factoryContract), StringEntry(keyVotingEmissionCandidateContract, votingEmissionCandidateContract), StringEntry(keyBoostingContract, boostingContract), StringEntry(keyStakingContract, stakingContract), IntegerEntry(keyEpochLength, epochLength)], unit)
366366 else throw("Strict value is not equal to itself.")
367367 }
368368
369369
370370
371371 @Callable(i)
372372 func create (amountAssetId,priceAssetId) = {
373373 let checks = [if ((toBase58String(i.caller.bytes) == valueOrElse(getString(this, keyVotingEmissionCandidateContract), "")))
374374 then true
375375 else mustManager(i)]
376376 if ((checks == checks))
377377 then {
378378 let pool = $Tuple2(amountAssetId, priceAssetId)
379379 let inListActions = ([BooleanEntry(keyInList(pool), true)] ++ insertNodeActions(poolsListName, poolToString(pool)))
380380 let currentEpochIsNotDefined = (getInteger(this, keyCurrentEpoch) == unit)
381381 let startHeightActions = if (currentEpochIsNotDefined)
382382 then {
383383 let epoch = 0
384384 [IntegerEntry(keyCurrentEpoch, epoch), IntegerEntry(keyStartHeightByEpoch(epoch), height), IntegerEntry(keyStartHeight, height), IntegerEntry(keyCurrentEpochUi, epoch), IntegerEntry(keyStartHeightUi, height)]
385385 }
386386 else nil
387387 $Tuple2((inListActions ++ startHeightActions), unit)
388388 }
389389 else throw("Strict value is not equal to itself.")
390390 }
391391
392392
393393
394394 @Callable(i)
395395 func vote (amountAssetId,priceAssetId,amount) = {
396396 let pool = $Tuple2(amountAssetId, priceAssetId)
397397 let epoch = valueOrElse(getInteger(this, keyCurrentEpoch), 0)
398398 let startHeight = getIntOrFail(this, keyStartHeightByEpoch(epoch))
399399 let epochLength = getIntOrFail(this, keyEpochLength)
400400 let endHeight = (startHeight + epochLength)
401401 let finalizationStageOrUnit = getInteger(this, keyFinalizationStage)
402402 let used = valueOrElse(getInteger(this, keyUsed(i.caller, epoch)), 0)
403403 let vote = valueOrElse(getInteger(this, keyVote(pool, i.caller, epoch)), 0)
404404 let poolResult = valueOrElse(getInteger(this, keyVotingResult(pool, epoch)), 0)
405405 let totalVotes = valueOrElse(getInteger(this, keyTotalVotes(epoch)), 0)
406406 let gwxAmountAtEndTotal = {
407407 let @ = invoke(this, "getUserGwxAmountAtHeight", [toBase58String(i.caller.bytes), endHeight], nil)
408408 if ($isInstanceOf(@, "Int"))
409409 then @
410410 else throw(($getType(@) + " couldn't be cast to Int"))
411411 }
412412 let available = (gwxAmountAtEndTotal - used)
413413 let newVote = (vote + amount)
414414 let wxEmission = checkWxEmissionPoolLabel(pool)
415415 let checks = [if ((getBoolean(keyInList(pool)) != unit))
416416 then true
417417 else throw("invalid assets"), if ((endHeight > height))
418418 then true
419419 else throw("invalid height"), if ((finalizationStageOrUnit == unit))
420420 then true
421421 else throw("finalization in progress"), if ((gwxAmountAtEndTotal > 0))
422422 then true
423423 else throw("you do not have gWX"), if (if ((amount > 0))
424424 then (available >= amount)
425425 else false)
426426 then true
427427 else throw("invalid amount"), if (wxEmission)
428428 then true
429429 else throw("pool hasn't WX_EMISSION label")]
430430 if ((checks == checks))
431431 then {
432432 let votesListName = getVotesListName(pool)
433433 let userAddressStr = toString(i.caller)
434434 let votesListActions = if (containsNode(votesListName, userAddressStr))
435435 then nil
436436 else insertNodeActions(votesListName, userAddressStr)
437437 $Tuple2(([IntegerEntry(keyUsed(i.caller, epoch), (used + amount)), IntegerEntry(keyVote(pool, i.caller, epoch), newVote), IntegerEntry(keyVotingResult(pool, epoch), (poolResult + amount)), IntegerEntry(keyTotalVotes(epoch), (totalVotes + amount))] ++ votesListActions), unit)
438438 }
439439 else throw("Strict value is not equal to itself.")
440440 }
441441
442442
443443
444444 @Callable(i)
445445 func cancelVote (amountAssetId,priceAssetId) = {
446446 let pool = $Tuple2(amountAssetId, priceAssetId)
447447 let epoch = valueOrElse(getInteger(this, keyCurrentEpoch), 0)
448448 let startHeight = getIntOrFail(this, keyStartHeightByEpoch(epoch))
449449 let epochLength = getIntOrFail(this, keyEpochLength)
450450 let endHeight = (startHeight + epochLength)
451451 let finalizationStageOrUnit = getInteger(this, keyFinalizationStage)
452452 let used = valueOrElse(getInteger(this, keyUsed(i.caller, epoch)), 0)
453453 let vote = valueOrElse(getInteger(this, keyVote(pool, i.caller, epoch)), 0)
454454 let poolResult = valueOrElse(getInteger(this, keyVotingResult(pool, epoch)), 0)
455455 let totalVotes = valueOrElse(getInteger(this, keyTotalVotes(epoch)), 0)
456456 let checks = [if ((getBoolean(keyInList(pool)) != unit))
457457 then true
458458 else throw("invalid assets"), if ((endHeight > height))
459459 then true
460460 else throw("invalid height"), if ((finalizationStageOrUnit == unit))
461461 then true
462462 else throw("finalization in progress"), if ((vote > 0))
463463 then true
464464 else throw("no vote")]
465465 if ((checks == checks))
466466 then {
467467 let votesListName = getVotesListName(pool)
468468 let userAddressStr = toString(i.caller)
469469 $Tuple2(([IntegerEntry(keyUsed(i.caller, epoch), max([(used - vote), 0])), DeleteEntry(keyVote(pool, i.caller, epoch)), IntegerEntry(keyVotingResult(pool, epoch), (poolResult - vote)), IntegerEntry(keyTotalVotes(epoch), (totalVotes - vote))] ++ deleteNodeActions(votesListName, userAddressStr)), unit)
470470 }
471471 else throw("Strict value is not equal to itself.")
472472 }
473473
474474
475475
476476 @Callable(i)
477477 func setEpochLength (newEpochLength) = {
478478 let checks = [mustManager(i), if ((newEpochLength > 0))
479479 then true
480480 else throw("invalid epoch length")]
481481 if ((checks == checks))
482482 then $Tuple2([IntegerEntry(keyEpochLengthNew, newEpochLength)], unit)
483483 else throw("Strict value is not equal to itself.")
484484 }
485485
486486
487487
488488 @Callable(i)
489489 func setMaxDepth (newMaxDepth) = {
490490 let checks = [mustManager(i), if ((newMaxDepth > 0))
491491 then true
492492 else throw("invalid max depth")]
493493 if ((checks == checks))
494494 then $Tuple2([IntegerEntry(keyMaxDepth, newMaxDepth)], unit)
495495 else throw("Strict value is not equal to itself.")
496496 }
497497
498498
499499
500500 @Callable(i)
501501 func processVoteINTERNAL (poolStr,userAddressStr) = {
502502 let checkCaller = mustThis(i)
503503 if ((checkCaller == checkCaller))
504504 then {
505505 let userAddress = valueOrErrorMessage(addressFromString(userAddressStr), ("processVoteINTERNAL: invalid user address " + userAddressStr))
506506 let epoch = getIntOrFail(this, keyCurrentEpoch)
507507 let epochPrevious = (epoch - 1)
508508 let epochLength = getIntOrFail(this, keyEpochLength)
509509 let startHeight = getIntOrFail(this, keyStartHeightByEpoch(epoch))
510510 let endHeight = (startHeight + epochLength)
511511 let startHeightPrevious = getIntOrFail(this, keyStartHeightByEpoch(epochPrevious))
512512 let epochLengthPrevious = getIntOrFail(this, keyEpochLengthByEpoch(epochPrevious))
513513 let endHeightPrevious = (startHeightPrevious + epochLengthPrevious)
514514 let checkTargetEpoch = if ((epochPrevious >= 0))
515515 then true
516516 else throw("processVoteINTERNAL: invalid previous epoch")
517517 if ((checkTargetEpoch == checkTargetEpoch))
518518 then {
519519 let pool = stringToPool(poolStr)
520- let $t01801718057 = pool
521- let amountAssetId = $t01801718057._1
522- let priceAssetId = $t01801718057._2
520+ let $t01798418024 = pool
521+ let amountAssetId = $t01798418024._1
522+ let priceAssetId = $t01798418024._2
523523 let wxEmission = checkWxEmissionPoolLabel(pool)
524524 let gwxAmountAtEndTotal = {
525525 let @ = invoke(this, "getUserGwxAmountAtHeight", [userAddressStr, endHeight], nil)
526526 if ($isInstanceOf(@, "Int"))
527527 then @
528528 else throw(($getType(@) + " couldn't be cast to Int"))
529529 }
530530 let gwxAmountAtEndTotalPrevious = {
531531 let @ = invoke(this, "getUserGwxAmountAtHeight", [userAddressStr, endHeightPrevious], nil)
532532 if ($isInstanceOf(@, "Int"))
533533 then @
534534 else throw(($getType(@) + " couldn't be cast to Int"))
535535 }
536536 let totalVotes = valueOrElse(getInteger(keyTotalVotes(epoch)), 0)
537537 let votingResult = valueOrElse(getInteger(keyVotingResult(pool, epoch)), 0)
538538 let votePrevious = valueOrErrorMessage(getInteger(keyVote(pool, userAddress, epochPrevious)), (((("processVoteINTERNAL " + poolStr) + " ") + userAddressStr) + ": no previous vote"))
539539 let used = valueOrElse(getInteger(this, keyUsed(userAddress, epoch)), 0)
540540 let lpAssetIdStr = getLpAssetByPoolAssets(amountAssetId, priceAssetId)
541541 let votingResultStakedPrevious = valueOrElse(getInteger(keyVotingResultStaked(lpAssetIdStr, epochPrevious)), 0)
542542 let stakingContract = addressFromStringValue(getStrOrFail(this, keyStakingContract))
543543 let stakedByUser = valueOrElse(getInteger(stakingContract, keyStakedByUser(lpAssetIdStr, userAddressStr)), 0)
544544 let votingResultStakedActions = if ((stakedByUser == 0))
545545 then nil
546546 else [IntegerEntry(keyVotingResultStaked(lpAssetIdStr, epochPrevious), (votingResultStakedPrevious + votePrevious))]
547547 let newVote = if ((gwxAmountAtEndTotalPrevious > 0))
548548 then fraction(votePrevious, gwxAmountAtEndTotal, gwxAmountAtEndTotalPrevious)
549549 else 0
550550 let actions = if (if ((newVote > 0))
551551 then wxEmission
552552 else false)
553553 then [IntegerEntry(keyVote(pool, userAddress, epoch), newVote), IntegerEntry(keyTotalVotes(epoch), (totalVotes + newVote)), IntegerEntry(keyVotingResult(pool, epoch), (votingResult + newVote)), IntegerEntry(keyUsed(userAddress, epoch), (used + newVote))]
554554 else deleteNodeActions(getVotesListName(pool), userAddressStr)
555555 $Tuple2((actions ++ votingResultStakedActions), unit)
556556 }
557557 else throw("Strict value is not equal to itself.")
558558 }
559559 else throw("Strict value is not equal to itself.")
560560 }
561561
562562
563563
564564 @Callable(i)
565565 func processPoolINTERNAL (poolStr,force) = {
566566 let checkCaller = mustThis(i)
567567 if ((checkCaller == checkCaller))
568568 then {
569569 let targetEpoch = {
570570 let currentEpoch = getIntOrFail(this, keyCurrentEpoch)
571571 if (force)
572572 then currentEpoch
573573 else (currentEpoch - 1)
574574 }
575575 let checkTargetEpoch = if ((targetEpoch >= 0))
576576 then true
577577 else throw("processPoolINTERNAL: invalid target epoch")
578578 if ((checkTargetEpoch == checkTargetEpoch))
579579 then {
580580 let pool = stringToPool(poolStr)
581- let $t02039920439 = pool
582- let amountAssetId = $t02039920439._1
583- let priceAssetId = $t02039920439._2
581+ let $t02036620406 = pool
582+ let amountAssetId = $t02036620406._1
583+ let priceAssetId = $t02036620406._2
584584 let stakingContract = addressFromStringValue(getStrOrFail(this, keyStakingContract))
585585 let factoryContract = addressFromStringValue(getStrOrFail(this, keyFactoryContract))
586586 let lpAssetId = getLpAssetByPoolAssets(amountAssetId, priceAssetId)
587587 let r = {
588588 let @ = invoke(stakingContract, "usersListTraversal", [lpAssetId], nil)
589589 if ($isInstanceOf(@, "Boolean"))
590590 then @
591591 else throw(($getType(@) + " couldn't be cast to Boolean"))
592592 }
593593 if ((r == r))
594594 then if (r)
595595 then $Tuple2(nil, true)
596596 else {
597597 let wxEmission = checkWxEmissionPoolLabel(pool)
598598 let totalVotes = valueOrElse(getInteger(this, keyTotalVotes(targetEpoch)), 0)
599599 let votingResult = valueOrElse(getInteger(this, keyVotingResult(pool, targetEpoch)), 0)
600600 let share = if (if ((totalVotes == 0))
601601 then true
602602 else !(wxEmission))
603603 then 0
604604 else fraction(votingResult, poolWeightMult, totalVotes)
605605 let modifyWeightInv = invoke(factoryContract, "modifyWeight", [lpAssetId, share], nil)
606606 if ((modifyWeightInv == modifyWeightInv))
607607 then {
608608 let poolsListActions = if (if (wxEmission)
609609 then true
610610 else force)
611611 then nil
612612 else ([DeleteEntry(keyInList(pool))] ++ deleteNodeActions(poolsListName, poolStr))
613613 $Tuple2(([IntegerEntry(keyPoolShare(pool, targetEpoch), share)] ++ poolsListActions), false)
614614 }
615615 else throw("Strict value is not equal to itself.")
616616 }
617617 else throw("Strict value is not equal to itself.")
618618 }
619619 else throw("Strict value is not equal to itself.")
620620 }
621621 else throw("Strict value is not equal to itself.")
622622 }
623623
624624
625625
626626 @Callable(i)
627627 func finalizeHelper () = {
628628 let force = valueOrElse(getBoolean(keyFinalizationShouldBeForced), false)
629629 let epoch = valueOrElse(getInteger(this, keyCurrentEpoch), 0)
630630 let previousEpoch = (epoch - 1)
631631 let startHeight = getIntOrFail(this, keyStartHeightByEpoch(epoch))
632632 let epochLength = getIntOrFail(this, keyEpochLength)
633633 let endHeight = (startHeight + epochLength)
634634 let finalizationStageOrUnit = getInteger(this, keyFinalizationStage)
635635 if (if (if ((height >= endHeight))
636636 then (finalizationStageOrUnit == unit)
637637 else false)
638638 then !(force)
639639 else false)
640640 then {
641641 let newEpoch = (epoch + 1)
642642 let newEpochLengthOption = getInteger(this, keyEpochLengthNew)
643643 let newEpochLengthActions = match newEpochLengthOption {
644644 case newEpochLength: Int =>
645645 [IntegerEntry(keyEpochLength, newEpochLength), DeleteEntry(keyEpochLengthNew)]
646646 case _: Unit =>
647647 nil
648648 case _ =>
649649 throw("Match error")
650650 }
651651 $Tuple2(([IntegerEntry(keyStartHeightByEpoch(newEpoch), height), IntegerEntry(keyStartHeight, height), IntegerEntry(keyCurrentEpoch, newEpoch), IntegerEntry(keyFinalizationStage, finalizationStageTotal), IntegerEntry(keyEpochLengthByEpoch(epoch), epochLength)] ++ newEpochLengthActions), true)
652652 }
653653 else if (if (force)
654654 then (finalizationStageOrUnit == unit)
655655 else false)
656656 then $Tuple2([IntegerEntry(keyFinalizationStage, finalizationStageShares)], true)
657657 else if ((finalizationStageOrUnit == unit))
658658 then $Tuple2(nil, false)
659659 else if ((finalizationStageOrUnit == finalizationStageTotal))
660660 then {
661661 let poolOrUnit = getString(keyNextPool)
662662 let userOrUnit = getString(keyNextUser)
663663 match poolOrUnit {
664664 case _: Unit =>
665665 match getString(keyListHead(poolsListName)) {
666666 case _: Unit =>
667667 $Tuple2([IntegerEntry(keyFinalizationStage, finalizationStageShares), DeleteEntry(keyNextPool), DeleteEntry(keyNextUser)], true)
668668 case poolsHeadStr: String =>
669669 $Tuple2([StringEntry(keyNextPool, poolsHeadStr)], true)
670670 case _ =>
671671 throw("Match error")
672672 }
673673 case poolStr: String =>
674674 let pool = stringToPool(poolStr)
675675 let nextUserOrUnit = match userOrUnit {
676676 case _: Unit =>
677677 getString(keyListHead(getVotesListName(pool)))
678678 case user: String =>
679679 let next = getString(keyListNext(getVotesListName(pool), user))
680680 if ((next == next))
681681 then {
682682 let processVoteInv = invoke(this, "processVoteINTERNAL", [poolStr, user], nil)
683683 if ((processVoteInv == processVoteInv))
684684 then next
685685 else throw("Strict value is not equal to itself.")
686686 }
687687 else throw("Strict value is not equal to itself.")
688688 case _ =>
689689 throw("Match error")
690690 }
691691 match nextUserOrUnit {
692692 case _: Unit =>
693693 let nextPoolOrUnit = getString(keyListNext(poolsListName, poolStr))
694694 match nextPoolOrUnit {
695695 case _: Unit =>
696696 $Tuple2([IntegerEntry(keyFinalizationStage, finalizationStageShares), DeleteEntry(keyNextPool), DeleteEntry(keyNextUser)], true)
697697 case s: String =>
698698 $Tuple2([StringEntry(keyNextPool, s), DeleteEntry(keyNextUser)], true)
699699 case _ =>
700700 throw("Match error")
701701 }
702702 case nextUser: String =>
703703 $Tuple2([StringEntry(keyNextUser, nextUser)], true)
704704 case _ =>
705705 throw("Match error")
706706 }
707707 case _ =>
708708 throw("Match error")
709709 }
710710 }
711711 else if ((finalizationStageOrUnit == finalizationStageShares))
712712 then {
713713 let poolOrUnit = getString(keyNextPool)
714714 match poolOrUnit {
715715 case _: Unit =>
716716 match getString(keyListHead(poolsListName)) {
717717 case _: Unit =>
718718 let actions = if (force)
719719 then [DeleteEntry(keyFinalizationStage), DeleteEntry(keyFinalizationShouldBeForced)]
720720 else [DeleteEntry(keyFinalizationStage), BooleanEntry(keyFinalized(previousEpoch), true), IntegerEntry(keyCurrentEpochUi, epoch), IntegerEntry(keyStartHeightUi, startHeight)]
721721 $Tuple2(actions, true)
722722 case nextPoolStr: String =>
723723 $Tuple2([StringEntry(keyNextPool, nextPoolStr)], true)
724724 case _ =>
725725 throw("Match error")
726726 }
727727 case poolStr: String =>
728728 let nextPoolOrUnit = getString(keyListNext(poolsListName, poolStr))
729729 if ((nextPoolOrUnit == nextPoolOrUnit))
730730 then {
731731 let r = {
732732 let @ = invoke(this, "processPoolINTERNAL", [poolStr, force], nil)
733733 if ($isInstanceOf(@, "Boolean"))
734734 then @
735735 else throw(($getType(@) + " couldn't be cast to Boolean"))
736736 }
737737 if ((r == r))
738738 then if (r)
739739 then $Tuple2(nil, true)
740740 else match nextPoolOrUnit {
741741 case _: Unit =>
742742 let actions = if (force)
743743 then [DeleteEntry(keyFinalizationStage), DeleteEntry(keyFinalizationShouldBeForced), DeleteEntry(keyNextPool)]
744744 else [DeleteEntry(keyFinalizationStage), BooleanEntry(keyFinalized(previousEpoch), true), IntegerEntry(keyCurrentEpochUi, epoch), IntegerEntry(keyStartHeightUi, startHeight), DeleteEntry(keyNextPool)]
745745 $Tuple2(actions, true)
746746 case nextPoolStr: String =>
747747 $Tuple2([StringEntry(keyNextPool, nextPoolStr)], true)
748748 case _ =>
749749 throw("Match error")
750750 }
751751 else throw("Strict value is not equal to itself.")
752752 }
753753 else throw("Strict value is not equal to itself.")
754754 case _ =>
755755 throw("Match error")
756756 }
757757 }
758758 else throw("finalization is broken")
759759 }
760760
761761
762762
763763 @Callable(i)
764764 func finalizeWrapper (counter) = {
765765 let result = {
766766 let @ = invoke(this, "finalizeHelper", nil, nil)
767767 if ($isInstanceOf(@, "Boolean"))
768768 then @
769769 else throw(($getType(@) + " couldn't be cast to Boolean"))
770770 }
771771 if ((result == result))
772772 then if (!(result))
773773 then if ((counter == 0))
774774 then throw("Current voting is not over yet")
775775 else $Tuple2(nil, unit)
776776 else {
777777 let maxDepth = valueOrElse(getInteger(this, keyMaxDepth), maxDepthDefault)
778778 if ((maxDepth > counter))
779779 then {
780780 let inv = invoke(this, "finalizeWrapper", [(counter + 1)], nil)
781781 if ((inv == inv))
782782 then $Tuple2(nil, unit)
783783 else throw("Strict value is not equal to itself.")
784784 }
785785 else $Tuple2(nil, unit)
786786 }
787787 else throw("Strict value is not equal to itself.")
788788 }
789789
790790
791791
792792 @Callable(i)
793793 func finalize () = {
794794 let inv = invoke(this, "finalizeWrapper", [0], nil)
795795 if ((inv == inv))
796796 then $Tuple2(nil, unit)
797797 else throw("Strict value is not equal to itself.")
798798 }
799799
800800
801801
802802 @Callable(i)
803803 func containsNodeREADONLY (listName,id) = $Tuple2(nil, containsNode(listName, id))
804804
805805
806806
807807 @Callable(i)
808808 func insertNode (listName,id) = {
809809 let checkCaller = mustManager(i)
810810 if ((checkCaller == checkCaller))
811811 then $Tuple2(insertNodeActions(listName, id), unit)
812812 else throw("Strict value is not equal to itself.")
813813 }
814814
815815
816816
817817 @Callable(i)
818818 func deleteNode (listName,id) = {
819819 let checkCaller = mustManager(i)
820820 if ((checkCaller == checkCaller))
821821 then $Tuple2(deleteNodeActions(listName, id), unit)
822822 else throw("Strict value is not equal to itself.")
823823 }
824824
825825
826826 @Verifier(tx)
827827 func verify () = {
828828 let targetPublicKey = match managerPublicKeyOrUnit() {
829829 case pk: ByteVector =>
830830 pk
831831 case _: Unit =>
832832 tx.senderPublicKey
833833 case _ =>
834834 throw("Match error")
835835 }
836836 sigVerify(tx.bodyBytes, tx.proofs[0], targetPublicKey)
837837 }
838838

github/deemru/w8io/c3f4982 
83.02 ms