tx · G6DQHkbRd85r8NeH8ignjkeF5js2fzJERmFyv5yxnV8q

3N9Zi6AcWcGbM23jQJ4fnzFsoHoTsxYDznr:  -0.02300000 Waves

2023.07.27 12:32 [2684077] smart account 3N9Zi6AcWcGbM23jQJ4fnzFsoHoTsxYDznr > SELF 0.00000000 Waves

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

github/deemru/w8io/169f3d6 
81.19 ms