tx · BmKZmrmXxxp1SJeRLq3jNn5chPNgNf5K1abCjz7raFEg

3P4uA5etnZi4AmBabKinq2bMiWU8KcnHZdH:  -0.01700000 Waves

2022.10.24 20:50 [3352221] smart account 3P4uA5etnZi4AmBabKinq2bMiWU8KcnHZdH > SELF 0.00000000 Waves

{ "type": 13, "id": "BmKZmrmXxxp1SJeRLq3jNn5chPNgNf5K1abCjz7raFEg", "fee": 1700000, "feeAssetId": null, "timestamp": 1666633910296, "version": 2, "chainId": 87, "sender": "3P4uA5etnZi4AmBabKinq2bMiWU8KcnHZdH", "senderPublicKey": "8DxbUxhy23djr6kUEE1Jzp7oVJXBsHNaATLRiABpkSde", "proofs": [ "4iNuvagp6P3VbQ5GePuxqcen6KHa2gzxoJHmMbW7PMfc1thzpdZ92WbXJyVuBNfHo9WpLYypNReFjcr1vVbYx8ro" ], "script": "base64:BgJMCAISBgoECAgICBIAEgQKAggBEgQKAggBEgASCAoGBAgBCAgIEgYKBAQIBAgSAwoBBBIECgIIBBIECgIIBBIDCgEEEgMKAQQSAwoBBBYABlNjYWxlOACAwtcvAAdTY2FsZTEwAIDIr6AlAAdTY2FsZTE2CQBoAgUGU2NhbGU4BQZTY2FsZTgAC3Jlc2VydmVGdW5kABQACWRheUJsb2NrcwCgCwENdHJ5R2V0SW50ZWdlcgEDa2V5BAckbWF0Y2gwCQCaCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgNJbnQEAWIFByRtYXRjaDAFAWIAAAENdHJ5R2V0Qm9vbGVhbgEDa2V5BAckbWF0Y2gwCQCbCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgdCb29sZWFuBAFiBQckbWF0Y2gwBQFiBwEMdHJ5R2V0U3RyaW5nAQNrZXkEByRtYXRjaDAJAJ0IAgUEdGhpcwUDa2V5AwkAAQIFByRtYXRjaDACBlN0cmluZwQBYgUHJG1hdGNoMAUBYgIAAQ5nZXRBc3NldFN0cmluZwEHYXNzZXRJZAQHJG1hdGNoMAUHYXNzZXRJZAMJAAECBQckbWF0Y2gwAgpCeXRlVmVjdG9yBAFiBQckbWF0Y2gwCQDYBAEFAWICBVdBVkVTAQ1nZXRBc3NldEJ5dGVzAQphc3NldElkU3RyAwkAAAIFCmFzc2V0SWRTdHICBVdBVkVTBQR1bml0CQDZBAEFCmFzc2V0SWRTdHIBCmdldEJhbGFuY2UBCmFzc2V0SWRTdHIDCQAAAgUKYXNzZXRJZFN0cgIFV0FWRVMICQDvBwEFBHRoaXMJYXZhaWxhYmxlCQDwBwIFBHRoaXMJANkEAQUKYXNzZXRJZFN0cgEPZ2V0TWFya2V0QXNzZXRzAAkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIMc2V0dXBfdG9rZW5zAgEsAQ1nZXRPdXRkYXRlZFVyAQphc3NldElkU3RyBARkb3duCQBrAwkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkBDXRyeUdldEludGVnZXIBCQCsAgIFCmFzc2V0SWRTdHICBl9zUmF0ZQUHU2NhbGUxNgMJAAACBQRkb3duAAAAAAkAawMFBlNjYWxlOAkAawMJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfYlJhdGUFB1NjYWxlMTYFBGRvd24BDGdldFJhdGVDdXJ2ZQEKYXNzZXRJZFN0cgQHJG1hdGNoMAUKYXNzZXRJZFN0cgMJAAACAiwzNE45WWNFRVRMV245M3FZUTY0RXNQMXg4OXRTcnVKVTQ0UnJFTVNYWEVQSgUHJG1hdGNoMAkAlgoEAICJegDA8PULAIDokiYAgMLXLwMJAAACAixERzJ4RmtQZER3S1VvQmt6R0FoUXRMcFNHemZYTGlDWVBFemVLSDJBZDI0cAUHJG1hdGNoMAkAlgoEAICJegDA8PULAIDokiYAgMLXLwMJAAACAixBanNvNm5UVGpwdHUyVUhMeDZoZlNYVnRIRnRSQkpDa0tZZDVTQXlqN3pmNQUHJG1hdGNoMAkAlgoEAICJegCAtIkTAIDokiYAgKPDRwMJAAACAixIRUI4UWF3OXhyV3BXczh0SHNpQVRZR0JXREJ0UDJTN2tjUEFMck11NDNBUwUHJG1hdGNoMAkAlgoEAAAAgNrECQCA6JImAIC0iRMDCQAAAgIFV0FWRVMFByRtYXRjaDAJAJYKBACAiXoAgIenDgCA6JImAIDh6xcJAJYKBAAAAIDaxAkAgOiSJgCA6JImAQtnZXRJbnRlcmVzdAEKYXNzZXRJZFN0cgQCdXIJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgQFY3VydmUJAQxnZXRSYXRlQ3VydmUBBQphc3NldElkU3RyBARyYXRlCQBkAggFBWN1cnZlAl8xAwkAZwIIBQVjdXJ2ZQJfMwUCdXIJAGsDBQJ1cggFBWN1cnZlAl8yCAUFY3VydmUCXzMJAGQCCAUFY3VydmUCXzIJAGsDCQBlAgUCdXIIBQVjdXJ2ZQJfMwkAZQIIBQVjdXJ2ZQJfNAgFBWN1cnZlAl8yCQBlAgCAwtcvCAUFY3VydmUCXzMJAJYDAQkAzAgCCQBrAwUEcmF0ZQUGU2NhbGU4CQBoAgUJZGF5QmxvY2tzAO0CCQDMCAIAAQUDbmlsARB0b2tlblJhdGVzUmVjYWxjAQphc3NldElkU3RyBAhpbnRlcmVzdAkBC2dldEludGVyZXN0AQUKYXNzZXRJZFN0cgQCdXIJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgQQbGFzdFJlY2FsY0hlaWdodAkBDXRyeUdldEludGVnZXIBAg5sYXN0UmF0ZUhlaWdodAQJbGFzdEJSYXRlCQCWAwEJAMwIAgkBDXRyeUdldEludGVnZXIBCQCsAgIFCmFzc2V0SWRTdHICBl9iUmF0ZQkAzAgCBQdTY2FsZTE2BQNuaWwECG5ld0JSYXRlCQBkAgUJbGFzdEJSYXRlCQBoAgkAZQIFBmhlaWdodAUQbGFzdFJlY2FsY0hlaWdodAUIaW50ZXJlc3QECWxhc3RTUmF0ZQkAlgMBCQDMCAIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfc1JhdGUJAMwIAgUHU2NhbGUxNgUDbmlsBAhuZXdTUmF0ZQkAZAIFCWxhc3RTUmF0ZQkAaQIJAGgCCQBoAgkAZQIFBmhlaWdodAUQbGFzdFJlY2FsY0hlaWdodAkAawMFCGludGVyZXN0BQJ1cgUGU2NhbGU4CQBlAgBkBQtyZXNlcnZlRnVuZABkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICBl9zUmF0ZQUIbmV3U1JhdGUJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIGX2JSYXRlBQhuZXdCUmF0ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgIObGFzdFJhdGVIZWlnaHQFBmhlaWdodAUDbmlsAQ1nZXRBY3R1YWxSYXRlAgphc3NldElkU3RyCHJhdGVUeXBlCgEBZgIFYWNjdW0FdG9rZW4EBnJlY2FsYwkBEHRva2VuUmF0ZXNSZWNhbGMBBQV0b2tlbgkAlAoCAwkBAiE9AgUFdG9rZW4FCmFzc2V0SWRTdHIIBQVhY2N1bQJfMQMJAAACBQhyYXRlVHlwZQIFc1JhdGUICQCRAwIFBnJlY2FsYwAABXZhbHVlCAkAkQMCBQZyZWNhbGMAAQV2YWx1ZQkAzggCCAUFYWNjdW0CXzIFBnJlY2FsYwoAAiRsCQEPZ2V0TWFya2V0QXNzZXRzAAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAJAJQKAgAABQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECE0xpc3Qgc2l6ZSBleGNlZWRzIDYJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYBBWdldFVyAQphc3NldElkU3RyBAVyYXRlcwkBEHRva2VuUmF0ZXNSZWNhbGMBBQphc3NldElkU3RyBARkb3duCQBrAwkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cggJAJEDAgUFcmF0ZXMAAAV2YWx1ZQUHU2NhbGUxNgkAawMFBlNjYWxlOAkAawMJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIICQCRAwIFBXJhdGVzAAEFdmFsdWUFB1NjYWxlMTYFBGRvd24BC3JhdGVzUmVjYWxjAAoBAWYCBWFjY3VtBXRva2VuCQDOCAIFBWFjY3VtCQEQdG9rZW5SYXRlc1JlY2FsYwEFBXRva2VuCgACJGwJAQ9nZXRNYXJrZXRBc3NldHMACgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhNMaXN0IHNpemUgZXhjZWVkcyA2CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAQ1nZXRUb2tlblByaWNlAQphc3NldElkU3RyBANpbnYJAPwHBAkBEUBleHRyTmF0aXZlKDEwNjIpAQIjM1A1QlR0ZGozMlNkMUR5aDFNZHczM3hRQUFja1NmTWZuS2YCCWdldFRXQVA2MAkAzAgCBQphc3NldElkU3RyCQDMCAIHBQNuaWwFA25pbAMJAAACBQNpbnYFA2ludgQHJG1hdGNoMAUDaW52AwkAAQIFByRtYXRjaDACCihJbnQsIEludCkEAXgFByRtYXRjaDAFAXgJAAIBAhVlcnJvciBvZiBwcmljZSBvcmFjbGUJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BDmNhbGNBc3NldFNjYWxlAQphc3NldElkU3RyBAhkZWNpbWFscwMJAAACBQphc3NldElkU3RyAgVXQVZFUwAICAkBBXZhbHVlAQkA7AcBCQDZBAEFCmFzc2V0SWRTdHIIZGVjaW1hbHMJAGwGAAoAAAUIZGVjaW1hbHMAAAAABQRET1dOARJjYWxjVXNlckNvbGxhdGVyYWwBB2FkZHJlc3MEFHVzZXJDb2xsYXRlcmFsSW52b2tlCQD8BwQFBHRoaXMCEWdldFVzZXJDb2xsYXRlcmFsCQDMCAIHCQDMCAIFB2FkZHJlc3MJAMwIAgYJAMwIAgIABQNuaWwFA25pbAMJAAACBRR1c2VyQ29sbGF0ZXJhbEludm9rZQUUdXNlckNvbGxhdGVyYWxJbnZva2UEE3VzZXJDb2xsYXRlcmFsVmFsdWUEByRtYXRjaDAFFHVzZXJDb2xsYXRlcmFsSW52b2tlAwkAAQIFByRtYXRjaDACA0ludAQBeAUHJG1hdGNoMAUBeAkAAgECJGlzc3VlIHdoaWxlIGRvaW5nIGluLWRhcHAgaW52b2NhdGlvbgMJAAACBRN1c2VyQ29sbGF0ZXJhbFZhbHVlBRN1c2VyQ29sbGF0ZXJhbFZhbHVlBRN1c2VyQ29sbGF0ZXJhbFZhbHVlCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuDQFpAQdwcmVJbml0BAZ0b2tlbnMEbHR2cwNsdHMJcGVuYWx0aWVzCgEBZgIFYWNjdW0FdG9rZW4JAM4IAgUFYWNjdW0JAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUFdG9rZW4CBl9iUmF0ZQUHU2NhbGUxNgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQV0b2tlbgIGX3NSYXRlBQdTY2FsZTE2BQNuaWwDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAgphZG1pbiBvbmx5BAVyYXRlcwoAAiRsCQC1CQIFBnRva2VucwIBLAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEBZgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQITTGlzdCBzaXplIGV4Y2VlZHMgNgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgkAzggCCQDMCAIJAQtTdHJpbmdFbnRyeQICDHNldHVwX3Rva2VucwUGdG9rZW5zCQDMCAIJAQtTdHJpbmdFbnRyeQICCnNldHVwX2x0dnMFBGx0dnMJAMwIAgkBC1N0cmluZ0VudHJ5AgIJc2V0dXBfbHRzBQNsdHMJAMwIAgkBC1N0cmluZ0VudHJ5AgIPc2V0dXBfcGVuYWx0aWVzBQlwZW5hbHRpZXMJAMwIAgkBDEJvb2xlYW5FbnRyeQICDHNldHVwX2FjdGl2ZQYFA25pbAUFcmF0ZXMBaQEGc3VwcGx5AAMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAMDCQECIT0CCQCQAwEIBQFpCHBheW1lbnRzAAEGCQAAAggJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQAAAkAAgECHDEgcGF5bWVudCBoYXMgdG8gYmUgYXR0YWNoZWQECmFzc2V0SWRTdHIJAQ5nZXRBc3NldFN0cmluZwEICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQLYXNzZXRBbW91bnQICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BAskdDA1NzAzNTc3MAkBDWdldEFjdHVhbFJhdGUCBQphc3NldElkU3RyAgVzUmF0ZQQFc1JhdGUIBQskdDA1NzAzNTc3MAJfMQQRcmF0ZXNSZWNhbGNSZXN1bHQIBQskdDA1NzAzNTc3MAJfMgQGYW1vdW50CQBuBAULYXNzZXRBbW91bnQFB1NjYWxlMTYFBXNSYXRlBQRET1dOBAdhZGRyZXNzCQClCAEIBQFpBmNhbGxlcgMJAAACCQCzCQIJAQx0cnlHZXRTdHJpbmcBAgxzZXR1cF90b2tlbnMFCmFzc2V0SWRTdHIFBHVuaXQJAAIBAil0aGlzIGFzc2V0IGlzIG5vdCBzdXBwb3J0ZWQgYnkgdGhlIG1hcmtldAkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkAZAIJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FCmFzc2V0SWRTdHIFBmFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBQZhbW91bnQFA25pbAURcmF0ZXNSZWNhbGNSZXN1bHQBaQEId2l0aGRyYXcCCmFzc2V0SWRTdHILYXNzZXRBbW91bnQECyR0MDYzNDE2NDA4CQENZ2V0QWN0dWFsUmF0ZQIFCmFzc2V0SWRTdHICBXNSYXRlBAVzUmF0ZQgFCyR0MDYzNDE2NDA4Al8xBBFyYXRlc1JlY2FsY1Jlc3VsdAgFCyR0MDYzNDE2NDA4Al8yBAZhbW91bnQJAG4EBQthc3NldEFtb3VudAUHU2NhbGUxNgUFc1JhdGUFBkhBTEZVUAQHYWRkcmVzcwkApQgBCAUBaQZjYWxsZXIEDWFzc2V0U3VwcGxpZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIEDWFzc2V0Qm9ycm93ZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIEEXVzZXJBc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQSY29sbGF0ZXJhbFZhbHVlSW52CQD8BwQFBHRoaXMCEWdldFVzZXJDb2xsYXRlcmFsCQDMCAIHCQDMCAIFB2FkZHJlc3MJAMwIAgYJAMwIAgkArAICCQCsAgIFCmFzc2V0SWRTdHICCixzdXBwbGllZCwJAKQDAQkBAS0BBQZhbW91bnQFA25pbAUDbmlsAwkAAAIFEmNvbGxhdGVyYWxWYWx1ZUludgUSY29sbGF0ZXJhbFZhbHVlSW52BA9jb2xsYXRlcmFsVmFsdWUEByRtYXRjaDAFEmNvbGxhdGVyYWxWYWx1ZUludgMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAh9jYW4ndCBnZXQgdXNlciBjb2xsYXRlcmFsIHZhbHVlAwkBASEBCQENdHJ5R2V0Qm9vbGVhbgECDHNldHVwX2FjdGl2ZQkAAgECEW1hcmtldCBpcyBzdG9wcGVkAwkAZgIAAAUPY29sbGF0ZXJhbFZhbHVlCQACAQIyeW91IGRvbnQgaGF2ZSBlbm91Z2ggY29sbGF0ZXJhbCBmb3IgdGhpcyBvcGVyYXRpb24DCQBmAgUGYW1vdW50CQBlAgUNYXNzZXRTdXBwbGllZAUNYXNzZXRCb3Jyb3dlZAkAAgECKnRoaXMgYW1vdW50IGlzIG5vdCBhdmFpbGFibGUgb24gdGhlIG1hcmtldAMJAGYCBQZhbW91bnQJAGUCBRF1c2VyQXNzZXRTdXBwbGllZAURdXNlckFzc2V0Qm9ycm93ZWQJAAIBAip0aGlzIGFtb3VudCBpcyBub3QgYXZhaWxhYmxlIGZvciB0aGlzIHVzZXIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FCmFzc2V0SWRTdHIJAGUCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyBQZhbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyCQBlAgkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgUGYW1vdW50CQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMIBQFpBmNhbGxlcgULYXNzZXRBbW91bnQJAQ1nZXRBc3NldEJ5dGVzAQUKYXNzZXRJZFN0cgUDbmlsBRFyYXRlc1JlY2FsY1Jlc3VsdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQZib3Jyb3cCCmFzc2V0SWRTdHILYXNzZXRBbW91bnQEB2FkZHJlc3MJAKUIAQgFAWkGY2FsbGVyBAskdDA3OTAwNzk2NwkBDWdldEFjdHVhbFJhdGUCBQphc3NldElkU3RyAgViUmF0ZQQFYlJhdGUIBQskdDA3OTAwNzk2NwJfMQQRcmF0ZXNSZWNhbGNSZXN1bHQIBQskdDA3OTAwNzk2NwJfMgQGYW1vdW50CQBuBAULYXNzZXRBbW91bnQFB1NjYWxlMTYFBWJSYXRlBQdDRUlMSU5HBBJjb2xsYXRlcmFsVmFsdWVJbnYJAPwHBAUEdGhpcwIRZ2V0VXNlckNvbGxhdGVyYWwJAMwIAgcJAMwIAgUHYWRkcmVzcwkAzAgCBgkAzAgCCQCsAgIJAKwCAgUKYXNzZXRJZFN0cgIKLGJvcnJvd2VkLAkApAMBBQZhbW91bnQFA25pbAUDbmlsAwkAAAIFEmNvbGxhdGVyYWxWYWx1ZUludgUSY29sbGF0ZXJhbFZhbHVlSW52BA9jb2xsYXRlcmFsVmFsdWUEByRtYXRjaDAFEmNvbGxhdGVyYWxWYWx1ZUludgMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAh9jYW4ndCBnZXQgdXNlciBjb2xsYXRlcmFsIHZhbHVlAwkBASEBCQENdHJ5R2V0Qm9vbGVhbgECDHNldHVwX2FjdGl2ZQkAAgECEW1hcmtldCBpcyBzdG9wcGVkAwkAZgIAAAUPY29sbGF0ZXJhbFZhbHVlCQACAQIheW91IGhhdmUgdG8gc3VwcGx5IG1vcmUgdG8gYm9ycm93BA1hc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBA1hc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgMJAGYCBQZhbW91bnQJAGUCBQ1hc3NldFN1cHBsaWVkBQ1hc3NldEJvcnJvd2VkCQACAQIcdGhpcyBhbW91bnQgaXMgbm90IGF2YWlsYWJsZQkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgkAZAIFEXVzZXJBc3NldEJvcnJvd2VkBQZhbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyCQBkAgUNYXNzZXRCb3Jyb3dlZAUGYW1vdW50CQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMIBQFpBmNhbGxlcgULYXNzZXRBbW91bnQJAQ1nZXRBc3NldEJ5dGVzAQUKYXNzZXRJZFN0cgUDbmlsBRFyYXRlc1JlY2FsY1Jlc3VsdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQVyZXBheQADCQEBIQEJAQ10cnlHZXRCb29sZWFuAQIMc2V0dXBfYWN0aXZlCQACAQIRbWFya2V0IGlzIHN0b3BwZWQDAwkBAiE9AgkAkAMBCAUBaQhwYXltZW50cwABBgkAAAIICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50AAAJAAIBAhwxIHBheW1lbnQgaGFzIHRvIGJlIGF0dGFjaGVkBAphc3NldElkU3RyCQEOZ2V0QXNzZXRTdHJpbmcBCAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQEC2Fzc2V0QW1vdW50CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAQLJHQwOTM5MTk0NTgJAQ1nZXRBY3R1YWxSYXRlAgUKYXNzZXRJZFN0cgIFYlJhdGUEBWJSYXRlCAULJHQwOTM5MTk0NTgCXzEEEXJhdGVzUmVjYWxjUmVzdWx0CAULJHQwOTM5MTk0NTgCXzIEBmFtb3VudAkAawMFC2Fzc2V0QW1vdW50BQdTY2FsZTE2BQViUmF0ZQQHYWRkcmVzcwkApQgBCAUBaQZjYWxsZXIEDWFzc2V0U3VwcGxpZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIEDWFzc2V0Qm9ycm93ZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIEEXVzZXJBc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyBAphbW91bnRMZWZ0CQBlAgURdXNlckFzc2V0Qm9ycm93ZWQFBmFtb3VudAQLcmVwYXlBbW91bnQDCQBnAgUKYW1vdW50TGVmdAAABQZhbW91bnQFEXVzZXJBc3NldEJvcnJvd2VkAwkAAAIJALMJAgkBDHRyeUdldFN0cmluZwECDHNldHVwX3Rva2VucwUKYXNzZXRJZFN0cgUEdW5pdAkAAgECKXRoaXMgYXNzZXQgaXMgbm90IHN1cHBvcnRlZCBieSB0aGUgbWFya2V0CQDOCAIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIJAGUCBRF1c2VyQXNzZXRCb3Jyb3dlZAULcmVwYXlBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyCQBlAgUNYXNzZXRCb3Jyb3dlZAULcmVwYXlBbW91bnQFA25pbAURcmF0ZXNSZWNhbGNSZXN1bHQDCQBnAgUKYW1vdW50TGVmdAAABQNuaWwJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCQEBLQEFCmFtb3VudExlZnQICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAUDbmlsAWkBCWxpcXVpZGF0ZQYFZGVidWcHYWRkcmVzcwthc3NldEFtb3VudAtzQXNzZXRJZFN0cgtiQXNzZXRJZFN0cghyb3V0ZVN0cgMJAQIhPQIIBQFpBmNhbGxlcgkBB0FkZHJlc3MBARoBV3eqztF4Vuipnl6rle+/6eFxWHVeMppxgQkAAgECJXRlbXBvcmFyaWx5IGxpc3RlZCBmb3Igd2hpdGVsaXN0IG9ubHkDCQEBIQEJAQ10cnlHZXRCb29sZWFuAQIMc2V0dXBfYWN0aXZlCQACAQIRbWFya2V0IGlzIHN0b3BwZWQEDnVzZXJDb2xsYXRlcmFsCQESY2FsY1VzZXJDb2xsYXRlcmFsAQUHYWRkcmVzcwMJAAACBQ51c2VyQ29sbGF0ZXJhbAUOdXNlckNvbGxhdGVyYWwEDSR0MDEwODIyMTA4OTEJAQ1nZXRBY3R1YWxSYXRlAgULc0Fzc2V0SWRTdHICBXNSYXRlBAVzUmF0ZQgFDSR0MDEwODIyMTA4OTECXzEEEnJhdGVzUmVjYWxjUmVzdWx0MQgFDSR0MDEwODIyMTA4OTECXzIEDSR0MDEwODk2MTA5NjUJAQ1nZXRBY3R1YWxSYXRlAgULYkFzc2V0SWRTdHICBWJSYXRlBAViUmF0ZQgFDSR0MDEwODk2MTA5NjUCXzEEEnJhdGVzUmVjYWxjUmVzdWx0MggFDSR0MDEwODk2MTA5NjUCXzIEDHNBc3NldEFtb3VudAkAawMFC2Fzc2V0QW1vdW50BQdTY2FsZTE2BQVzUmF0ZQQQY3VycmVudFNQb3NpdGlvbgkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwULc0Fzc2V0SWRTdHIEEGN1cnJlbnRCUG9zaXRpb24JAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FC2JBc3NldElkU3RyAwkAZgIFDnVzZXJDb2xsYXRlcmFsAAAJAAIBAhh1c2VyIGNhbid0IGJlIGxpcXVpZGF0ZWQDCQBmAgUMc0Fzc2V0QW1vdW50BRBjdXJyZW50U1Bvc2l0aW9uCQACAQIycG9zaXRpb24gdG8gbGlxdWlkYXRlIGlzIGJpZ2dlciB0aGFuIHVzZXIncyBzdXBwbHkEEWFnZ3JlZ2F0b3JBZGRyZXNzCQEHQWRkcmVzcwEBGgFXnQyqxhNRqW7LgPdjfcFkeOLvck2oDLrTBA5iYWxhbmNlMEJlZm9yZQkBCmdldEJhbGFuY2UBBQtzQXNzZXRJZFN0cgMJAAACBQ5iYWxhbmNlMEJlZm9yZQUOYmFsYW5jZTBCZWZvcmUEDmJhbGFuY2UxQmVmb3JlCQEKZ2V0QmFsYW5jZQEFC2JBc3NldElkU3RyAwkAAAIFDmJhbGFuY2UxQmVmb3JlBQ5iYWxhbmNlMUJlZm9yZQQOZXhjaGFuZ2VJbnZva2UJAPwHBAURYWdncmVnYXRvckFkZHJlc3MCBHN3YXAJAMwIAgUIcm91dGVTdHIJAMwIAgAABQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQ1nZXRBc3NldEJ5dGVzAQULc0Fzc2V0SWRTdHIFC2Fzc2V0QW1vdW50BQNuaWwDCQAAAgUOZXhjaGFuZ2VJbnZva2UFDmV4Y2hhbmdlSW52b2tlBAphc3NldDBTb2xkCQBlAgUOYmFsYW5jZTBCZWZvcmUJAQpnZXRCYWxhbmNlAQULc0Fzc2V0SWRTdHIDCQAAAgUKYXNzZXQwU29sZAUKYXNzZXQwU29sZAQMYXNzZXQxQm91Z2h0CQBlAgkBCmdldEJhbGFuY2UBBQtiQXNzZXRJZFN0cgUOYmFsYW5jZTFCZWZvcmUDCQAAAgUMYXNzZXQxQm91Z2h0BQxhc3NldDFCb3VnaHQEC2Fzc2V0MFByaWNlCAkBDWdldFRva2VuUHJpY2UBBQtzQXNzZXRJZFN0cgJfMgQLYXNzZXQwU2NhbGUJAQ5jYWxjQXNzZXRTY2FsZQEFC3NBc3NldElkU3RyBAlhc3NldDBVc2QJAGsDBQphc3NldDBTb2xkBQthc3NldDBQcmljZQULYXNzZXQwU2NhbGUEC2Fzc2V0MVByaWNlCAkBDWdldFRva2VuUHJpY2UBBQtiQXNzZXRJZFN0cgJfMQQLYXNzZXQxU2NhbGUJAQ5jYWxjQXNzZXRTY2FsZQEFC2JBc3NldElkU3RyBAlhc3NldDFVc2QJAGsDBQxhc3NldDFCb3VnaHQFC2Fzc2V0MVByaWNlBQthc3NldDFTY2FsZQQHcGVuYWx0eQkBDXBhcnNlSW50VmFsdWUBCQCRAwIJALUJAgkBDHRyeUdldFN0cmluZwECD3NldHVwX3BlbmFsdGllcwIBLAkBBXZhbHVlAQkAzwgCCQEPZ2V0TWFya2V0QXNzZXRzAAULYkFzc2V0SWRTdHIEEWxpcXVpZGF0aW9uUHJvZml0CQBlAgUJYXNzZXQxVXNkCQBrAwUJYXNzZXQwVXNkCQBlAgUGU2NhbGU4BQdwZW5hbHR5BQZTY2FsZTgEDHNBc3NldENoYW5nZQkAawMFCmFzc2V0MFNvbGQFB1NjYWxlMTYFBXNSYXRlBAxiQXNzZXRDaGFuZ2UJAGsDBQxhc3NldDFCb3VnaHQFB1NjYWxlMTYFBWJSYXRlAwkAZgIFCmFzc2V0MFNvbGQFC2Fzc2V0QW1vdW50CQACAQIjbW9yZSBhc3NldHMgZXhjaGFuZ2VkIHRoYW4gZXhwZWN0ZWQDCQBmAgAABRFsaXF1aWRhdGlvblByb2ZpdAkAAgECL3ByaWNlIGltcGFjdCBpcyBiaWdnZXIgdGhhbiBsaXF1aWRhdGlvbiBwZW5hbHR5CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwULc0Fzc2V0SWRTdHIJAGUCBRBjdXJyZW50U1Bvc2l0aW9uBQxzQXNzZXRDaGFuZ2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgkAZQIFEGN1cnJlbnRCUG9zaXRpb24FDGJBc3NldENoYW5nZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9zdXBwbGllZF8FC3NBc3NldElkU3RyCQBlAgkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX3N1cHBsaWVkXwULc0Fzc2V0SWRTdHIFDHNBc3NldENoYW5nZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9ib3Jyb3dlZF8FC2JBc3NldElkU3RyCQBlAgkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX2JvcnJvd2VkXwULYkFzc2V0SWRTdHIFDGJBc3NldENoYW5nZQkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIFEWxpcXVpZGF0aW9uUHJvZml0CQENZ2V0QXNzZXRCeXRlcwEFC2JBc3NldElkU3RyBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQERZ2V0VXNlckNvbGxhdGVyYWwEBWRlYnVnB2FkZHJlc3MNbWludXNCb3Jyb3dlZAthZnRlckNoYW5nZQQGYXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAQEbHR2cwkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIKc2V0dXBfbHR2cwIBLAQDbHRzCQC1CQIJAQx0cnlHZXRTdHJpbmcBAglzZXR1cF9sdHMCASwEBXJhdGVzCAkBDWdldEFjdHVhbFJhdGUCCQCRAwIFBmFzc2V0cwAAAgVzUmF0ZQJfMgQNY2hhbmdlSGFuZGxlcgkAtQkCBQthZnRlckNoYW5nZQIBLAoBAWYCBWFjY3VtBG5leHQDCQBnAgUEbmV4dAkAkAMBBQZhc3NldHMFBWFjY3VtBAhkZWNpbWFscwMJAAACCQCRAwIFBmFzc2V0cwUEbmV4dAIFV0FWRVMACAgJAQV2YWx1ZQEJAOwHAQkA2QQBCQCRAwIFBmFzc2V0cwUEbmV4dAhkZWNpbWFscwQKYXNzZXRTY2FsZQkAbAYACgAABQhkZWNpbWFscwAAAAAFBERPV04ECmFzc2V0UHJpY2UJAQ1nZXRUb2tlblByaWNlAQkAkQMCBQZhc3NldHMFBG5leHQJAGUCCQBkAgUFYWNjdW0JAGsDCQBrAwkAawMJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfCQCRAwIFBmFzc2V0cwUEbmV4dAMDAwkBAiE9AgULYWZ0ZXJDaGFuZ2UCAAkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgAACQCRAwIFBmFzc2V0cwUEbmV4dAcJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAQIIc3VwcGxpZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgACAAAICQCRAwIFBXJhdGVzCQBoAgUEbmV4dAADBXZhbHVlBQdTY2FsZTE2CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUEbHR2cwUEbmV4dAUGU2NhbGU4CAUKYXNzZXRQcmljZQJfMQUKYXNzZXRTY2FsZQMFDW1pbnVzQm9ycm93ZWQJAGsDCQBrAwkAawMJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfCQCRAwIFBmFzc2V0cwUEbmV4dAMDAwkBAiE9AgULYWZ0ZXJDaGFuZ2UCAAkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgAACQCRAwIFBmFzc2V0cwUEbmV4dAcJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAQIIYm9ycm93ZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgACAAAICQCRAwIFBXJhdGVzCQBkAgkAaAIFBG5leHQAAwABBXZhbHVlBQdTY2FsZTE2BQZTY2FsZTgJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQNsdHMFBG5leHQIBQphc3NldFByaWNlAl8yBQphc3NldFNjYWxlAAAEBnJlc3VsdAoAAiRsCQDMCAIAAAkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFBQNuaWwKAAIkcwkAkAMBBQIkbAoABSRhY2MwAAAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECE0xpc3Qgc2l6ZSBleGNlZWRzIDYJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYDBQVkZWJ1ZwkAAgEJAKQDAQUGcmVzdWx0CQCUCgIFBXJhdGVzBQZyZXN1bHQBaQEJZ2V0UHJpY2VzAQVkZWJ1ZwQGYXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAoBAWYCBWFjY3VtBG5leHQDCQBnAgUEbmV4dAkAkAMBBQZhc3NldHMFBWFjY3VtBAphc3NldFByaWNlCQENZ2V0VG9rZW5QcmljZQEJAJEDAgUGYXNzZXRzBQRuZXh0CQCsAgIJAKwCAgkArAICCQCsAgIFBWFjY3VtCQCkAwEIBQphc3NldFByaWNlAl8xAgEsCQCkAwEIBQphc3NldFByaWNlAl8yAgF8BAZyZXN1bHQKAAIkbAkAzAgCAAAJAMwIAgABCQDMCAIAAgkAzAgCAAMJAMwIAgAECQDMCAIABQUDbmlsCgACJHMJAJADAQUCJGwKAAUkYWNjMAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhNMaXN0IHNpemUgZXhjZWVkcyA2CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAwUFZGVidWcJAAIBBQZyZXN1bHQJAJQKAgUDbmlsBQZyZXN1bHQBaQEZY2FsY3VsYXRlVXRpbGl6YXRpb25SYXRpbwIKYXNzZXRJZFN0cgVkZWJ1ZwMFBWRlYnVnCQACAQkApAMBCQEFZ2V0VXIBBQphc3NldElkU3RyCQCUCgIFA25pbAkBBWdldFVyAQUKYXNzZXRJZFN0cgFpARNjYWxjdWxhdGVPdXRkYXRlZFVSAgphc3NldElkU3RyBWRlYnVnAwUFZGVidWcJAAIBCQCkAwEJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgkAlAoCBQNuaWwJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgFpARNjYWxjdWxhdGVUb2tlblJhdGVzAQVkZWJ1ZwoBAWYCBWFjY3VtCmFzc2V0SWRTdHIEBXJhdGVzCQEQdG9rZW5SYXRlc1JlY2FsYwEFCmFzc2V0SWRTdHIJAJQKAgkArAICCQCsAgIJAKwCAgkArAICCAUFYWNjdW0CXzEJAKQDAQgJAJEDAgUFcmF0ZXMAAQV2YWx1ZQIBfAkApAMBCAkAkQMCBQVyYXRlcwAABXZhbHVlAgEsCQDOCAIIBQVhY2N1bQJfMgUFcmF0ZXMECXBhcmFtZXRlcgoAAiRsCQEPZ2V0TWFya2V0QXNzZXRzAAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAJAJQKAgIABQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECE0xpc3Qgc2l6ZSBleGNlZWRzIDYJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYDBQVkZWJ1ZwkAAgEIBQlwYXJhbWV0ZXICXzEJAJQKAggFCXBhcmFtZXRlcgJfMggFCXBhcmFtZXRlcgJfMQFpARdjYWxjdWxhdGVUb2tlbnNJbnRlcmVzdAEFZGVidWcKAQFmAgVhY2N1bQphc3NldElkU3RyBARyYXRlCQBrAwkBC2dldEludGVyZXN0AQUKYXNzZXRJZFN0cgUJZGF5QmxvY2tzBQZTY2FsZTgJAKwCAgkArAICBQVhY2N1bQkApAMBBQRyYXRlAgEsBAlwYXJhbWV0ZXIKAAIkbAkBD2dldE1hcmtldEFzc2V0cwAKAAIkcwkAkAMBBQIkbAoABSRhY2MwAgAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECE0xpc3Qgc2l6ZSBleGNlZWRzIDYJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYDBQVkZWJ1ZwkAAgEFCXBhcmFtZXRlcgkAlAoCBQNuaWwFCXBhcmFtZXRlcgFpAQhzaHV0ZG93bgEIc2h1dGRvd24ECXdoaXRlbGlzdAkAzAgCARoBV9fhI1GOfKuIWtXXKJd12Ih104+uIwWm4QkAzAgCARoBV2MNA4Gmf1tU0hy55Cgq86eANPDYtY3VZQkAzAgCARoBV7PVklRJCZoqEPvG+J8QdqvvPw9ypYkVjAkAzAgCARoBV/j6+gImwFGL7Go7RjNJ0hxxsHWL+aTKmgkAzAgCARoBV6Hfrr5FnbmGEdOYYnGJhfRbeUsgh/NHawkAzAgCARoBVzA5HNGkZhg7fn/j2N6i/bhnmP9XTREmnQUDbmlsAwkAAAIJAM8IAgUJd2hpdGVsaXN0CAgFAWkGY2FsbGVyBWJ5dGVzBQR1bml0CQACAQIXdXNlciBub3QgaW4gYSB3aGl0ZWxpc3QJAMwIAgkBDEJvb2xlYW5FbnRyeQICDHNldHVwX2FjdGl2ZQkBASEBBQhzaHV0ZG93bgUDbmlsAQJ0eAEGdmVyaWZ5AAkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAgFAnR4D3NlbmRlclB1YmxpY0tleRDqVuc=", "height": 3352221, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: BUwDgXtemQy5xQcb9xa4ge19xacYtyVkYwroY9Sqze3p Next: BFGtB1XMS8qRi7vdpThMPahCH8XtH4nMZN4fyCQGo9Gg Diff:
OldNewDifferences
6464 }
6565
6666
67+func getRateCurve (assetIdStr) = match assetIdStr {
68+ case _ =>
69+ if (("34N9YcEETLWn93qYQ64EsP1x89tSruJU44RrEMSXXEPJ" == $match0))
70+ then $Tuple4(2000000, 25000000, 80000000, 100000000)
71+ else if (("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p" == $match0))
72+ then $Tuple4(2000000, 25000000, 80000000, 100000000)
73+ else if (("Ajso6nTTjptu2UHLx6hfSXVtHFtRBJCkKYd5SAyj7zf5" == $match0))
74+ then $Tuple4(2000000, 40000000, 80000000, 150000000)
75+ else if (("HEB8Qaw9xrWpWs8tHsiATYGBWDBtP2S7kcPALrMu43AS" == $match0))
76+ then $Tuple4(0, 20000000, 80000000, 40000000)
77+ else if (("WAVES" == $match0))
78+ then $Tuple4(2000000, 30000000, 80000000, 50000000)
79+ else $Tuple4(0, 20000000, 80000000, 80000000)
80+}
81+
82+
6783 func getInterest (assetIdStr) = {
6884 let ur = getOutdatedUr(assetIdStr)
69- let dailyInterest = if ((80000000 >= ur))
70- then fraction(ur, 90000, 80000000)
71- else (80000 + fraction((ur - 80000000), 80000, 20000000))
72- max([fraction(dailyInterest, Scale8, dayBlocks), 1])
85+ let curve = getRateCurve(assetIdStr)
86+ let rate = (curve._1 + (if ((curve._3 >= ur))
87+ then fraction(ur, curve._2, curve._3)
88+ else (curve._2 + fraction((ur - curve._3), (curve._4 - curve._2), (100000000 - curve._3)))))
89+ max([fraction(rate, Scale8, (dayBlocks * 365)), 1])
7390 }
7491
7592
211228 else {
212229 let assetIdStr = getAssetString(i.payments[0].assetId)
213230 let assetAmount = i.payments[0].amount
214- let $t050025069 = getActualRate(assetIdStr, "sRate")
215- let sRate = $t050025069._1
216- let ratesRecalcResult = $t050025069._2
231+ let $t057035770 = getActualRate(assetIdStr, "sRate")
232+ let sRate = $t057035770._1
233+ let ratesRecalcResult = $t057035770._2
217234 let amount = fraction(assetAmount, Scale16, sRate, DOWN)
218235 let address = toString(i.caller)
219236 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
225242
226243 @Callable(i)
227244 func withdraw (assetIdStr,assetAmount) = {
228- let $t056405707 = getActualRate(assetIdStr, "sRate")
229- let sRate = $t056405707._1
230- let ratesRecalcResult = $t056405707._2
231- let amount = fraction(assetAmount, Scale16, sRate, CEILING)
245+ let $t063416408 = getActualRate(assetIdStr, "sRate")
246+ let sRate = $t063416408._1
247+ let ratesRecalcResult = $t063416408._2
248+ let amount = fraction(assetAmount, Scale16, sRate, HALFUP)
232249 let address = toString(i.caller)
233250 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
234251 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
261278 @Callable(i)
262279 func borrow (assetIdStr,assetAmount) = {
263280 let address = toString(i.caller)
264- let $t072007267 = getActualRate(assetIdStr, "bRate")
265- let bRate = $t072007267._1
266- let ratesRecalcResult = $t072007267._2
281+ let $t079007967 = getActualRate(assetIdStr, "bRate")
282+ let bRate = $t079007967._1
283+ let ratesRecalcResult = $t079007967._2
267284 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
268285 let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",borrowed,") + toString(amount))], nil)
269286 if ((collateralValueInv == collateralValueInv))
302319 else {
303320 let assetIdStr = getAssetString(i.payments[0].assetId)
304321 let assetAmount = i.payments[0].amount
305- let $t086918758 = getActualRate(assetIdStr, "bRate")
306- let bRate = $t086918758._1
307- let ratesRecalcResult = $t086918758._2
322+ let $t093919458 = getActualRate(assetIdStr, "bRate")
323+ let bRate = $t093919458._1
324+ let ratesRecalcResult = $t093919458._2
308325 let amount = fraction(assetAmount, Scale16, bRate)
309326 let address = toString(i.caller)
310327 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
326343 @Callable(i)
327344 func liquidate (debug,address,assetAmount,sAssetIdStr,bAssetIdStr,routeStr) = if ((i.caller != Address(base58'3PCqdm1mAoQqR46oZotFanmqb5CLUvrKEo2')))
328345 then throw("temporarily listed for whitelist only")
329- else {
330- let userCollateral = calcUserCollateral(address)
331- if ((userCollateral == userCollateral))
332- then {
333- let $t01004510114 = getActualRate(sAssetIdStr, "sRate")
334- let sRate = $t01004510114._1
335- let ratesRecalcResult1 = $t01004510114._2
336- let $t01011910188 = getActualRate(bAssetIdStr, "bRate")
337- let bRate = $t01011910188._1
338- let ratesRecalcResult2 = $t01011910188._2
339- let sAssetAmount = fraction(assetAmount, Scale16, sRate)
340- let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
341- let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
342- if ((userCollateral > 0))
343- then throw("user can't be liquidated")
344- else if ((sAssetAmount > currentSPosition))
345- then throw("position to liquidate is bigger than user's supply")
346- else {
347- let aggregatorAddress = Address(base58'3PGFHzVGT4NTigwCKP1NcwoXkodVZwvBuuU')
348- let balance0Before = getBalance(sAssetIdStr)
349- if ((balance0Before == balance0Before))
350- then {
351- let balance1Before = getBalance(bAssetIdStr)
352- if ((balance1Before == balance1Before))
353- then {
354- let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
355- if ((exchangeInvoke == exchangeInvoke))
356- then {
357- let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
358- if ((asset0Sold == asset0Sold))
359- then {
360- let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
361- if ((asset1Bought == asset1Bought))
362- then {
363- let asset0Price = getTokenPrice(sAssetIdStr)._2
364- let asset0Scale = calcAssetScale(sAssetIdStr)
365- let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
366- let asset1Price = getTokenPrice(bAssetIdStr)._1
367- let asset1Scale = calcAssetScale(bAssetIdStr)
368- let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
369- let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
370- let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
371- let sAssetChange = fraction(asset0Sold, Scale16, sRate)
372- let bAssetChange = fraction(asset1Bought, Scale16, bRate)
373- if ((asset0Sold > assetAmount))
374- then throw("more assets exchanged than expected")
375- else if ((0 > liquidationProfit))
376- then throw("price impact is bigger than liquidation penalty")
377- else [IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAssetChange)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - bAssetChange)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAssetChange)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - bAssetChange)), ScriptTransfer(i.caller, liquidationProfit, getAssetBytes(bAssetIdStr))]
378- }
379- else throw("Strict value is not equal to itself.")
380- }
381- else throw("Strict value is not equal to itself.")
382- }
383- else throw("Strict value is not equal to itself.")
384- }
385- else throw("Strict value is not equal to itself.")
386- }
387- else throw("Strict value is not equal to itself.")
388- }
389- }
390- else throw("Strict value is not equal to itself.")
391- }
346+ else if (!(tryGetBoolean("setup_active")))
347+ then throw("market is stopped")
348+ else {
349+ let userCollateral = calcUserCollateral(address)
350+ if ((userCollateral == userCollateral))
351+ then {
352+ let $t01082210891 = getActualRate(sAssetIdStr, "sRate")
353+ let sRate = $t01082210891._1
354+ let ratesRecalcResult1 = $t01082210891._2
355+ let $t01089610965 = getActualRate(bAssetIdStr, "bRate")
356+ let bRate = $t01089610965._1
357+ let ratesRecalcResult2 = $t01089610965._2
358+ let sAssetAmount = fraction(assetAmount, Scale16, sRate)
359+ let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
360+ let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
361+ if ((userCollateral > 0))
362+ then throw("user can't be liquidated")
363+ else if ((sAssetAmount > currentSPosition))
364+ then throw("position to liquidate is bigger than user's supply")
365+ else {
366+ let aggregatorAddress = Address(base58'3PGFHzVGT4NTigwCKP1NcwoXkodVZwvBuuU')
367+ let balance0Before = getBalance(sAssetIdStr)
368+ if ((balance0Before == balance0Before))
369+ then {
370+ let balance1Before = getBalance(bAssetIdStr)
371+ if ((balance1Before == balance1Before))
372+ then {
373+ let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
374+ if ((exchangeInvoke == exchangeInvoke))
375+ then {
376+ let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
377+ if ((asset0Sold == asset0Sold))
378+ then {
379+ let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
380+ if ((asset1Bought == asset1Bought))
381+ then {
382+ let asset0Price = getTokenPrice(sAssetIdStr)._2
383+ let asset0Scale = calcAssetScale(sAssetIdStr)
384+ let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
385+ let asset1Price = getTokenPrice(bAssetIdStr)._1
386+ let asset1Scale = calcAssetScale(bAssetIdStr)
387+ let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
388+ let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
389+ let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
390+ let sAssetChange = fraction(asset0Sold, Scale16, sRate)
391+ let bAssetChange = fraction(asset1Bought, Scale16, bRate)
392+ if ((asset0Sold > assetAmount))
393+ then throw("more assets exchanged than expected")
394+ else if ((0 > liquidationProfit))
395+ then throw("price impact is bigger than liquidation penalty")
396+ else [IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAssetChange)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - bAssetChange)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAssetChange)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - bAssetChange)), ScriptTransfer(i.caller, liquidationProfit, getAssetBytes(bAssetIdStr))]
397+ }
398+ else throw("Strict value is not equal to itself.")
399+ }
400+ else throw("Strict value is not equal to itself.")
401+ }
402+ else throw("Strict value is not equal to itself.")
403+ }
404+ else throw("Strict value is not equal to itself.")
405+ }
406+ else throw("Strict value is not equal to itself.")
407+ }
408+ }
409+ else throw("Strict value is not equal to itself.")
410+ }
392411
393412
394413
545564 }
546565
547566
567+
568+@Callable(i)
569+func shutdown (shutdown) = {
570+ let whitelist = [base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL', base58'3PAxdDSmN758L5SHSGRC5apEtQE2aApZotG', base58'3PJKmXoHJvVeQXjSJdhtkUcFDtdiQqMbUTD', base58'3PQdNxynJy5mche2kxMVc5shXWzK8Gstq3o', base58'3PGgoUsQX3a5zGCc4e2nEnDCWAkzJ1jASzv', base58'3P6Ksahs71SiKQgQ4qaZuFAVhqncdi2nvJQ']
571+ if ((indexOf(whitelist, i.caller.bytes) == unit))
572+ then throw("user not in a whitelist")
573+ else [BooleanEntry("setup_active", !(shutdown))]
574+ }
575+
576+
548577 @Verifier(tx)
549578 func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
550579
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let Scale8 = 100000000
55
66 let Scale10 = 10000000000
77
88 let Scale16 = (Scale8 * Scale8)
99
1010 let reserveFund = 20
1111
1212 let dayBlocks = 1440
1313
1414 func tryGetInteger (key) = match getInteger(this, key) {
1515 case b: Int =>
1616 b
1717 case _ =>
1818 0
1919 }
2020
2121
2222 func tryGetBoolean (key) = match getBoolean(this, key) {
2323 case b: Boolean =>
2424 b
2525 case _ =>
2626 false
2727 }
2828
2929
3030 func tryGetString (key) = match getString(this, key) {
3131 case b: String =>
3232 b
3333 case _ =>
3434 ""
3535 }
3636
3737
3838 func getAssetString (assetId) = match assetId {
3939 case b: ByteVector =>
4040 toBase58String(b)
4141 case _ =>
4242 "WAVES"
4343 }
4444
4545
4646 func getAssetBytes (assetIdStr) = if ((assetIdStr == "WAVES"))
4747 then unit
4848 else fromBase58String(assetIdStr)
4949
5050
5151 func getBalance (assetIdStr) = if ((assetIdStr == "WAVES"))
5252 then wavesBalance(this).available
5353 else assetBalance(this, fromBase58String(assetIdStr))
5454
5555
5656 func getMarketAssets () = split(tryGetString("setup_tokens"), ",")
5757
5858
5959 func getOutdatedUr (assetIdStr) = {
6060 let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), tryGetInteger((assetIdStr + "_sRate")), Scale16)
6161 if ((down == 0))
6262 then 0
6363 else fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), tryGetInteger((assetIdStr + "_bRate")), Scale16), down)
6464 }
6565
6666
67+func getRateCurve (assetIdStr) = match assetIdStr {
68+ case _ =>
69+ if (("34N9YcEETLWn93qYQ64EsP1x89tSruJU44RrEMSXXEPJ" == $match0))
70+ then $Tuple4(2000000, 25000000, 80000000, 100000000)
71+ else if (("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p" == $match0))
72+ then $Tuple4(2000000, 25000000, 80000000, 100000000)
73+ else if (("Ajso6nTTjptu2UHLx6hfSXVtHFtRBJCkKYd5SAyj7zf5" == $match0))
74+ then $Tuple4(2000000, 40000000, 80000000, 150000000)
75+ else if (("HEB8Qaw9xrWpWs8tHsiATYGBWDBtP2S7kcPALrMu43AS" == $match0))
76+ then $Tuple4(0, 20000000, 80000000, 40000000)
77+ else if (("WAVES" == $match0))
78+ then $Tuple4(2000000, 30000000, 80000000, 50000000)
79+ else $Tuple4(0, 20000000, 80000000, 80000000)
80+}
81+
82+
6783 func getInterest (assetIdStr) = {
6884 let ur = getOutdatedUr(assetIdStr)
69- let dailyInterest = if ((80000000 >= ur))
70- then fraction(ur, 90000, 80000000)
71- else (80000 + fraction((ur - 80000000), 80000, 20000000))
72- max([fraction(dailyInterest, Scale8, dayBlocks), 1])
85+ let curve = getRateCurve(assetIdStr)
86+ let rate = (curve._1 + (if ((curve._3 >= ur))
87+ then fraction(ur, curve._2, curve._3)
88+ else (curve._2 + fraction((ur - curve._3), (curve._4 - curve._2), (100000000 - curve._3)))))
89+ max([fraction(rate, Scale8, (dayBlocks * 365)), 1])
7390 }
7491
7592
7693 func tokenRatesRecalc (assetIdStr) = {
7794 let interest = getInterest(assetIdStr)
7895 let ur = getOutdatedUr(assetIdStr)
7996 let lastRecalcHeight = tryGetInteger("lastRateHeight")
8097 let lastBRate = max([tryGetInteger((assetIdStr + "_bRate")), Scale16])
8198 let newBRate = (lastBRate + ((height - lastRecalcHeight) * interest))
8299 let lastSRate = max([tryGetInteger((assetIdStr + "_sRate")), Scale16])
83100 let newSRate = (lastSRate + ((((height - lastRecalcHeight) * fraction(interest, ur, Scale8)) * (100 - reserveFund)) / 100))
84101 [IntegerEntry((assetIdStr + "_sRate"), newSRate), IntegerEntry((assetIdStr + "_bRate"), newBRate), IntegerEntry("lastRateHeight", height)]
85102 }
86103
87104
88105 func getActualRate (assetIdStr,rateType) = {
89106 func f (accum,token) = {
90107 let recalc = tokenRatesRecalc(token)
91108 $Tuple2(if ((token != assetIdStr))
92109 then accum._1
93110 else if ((rateType == "sRate"))
94111 then recalc[0].value
95112 else recalc[1].value, (accum._2 ++ recalc))
96113 }
97114
98115 let $l = getMarketAssets()
99116 let $s = size($l)
100117 let $acc0 = $Tuple2(0, nil)
101118 func $f0_1 ($a,$i) = if (($i >= $s))
102119 then $a
103120 else f($a, $l[$i])
104121
105122 func $f0_2 ($a,$i) = if (($i >= $s))
106123 then $a
107124 else throw("List size exceeds 6")
108125
109126 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
110127 }
111128
112129
113130 func getUr (assetIdStr) = {
114131 let rates = tokenRatesRecalc(assetIdStr)
115132 let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[0].value, Scale16)
116133 fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[1].value, Scale16), down)
117134 }
118135
119136
120137 func ratesRecalc () = {
121138 func f (accum,token) = (accum ++ tokenRatesRecalc(token))
122139
123140 let $l = getMarketAssets()
124141 let $s = size($l)
125142 let $acc0 = nil
126143 func $f0_1 ($a,$i) = if (($i >= $s))
127144 then $a
128145 else f($a, $l[$i])
129146
130147 func $f0_2 ($a,$i) = if (($i >= $s))
131148 then $a
132149 else throw("List size exceeds 6")
133150
134151 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
135152 }
136153
137154
138155 func getTokenPrice (assetIdStr) = {
139156 let inv = invoke(addressFromStringValue("3P5BTtdj32Sd1Dyh1Mdw33xQAAckSfMfnKf"), "getTWAP60", [assetIdStr, false], nil)
140157 if ((inv == inv))
141158 then match inv {
142159 case x: (Int, Int) =>
143160 x
144161 case _ =>
145162 throw("error of price oracle")
146163 }
147164 else throw("Strict value is not equal to itself.")
148165 }
149166
150167
151168 func calcAssetScale (assetIdStr) = {
152169 let decimals = if ((assetIdStr == "WAVES"))
153170 then 8
154171 else value(assetInfo(fromBase58String(assetIdStr))).decimals
155172 pow(10, 0, decimals, 0, 0, DOWN)
156173 }
157174
158175
159176 func calcUserCollateral (address) = {
160177 let userCollateralInvoke = invoke(this, "getUserCollateral", [false, address, true, ""], nil)
161178 if ((userCollateralInvoke == userCollateralInvoke))
162179 then {
163180 let userCollateralValue = match userCollateralInvoke {
164181 case x: Int =>
165182 x
166183 case _ =>
167184 throw("issue while doing in-dapp invocation")
168185 }
169186 if ((userCollateralValue == userCollateralValue))
170187 then userCollateralValue
171188 else throw("Strict value is not equal to itself.")
172189 }
173190 else throw("Strict value is not equal to itself.")
174191 }
175192
176193
177194 @Callable(i)
178195 func preInit (tokens,ltvs,lts,penalties) = {
179196 func f (accum,token) = (accum ++ [IntegerEntry((token + "_bRate"), Scale16), IntegerEntry((token + "_sRate"), Scale16)])
180197
181198 if ((i.caller != this))
182199 then throw("admin only")
183200 else {
184201 let rates = {
185202 let $l = split(tokens, ",")
186203 let $s = size($l)
187204 let $acc0 = nil
188205 func $f0_1 ($a,$i) = if (($i >= $s))
189206 then $a
190207 else f($a, $l[$i])
191208
192209 func $f0_2 ($a,$i) = if (($i >= $s))
193210 then $a
194211 else throw("List size exceeds 6")
195212
196213 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
197214 }
198215 ([StringEntry("setup_tokens", tokens), StringEntry("setup_ltvs", ltvs), StringEntry("setup_lts", lts), StringEntry("setup_penalties", penalties), BooleanEntry("setup_active", true)] ++ rates)
199216 }
200217 }
201218
202219
203220
204221 @Callable(i)
205222 func supply () = if (!(tryGetBoolean("setup_active")))
206223 then throw("market is stopped")
207224 else if (if ((size(i.payments) != 1))
208225 then true
209226 else (i.payments[0].amount == 0))
210227 then throw("1 payment has to be attached")
211228 else {
212229 let assetIdStr = getAssetString(i.payments[0].assetId)
213230 let assetAmount = i.payments[0].amount
214- let $t050025069 = getActualRate(assetIdStr, "sRate")
215- let sRate = $t050025069._1
216- let ratesRecalcResult = $t050025069._2
231+ let $t057035770 = getActualRate(assetIdStr, "sRate")
232+ let sRate = $t057035770._1
233+ let ratesRecalcResult = $t057035770._2
217234 let amount = fraction(assetAmount, Scale16, sRate, DOWN)
218235 let address = toString(i.caller)
219236 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
220237 then throw("this asset is not supported by the market")
221238 else ([IntegerEntry(((address + "_supplied_") + assetIdStr), (tryGetInteger(((address + "_supplied_") + assetIdStr)) + amount)), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) + amount))] ++ ratesRecalcResult)
222239 }
223240
224241
225242
226243 @Callable(i)
227244 func withdraw (assetIdStr,assetAmount) = {
228- let $t056405707 = getActualRate(assetIdStr, "sRate")
229- let sRate = $t056405707._1
230- let ratesRecalcResult = $t056405707._2
231- let amount = fraction(assetAmount, Scale16, sRate, CEILING)
245+ let $t063416408 = getActualRate(assetIdStr, "sRate")
246+ let sRate = $t063416408._1
247+ let ratesRecalcResult = $t063416408._2
248+ let amount = fraction(assetAmount, Scale16, sRate, HALFUP)
232249 let address = toString(i.caller)
233250 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
234251 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
235252 let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr))
236253 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
237254 let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
238255 if ((collateralValueInv == collateralValueInv))
239256 then {
240257 let collateralValue = match collateralValueInv {
241258 case x: Int =>
242259 x
243260 case _ =>
244261 throw("can't get user collateral value")
245262 }
246263 if (!(tryGetBoolean("setup_active")))
247264 then throw("market is stopped")
248265 else if ((0 > collateralValue))
249266 then throw("you dont have enough collateral for this operation")
250267 else if ((amount > (assetSupplied - assetBorrowed)))
251268 then throw("this amount is not available on the market")
252269 else if ((amount > (userAssetSupplied - userAssetBorrowed)))
253270 then throw("this amount is not available for this user")
254271 else ([IntegerEntry(((address + "_supplied_") + assetIdStr), (tryGetInteger(((address + "_supplied_") + assetIdStr)) - amount)), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) - amount)), ScriptTransfer(i.caller, assetAmount, getAssetBytes(assetIdStr))] ++ ratesRecalcResult)
255272 }
256273 else throw("Strict value is not equal to itself.")
257274 }
258275
259276
260277
261278 @Callable(i)
262279 func borrow (assetIdStr,assetAmount) = {
263280 let address = toString(i.caller)
264- let $t072007267 = getActualRate(assetIdStr, "bRate")
265- let bRate = $t072007267._1
266- let ratesRecalcResult = $t072007267._2
281+ let $t079007967 = getActualRate(assetIdStr, "bRate")
282+ let bRate = $t079007967._1
283+ let ratesRecalcResult = $t079007967._2
267284 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
268285 let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",borrowed,") + toString(amount))], nil)
269286 if ((collateralValueInv == collateralValueInv))
270287 then {
271288 let collateralValue = match collateralValueInv {
272289 case x: Int =>
273290 x
274291 case _ =>
275292 throw("can't get user collateral value")
276293 }
277294 if (!(tryGetBoolean("setup_active")))
278295 then throw("market is stopped")
279296 else if ((0 > collateralValue))
280297 then throw("you have to supply more to borrow")
281298 else {
282299 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
283300 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
284301 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
285302 if ((amount > (assetSupplied - assetBorrowed)))
286303 then throw("this amount is not available")
287304 else ([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed + amount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed + amount)), ScriptTransfer(i.caller, assetAmount, getAssetBytes(assetIdStr))] ++ ratesRecalcResult)
288305 }
289306 }
290307 else throw("Strict value is not equal to itself.")
291308 }
292309
293310
294311
295312 @Callable(i)
296313 func repay () = if (!(tryGetBoolean("setup_active")))
297314 then throw("market is stopped")
298315 else if (if ((size(i.payments) != 1))
299316 then true
300317 else (i.payments[0].amount == 0))
301318 then throw("1 payment has to be attached")
302319 else {
303320 let assetIdStr = getAssetString(i.payments[0].assetId)
304321 let assetAmount = i.payments[0].amount
305- let $t086918758 = getActualRate(assetIdStr, "bRate")
306- let bRate = $t086918758._1
307- let ratesRecalcResult = $t086918758._2
322+ let $t093919458 = getActualRate(assetIdStr, "bRate")
323+ let bRate = $t093919458._1
324+ let ratesRecalcResult = $t093919458._2
308325 let amount = fraction(assetAmount, Scale16, bRate)
309326 let address = toString(i.caller)
310327 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
311328 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
312329 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
313330 let amountLeft = (userAssetBorrowed - amount)
314331 let repayAmount = if ((amountLeft >= 0))
315332 then amount
316333 else userAssetBorrowed
317334 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
318335 then throw("this asset is not supported by the market")
319336 else (([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ (if ((amountLeft >= 0))
320337 then nil
321338 else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]))
322339 }
323340
324341
325342
326343 @Callable(i)
327344 func liquidate (debug,address,assetAmount,sAssetIdStr,bAssetIdStr,routeStr) = if ((i.caller != Address(base58'3PCqdm1mAoQqR46oZotFanmqb5CLUvrKEo2')))
328345 then throw("temporarily listed for whitelist only")
329- else {
330- let userCollateral = calcUserCollateral(address)
331- if ((userCollateral == userCollateral))
332- then {
333- let $t01004510114 = getActualRate(sAssetIdStr, "sRate")
334- let sRate = $t01004510114._1
335- let ratesRecalcResult1 = $t01004510114._2
336- let $t01011910188 = getActualRate(bAssetIdStr, "bRate")
337- let bRate = $t01011910188._1
338- let ratesRecalcResult2 = $t01011910188._2
339- let sAssetAmount = fraction(assetAmount, Scale16, sRate)
340- let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
341- let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
342- if ((userCollateral > 0))
343- then throw("user can't be liquidated")
344- else if ((sAssetAmount > currentSPosition))
345- then throw("position to liquidate is bigger than user's supply")
346- else {
347- let aggregatorAddress = Address(base58'3PGFHzVGT4NTigwCKP1NcwoXkodVZwvBuuU')
348- let balance0Before = getBalance(sAssetIdStr)
349- if ((balance0Before == balance0Before))
350- then {
351- let balance1Before = getBalance(bAssetIdStr)
352- if ((balance1Before == balance1Before))
353- then {
354- let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
355- if ((exchangeInvoke == exchangeInvoke))
356- then {
357- let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
358- if ((asset0Sold == asset0Sold))
359- then {
360- let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
361- if ((asset1Bought == asset1Bought))
362- then {
363- let asset0Price = getTokenPrice(sAssetIdStr)._2
364- let asset0Scale = calcAssetScale(sAssetIdStr)
365- let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
366- let asset1Price = getTokenPrice(bAssetIdStr)._1
367- let asset1Scale = calcAssetScale(bAssetIdStr)
368- let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
369- let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
370- let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
371- let sAssetChange = fraction(asset0Sold, Scale16, sRate)
372- let bAssetChange = fraction(asset1Bought, Scale16, bRate)
373- if ((asset0Sold > assetAmount))
374- then throw("more assets exchanged than expected")
375- else if ((0 > liquidationProfit))
376- then throw("price impact is bigger than liquidation penalty")
377- else [IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAssetChange)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - bAssetChange)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAssetChange)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - bAssetChange)), ScriptTransfer(i.caller, liquidationProfit, getAssetBytes(bAssetIdStr))]
378- }
379- else throw("Strict value is not equal to itself.")
380- }
381- else throw("Strict value is not equal to itself.")
382- }
383- else throw("Strict value is not equal to itself.")
384- }
385- else throw("Strict value is not equal to itself.")
386- }
387- else throw("Strict value is not equal to itself.")
388- }
389- }
390- else throw("Strict value is not equal to itself.")
391- }
346+ else if (!(tryGetBoolean("setup_active")))
347+ then throw("market is stopped")
348+ else {
349+ let userCollateral = calcUserCollateral(address)
350+ if ((userCollateral == userCollateral))
351+ then {
352+ let $t01082210891 = getActualRate(sAssetIdStr, "sRate")
353+ let sRate = $t01082210891._1
354+ let ratesRecalcResult1 = $t01082210891._2
355+ let $t01089610965 = getActualRate(bAssetIdStr, "bRate")
356+ let bRate = $t01089610965._1
357+ let ratesRecalcResult2 = $t01089610965._2
358+ let sAssetAmount = fraction(assetAmount, Scale16, sRate)
359+ let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
360+ let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
361+ if ((userCollateral > 0))
362+ then throw("user can't be liquidated")
363+ else if ((sAssetAmount > currentSPosition))
364+ then throw("position to liquidate is bigger than user's supply")
365+ else {
366+ let aggregatorAddress = Address(base58'3PGFHzVGT4NTigwCKP1NcwoXkodVZwvBuuU')
367+ let balance0Before = getBalance(sAssetIdStr)
368+ if ((balance0Before == balance0Before))
369+ then {
370+ let balance1Before = getBalance(bAssetIdStr)
371+ if ((balance1Before == balance1Before))
372+ then {
373+ let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
374+ if ((exchangeInvoke == exchangeInvoke))
375+ then {
376+ let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
377+ if ((asset0Sold == asset0Sold))
378+ then {
379+ let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
380+ if ((asset1Bought == asset1Bought))
381+ then {
382+ let asset0Price = getTokenPrice(sAssetIdStr)._2
383+ let asset0Scale = calcAssetScale(sAssetIdStr)
384+ let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
385+ let asset1Price = getTokenPrice(bAssetIdStr)._1
386+ let asset1Scale = calcAssetScale(bAssetIdStr)
387+ let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
388+ let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
389+ let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
390+ let sAssetChange = fraction(asset0Sold, Scale16, sRate)
391+ let bAssetChange = fraction(asset1Bought, Scale16, bRate)
392+ if ((asset0Sold > assetAmount))
393+ then throw("more assets exchanged than expected")
394+ else if ((0 > liquidationProfit))
395+ then throw("price impact is bigger than liquidation penalty")
396+ else [IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAssetChange)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - bAssetChange)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAssetChange)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - bAssetChange)), ScriptTransfer(i.caller, liquidationProfit, getAssetBytes(bAssetIdStr))]
397+ }
398+ else throw("Strict value is not equal to itself.")
399+ }
400+ else throw("Strict value is not equal to itself.")
401+ }
402+ else throw("Strict value is not equal to itself.")
403+ }
404+ else throw("Strict value is not equal to itself.")
405+ }
406+ else throw("Strict value is not equal to itself.")
407+ }
408+ }
409+ else throw("Strict value is not equal to itself.")
410+ }
392411
393412
394413
395414 @Callable(i)
396415 func getUserCollateral (debug,address,minusBorrowed,afterChange) = {
397416 let assets = getMarketAssets()
398417 let ltvs = split(tryGetString("setup_ltvs"), ",")
399418 let lts = split(tryGetString("setup_lts"), ",")
400419 let rates = getActualRate(assets[0], "sRate")._2
401420 let changeHandler = split(afterChange, ",")
402421 func f (accum,next) = if ((next >= size(assets)))
403422 then accum
404423 else {
405424 let decimals = if ((assets[next] == "WAVES"))
406425 then 8
407426 else value(assetInfo(fromBase58String(assets[next]))).decimals
408427 let assetScale = pow(10, 0, decimals, 0, 0, DOWN)
409428 let assetPrice = getTokenPrice(assets[next])
410429 ((accum + fraction(fraction(fraction((tryGetInteger(((address + "_supplied_") + assets[next])) + (if (if (if ((afterChange != ""))
411430 then (changeHandler[0] == assets[next])
412431 else false)
413432 then (changeHandler[1] == "supplied")
414433 else false)
415434 then parseIntValue(changeHandler[2])
416435 else 0)), rates[(next * 3)].value, Scale16), parseIntValue(ltvs[next]), Scale8), assetPrice._1, assetScale)) - (if (minusBorrowed)
417436 then fraction(fraction(fraction((tryGetInteger(((address + "_borrowed_") + assets[next])) + (if (if (if ((afterChange != ""))
418437 then (changeHandler[0] == assets[next])
419438 else false)
420439 then (changeHandler[1] == "borrowed")
421440 else false)
422441 then parseIntValue(changeHandler[2])
423442 else 0)), rates[((next * 3) + 1)].value, Scale16), Scale8, parseIntValue(lts[next])), assetPrice._2, assetScale)
424443 else 0))
425444 }
426445
427446 let result = {
428447 let $l = [0, 1, 2, 3, 4, 5]
429448 let $s = size($l)
430449 let $acc0 = 0
431450 func $f0_1 ($a,$i) = if (($i >= $s))
432451 then $a
433452 else f($a, $l[$i])
434453
435454 func $f0_2 ($a,$i) = if (($i >= $s))
436455 then $a
437456 else throw("List size exceeds 6")
438457
439458 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
440459 }
441460 if (debug)
442461 then throw(toString(result))
443462 else $Tuple2(rates, result)
444463 }
445464
446465
447466
448467 @Callable(i)
449468 func getPrices (debug) = {
450469 let assets = getMarketAssets()
451470 func f (accum,next) = if ((next >= size(assets)))
452471 then accum
453472 else {
454473 let assetPrice = getTokenPrice(assets[next])
455474 ((((accum + toString(assetPrice._1)) + ",") + toString(assetPrice._2)) + "|")
456475 }
457476
458477 let result = {
459478 let $l = [0, 1, 2, 3, 4, 5]
460479 let $s = size($l)
461480 let $acc0 = ""
462481 func $f0_1 ($a,$i) = if (($i >= $s))
463482 then $a
464483 else f($a, $l[$i])
465484
466485 func $f0_2 ($a,$i) = if (($i >= $s))
467486 then $a
468487 else throw("List size exceeds 6")
469488
470489 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
471490 }
472491 if (debug)
473492 then throw(result)
474493 else $Tuple2(nil, result)
475494 }
476495
477496
478497
479498 @Callable(i)
480499 func calculateUtilizationRatio (assetIdStr,debug) = if (debug)
481500 then throw(toString(getUr(assetIdStr)))
482501 else $Tuple2(nil, getUr(assetIdStr))
483502
484503
485504
486505 @Callable(i)
487506 func calculateOutdatedUR (assetIdStr,debug) = if (debug)
488507 then throw(toString(getOutdatedUr(assetIdStr)))
489508 else $Tuple2(nil, getOutdatedUr(assetIdStr))
490509
491510
492511
493512 @Callable(i)
494513 func calculateTokenRates (debug) = {
495514 func f (accum,assetIdStr) = {
496515 let rates = tokenRatesRecalc(assetIdStr)
497516 $Tuple2(((((accum._1 + toString(rates[1].value)) + "|") + toString(rates[0].value)) + ","), (accum._2 ++ rates))
498517 }
499518
500519 let parameter = {
501520 let $l = getMarketAssets()
502521 let $s = size($l)
503522 let $acc0 = $Tuple2("", nil)
504523 func $f0_1 ($a,$i) = if (($i >= $s))
505524 then $a
506525 else f($a, $l[$i])
507526
508527 func $f0_2 ($a,$i) = if (($i >= $s))
509528 then $a
510529 else throw("List size exceeds 6")
511530
512531 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
513532 }
514533 if (debug)
515534 then throw(parameter._1)
516535 else $Tuple2(parameter._2, parameter._1)
517536 }
518537
519538
520539
521540 @Callable(i)
522541 func calculateTokensInterest (debug) = {
523542 func f (accum,assetIdStr) = {
524543 let rate = fraction(getInterest(assetIdStr), dayBlocks, Scale8)
525544 ((accum + toString(rate)) + ",")
526545 }
527546
528547 let parameter = {
529548 let $l = getMarketAssets()
530549 let $s = size($l)
531550 let $acc0 = ""
532551 func $f0_1 ($a,$i) = if (($i >= $s))
533552 then $a
534553 else f($a, $l[$i])
535554
536555 func $f0_2 ($a,$i) = if (($i >= $s))
537556 then $a
538557 else throw("List size exceeds 6")
539558
540559 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6)
541560 }
542561 if (debug)
543562 then throw(parameter)
544563 else $Tuple2(nil, parameter)
545564 }
546565
547566
567+
568+@Callable(i)
569+func shutdown (shutdown) = {
570+ let whitelist = [base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL', base58'3PAxdDSmN758L5SHSGRC5apEtQE2aApZotG', base58'3PJKmXoHJvVeQXjSJdhtkUcFDtdiQqMbUTD', base58'3PQdNxynJy5mche2kxMVc5shXWzK8Gstq3o', base58'3PGgoUsQX3a5zGCc4e2nEnDCWAkzJ1jASzv', base58'3P6Ksahs71SiKQgQ4qaZuFAVhqncdi2nvJQ']
571+ if ((indexOf(whitelist, i.caller.bytes) == unit))
572+ then throw("user not in a whitelist")
573+ else [BooleanEntry("setup_active", !(shutdown))]
574+ }
575+
576+
548577 @Verifier(tx)
549578 func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
550579

github/deemru/w8io/6500d08 
89.19 ms