tx · Gpw77SAtCHVib1Qp9Gjpg3vjGqcW8KRpNEy1xeELeSSq

3P69uU8Hdxb77iYBjfzZgDDdPNhXtZaZxSf:  -0.05000000 Waves

2023.08.29 12:23 [3797435] smart account 3P69uU8Hdxb77iYBjfzZgDDdPNhXtZaZxSf > SELF 0.00000000 Waves

{ "type": 13, "id": "Gpw77SAtCHVib1Qp9Gjpg3vjGqcW8KRpNEy1xeELeSSq", "fee": 5000000, "feeAssetId": null, "timestamp": 1693300202736, "version": 2, "chainId": 87, "sender": "3P69uU8Hdxb77iYBjfzZgDDdPNhXtZaZxSf", "senderPublicKey": "6MTghGyV6jjNJY5WHhQcWHMhByuzn5iuNb1hjNQ5qqqb", "proofs": [ "5UFsyeZg5XTaNy2VcYLWELBRhPnM4r54nrfKSW75UK1dStCZyq6gdDy5xzfjD1Mjnc7YNZDze8qEs21oxb9TsqDV", "5x7cwo13RBaD17SARXNNstcMDg2rxH1f1bnbXBYMH6Ag7ubUvAspc9e447AUjsZ66w6AJrZnJ9mdVQpBBAi9iUo4" ], "script": "base64:BgJ/CAISCQoHCAgIAQgICBIAEgQKAggBEgASAwoBCBIAEgYKBAgICAgSBgoECAgICBIECgIICBIECgIICBIDCgEEEgMKAQQSCAoGBAgBCAgIEgUKAwQICBIGCgQECAQIEgUKAwQICBIDCgEEEgQKAggEEgQKAggEEgMKAQQSAwoBBCQAC2F4bHlBZGRyZXNzCQEHQWRkcmVzcwEBGgFXz8h13gt2/Imb93uvCUhEZwkqUEc/D0ItAAdheGx5TFBzCQDMCAICLDdLWmJKclZvcHdKaGtkd2JlMWVGREJiZXg0ZGtZNjNNeGpUTmpxWHRyemoxCQDMCAICLEJ0dzNHMWo0d1FnZHA0OVBUeGFGa052bjc1ZFF0cUdETTdlalFwcEhuV0MxCQDMCAICLEJpU3pGZThuU0w3OG9aYWViZm9pbjV2Qlo1UHplNmQ3a2FlaWpMcXI1eFplCQDMCAICLEYyQUtrQTUxM2s1eUhFSmtMc1U2dld4Q1lZazgxMUdwakxod0V2MldHd1o5CQDMCAICLDRDUTVDUEdMWExiV0JVczJKQmpLVWFScUY0OUNtS0hrd3p2UGdTdlFwQVFWBQNuaWwAEmF4bHlGYXJtaW5nQWRkcmVzcwkBB0FkZHJlc3MBARoBVw0BWZY1FHCZ5lzdPjsnXRNYpbnNNV3lwgAScmVzZXJ2ZUZ1bmRBZGRyZXNzCQEHQWRkcmVzcwEBGgFXHuH7QDFrgebsS1YbBSxRoZNu3wmxPNLBAAtyZXNlcnZlRnVuZAAUABFhZ2dyZWdhdG9yQWRkcmVzcwkBB0FkZHJlc3MBARoBV50MqsYTUaluy4D3Y33BZHji73JNqAy60wARc2h1dGRvd25XaGl0ZWxpc3QJAMwIAgEaAVfX4SNRjnyriFrV1yiXddiIddOPriMFpuEJAMwIAgEaAVdjDQOBpn9bVNIcueQoKvOngDTw2LWN1WUJAMwIAgEaAVez1ZJUSQmaKhD7xvifEHar7z8PcqWJFYwJAMwIAgEaAVf4+voCJsBRi+xqO0YzSdIccbB1i/mkypoJAMwIAgEaAVer3aoi85i5H0XYLNSn6vkSYq+JNmpQO3IJAMwIAgEaAVcwORzRpGYYO35/49jeov24Z5j/V00RJp0FA25pbAEWdmVyaWZ5TGlxdWlkYXRvclJpZ2h0cwEHYWRkcmVzcwkBASEBAwMJAQIhPQIFB2FkZHJlc3MJAQdBZGRyZXNzAQEaAVd3qs7ReFboqZ5eq5Xvv+nhcVh1XjKacYEJAQIhPQIFB2FkZHJlc3MJAQdBZGRyZXNzAQEaAVfX4SNRjnyriFrV1yiXddiIddOPriMFpuEHCQECIT0CBQdhZGRyZXNzCQEHQWRkcmVzcwEBGgFXq92qIvOYuR9F2CzUp+r5EmKviTZqUDtyBwEUdmVyaWZ5U2VudGluZWxSaWdodHMBB2FkZHJlc3MJAQEhAQMJAQIhPQIFB2FkZHJlc3MFEnJlc2VydmVGdW5kQWRkcmVzcwkBAiE9AgUHYWRkcmVzcwULYXhseUFkZHJlc3MHAQxnZXRSYXRlQ3VydmUBCmFzc2V0SWRTdHIEByRtYXRjaDAFCmFzc2V0SWRTdHIDCQAAAgIsOXdjM0xYTkE0VEVCc1h5S3RvTEU5bXJiREQ3V01IWHZYckNqWnZhYkxBc2kFByRtYXRjaDAJAJYKBACAiXoAwPD1CwCA6JImAIDC1y8DCQAAAgIsSEdnYWJUcVVTOFd0VkZVSnpmbXJURE1nRWNjSnVaTEJQaEZnUUZ4dm5zb1cFByRtYXRjaDAJAJYKBACAiXoAwPD1CwCA6JImAIDC1y8DCQAAAgIsMzROOVljRUVUTFduOTNxWVE2NEVzUDF4ODl0U3J1SlU0NFJyRU1TWFhFUEoFByRtYXRjaDAJAJYKBACAiXoAwPD1CwCA6JImAIDC1y8DCQAAAgIsNlh0SGpwWGJzOVJSSlAyU3I5R1V5VnF6QUNjYnk5VGtUaEhYbmpWQzVDREoFByRtYXRjaDAJAJYKBACAiXoAwPD1CwCA6JImAIDC1y8DCQAAAgIsREcyeEZrUGREd0tVb0JrekdBaFF0THBTR3pmWExpQ1lQRXplS0gyQWQyNHAFByRtYXRjaDAJAJYKBACAiXoAwPD1CwCA6JImAIDC1y8DCQAAAgIsQWpzbzZuVFRqcHR1MlVITHg2aGZTWFZ0SEZ0UkJKQ2tLWWQ1U0F5ajd6ZjUFByRtYXRjaDAJAJYKBACAiXoAgLSJEwCA6JImAICjw0cDCQAAAgIsSEVCOFFhdzl4cldwV3M4dEhzaUFUWUdCV0RCdFAyUzdrY1BBTHJNdTQzQVMFByRtYXRjaDAJAJYKBAAAAIDaxAkAgOiSJgCAtIkTAwkAAAICBVdBVkVTBQckbWF0Y2gwCQCWCgQAgIl6AICHpw4AgOiSJgCA4esXAwkAAAICLEF0cXY1OUVZempGR3VpdEtWbk1SazZIOEZ1a2pvVjNrdFBvcmJFeXMyNW9uBQckbWF0Y2gwCQCWCgQAAACA2sQJAIDokiYAgLSJEwMJAAACAixEU2JiaExzU1RlRGc1THNpdWZrMkFuZWgzRGpWcUp1UHIyTTl1VTFnd3k1cAUHJG1hdGNoMAkAlgoEAAAAgNrECQCA6JImAIDC1y8DCQAAAgIsOHQ0RFBXVHdQenBhdEhBOUFrVHhXQUI0N1RIbll6QnNEbm9ZN2ZRcWJHOTEFByRtYXRjaDAJAJYKBAAAAICHpw4AgOiSJgCAtIkTAwkAAAICLEF0OEQ2TkZGcGhlQ2J2S1ZualZvZUxMODRFbzhOWm42b3ZNYW54ZkxhRldMBQckbWF0Y2gwCQCWCgQAAADA8PULAIDokiYAgLSJEwMJAAACAiw4TFFXOGY3UDVkNVBaTTdHdFpFQmdhcVJQR1N6UzNEZlB1aVhyVVJKNEFKUwUHJG1hdGNoMAkAlgoEAICJegCAh6cOAIDokiYAgOHrFwMJAAACAiw0NzRqVGVZeDJyMlZhMzU3OTR0Q1NjQVhXSkc5aFUySGNneHpNb3dhWlVudQUHJG1hdGNoMAkAlgoEAICJegCAh6cOAIDokiYAgOHrFwMJAAACAiw1VVlCUHBxNFdvVTVuNE13cEZrZ0puVzNGcTRCMXUzdWtwSzMzaWs0UWVyUgUHJG1hdGNoMAkAlgoEAICJegCAh6cOAIDokiYAgOHrFwMJAAACAiwydGhzQUN1SG16RE11TmV6UE0zMndnOWEzQndVekJXRGVTS2FrZ3ozY3cyMQUHJG1hdGNoMAkAlgoEAICJegCAtIkTAIDokiYAgMLXLwMJAAACAitZaU5ib2ZGekMxN2pFSEhDTXdyUmNweTlNcnJqYWJNTUxaeGc4ZzV4bWY3BQckbWF0Y2gwCQCWCgQAgIl6AICHpw4AgOiSJgCA6JImCQCWCgQAAACA2sQJAIDokiYAgOiSJgAGU2NhbGU4AIDC1y8AB1NjYWxlMTAAgMivoCUAB1NjYWxlMTYJAGgCBQZTY2FsZTgFBlNjYWxlOAAJZGF5QmxvY2tzAKALAQpsaUludFRvU3RyAQJsaQoBAWYCBWFjY3VtBG5leHQJAKwCAgkArAICBQVhY2N1bQkApAMBBQRuZXh0AgEsCgACJGwFAmxpCgACJHMJAJADAQUCJGwKAAUkYWNjMAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAENdHJ5R2V0SW50ZWdlcgEDa2V5BAckbWF0Y2gwCQCaCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgNJbnQEAWIFByRtYXRjaDAFAWIAAAENdHJ5R2V0Qm9vbGVhbgEDa2V5BAckbWF0Y2gwCQCbCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgdCb29sZWFuBAFiBQckbWF0Y2gwBQFiBwEMdHJ5R2V0U3RyaW5nAQNrZXkEByRtYXRjaDAJAJ0IAgUEdGhpcwUDa2V5AwkAAQIFByRtYXRjaDACBlN0cmluZwQBYgUHJG1hdGNoMAUBYgIAAQx0cnlHZXRCaW5hcnkBA2tleQQHJG1hdGNoMAkAnAgCBQR0aGlzBQNrZXkDCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQBYgUHJG1hdGNoMAUBYgEAAQ5nZXRBc3NldFN0cmluZwEHYXNzZXRJZAQHJG1hdGNoMAUHYXNzZXRJZAMJAAECBQckbWF0Y2gwAgpCeXRlVmVjdG9yBAFiBQckbWF0Y2gwCQDYBAEFAWICBVdBVkVTAQ1nZXRBc3NldEJ5dGVzAQphc3NldElkU3RyAwkAAAIFCmFzc2V0SWRTdHICBVdBVkVTBQR1bml0CQDZBAEFCmFzc2V0SWRTdHIBCmdldEJhbGFuY2UBCmFzc2V0SWRTdHIDCQAAAgUKYXNzZXRJZFN0cgIFV0FWRVMICQDvBwEFBHRoaXMJYXZhaWxhYmxlCQDwBwIFBHRoaXMJANkEAQUKYXNzZXRJZFN0cgEPZ2V0TWFya2V0QXNzZXRzAAkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIMc2V0dXBfdG9rZW5zAgEsARJnZXRBc3NldHNNYXhTdXBwbHkABAFzCQEMdHJ5R2V0U3RyaW5nAQIPc2V0dXBfbWF4c3VwcGx5AwkAAAIFAXMCAAkAzAgCAP///////////wEJAMwIAgD///////////8BCQDMCAIA////////////AQkAzAgCAP///////////wEJAMwIAgD///////////8BCQDMCAIA////////////AQkAzAgCAP///////////wEFA25pbAkAtQkCBQFzAgEsAQ1nZXRPdXRkYXRlZFVyAQphc3NldElkU3RyBARkb3duCQBrAwkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkBDXRyeUdldEludGVnZXIBCQCsAgIFCmFzc2V0SWRTdHICBl9zUmF0ZQUHU2NhbGUxNgMJAAACBQRkb3duAAAAAAkAawMFBlNjYWxlOAkAawMJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfYlJhdGUFB1NjYWxlMTYFBGRvd24BC2dldEludGVyZXN0AQphc3NldElkU3RyBAJ1cgkBDWdldE91dGRhdGVkVXIBBQphc3NldElkU3RyBAVjdXJ2ZQkBDGdldFJhdGVDdXJ2ZQEFCmFzc2V0SWRTdHIEBHJhdGUJAGQCCAUFY3VydmUCXzEDCQBnAggFBWN1cnZlAl8zBQJ1cgkAawMFAnVyCAUFY3VydmUCXzIIBQVjdXJ2ZQJfMwkAZAIIBQVjdXJ2ZQJfMgkAawMJAGUCBQJ1cggFBWN1cnZlAl8zCAUFY3VydmUCXzQJAGUCAIDC1y8IBQVjdXJ2ZQJfMwkAlgMBCQDMCAIJAGsDBQRyYXRlBQZTY2FsZTgJAGgCBQlkYXlCbG9ja3MA7QIJAMwIAgABBQNuaWwBEHRva2VuUmF0ZXNSZWNhbGMBCmFzc2V0SWRTdHIDCQECIT0CCQDPCAIFB2F4bHlMUHMFCmFzc2V0SWRTdHIFBHVuaXQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIGX3NSYXRlBQdTY2FsZTE2CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICBl9iUmF0ZQUHU2NhbGUxNgkAzAgCCQEMSW50ZWdlckVudHJ5AgIObGFzdFJhdGVIZWlnaHQFBmhlaWdodAUDbmlsBAhpbnRlcmVzdAkBC2dldEludGVyZXN0AQUKYXNzZXRJZFN0cgQCdXIJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgQQbGFzdFJlY2FsY0hlaWdodAkBDXRyeUdldEludGVnZXIBAg5sYXN0UmF0ZUhlaWdodAQJbGFzdEJSYXRlCQCWAwEJAMwIAgkBDXRyeUdldEludGVnZXIBCQCsAgIFCmFzc2V0SWRTdHICBl9iUmF0ZQkAzAgCBQdTY2FsZTE2BQNuaWwECG5ld0JSYXRlCQBkAgUJbGFzdEJSYXRlCQBoAgkAZQIFBmhlaWdodAUQbGFzdFJlY2FsY0hlaWdodAUIaW50ZXJlc3QECWxhc3RTUmF0ZQkAlgMBCQDMCAIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfc1JhdGUJAMwIAgUHU2NhbGUxNgUDbmlsBAhuZXdTUmF0ZQkAZAIFCWxhc3RTUmF0ZQkAaQIJAGgCCQBoAgkAZQIFBmhlaWdodAUQbGFzdFJlY2FsY0hlaWdodAkAawMFCGludGVyZXN0BQJ1cgUGU2NhbGU4CQBlAgBkBQtyZXNlcnZlRnVuZABkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICBl9zUmF0ZQUIbmV3U1JhdGUJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIGX2JSYXRlBQhuZXdCUmF0ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgIObGFzdFJhdGVIZWlnaHQFBmhlaWdodAUDbmlsAQ1nZXRBY3R1YWxSYXRlAgphc3NldElkU3RyCHJhdGVUeXBlCgEBZgIFYWNjdW0FdG9rZW4EBnJlY2FsYwkBEHRva2VuUmF0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Y2VzIGRvbid0IG1hdGNoOiAJAKQDAQUFcHJpY2UCIyBpcyB0aGUgcHJpY2UsIGJ1dCByaXNrIGlzIHRvbyBoaWdoAQ1nZXRUb2tlblByaWNlAQphc3NldElkU3RyCQEVZ2V0VG9rZW5QcmljZVdpdGhSaXNrAgUKYXNzZXRJZFN0cgABAQ5jYWxjQXNzZXRTY2FsZQEKYXNzZXRJZFN0cgQIZGVjaW1hbHMDCQAAAgUKYXNzZXRJZFN0cgIFV0FWRVMACAgJAQV2YWx1ZQEJAOwHAQkA2QQBBQphc3NldElkU3RyCGRlY2ltYWxzCQBsBgAKAAAFCGRlY2ltYWxzAAAAAAUERE9XTgESY2FsY1VzZXJDb2xsYXRlcmFsAQdhZGRyZXNzBBR1c2VyQ29sbGF0ZXJhbEludm9rZQkA/AcEBQR0aGlzAhFnZXRVc2VyQ29sbGF0ZXJhbAkAzAgCBwkAzAgCBQdhZGRyZXNzCQDMCAIGCQDMCAICAAUDbmlsBQNuaWwDCQAAAgUUdXNlckNvbGxhdGVyYWxJbnZva2UFFHVzZXJDb2xsYXRlcmFsSW52b2tlBBN1c2VyQ29sbGF0ZXJhbFZhbHVlBAckbWF0Y2gwBRR1c2VyQ29sbGF0ZXJhbEludm9rZQMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAiRpc3N1ZSB3aGlsZSBkb2luZyBpbi1kYXBwIGludm9jYXRpb24DCQAAAgUTdXNlckNvbGxhdGVyYWxWYWx1ZQUTdXNlckNvbGxhdGVyYWxWYWx1ZQUTdXNlckNvbGxhdGVyYWxWYWx1ZQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgEOc3VwcGx5SW50ZXJuYWwDCmFzc2V0SWRTdHILYXNzZXRBbW91bnQHYWRkcmVzcwMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAQLJHQwODkxMzg5ODAJAQ1nZXRBY3R1YWxSYXRlAgUKYXNzZXRJZFN0cgIFc1JhdGUEBXNSYXRlCAULJHQwODkxMzg5ODACXzEEEXJhdGVzUmVjYWxjUmVzdWx0CAULJHQwODkxMzg5ODACXzIEBmFtb3VudAkAbgQFC2Fzc2V0QW1vdW50BQdTY2FsZTE2BQVzUmF0ZQUERE9XTgQJbWF4U3VwcGx5BAckbWF0Y2gwCQCiCAEJAKwCAgIQc2V0dXBfbWF4U3VwcGx5XwUKYXNzZXRJZFN0cgMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAXgFByRtYXRjaDAJAQ1wYXJzZUludFZhbHVlAQUBeAAABAphc3NldFByaWNlCQENZ2V0VG9rZW5QcmljZQEFCmFzc2V0SWRTdHIEEG5ld1RvdGFsU3VwcGxpZWQJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyBQZhbW91bnQEBHJhdGUICQENZ2V0QWN0dWFsUmF0ZQIFCmFzc2V0SWRTdHICBXNSYXRlAl8xBAphc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQphc3NldElkU3RyBBNuZXdUb3RhbFN1cHBsaWVkVXNkCQBrAwkAawMFEG5ld1RvdGFsU3VwcGxpZWQFBHJhdGUFB1NjYWxlMTYIBQphc3NldFByaWNlAl8xBQphc3NldFNjYWxlAwkAAAIJALMJAgkBDHRyeUdldFN0cmluZwECDHNldHVwX3Rva2VucwUKYXNzZXRJZFN0cgUEdW5pdAkAAgECKXRoaXMgYXNzZXQgaXMgbm90IHN1cHBvcnRlZCBieSB0aGUgbWFya2V0AwMJAQIhPQIFCW1heFN1cHBseQAACQBmAgUTbmV3VG90YWxTdXBwbGllZFVzZAUJbWF4U3VwcGx5BwkAAgECM21heCB0b3RhbCBzdXBwbHkgZm9yIHRoaXMgdG9rZW4gcmVhY2hlZCBpbiB0aGUgcG9vbAkAlAoCCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyBRBuZXdUb3RhbFN1cHBsaWVkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkAZAIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIFBmFtb3VudAUDbmlsBRFyYXRlc1JlY2FsY1Jlc3VsdAULYXNzZXRBbW91bnQBDmJvcnJvd0ludGVybmFsBAphc3NldElkU3RyC2Fzc2V0QW1vdW50B2FkZHJlc3MLc3VwcGx5TGF0ZXIEDSR0MDEwMjAyMTAyNjkJAQ1nZXRBY3R1YWxSYXRlAgUKYXNzZXRJZFN0cgIFYlJhdGUEBWJSYXRlCAUNJHQwMTAyMDIxMDI2OQJfMQQRcmF0ZXNSZWNhbGNSZXN1bHQIBQ0kdDAxMDIwMjEwMjY5Al8yBAZhbW91bnQJAG4EBQthc3NldEFtb3VudAUHU2NhbGUxNgUFYlJhdGUFB0NFSUxJTkcEEmNvbGxhdGVyYWxWYWx1ZUludgkA/AcEBQR0aGlzAhFnZXRVc2VyQ29sbGF0ZXJhbAkAzAgCBwkAzAgCBQdhZGRyZXNzCQDMCAIGCQDMCAIJAKwCAgkArAICBQphc3NldElkU3RyAgosYm9ycm93ZWQsCQCkAwEFBmFtb3VudAUDbmlsBQNuaWwDCQAAAgUSY29sbGF0ZXJhbFZhbHVlSW52BRJjb2xsYXRlcmFsVmFsdWVJbnYED2NvbGxhdGVyYWxWYWx1ZQQHJG1hdGNoMAUSY29sbGF0ZXJhbFZhbHVlSW52AwkAAQIFByRtYXRjaDACA0ludAQBeAUHJG1hdGNoMAUBeAkAAgECH2Nhbid0IGdldCB1c2VyIGNvbGxhdGVyYWwgdmFsdWUDCQEBIQEJAQ10cnlHZXRCb29sZWFuAQIMc2V0dXBfYWN0aXZlCQACAQIRbWFya2V0IGlzIHN0b3BwZWQDCQECIT0CCQDPCAIFB2F4bHlMUHMFCmFzc2V0SWRTdHIFBHVuaXQJAAIBAh10aGlzIHRva2VuIGNhbm5vdCBiZSBib3Jyb3dlZAMDCQEBIQEFC3N1cHBseUxhdGVyCQBmAgAABQ9jb2xsYXRlcmFsVmFsdWUHCQACAQIheW91IGhhdmUgdG8gc3VwcGx5IG1vcmUgdG8gYm9ycm93BA1hc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBA1hc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgMJAGYCBQZhbW91bnQJAGUCBQ1hc3NldFN1cHBsaWVkBQ1hc3NldEJvcnJvd2VkCQACAQIcdGhpcyBhbW91bnQgaXMgbm90IGF2YWlsYWJsZQkAlAoCCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyCQBkAgURdXNlckFzc2V0Qm9ycm93ZWQFBmFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIJAGQCBQ1hc3NldEJvcnJvd2VkBQZhbW91bnQFA25pbAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgZfYlJhdGUFBWJSYXRlBQNuaWwFC2Fzc2V0QW1vdW50CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuFQFpAQ1mbGFzaFBvc2l0aW9uBwR1c2VyC3NBc3NldElkU3RyC2JBc3NldElkU3RyB2JBbW91bnQPY2FsbGJhY2tBZGRyZXNzEGNhbGxiYWNrRnVuY3Rpb24MY2FsbGJhY2tBcmdzAwkBAiE9AggFAWkGY2FsbGVyBQtheGx5QWRkcmVzcwkAAgECIGF2YWlsYWJsZSBvbmx5IGZvciBBeGx5IHByb3RvY29sBAlib3Jyb3dSZXMJAQ5ib3Jyb3dJbnRlcm5hbAQFC2JBc3NldElkU3RyBQdiQW1vdW50BQR1c2VyBgQIYkFzc2V0SWQJAQ1nZXRBc3NldEJ5dGVzAQULYkFzc2V0SWRTdHIECGNhbGxiYWNrCQD9BwQJARFAZXh0ck5hdGl2ZSgxMDYyKQEFD2NhbGxiYWNrQWRkcmVzcwUQY2FsbGJhY2tGdW5jdGlvbgkAtQkCBQxjYWxsYmFja0FyZ3MCASwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIFCGJBc3NldElkBQdiQW1vdW50BQNuaWwDCQAAAgUIY2FsbGJhY2sFCGNhbGxiYWNrBAdzQW1vdW50CgABQAUIY2FsbGJhY2sDCQABAgUBQAIDSW50BQFACQACAQkArAICCQADAQUBQAIYIGNvdWxkbid0IGJlIGNhc3QgdG8gSW50AwkAAAIFB3NBbW91bnQFB3NBbW91bnQECXN1cHBseVJlcwkBDnN1cHBseUludGVybmFsAwULc0Fzc2V0SWRTdHIFB3NBbW91bnQFBHVzZXIED2NvbGxhdGVyYWxWYWx1ZQoAAUAJAPwHBAUEdGhpcwIRZ2V0VXNlckNvbGxhdGVyYWwJAMwIAgcJAMwIAgUEdXNlcgkAzAgCBgkAzAgCCQCsAgIJAKwCAgULc0Fzc2V0SWRTdHICCixzdXBwbGllZCwJAKQDAQUHc0Ftb3VudAUDbmlsBQNuaWwDCQABAgUBQAIDSW50BQFACQACAQkArAICCQADAQUBQAIYIGNvdWxkbid0IGJlIGNhc3QgdG8gSW50AwkAAAIFD2NvbGxhdGVyYWxWYWx1ZQUPY29sbGF0ZXJhbFZhbHVlBAtib3Jyb3dWYWx1ZQoAAUAJAPwHBAUEdGhpcwIRZ2V0VXNlckNvbGxhdGVyYWwJAMwIAgcJAMwIAgUEdXNlcgkAzAgCBgkAzAgCCQCsAgIJAKwCAgULYkFzc2V0SWRTdHICCixib3Jyb3dlZCwJAKQDAQUHYkFtb3VudAUDbmlsBQNuaWwDCQABAgUBQAIDSW50BQFACQACAQkArAICCQADAQUBQAIYIGNvdWxkbid0IGJlIGNhc3QgdG8gSW50AwkAAAIFC2JvcnJvd1ZhbHVlBQtib3Jyb3dWYWx1ZQMJAGYCBQtib3Jyb3dWYWx1ZQUPY29sbGF0ZXJhbFZhbHVlCQACAQIebm90IGVub3VnaCBjb2xsYXRlcmFsIHByb3ZpZGVkCQDOCAIIBQlib3Jyb3dSZXMCXzEIBQlzdXBwbHlSZXMCXzEJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEGc3VwcGx5AAMDCQECIT0CCQCQAwEIBQFpCHBheW1lbnRzAAEGCQAAAggJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQAAAkAAgECHDEgcGF5bWVudCBoYXMgdG8gYmUgYXR0YWNoZWQECmFzc2V0SWRTdHIJAQ5nZXRBc3NldFN0cmluZwEICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQLYXNzZXRBbW91bnQICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BApheGx5Tm90aWZ5CQD8BwQFEmF4bHlGYXJtaW5nQWRkcmVzcwIKbGVuZEFjdGlvbgkAzAgCCQClCAEIBQFpBmNhbGxlcgkAzAgCBQphc3NldElkU3RyBQNuaWwFA25pbAMJAAACBQpheGx5Tm90aWZ5BQpheGx5Tm90aWZ5CQEOc3VwcGx5SW50ZXJuYWwDBQphc3NldElkU3RyBQthc3NldEFtb3VudAkApQgBCAUBaQZjYWxsZXIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEId2l0aGRyYXcCCmFzc2V0SWRTdHILYXNzZXRBbW91bnQEDSR0MDEzMTAzMTMxNzAJAQ1nZXRBY3R1YWxSYXRlAgUKYXNzZXRJZFN0cgIFc1JhdGUEBXNSYXRlCAUNJHQwMTMxMDMxMzE3MAJfMQQRcmF0ZXNSZWNhbGNSZXN1bHQIBQ0kdDAxMzEwMzEzMTcwAl8yBAZhbW91bnQJAG4EBQthc3NldEFtb3VudAUHU2NhbGUxNgUFc1JhdGUFB0NFSUxJTkcEB2FkZHJlc3MJAKUIAQgFAWkGY2FsbGVyBA1hc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBA1hc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRTdXBwbGllZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgQRdXNlckFzc2V0Qm9ycm93ZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIECmF4bHlOb3RpZnkJAPwHBAUSYXhseUZhcm1pbmdBZGRyZXNzAgpsZW5kQWN0aW9uCQDMCAIJAKUIAQgFAWkGY2FsbGVyCQDMCAIFCmFzc2V0SWRTdHIFA25pbAUDbmlsAwkAAAIFCmF4bHlOb3RpZnkFCmF4bHlOb3RpZnkEEmNvbGxhdGVyYWxWYWx1ZUludgkA/AcEBQR0aGlzAhFnZXRVc2VyQ29sbGF0ZXJhbAkAzAgCBwkAzAgCBQdhZGRyZXNzCQDMCAIGCQDMCAIJAKwCAgkArAICBQphc3NldElkU3RyAgosc3VwcGxpZWQsCQCkAwEJAQEtAQUGYW1vdW50BQNuaWwFA25pbAMJAAACBRJjb2xsYXRlcmFsVmFsdWVJbnYFEmNvbGxhdGVyYWxWYWx1ZUludgQPY29sbGF0ZXJhbFZhbHVlBAckbWF0Y2gwBRJjb2xsYXRlcmFsVmFsdWVJbnYDCQABAgUHJG1hdGNoMAIDSW50BAF4BQckbWF0Y2gwBQF4CQACAQIfY2FuJ3QgZ2V0IHVzZXIgY29sbGF0ZXJhbCB2YWx1ZQMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAMJAGYCAAAFD2NvbGxhdGVyYWxWYWx1ZQkAAgECMnlvdSBkb250IGhhdmUgZW5vdWdoIGNvbGxhdGVyYWwgZm9yIHRoaXMgb3BlcmF0aW9uAwkAZgIFBmFtb3VudAkAZQIFDWFzc2V0U3VwcGxpZWQFDWFzc2V0Qm9ycm93ZWQJAAIBAip0aGlzIGFtb3VudCBpcyBub3QgYXZhaWxhYmxlIG9uIHRoZSBtYXJrZXQDCQBmAgUGYW1vdW50CQBlAgURdXNlckFzc2V0U3VwcGxpZWQFEXVzZXJBc3NldEJvcnJvd2VkCQACAQIqdGhpcyBhbW91bnQgaXMgbm90IGF2YWlsYWJsZSBmb3IgdGhpcyB1c2VyCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyCQBlAgkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgUGYW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIFBmFtb3VudAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIFC2Fzc2V0QW1vdW50CQENZ2V0QXNzZXRCeXRlcwEFCmFzc2V0SWRTdHIFA25pbAURcmF0ZXNSZWNhbGNSZXN1bHQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEFcmVwYXkAAwkBASEBCQENdHJ5R2V0Qm9vbGVhbgECDHNldHVwX2FjdGl2ZQkAAgECEW1hcmtldCBpcyBzdG9wcGVkAwMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQYJAAACCAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAAACQACAQIcMSBwYXltZW50IGhhcyB0byBiZSBhdHRhY2hlZAQKYXNzZXRJZFN0cgkBDmdldEFzc2V0U3RyaW5nAQgJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBAthc3NldEFtb3VudAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQEDSR0MDE0OTg3MTUwNTQJAQ1nZXRBY3R1YWxSYXRlAgUKYXNzZXRJZFN0cgIFYlJhdGUEBWJSYXRlCAUNJHQwMTQ5ODcxNTA1NAJfMQQRcmF0ZXNSZWNhbGNSZXN1bHQIBQ0kdDAxNDk4NzE1MDU0Al8yBAZhbW91bnQJAG4EBQthc3NldEFtb3VudAUHU2NhbGUxNgUFYlJhdGUFB0NFSUxJTkcEB2FkZHJlc3MJAKUIAQgFAWkGY2FsbGVyBA1hc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBA1hc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQKYW1vdW50TGVmdAkAZQIFEXVzZXJBc3NldEJvcnJvd2VkBQZhbW91bnQEC3JlcGF5QW1vdW50AwkAZwIFCmFtb3VudExlZnQAAAUGYW1vdW50BRF1c2VyQXNzZXRCb3Jyb3dlZAMJAAACCQCzCQIJAQx0cnlHZXRTdHJpbmcBAgxzZXR1cF90b2tlbnMFCmFzc2V0SWRTdHIFBHVuaXQJAAIBAil0aGlzIGFzc2V0IGlzIG5vdCBzdXBwb3J0ZWQgYnkgdGhlIG1hcmtldAkAzggCCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyCQBlAgURdXNlckFzc2V0Qm9ycm93ZWQFC3JlcGF5QW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgkAZQIFDWFzc2V0Qm9ycm93ZWQFC3JlcGF5QW1vdW50BQNuaWwFEXJhdGVzUmVjYWxjUmVzdWx0AwkAZwIFCmFtb3VudExlZnQAAAUDbmlsCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMIBQFpBmNhbGxlcgkBAS0BBQphbW91bnRMZWZ0CAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQFA25pbAFpAQhyZXBheUZvcgEHYWRkcmVzcwMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAMJAQEhAQkBFHZlcmlmeVNlbnRpbmVsUmlnaHRzAQgFAWkGY2FsbGVyCQACAQIcYXZhaWxhYmxlIG9ubHkgZm9yIHdoaXRlbGlzdAMJAAACBQdhZGRyZXNzAgZnbG9iYWwJAAIBAiB5b3UgY2FuJ3QgcmVwYXkgZm9yIGV2ZXJ5b25lIDpfKQMDCQECIT0CCQCQAwEIBQFpCHBheW1lbnRzAAEGCQAAAggJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQAAAkAAgECHDEgcGF5bWVudCBoYXMgdG8gYmUgYXR0YWNoZWQECmFzc2V0SWRTdHIJAQ5nZXRBc3NldFN0cmluZwEICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQLYXNzZXRBbW91bnQICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BA0kdDAxNjUxNzE2NTg0CQENZ2V0QWN0dWFsUmF0ZQIFCmFzc2V0SWRTdHICBWJSYXRlBAViUmF0ZQgFDSR0MDE2NTE3MTY1ODQCXzEEEXJhdGVzUmVjYWxjUmVzdWx0CAUNJHQwMTY1MTcxNjU4NAJfMgQGYW1vdW50CQBuBAULYXNzZXRBbW91bnQFB1NjYWxlMTYFBWJSYXRlBQdDRUlMSU5HBA1hc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBA1hc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQKYW1vdW50TGVmdAkAZQIFEXVzZXJBc3NldEJvcnJvd2VkBQZhbW91bnQEC3JlcGF5QW1vdW50AwkAZwIFCmFtb3VudExlZnQAAAUGYW1vdW50BRF1c2VyQXNzZXRCb3Jyb3dlZAQRZXh0cmFBbW91bnRSZXN1bHQDCQBnAgUKYW1vdW50TGVmdAAABQNuaWwJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCQEBLQEFCmFtb3VudExlZnQICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAUDbmlsBA53aXRoZHJhd1Jlc3VsdAMJAAACCAUBaQZjYWxsZXIFC2F4bHlBZGRyZXNzCgEEZm9sZAIFYWNjdW0KYXNzZXRJZFN0cgQMdXNlclN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyAwkAZgIFDHVzZXJTdXBwbGllZAAACQDOCAIFBWFjY3VtCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgAACQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIFDHVzZXJTdXBwbGllZAUDbmlsBQVhY2N1bQoAAiRsBQdheGx5TFBzCgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQRmb2xkAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhNMaXN0IHNpemUgZXhjZWVkcyA4CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAUDbmlsAwkAAAIJALMJAgkBDHRyeUdldFN0cmluZwECDHNldHVwX3Rva2VucwUKYXNzZXRJZFN0cgUEdW5pdAkAAgECKXRoaXMgYXNzZXQgaXMgbm90IHN1cHBvcnRlZCBieSB0aGUgbWFya2V0CQDOCAIJAM4IAgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgkAZQIFEXVzZXJBc3NldEJvcnJvd2VkBQtyZXBheUFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIJAGUCBQ1hc3NldEJvcnJvd2VkBQtyZXBheUFtb3VudAUDbmlsBRFyYXRlc1JlY2FsY1Jlc3VsdAURZXh0cmFBbW91bnRSZXN1bHQFDndpdGhkcmF3UmVzdWx0AWkBE2FkZEludGVyZXN0RVhURVJOQUwABAZhbW91bnQJAGsDCAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudABQAGQEB2Fzc2V0SWQICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQKYXNzZXRJZFN0cgkBDmdldEFzc2V0U3RyaW5nAQUHYXNzZXRJZAQGZWFybmVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIVYXV0b3N0YWtlX2xhc3RFYXJuZWRfBQphc3NldElkU3RyBApsYXN0SGVpZ2h0CQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIUYXV0b3N0YWtlX2xhc3RCbG9ja18FCmFzc2V0SWRTdHIEDHN0YXRlQ2hhbmdlcwMDCQAAAgUKbGFzdEhlaWdodAUGaGVpZ2h0BgkAAAIFBmFtb3VudAAABQNuaWwJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIYYXV0b3N0YWtlX3ByZUxhc3RFYXJuZWRfBQphc3NldElkU3RyBQZlYXJuZWQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIXYXV0b3N0YWtlX3ByZUxhc3RCbG9ja18FCmFzc2V0SWRTdHIFCmxhc3RIZWlnaHQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIVYXV0b3N0YWtlX2xhc3RFYXJuZWRfBQphc3NldElkU3RyCQBkAgUGZWFybmVkBQZhbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIUYXV0b3N0YWtlX2xhc3RCbG9ja18FCmFzc2V0SWRTdHIFBmhlaWdodAUDbmlsCQDOCAIFDHN0YXRlQ2hhbmdlcwkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgZfc1JhdGUJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgUKYXNzZXRJZFN0cgIGX3NSYXRlCQBrAwUHU2NhbGUxNgUGYW1vdW50CQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBQNuaWwBaQEHcHJlSW5pdAQGdG9rZW5zBGx0dnMDbHRzCXBlbmFsdGllcwoBAWYCBWFjY3VtBXRva2VuCQDOCAIFBWFjY3VtCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBXRva2VuAgZfYlJhdGUFB1NjYWxlMTYJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUFdG9rZW4CBl9zUmF0ZQUHU2NhbGUxNgUDbmlsAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIKYWRtaW4gb25seQQFcmF0ZXMKAAIkbAkAtQkCBQZ0b2tlbnMCASwKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEyCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMCQDOCAIJAMwIAgkBC1N0cmluZ0VudHJ5AgIMc2V0dXBfdG9rZW5zBQZ0b2tlbnMJAMwIAgkBC1N0cmluZ0VudHJ5AgIKc2V0dXBfbHR2cwUEbHR2cwkAzAgCCQELU3RyaW5nRW50cnkCAglzZXR1cF9sdHMFA2x0cwkAzAgCCQELU3RyaW5nRW50cnkCAg9zZXR1cF9wZW5hbHRpZXMFCXBlbmFsdGllcwkAzAgCCQEMQm9vbGVhbkVudHJ5AgIMc2V0dXBfYWN0aXZlBgUDbmlsBQVyYXRlcwFpAQxpbml0TmV3VG9rZW4EBXRva2VuA2x0dgJsdAdwZW5hbHR5AwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIKYWRtaW4gb25seQkAzAgCCQELU3RyaW5nRW50cnkCAgxzZXR1cF90b2tlbnMJAKwCAgkArAICCQEMdHJ5R2V0U3RyaW5nAQIMc2V0dXBfdG9rZW5zAgEsBQV0b2tlbgkAzAgCCQELU3RyaW5nRW50cnkCAgpzZXR1cF9sdHZzCQCsAgIJAKwCAgkBDHRyeUdldFN0cmluZwECCnNldHVwX2x0dnMCASwFA2x0dgkAzAgCCQELU3RyaW5nRW50cnkCAglzZXR1cF9sdHMJAKwCAgkArAICCQEMdHJ5R2V0U3RyaW5nAQIJc2V0dXBfbHRzAgEsBQJsdAkAzAgCCQELU3RyaW5nRW50cnkCAg9zZXR1cF9wZW5hbHRpZXMJAKwCAgkArAICCQEMdHJ5R2V0U3RyaW5nAQIPc2V0dXBfcGVuYWx0aWVzAgEsBQdwZW5hbHR5CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBXRva2VuAgZfYlJhdGUFB1NjYWxlMTYJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUFdG9rZW4CBl9zUmF0ZQUHU2NhbGUxNgUDbmlsAWkBD3VwZGF0ZVBhcmFtZXRlcgIDa2V5A3ZhbAMDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAQIhPQIIBQFpBmNhbGxlcgkBB0FkZHJlc3MBARoBVxR5WIXA+hpn5xQ0jQz3xPBtRe32HPk9QQcJAAIBAgphZG1pbiBvbmx5CQDMCAIJAQxJbnRlZ2VyRW50cnkCBQNrZXkJAQ1wYXJzZUludFZhbHVlAQUDdmFsBQNuaWwBaQEMdXBkYXRlU3RyaW5nAgNrZXkDdmFsAwMJAQIhPQIIBQFpBmNhbGxlcgUEdGhpcwkBAiE9AggFAWkGY2FsbGVyCQEHQWRkcmVzcwEBGgFXFHlYhcD6GmfnFDSNDPfE8G1F7fYc+T1BBwkAAgECCmFkbWluIG9ubHkJAMwIAgkBC1N0cmluZ0VudHJ5AgUDa2V5BQN2YWwFA25pbAFpARJjbGFpbVRvUmVzZXJ2ZUZ1bmQBBWRlYnVnBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMABAVyYXRlcwgJAQ1nZXRBY3R1YWxSYXRlAgkAkQMCBQZhc3NldHMAAAIFc1JhdGUCXzIEAmxpCQDMCAIAAAkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFCQDMCAIABgkAzAgCAAcJAMwIAgAICQDMCAIACQkAzAgCAAoJAMwIAgALBQNuaWwKAQFmAgVhY2N1bQFuAwkAZwIFAW4JAJADAQUGYXNzZXRzBQVhY2N1bQQKYXNzZXRJZFN0cgkAkQMCBQZhc3NldHMFAW4ED2F1dG9zdGFrZUFtb3VudAkBDHRyeUdldFN0cmluZwEJAKwCAgIRYXV0b3N0YWtlX2Ftb3VudF8FCmFzc2V0SWRTdHIEBmFtb3VudAkAZQIJAGQCCQBkAgkAZAIJAQpnZXRCYWxhbmNlAQUKYXNzZXRJZFN0cgkBDXRyeUdldEludGVnZXIBCQCsAgICEWF1dG9zdGFrZV9hbW91bnRfBQphc3NldElkU3RyAwkBAiE9AgUPYXV0b3N0YWtlQW1vdW50AgAJAQ1wYXJzZUludFZhbHVlAQUPYXV0b3N0YWtlQW1vdW50AAAJAGsDCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyCAkAkQMCBQVyYXRlcwkAZAIJAGgCBQFuAAMAAQV2YWx1ZQUHU2NhbGUxNgkAawMJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIICQCRAwIFBXJhdGVzCQBoAgUBbgADBXZhbHVlBQdTY2FsZTE2BANpbnYDCQBmAgkBDXRyeUdldEludGVnZXIBCQCsAgICEWF1dG9zdGFrZV9hbW91bnRfBQphc3NldElkU3RyAAAJAPwHBAUEdGhpcwIMdW5zdGFrZVRva2VuCQDMCAIFCmFzc2V0SWRTdHIJAMwIAgkAlgMBCQDMCAIFBmFtb3VudAkAzAgCAAAFA25pbAUDbmlsBQNuaWwAAAMJAAACBQNpbnYFA2ludgkAzggCBQVhY2N1bQkAzAgCBQZhbW91bnQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgQJcGFyYW1ldGVyCgACJGwFAmxpCgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAoBAmYyAgVhY2N1bQFuAwkAZwIFAW4JAJADAQUGYXNzZXRzBQVhY2N1bQQKYXNzZXRJZFN0cgkAkQMCBQZhc3NldHMFAW4JAM4IAgUFYWNjdW0JAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwUScmVzZXJ2ZUZ1bmRBZGRyZXNzCQCWAwEJAMwIAgkAkQMCBQlwYXJhbWV0ZXIFAW4JAMwIAgAABQNuaWwJAQ1nZXRBc3NldEJ5dGVzAQUKYXNzZXRJZFN0cgUDbmlsAwUFZGVidWcJAAIBCQEKbGlJbnRUb1N0cgEFCXBhcmFtZXRlcgkAlAoCCgACJGwFAmxpCgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYxXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQJmMgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMV8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTIJAQUkZjFfMgIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgALAAwFCXBhcmFtZXRlcgFpAQhzaHV0ZG93bgEIc2h1dGRvd24DCQAAAgkAzwgCBRFzaHV0ZG93bldoaXRlbGlzdAgIBQFpBmNhbGxlcgVieXRlcwUEdW5pdAkAAgECF3VzZXIgbm90IGluIGEgd2hpdGVsaXN0CQDMCAIJAQxCb29sZWFuRW50cnkCAgxzZXR1cF9hY3RpdmUJAQEhAQUIc2h1dGRvd24FA25pbAFpAQlsaXF1aWRhdGUGBWRlYnVnB2FkZHJlc3MLYXNzZXRBbW91bnQLc0Fzc2V0SWRTdHILYkFzc2V0SWRTdHIIcm91dGVTdHIDCQEBIQEJARZ2ZXJpZnlMaXF1aWRhdG9yUmlnaHRzAQgFAWkGY2FsbGVyCQACAQIcYXZhaWxhYmxlIGZvciB3aGl0ZWxpc3Qgb25seQMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAQOdXNlckNvbGxhdGVyYWwJARJjYWxjVXNlckNvbGxhdGVyYWwBBQdhZGRyZXNzAwkAAAIFDnVzZXJDb2xsYXRlcmFsBQ51c2VyQ29sbGF0ZXJhbAQNJHQwMjI2NTIyMjcxNAkBDWdldEFjdHVhbFJhdGUCBQtzQXNzZXRJZFN0cgIFc1JhdGUEBXNSYXRlCAUNJHQwMjI2NTIyMjcxNAJfMQQLcmF0ZXNSZXN1bHQIBQ0kdDAyMjY1MjIyNzE0Al8yBA0kdDAyMjcxOTIyNzg4CQENZ2V0QWN0dWFsUmF0ZQIFC2JBc3NldElkU3RyAgViUmF0ZQQFYlJhdGUIBQ0kdDAyMjcxOTIyNzg4Al8xBBJyYXRlc1JlY2FsY1Jlc3VsdDIIBQ0kdDAyMjcxOTIyNzg4Al8yBAxzQXNzZXRBbW91bnQJAGsDBQthc3NldEFtb3VudAUHU2NhbGUxNgUFc1JhdGUEEGN1cnJlbnRTUG9zaXRpb24JAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FC3NBc3NldElkU3RyBBNjdXJyZW50QlBvc2l0aW9uVmFsCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgQQY3VycmVudEJQb3NpdGlvbgMJAGYCBRNjdXJyZW50QlBvc2l0aW9uVmFsAAAFE2N1cnJlbnRCUG9zaXRpb25WYWwJAAIBAiB1c2VyIGhhcyBubyBib3Jyb3cgaW4gdGhpcyB0b2tlbgMJAGYCBQ51c2VyQ29sbGF0ZXJhbAAACQACAQIYdXNlciBjYW4ndCBiZSBsaXF1aWRhdGVkAwkAZgIFDHNBc3NldEFtb3VudAUQY3VycmVudFNQb3NpdGlvbgkAAgECMnBvc2l0aW9uIHRvIGxpcXVpZGF0ZSBpcyBiaWdnZXIgdGhhbiB1c2VyJ3Mgc3VwcGx5BANpbnYDCQECIT0CCQDPCAIFB2F4bHlMUHMFC3NBc3NldElkU3RyBQR1bml0BAVwb3NJZAkAkQMCCQC1CQIFB2FkZHJlc3MCAV8AAQQJYWRkcmVzc0lkCQCRAwIJALUJAgUHYWRkcmVzcwIBXwAACQD9BwQFC2F4bHlBZGRyZXNzAglsaXF1aWRhdGUJAMwIAgUJYWRkcmVzc0lkCQDMCAIFBXBvc0lkCQDMCAIFDHNBc3NldEFtb3VudAUDbmlsBQNuaWwFA25pbAMJAAACBQNpbnYFA2ludgQOYmFsYW5jZTBCZWZvcmUJAQpnZXRCYWxhbmNlAQULc0Fzc2V0SWRTdHIDCQAAAgUOYmFsYW5jZTBCZWZvcmUFDmJhbGFuY2UwQmVmb3JlBA5iYWxhbmNlMUJlZm9yZQkBCmdldEJhbGFuY2UBBQtiQXNzZXRJZFN0cgMJAAACBQ5iYWxhbmNlMUJlZm9yZQUOYmFsYW5jZTFCZWZvcmUEDmV4Y2hhbmdlSW52b2tlCQD8BwQFEWFnZ3JlZ2F0b3JBZGRyZXNzAgRzd2FwCQDMCAIFCHJvdXRlU3RyCQDMCAIAAAUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQENZ2V0QXNzZXRCeXRlcwEFC3NBc3NldElkU3RyBQthc3NldEFtb3VudAUDbmlsAwkAAAIFDmV4Y2hhbmdlSW52b2tlBQ5leGNoYW5nZUludm9rZQQKYXNzZXQwU29sZAkAZQIFDmJhbGFuY2UwQmVmb3JlCQEKZ2V0QmFsYW5jZQEFC3NBc3NldElkU3RyAwkAAAIFCmFzc2V0MFNvbGQFCmFzc2V0MFNvbGQEDGFzc2V0MUJvdWdodAkAZQIJAQpnZXRCYWxhbmNlAQULYkFzc2V0SWRTdHIFDmJhbGFuY2UxQmVmb3JlAwkAAAIFDGFzc2V0MUJvdWdodAUMYXNzZXQxQm91Z2h0BAthc3NldDBQcmljZQgJAQ1nZXRUb2tlblByaWNlAQULc0Fzc2V0SWRTdHICXzEEC2Fzc2V0MFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQtzQXNzZXRJZFN0cgQJYXNzZXQwVXNkCQBrAwUKYXNzZXQwU29sZAULYXNzZXQwUHJpY2UFC2Fzc2V0MFNjYWxlBAthc3NldDFQcmljZQgJAQ1nZXRUb2tlblByaWNlAQULYkFzc2V0SWRTdHICXzIEC2Fzc2V0MVNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQtiQXNzZXRJZFN0cgQJYXNzZXQxVXNkCQBrAwUMYXNzZXQxQm91Z2h0BQthc3NldDFQcmljZQULYXNzZXQxU2NhbGUEB3BlbmFsdHkJAQ1wYXJzZUludFZhbHVlAQkAkQMCCQC1CQIJAQx0cnlHZXRTdHJpbmcBAg9zZXR1cF9wZW5hbHRpZXMCASwJAQV2YWx1ZQEJAM8IAgkBD2dldE1hcmtldEFzc2V0cwAFC2JBc3NldElkU3RyBBFsaXF1aWRhdGlvblByb2ZpdAkAZQIFCWFzc2V0MVVzZAkAawMFCWFzc2V0MFVzZAkAZQIFBlNjYWxlOAUHcGVuYWx0eQUGU2NhbGU4BAxzQXNzZXRDaGFuZ2UJAGsDBQphc3NldDBTb2xkBQdTY2FsZTE2BQVzUmF0ZQQMYkFzc2V0Q2hhbmdlCQBrAwkAawMFDGFzc2V0MUJvdWdodAUHU2NhbGUxNgUFYlJhdGUJAGUCBQZTY2FsZTgJAGsDBRFsaXF1aWRhdGlvblByb2ZpdAUGU2NhbGU4BQlhc3NldDFVc2QFBlNjYWxlOAMJAGYCBQphc3NldDBTb2xkBQthc3NldEFtb3VudAkAAgECI21vcmUgYXNzZXRzIGV4Y2hhbmdlZCB0aGFuIGV4cGVjdGVkAwkAZgIAAAURbGlxdWlkYXRpb25Qcm9maXQJAAIBAi9wcmljZSBpbXBhY3QgaXMgYmlnZ2VyIHRoYW4gbGlxdWlkYXRpb24gcGVuYWx0eQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FC3NBc3NldElkU3RyCQBlAgUQY3VycmVudFNQb3NpdGlvbgUMc0Fzc2V0Q2hhbmdlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwULYkFzc2V0SWRTdHIJAGUCBRBjdXJyZW50QlBvc2l0aW9uBQxiQXNzZXRDaGFuZ2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQtzQXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FC3NBc3NldElkU3RyBQxzQXNzZXRDaGFuZ2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FC2JBc3NldElkU3RyBQxiQXNzZXRDaGFuZ2UFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQtsaXF1aWRhdGVWMgMFZGVidWcHYWRkcmVzcwtzQXNzZXRJZFN0cgMJAQEhAQkBFnZlcmlmeUxpcXVpZGF0b3JSaWdodHMBCAUBaQZjYWxsZXIJAAIBAhxhdmFpbGFibGUgZm9yIHdoaXRlbGlzdCBvbmx5AwkBASEBCQENdHJ5R2V0Qm9vbGVhbgECDHNldHVwX2FjdGl2ZQkAAgECEW1hcmtldCBpcyBzdG9wcGVkBAhiQXNzZXRJZAgJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBAtiQXNzZXRJZFN0cgkBDmdldEFzc2V0U3RyaW5nAQUIYkFzc2V0SWQEDGJBc3NldEFtb3VudAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQEDnVzZXJDb2xsYXRlcmFsCQESY2FsY1VzZXJDb2xsYXRlcmFsAQUHYWRkcmVzcwMJAAACBQ51c2VyQ29sbGF0ZXJhbAUOdXNlckNvbGxhdGVyYWwDCQBmAgUOdXNlckNvbGxhdGVyYWwAAAkAAgECGHVzZXIgY2FuJ3QgYmUgbGlxdWlkYXRlZAQMbWFya2V0QXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAQJYXNzZXQxTnVtCQEFdmFsdWUBCQDPCAIFDG1hcmtldEFzc2V0cwULYkFzc2V0SWRTdHIECWFzc2V0ME51bQkBBXZhbHVlAQkAzwgCBQxtYXJrZXRBc3NldHMFC3NBc3NldElkU3RyBA0kdDAyNjEwOTI2MTcxCQENZ2V0QWN0dWFsUmF0ZQIFC2JBc3NldElkU3RyAgViUmF0ZQQFYlJhdGUIBQ0kdDAyNjEwOTI2MTcxAl8xBAtyYXRlc1Jlc3VsdAgFDSR0MDI2MTA5MjYxNzECXzIEC2Fzc2V0MVByaWNlCAkBDWdldFRva2VuUHJpY2UBBQtiQXNzZXRJZFN0cgJfMgQLYXNzZXQxU2NhbGUJAQ5jYWxjQXNzZXRTY2FsZQEFC2JBc3NldElkU3RyBApiQW1vdW50VXNkCQBrAwUMYkFzc2V0QW1vdW50BQthc3NldDFQcmljZQULYXNzZXQxU2NhbGUEB3BlbmFsdHkJAQ1wYXJzZUludFZhbHVlAQkBBXZhbHVlAQkAkQMCCQC1CQIJAQx0cnlHZXRTdHJpbmcBAg9zZXR1cF9wZW5hbHRpZXMCASwFCWFzc2V0MU51bQQLYXNzZXQwUHJpY2UICQENZ2V0VG9rZW5QcmljZQEFC3NBc3NldElkU3RyAl8xBAthc3NldDBTY2FsZQkBDmNhbGNBc3NldFNjYWxlAQULc0Fzc2V0SWRTdHIECnNBbW91bnRVc2QJAGsDBQpiQW1vdW50VXNkCQBkAgUGU2NhbGU4BQdwZW5hbHR5BQZTY2FsZTgEDHNBc3NldEFtb3VudAkAawMFCnNBbW91bnRVc2QFC2Fzc2V0MFNjYWxlBQthc3NldDBQcmljZQQHYkFtb3VudAkAawMFDGJBc3NldEFtb3VudAUHU2NhbGUxNgUFYlJhdGUEB3NBbW91bnQJAGsDBQxzQXNzZXRBbW91bnQFB1NjYWxlMTYICQCRAwIFC3JhdGVzUmVzdWx0CQBkAgkAaAIFCWFzc2V0ME51bQADAAEFdmFsdWUEA2ludgMJAQIhPQIJAM8IAgUHYXhseUxQcwULc0Fzc2V0SWRTdHIFBHVuaXQEBXBvc0lkCQCRAwIJALUJAgUHYWRkcmVzcwIBXwABBAlhZGRyZXNzSWQJAJEDAgkAtQkCBQdhZGRyZXNzAgFfAAAJAP0HBAULYXhseUFkZHJlc3MCCWxpcXVpZGF0ZQkAzAgCBQlhZGRyZXNzSWQJAMwIAgUFcG9zSWQJAMwIAgUHc0Ftb3VudAUDbmlsBQNuaWwFA25pbAMJAAACBQNpbnYFA2ludgQQY3VycmVudFNQb3NpdGlvbgkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwULc0Fzc2V0SWRTdHIEE2N1cnJlbnRCUG9zaXRpb25WYWwJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FC2JBc3NldElkU3RyBBBjdXJyZW50QlBvc2l0aW9uAwkAZgIFE2N1cnJlbnRCUG9zaXRpb25WYWwAAAUTY3VycmVudEJQb3NpdGlvblZhbAkAAgECIHVzZXIgaGFzIG5vIGJvcnJvdyBpbiB0aGlzIHRva2VuAwkAZgIFB3NBbW91bnQFEGN1cnJlbnRTUG9zaXRpb24JAAIBAjJwb3NpdGlvbiB0byBsaXF1aWRhdGUgaXMgYmlnZ2VyIHRoYW4gdXNlcidzIHN1cHBseQMFBWRlYnVnCQACAQIVbGlxdWlkYXRpb24gd2lsbCBwYXNzCQDOCAIJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyBQxzQXNzZXRBbW91bnQJAQ1nZXRBc3NldEJ5dGVzAQULc0Fzc2V0SWRTdHIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQtzQXNzZXRJZFN0cgkAZQIFEGN1cnJlbnRTUG9zaXRpb24FB3NBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgkAZQIFEGN1cnJlbnRCUG9zaXRpb24FB2JBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQtzQXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FC3NBc3NldElkU3RyBQdzQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX2JvcnJvd2VkXwULYkFzc2V0SWRTdHIJAGUCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgUHYkFtb3VudAUDbmlsBQtyYXRlc1Jlc3VsdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARFnZXRVc2VyQ29sbGF0ZXJhbAQFZGVidWcHYWRkcmVzcw1taW51c0JvcnJvd2VkC2FmdGVyQ2hhbmdlBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMABARsdHZzCQC1CQIJAQx0cnlHZXRTdHJpbmcBAgpzZXR1cF9sdHZzAgEsBANsdHMJALUJAgkBDHRyeUdldFN0cmluZwECCXNldHVwX2x0cwIBLAQFcmF0ZXMICQENZ2V0QWN0dWFsUmF0ZQIJAJEDAgUGYXNzZXRzAAACBXNSYXRlAl8yBA1jaGFuZ2VIYW5kbGVyCQC1CQIFC2FmdGVyQ2hhbmdlAgEsCgEBZgIFYWNjdW0EbmV4dAMJAGcCBQRuZXh0CQCQAwEFBmFzc2V0cwUFYWNjdW0EDHVzZXJTdXBwbGllZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwkAkQMCBQZhc3NldHMFBG5leHQEDHVzZXJCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwkAkQMCBQZhc3NldHMFBG5leHQEE25lZWRUb2tlbkFjY291bnRpbmcDCQAAAgULYWZ0ZXJDaGFuZ2UCAAMDCQECIT0CBQx1c2VyQm9ycm93ZWQAAAYJAQIhPQIFDHVzZXJTdXBwbGllZAAABgcGAwUTbmVlZFRva2VuQWNjb3VudGluZwQKYXNzZXRTY2FsZQkBDmNhbGNBc3NldFNjYWxlAQkAkQMCBQZhc3NldHMFBG5leHQECmFzc2V0UHJpY2UJAQ1nZXRUb2tlblByaWNlAQkAkQMCBQZhc3NldHMFBG5leHQJAGUCCQBkAgUFYWNjdW0JAGsDCQBrAwkAawMJAGQCBQx1c2VyU3VwcGxpZWQDAwMJAQIhPQIFC2FmdGVyQ2hhbmdlAgAJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAAkAkQMCBQZhc3NldHMFBG5leHQHCQAAAgkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAECCHN1cHBsaWVkBwkBDXBhcnNlSW50VmFsdWUBCQCRAwIFDWNoYW5nZUhhbmRsZXIAAgAACAkAkQMCBQVyYXRlcwkAaAIFBG5leHQAAwV2YWx1ZQUHU2NhbGUxNgkBDXBhcnNlSW50VmFsdWUBCQCRAwIFBGx0dnMFBG5leHQFBlNjYWxlOAgFCmFzc2V0UHJpY2UCXzEFCmFzc2V0U2NhbGUDBQ1taW51c0JvcnJvd2VkCQBrAwkAawMJAGsDCQBkAgUMdXNlckJvcnJvd2VkAwMDCQECIT0CBQthZnRlckNoYW5nZQIACQAAAgkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAAJAJEDAgUGYXNzZXRzBQRuZXh0BwkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgABAghib3Jyb3dlZAcJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAIAAAgJAJEDAgUFcmF0ZXMJAGQCCQBoAgUEbmV4dAADAAEFdmFsdWUFB1NjYWxlMTYFBlNjYWxlOAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFA2x0cwUEbmV4dAgFCmFzc2V0UHJpY2UCXzIFCmFzc2V0U2NhbGUAAAUFYWNjdW0EBnJlc3VsdAoAAiRsCQDMCAIAAAkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFCQDMCAIABgkAzAgCAAcJAMwIAgAICQDMCAIACQkAzAgCAAoJAMwIAgALBQNuaWwKAAIkcwkAkAMBBQIkbAoABSRhY2MwAAAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEyCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAwUFZGVidWcJAAIBCQCkAwEFBnJlc3VsdAkAlAoCBQNuaWwFBnJlc3VsdAFpAQxnZXRBc3NldERlYnQDBWRlYnVnB2FkZHJlc3MKYXNzZXRJZFN0cgQMdXNlckJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyBAphc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQphc3NldElkU3RyBARyYXRlCAkBDWdldEFjdHVhbFJhdGUCBQphc3NldElkU3RyAgViUmF0ZQJfMQQGcmVzdWx0CQBrAwUMdXNlckJvcnJvd2VkBQRyYXRlBQdTY2FsZTE2AwUFZGVidWcJAAIBCQCkAwEFBnJlc3VsdAkAlAoCBQNuaWwFBnJlc3VsdAFpAQlnZXRQcmljZXMBBWRlYnVnBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMACgEBZgIFYWNjdW0EbmV4dAMJAGcCBQRuZXh0CQCQAwEFBmFzc2V0cwUFYWNjdW0ECmFzc2V0UHJpY2UJARVnZXRUb2tlblByaWNlV2l0aFJpc2sCCQCRAwIFBmFzc2V0cwUEbmV4dAADCQCsAgIJAKwCAgkArAICCQCsAgIFBWFjY3VtCQCkAwEIBQphc3NldFByaWNlAl8xAgEsCQCkAwEIBQphc3NldFByaWNlAl8yAgF8BAZyZXN1bHQKAAIkbAkAzAgCAAAJAMwIAgABCQDMCAIAAgkAzAgCAAMJAMwIAgAECQDMCAIABQkAzAgCAAYJAMwIAgAHCQDMCAIACAkAzAgCAAkJAMwIAgAKCQDMCAIACwUDbmlsCgACJHMJAJADAQUCJGwKAAUkYWNjMAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkA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", "height": 3797435, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: DuHkNHthJQXGooFtnTidsXLNErSvT4Usd8ACsKHXimX7 Next: 7FfDoiT8vcc3TU3MeWUCLh7Gkee4Zp1gjvRF1Yjj2Cj2 Diff:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4-let flashLoanFee = 1000000
5-
6-let axlyAddress = Address(base58'3PHQvxHa4AzNtdKGrAFWwK31kGrETVi9TF5')
4+let axlyAddress = Address(base58'3PLsYkBw7taejV1J3qWPCN2yeyVRu31d5HW')
75
86 let axlyLPs = ["7KZbJrVopwJhkdwbe1eFDBbex4dkY63MxjTNjqXtrzj1", "Btw3G1j4wQgdp49PTxaFkNvn75dQtqGDM7ejQppHnWC1", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "F2AKkA513k5yHEJkLsU6vWxCYYk811GpjLhwEv2WGwZ9", "4CQ5CPGLXLbWBUs2JBjKUaRqF49CmKHkwzvPgSvQpAQV"]
7+
8+let axlyFarmingAddress = Address(base58'3P37f8Y3rNpKzSkHNVZMAStXdpmNBgftHN9')
99
1010 let reserveFundAddress = Address(base58'3P4kBiU4wr2yV1S5gMfu3MdkVvy7kxXHsKe')
1111
293293 func supplyInternal (assetIdStr,assetAmount,address) = if (!(tryGetBoolean("setup_active")))
294294 then throw("market is stopped")
295295 else {
296- let $t088398906 = getActualRate(assetIdStr, "sRate")
297- let sRate = $t088398906._1
298- let ratesRecalcResult = $t088398906._2
296+ let $t089138980 = getActualRate(assetIdStr, "sRate")
297+ let sRate = $t089138980._1
298+ let ratesRecalcResult = $t089138980._2
299299 let amount = fraction(assetAmount, Scale16, sRate, DOWN)
300300 let maxSupply = match getString(("setup_maxSupply_" + assetIdStr)) {
301301 case x: String =>
319319
320320
321321 func borrowInternal (assetIdStr,assetAmount,address,supplyLater) = {
322- let $t01020710274 = getActualRate(assetIdStr, "bRate")
323- let bRate = $t01020710274._1
324- let ratesRecalcResult = $t01020710274._2
322+ let $t01020210269 = getActualRate(assetIdStr, "bRate")
323+ let bRate = $t01020210269._1
324+ let ratesRecalcResult = $t01020210269._2
325325 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
326326 let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",borrowed,") + toString(amount))], nil)
327327 if ((collateralValueInv == collateralValueInv))
408408 else {
409409 let assetIdStr = getAssetString(i.payments[0].assetId)
410410 let assetAmount = i.payments[0].amount
411- supplyInternal(assetIdStr, assetAmount, toString(i.caller))
411+ let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
412+ if ((axlyNotify == axlyNotify))
413+ then supplyInternal(assetIdStr, assetAmount, toString(i.caller))
414+ else throw("Strict value is not equal to itself.")
412415 }
413416
414417
415418
416419 @Callable(i)
417420 func withdraw (assetIdStr,assetAmount) = {
418- let $t01316513232 = getActualRate(assetIdStr, "sRate")
419- let sRate = $t01316513232._1
420- let ratesRecalcResult = $t01316513232._2
421+ let $t01310313170 = getActualRate(assetIdStr, "sRate")
422+ let sRate = $t01310313170._1
423+ let ratesRecalcResult = $t01310313170._2
421424 let amount = fraction(assetAmount, Scale16, sRate, CEILING)
422425 let address = toString(i.caller)
423426 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
424427 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
425428 let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr))
426429 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
427- let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
428- if ((collateralValueInv == collateralValueInv))
430+ let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
431+ if ((axlyNotify == axlyNotify))
429432 then {
430- let collateralValue = match collateralValueInv {
431- case x: Int =>
432- x
433- case _ =>
434- throw("can't get user collateral value")
435- }
436- if (!(tryGetBoolean("setup_active")))
437- then throw("market is stopped")
438- else if ((0 > collateralValue))
439- then throw("you dont have enough collateral for this operation")
440- else if ((amount > (assetSupplied - assetBorrowed)))
441- then throw("this amount is not available on the market")
442- else if ((amount > (userAssetSupplied - userAssetBorrowed)))
443- then throw("this amount is not available for this user")
444- 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)
433+ let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
434+ if ((collateralValueInv == collateralValueInv))
435+ then {
436+ let collateralValue = match collateralValueInv {
437+ case x: Int =>
438+ x
439+ case _ =>
440+ throw("can't get user collateral value")
441+ }
442+ if (!(tryGetBoolean("setup_active")))
443+ then throw("market is stopped")
444+ else if ((0 > collateralValue))
445+ then throw("you dont have enough collateral for this operation")
446+ else if ((amount > (assetSupplied - assetBorrowed)))
447+ then throw("this amount is not available on the market")
448+ else if ((amount > (userAssetSupplied - userAssetBorrowed)))
449+ then throw("this amount is not available for this user")
450+ 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)
451+ }
452+ else throw("Strict value is not equal to itself.")
445453 }
446454 else throw("Strict value is not equal to itself.")
447- }
448-
449-
450-
451-@Callable(i)
452-func borrow (assetIdStr,assetAmount) = {
453- let address = toString(i.caller)
454- let res = borrowInternal(assetIdStr, assetAmount, address, false)
455- let amount = res._2
456- (res._1 ++ [ScriptTransfer(i.caller, amount, getAssetBytes(assetIdStr))])
457455 }
458456
459457
468466 else {
469467 let assetIdStr = getAssetString(i.payments[0].assetId)
470468 let assetAmount = i.payments[0].amount
471- let $t01530615373 = getActualRate(assetIdStr, "bRate")
472- let bRate = $t01530615373._1
473- let ratesRecalcResult = $t01530615373._2
469+ let $t01498715054 = getActualRate(assetIdStr, "bRate")
470+ let bRate = $t01498715054._1
471+ let ratesRecalcResult = $t01498715054._2
474472 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
475473 let address = toString(i.caller)
476474 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
503501 else {
504502 let assetIdStr = getAssetString(i.payments[0].assetId)
505503 let assetAmount = i.payments[0].amount
506- let $t01696217029 = getActualRate(assetIdStr, "bRate")
507- let bRate = $t01696217029._1
508- let ratesRecalcResult = $t01696217029._2
504+ let $t01651716584 = getActualRate(assetIdStr, "bRate")
505+ let bRate = $t01651716584._1
506+ let ratesRecalcResult = $t01651716584._2
509507 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
510508 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
511509 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
514512 let repayAmount = if ((amountLeft >= 0))
515513 then amount
516514 else userAssetBorrowed
515+ let extraAmountResult = if ((amountLeft >= 0))
516+ then nil
517+ else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]
518+ let withdrawResult = if ((i.caller == axlyAddress))
519+ then {
520+ func fold (accum,assetIdStr) = {
521+ let userSupplied = tryGetInteger(((address + "_borrowed_") + assetIdStr))
522+ if ((userSupplied > 0))
523+ then (accum ++ [IntegerEntry(((address + "_supplied_") + assetIdStr), 0), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) - userSupplied))])
524+ else accum
525+ }
526+
527+ let $l = axlyLPs
528+ let $s = size($l)
529+ let $acc0 = nil
530+ func $f0_1 ($a,$i) = if (($i >= $s))
531+ then $a
532+ else fold($a, $l[$i])
533+
534+ func $f0_2 ($a,$i) = if (($i >= $s))
535+ then $a
536+ else throw("List size exceeds 8")
537+
538+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8)
539+ }
540+ else nil
517541 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
518542 then throw("this asset is not supported by the market")
519- else (([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ (if ((amountLeft >= 0))
520- then nil
521- else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]))
543+ else ((([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ extraAmountResult) ++ withdrawResult)
522544 }
523-
524-
525-
526-@Callable(i)
527-func stakeTokenAll (assetIdStr) = if ((i.caller != this))
528- then throw("only for internal smart contract invocations")
529- else {
530- let amount = getBalance(assetIdStr)
531- let inv = invoke(this, "stakeToken", [assetIdStr, amount], nil)
532- if ((inv == inv))
533- then nil
534- else throw("Strict value is not equal to itself.")
535- }
536-
537-
538-
539-@Callable(i)
540-func stakeToken (assetIdStr,amount) = if ((i.caller != this))
541- then throw("only for internal smart contract invocations")
542- else if ((assetIdStr == "DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p"))
543- then {
544- let amountStaked = tryGetInteger("autostake_amount_DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p")
545- let inv = invoke(Address(base58'3PNikM6yp4NqcSU8guxQtmR5onr2D4e8yTJ'), "stake", nil, [AttachedPayment(base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p', amount)])
546- if ((inv == inv))
547- then [IntegerEntry("autostake_amount_DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p", (amountStaked + amount))]
548- else throw("Strict value is not equal to itself.")
549- }
550- else if ((assetIdStr == "8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91"))
551- then {
552- let amountStaked = tryGetInteger("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91")
553- let inv = invoke(Address(base58'3PQTM38wDmAY9vWonK6ha7QL3PAycLz5oPP'), "stake", nil, [AttachedPayment(base58'8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91', amount)])
554- if ((inv == inv))
555- then [IntegerEntry("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91", (amountStaked + amount))]
556- else throw("Strict value is not equal to itself.")
557- }
558- else if ((assetIdStr == "At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL"))
559- then {
560- let amountStaked = tryGetInteger("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL")
561- let inv = invoke(Address(base58'3PBiotFpqjRMkkeFBccnQNUXUopy7KFez5C'), "stake", nil, [AttachedPayment(base58'At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL', amount)])
562- if ((inv == inv))
563- then [IntegerEntry("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", (amountStaked + amount))]
564- else throw("Strict value is not equal to itself.")
565- }
566- else nil
567-
568-
569-
570-@Callable(i)
571-func unstakeToken (assetIdStr,amount) = if ((i.caller != this))
572- then throw("only for internal smart contract invocations")
573- else if ((assetIdStr == "8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91"))
574- then {
575- let amountStaked = tryGetInteger("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91")
576- let inv = invoke(Address(base58'3PQTM38wDmAY9vWonK6ha7QL3PAycLz5oPP'), "unStake", [amount], nil)
577- if ((inv == inv))
578- then [IntegerEntry("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91", (amountStaked - amount))]
579- else throw("Strict value is not equal to itself.")
580- }
581- else if ((assetIdStr == "At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL"))
582- then {
583- let amountStaked = tryGetInteger("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL")
584- let inv = invoke(Address(base58'3PBiotFpqjRMkkeFBccnQNUXUopy7KFez5C'), "unstake", [amount], nil)
585- if ((inv == inv))
586- then {
587- let bal0 = getBalance("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL")
588- if ((bal0 == bal0))
589- then {
590- let inv2 = invoke(Address(base58'3PQrVbTVpqXHqpVKftkNdjy3zZAh4dsRzN6'), "gnsbtRewardsSYSREADONLY", [toString(this)], nil)
591- if ((inv2 == inv2))
592- then {
593- let topupRewards = match inv2 {
594- case x: List[Any] =>
595- let secondEl = x[1]
596- match secondEl {
597- case secondEl: String =>
598- let usdnValue = parseIntValue(split(split(secondEl, "_")[1], ":")[1])
599- let wavesValue = parseIntValue(split(split(secondEl, "_")[0], ":")[1])
600- if (if ((usdnValue != 0))
601- then true
602- else (wavesValue != 0))
603- then {
604- let usdnBal0 = getBalance("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p")
605- if ((usdnBal0 == usdnBal0))
606- then {
607- let wavesBal0 = getBalance("WAVES")
608- if ((wavesBal0 == wavesBal0))
609- then {
610- let inv3 = invoke(Address(base58'3PBiotFpqjRMkkeFBccnQNUXUopy7KFez5C'), "claimRewards", nil, nil)
611- if ((inv3 == inv3))
612- then {
613- let wavesBal1 = getBalance("WAVES")
614- if ((wavesBal1 == wavesBal1))
615- then {
616- let inv4 = invoke(Address(base58'3PLiXyywNThdvf3vVEUxwc7TJTucjZvuegh'), "swap", ["DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p", 0], [AttachedPayment(unit, (wavesBal1 - wavesBal0))])
617- if ((inv4 == inv4))
618- then {
619- let usdnBal1 = getBalance("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p")
620- if ((usdnBal1 == usdnBal1))
621- then {
622- let inv5 = invoke(Address(base58'3P7r93vXHuusageNJVGwzqaz3WMotAu49Yz'), "swap", ["At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", 0], [AttachedPayment(base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p', (usdnBal1 - usdnBal0))])
623- if ((inv5 == inv5))
624- then {
625- let inv6 = invoke(this, "addInterest", ["At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", (getBalance("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL") - bal0)], nil)
626- if ((inv6 == inv6))
627- then 2
628- else throw("Strict value is not equal to itself.")
629- }
630- else throw("Strict value is not equal to itself.")
631- }
632- else throw("Strict value is not equal to itself.")
633- }
634- else throw("Strict value is not equal to itself.")
635- }
636- else throw("Strict value is not equal to itself.")
637- }
638- else throw("Strict value is not equal to itself.")
639- }
640- else throw("Strict value is not equal to itself.")
641- }
642- else throw("Strict value is not equal to itself.")
643- }
644- else 1
645- case _ =>
646- 1
647- }
648- case _ =>
649- 0
650- }
651- if ((topupRewards == topupRewards))
652- then [IntegerEntry("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", (amountStaked - amount))]
653- else throw("Strict value is not equal to itself.")
654- }
655- else throw("Strict value is not equal to itself.")
656- }
657- else throw("Strict value is not equal to itself.")
658- }
659- else throw("Strict value is not equal to itself.")
660- }
661- else nil
662-
663-
664-
665-@Callable(i)
666-func addInterest (assetIdStr,amount) = if ((i.caller != this))
667- then throw("only for self invocation")
668- else {
669- let earned = tryGetInteger(("autostake_lastEarned_" + assetIdStr))
670- let lastHeight = tryGetInteger(("autostake_lastBlock_" + assetIdStr))
671- let cleanAmount = fraction(amount, 80, 100)
672- let stateChanges = if (if ((lastHeight == height))
673- then true
674- else (amount == 0))
675- then nil
676- else [IntegerEntry(("autostake_preLastEarned_" + assetIdStr), earned), IntegerEntry(("autostake_preLastBlock_" + assetIdStr), lastHeight), IntegerEntry(("autostake_lastEarned_" + assetIdStr), (earned + cleanAmount)), IntegerEntry(("autostake_lastBlock_" + assetIdStr), height)]
677- (stateChanges ++ [IntegerEntry((assetIdStr + "_sRate"), (tryGetInteger((assetIdStr + "_sRate")) + fraction(Scale16, cleanAmount, tryGetInteger(("total_supplied_" + assetIdStr)))))])
678- }
679545
680546
681547
731597
732598
733599 @Callable(i)
734-func updateParameter (key,val) = if (if (if ((i.caller != this))
600+func updateParameter (key,val) = if (if ((i.caller != this))
735601 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
736- else false)
737- then (i.caller != Address(base58'3N8ZpuFLVKwxbpVFe1Pcs14hLCrujizpQsa'))
738602 else false)
739603 then throw("admin only")
740604 else [IntegerEntry(key, parseIntValue(val))]
742606
743607
744608 @Callable(i)
745-func updateString (key,val) = if (if (if ((i.caller != this))
609+func updateString (key,val) = if (if ((i.caller != this))
746610 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
747- else false)
748- then (i.caller != Address(base58'3N8ZpuFLVKwxbpVFe1Pcs14hLCrujizpQsa'))
749611 else false)
750612 then throw("admin only")
751613 else [StringEntry(key, val)]
815677
816678
817679 @Callable(i)
818-func reSetup (assetIdStr) = {
819- let lastResetup = tryGetInteger("resetup_lastUpdate")
820- if ((dayBlocks > (height - lastResetup)))
821- then throw("can be updated only once per day")
822- else {
823- let lts = split(tryGetString("setup_lts"), ",")
824- let assets = getMarketAssets()
825- let ur = getUr(assetIdStr)
826- let tempLT = tryGetInteger((("setup_" + assetIdStr) + "_tempLT"))
827- let lt = parseIntValue(assets[value(indexOf(assets, assetIdStr))])
828- if ((ur > 90000000))
829- then [IntegerEntry((("setup_" + assetIdStr) + "_tempLT"), fraction(tempLT, 9975, 10000))]
830- else if (if ((lt > tempLT))
831- then (90000000 > ur)
832- else false)
833- then [IntegerEntry((("setup_" + assetIdStr) + "_tempLT"), fraction(tempLT, 10025, 10000))]
834- else nil
835- }
836- }
837-
838-
839-
840-@Callable(i)
841680 func shutdown (shutdown) = if ((indexOf(shutdownWhitelist, i.caller.bytes) == unit))
842681 then throw("user not in a whitelist")
843682 else [BooleanEntry("setup_active", !(shutdown))]
846685
847686 @Callable(i)
848687 func liquidate (debug,address,assetAmount,sAssetIdStr,bAssetIdStr,routeStr) = if (!(verifyLiquidatorRights(i.caller)))
849- then throw("temporarily available for whitelist only")
688+ then throw("available for whitelist only")
850689 else if (!(tryGetBoolean("setup_active")))
851690 then throw("market is stopped")
852691 else {
853692 let userCollateral = calcUserCollateral(address)
854693 if ((userCollateral == userCollateral))
855694 then {
856- let $t02907829140 = getActualRate(sAssetIdStr, "sRate")
857- let sRate = $t02907829140._1
858- let ratesResult = $t02907829140._2
859- let $t02914529214 = getActualRate(bAssetIdStr, "bRate")
860- let bRate = $t02914529214._1
861- let ratesRecalcResult2 = $t02914529214._2
695+ let $t02265222714 = getActualRate(sAssetIdStr, "sRate")
696+ let sRate = $t02265222714._1
697+ let ratesResult = $t02265222714._2
698+ let $t02271922788 = getActualRate(bAssetIdStr, "bRate")
699+ let bRate = $t02271922788._1
700+ let ratesRecalcResult2 = $t02271922788._2
862701 let sAssetAmount = fraction(assetAmount, Scale16, sRate)
863702 let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
864703 let currentBPositionVal = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
870709 else if ((sAssetAmount > currentSPosition))
871710 then throw("position to liquidate is bigger than user's supply")
872711 else {
873- let balance0Before = getBalance(sAssetIdStr)
874- if ((balance0Before == balance0Before))
712+ let inv = if ((indexOf(axlyLPs, sAssetIdStr) != unit))
875713 then {
876- let balance1Before = getBalance(bAssetIdStr)
877- if ((balance1Before == balance1Before))
714+ let posId = split(address, "_")[1]
715+ let addressId = split(address, "_")[0]
716+ reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, sAssetAmount], nil)
717+ }
718+ else nil
719+ if ((inv == inv))
720+ then {
721+ let balance0Before = getBalance(sAssetIdStr)
722+ if ((balance0Before == balance0Before))
878723 then {
879- let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
880- if ((exchangeInvoke == exchangeInvoke))
724+ let balance1Before = getBalance(bAssetIdStr)
725+ if ((balance1Before == balance1Before))
881726 then {
882- let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
883- if ((asset0Sold == asset0Sold))
727+ let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
728+ if ((exchangeInvoke == exchangeInvoke))
884729 then {
885- let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
886- if ((asset1Bought == asset1Bought))
730+ let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
731+ if ((asset0Sold == asset0Sold))
887732 then {
888- let asset0Price = getTokenPrice(sAssetIdStr)._1
889- let asset0Scale = calcAssetScale(sAssetIdStr)
890- let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
891- let asset1Price = getTokenPrice(bAssetIdStr)._2
892- let asset1Scale = calcAssetScale(bAssetIdStr)
893- let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
894- let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
895- let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
896- let sAssetChange = fraction(asset0Sold, Scale16, sRate)
897- let bAssetChange = fraction(fraction(asset1Bought, Scale16, bRate), (Scale8 - fraction(liquidationProfit, Scale8, asset1Usd)), Scale8)
898- if ((asset0Sold > assetAmount))
899- then throw("more assets exchanged than expected")
900- else if ((0 > liquidationProfit))
901- then throw("price impact is bigger than liquidation penalty")
902- 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))]
733+ let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
734+ if ((asset1Bought == asset1Bought))
735+ then {
736+ let asset0Price = getTokenPrice(sAssetIdStr)._1
737+ let asset0Scale = calcAssetScale(sAssetIdStr)
738+ let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
739+ let asset1Price = getTokenPrice(bAssetIdStr)._2
740+ let asset1Scale = calcAssetScale(bAssetIdStr)
741+ let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
742+ let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
743+ let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
744+ let sAssetChange = fraction(asset0Sold, Scale16, sRate)
745+ let bAssetChange = fraction(fraction(asset1Bought, Scale16, bRate), (Scale8 - fraction(liquidationProfit, Scale8, asset1Usd)), Scale8)
746+ if ((asset0Sold > assetAmount))
747+ then throw("more assets exchanged than expected")
748+ else if ((0 > liquidationProfit))
749+ then throw("price impact is bigger than liquidation penalty")
750+ 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))]
751+ }
752+ else throw("Strict value is not equal to itself.")
903753 }
904754 else throw("Strict value is not equal to itself.")
905755 }
919769
920770 @Callable(i)
921771 func liquidateV2 (debug,address,sAssetIdStr) = if (!(verifyLiquidatorRights(i.caller)))
922- then throw("temporarily available for whitelist only")
772+ then throw("available for whitelist only")
923773 else if (!(tryGetBoolean("setup_active")))
924774 then throw("market is stopped")
925775 else {
934784 let marketAssets = getMarketAssets()
935785 let asset1Num = value(indexOf(marketAssets, bAssetIdStr))
936786 let asset0Num = value(indexOf(marketAssets, sAssetIdStr))
937- let $t03236932431 = getActualRate(bAssetIdStr, "bRate")
938- let bRate = $t03236932431._1
939- let ratesResult = $t03236932431._2
787+ let $t02610926171 = getActualRate(bAssetIdStr, "bRate")
788+ let bRate = $t02610926171._1
789+ let ratesResult = $t02610926171._2
940790 let asset1Price = getTokenPrice(bAssetIdStr)._2
941791 let asset1Scale = calcAssetScale(bAssetIdStr)
942792 let bAmountUsd = fraction(bAssetAmount, asset1Price, asset1Scale)
11511001
11521002
11531003 @Verifier(tx)
1154-func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
1004+func verify () = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'J4QQBsh6FGgZbijQPMDZ7GtNALx2N15K3eHvGhjq43Lt'))
1005+ then true
1006+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS'))
1007+ then true
1008+ else if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'CYCxuKusNeomQkH9wbcEd4rRP1mDYMc7bw8pr82Agi4N'))
1009+ then sigVerify(tx.bodyBytes, tx.proofs[0], base58'3oqguA2LUTZFFjJCJ62UybbXFTLwv88tSJ3oMrNSGhiA')
1010+ else false)
1011+ then true
1012+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'9t2BGHP6EddYu5xvArqDkhHPGbf8ZiYUaWgFcEPtRZhE')
11551013
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4-let flashLoanFee = 1000000
5-
6-let axlyAddress = Address(base58'3PHQvxHa4AzNtdKGrAFWwK31kGrETVi9TF5')
4+let axlyAddress = Address(base58'3PLsYkBw7taejV1J3qWPCN2yeyVRu31d5HW')
75
86 let axlyLPs = ["7KZbJrVopwJhkdwbe1eFDBbex4dkY63MxjTNjqXtrzj1", "Btw3G1j4wQgdp49PTxaFkNvn75dQtqGDM7ejQppHnWC1", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "F2AKkA513k5yHEJkLsU6vWxCYYk811GpjLhwEv2WGwZ9", "4CQ5CPGLXLbWBUs2JBjKUaRqF49CmKHkwzvPgSvQpAQV"]
7+
8+let axlyFarmingAddress = Address(base58'3P37f8Y3rNpKzSkHNVZMAStXdpmNBgftHN9')
99
1010 let reserveFundAddress = Address(base58'3P4kBiU4wr2yV1S5gMfu3MdkVvy7kxXHsKe')
1111
1212 let reserveFund = 20
1313
1414 let aggregatorAddress = Address(base58'3PGFHzVGT4NTigwCKP1NcwoXkodVZwvBuuU')
1515
1616 let shutdownWhitelist = [base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL', base58'3PAxdDSmN758L5SHSGRC5apEtQE2aApZotG', base58'3PJKmXoHJvVeQXjSJdhtkUcFDtdiQqMbUTD', base58'3PQdNxynJy5mche2kxMVc5shXWzK8Gstq3o', base58'3PHbdpaKzz8EiAngGHaFu2hVuNCdsC67qh3', base58'3P6Ksahs71SiKQgQ4qaZuFAVhqncdi2nvJQ']
1717
1818 func verifyLiquidatorRights (address) = !(if (if ((address != Address(base58'3PCqdm1mAoQqR46oZotFanmqb5CLUvrKEo2')))
1919 then (address != Address(base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL'))
2020 else false)
2121 then (address != Address(base58'3PHbdpaKzz8EiAngGHaFu2hVuNCdsC67qh3'))
2222 else false)
2323
2424
2525 func verifySentinelRights (address) = !(if ((address != reserveFundAddress))
2626 then (address != axlyAddress)
2727 else false)
2828
2929
3030 func getRateCurve (assetIdStr) = match assetIdStr {
3131 case _ =>
3232 if (("9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi" == $match0))
3333 then $Tuple4(2000000, 25000000, 80000000, 100000000)
3434 else if (("HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW" == $match0))
3535 then $Tuple4(2000000, 25000000, 80000000, 100000000)
3636 else if (("34N9YcEETLWn93qYQ64EsP1x89tSruJU44RrEMSXXEPJ" == $match0))
3737 then $Tuple4(2000000, 25000000, 80000000, 100000000)
3838 else if (("6XtHjpXbs9RRJP2Sr9GUyVqzACcby9TkThHXnjVC5CDJ" == $match0))
3939 then $Tuple4(2000000, 25000000, 80000000, 100000000)
4040 else if (("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p" == $match0))
4141 then $Tuple4(2000000, 25000000, 80000000, 100000000)
4242 else if (("Ajso6nTTjptu2UHLx6hfSXVtHFtRBJCkKYd5SAyj7zf5" == $match0))
4343 then $Tuple4(2000000, 40000000, 80000000, 150000000)
4444 else if (("HEB8Qaw9xrWpWs8tHsiATYGBWDBtP2S7kcPALrMu43AS" == $match0))
4545 then $Tuple4(0, 20000000, 80000000, 40000000)
4646 else if (("WAVES" == $match0))
4747 then $Tuple4(2000000, 30000000, 80000000, 50000000)
4848 else if (("Atqv59EYzjFGuitKVnMRk6H8FukjoV3ktPorbEys25on" == $match0))
4949 then $Tuple4(0, 20000000, 80000000, 40000000)
5050 else if (("DSbbhLsSTeDg5Lsiufk2Aneh3DjVqJuPr2M9uU1gwy5p" == $match0))
5151 then $Tuple4(0, 20000000, 80000000, 100000000)
5252 else if (("8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91" == $match0))
5353 then $Tuple4(0, 30000000, 80000000, 40000000)
5454 else if (("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL" == $match0))
5555 then $Tuple4(0, 25000000, 80000000, 40000000)
5656 else if (("8LQW8f7P5d5PZM7GtZEBgaqRPGSzS3DfPuiXrURJ4AJS" == $match0))
5757 then $Tuple4(2000000, 30000000, 80000000, 50000000)
5858 else if (("474jTeYx2r2Va35794tCScAXWJG9hU2HcgxzMowaZUnu" == $match0))
5959 then $Tuple4(2000000, 30000000, 80000000, 50000000)
6060 else if (("5UYBPpq4WoU5n4MwpFkgJnW3Fq4B1u3ukpK33ik4QerR" == $match0))
6161 then $Tuple4(2000000, 30000000, 80000000, 50000000)
6262 else if (("2thsACuHmzDMuNezPM32wg9a3BwUzBWDeSKakgz3cw21" == $match0))
6363 then $Tuple4(2000000, 40000000, 80000000, 100000000)
6464 else if (("YiNbofFzC17jEHHCMwrRcpy9MrrjabMMLZxg8g5xmf7" == $match0))
6565 then $Tuple4(2000000, 30000000, 80000000, 80000000)
6666 else $Tuple4(0, 20000000, 80000000, 80000000)
6767 }
6868
6969
7070 let Scale8 = 100000000
7171
7272 let Scale10 = 10000000000
7373
7474 let Scale16 = (Scale8 * Scale8)
7575
7676 let dayBlocks = 1440
7777
7878 func liIntToStr (li) = {
7979 func f (accum,next) = ((accum + toString(next)) + ",")
8080
8181 let $l = li
8282 let $s = size($l)
8383 let $acc0 = ""
8484 func $f0_1 ($a,$i) = if (($i >= $s))
8585 then $a
8686 else f($a, $l[$i])
8787
8888 func $f0_2 ($a,$i) = if (($i >= $s))
8989 then $a
9090 else throw("List size exceeds 12")
9191
9292 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
9393 }
9494
9595
9696 func tryGetInteger (key) = match getInteger(this, key) {
9797 case b: Int =>
9898 b
9999 case _ =>
100100 0
101101 }
102102
103103
104104 func tryGetBoolean (key) = match getBoolean(this, key) {
105105 case b: Boolean =>
106106 b
107107 case _ =>
108108 false
109109 }
110110
111111
112112 func tryGetString (key) = match getString(this, key) {
113113 case b: String =>
114114 b
115115 case _ =>
116116 ""
117117 }
118118
119119
120120 func tryGetBinary (key) = match getBinary(this, key) {
121121 case b: ByteVector =>
122122 b
123123 case _ =>
124124 base58''
125125 }
126126
127127
128128 func getAssetString (assetId) = match assetId {
129129 case b: ByteVector =>
130130 toBase58String(b)
131131 case _ =>
132132 "WAVES"
133133 }
134134
135135
136136 func getAssetBytes (assetIdStr) = if ((assetIdStr == "WAVES"))
137137 then unit
138138 else fromBase58String(assetIdStr)
139139
140140
141141 func getBalance (assetIdStr) = if ((assetIdStr == "WAVES"))
142142 then wavesBalance(this).available
143143 else assetBalance(this, fromBase58String(assetIdStr))
144144
145145
146146 func getMarketAssets () = split(tryGetString("setup_tokens"), ",")
147147
148148
149149 func getAssetsMaxSupply () = {
150150 let s = tryGetString("setup_maxsupply")
151151 if ((s == ""))
152152 then [-1, -1, -1, -1, -1, -1, -1]
153153 else split(s, ",")
154154 }
155155
156156
157157 func getOutdatedUr (assetIdStr) = {
158158 let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), tryGetInteger((assetIdStr + "_sRate")), Scale16)
159159 if ((down == 0))
160160 then 0
161161 else fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), tryGetInteger((assetIdStr + "_bRate")), Scale16), down)
162162 }
163163
164164
165165 func getInterest (assetIdStr) = {
166166 let ur = getOutdatedUr(assetIdStr)
167167 let curve = getRateCurve(assetIdStr)
168168 let rate = (curve._1 + (if ((curve._3 >= ur))
169169 then fraction(ur, curve._2, curve._3)
170170 else (curve._2 + fraction((ur - curve._3), curve._4, (100000000 - curve._3)))))
171171 max([fraction(rate, Scale8, (dayBlocks * 365)), 1])
172172 }
173173
174174
175175 func tokenRatesRecalc (assetIdStr) = if ((indexOf(axlyLPs, assetIdStr) != unit))
176176 then [IntegerEntry((assetIdStr + "_sRate"), Scale16), IntegerEntry((assetIdStr + "_bRate"), Scale16), IntegerEntry("lastRateHeight", height)]
177177 else {
178178 let interest = getInterest(assetIdStr)
179179 let ur = getOutdatedUr(assetIdStr)
180180 let lastRecalcHeight = tryGetInteger("lastRateHeight")
181181 let lastBRate = max([tryGetInteger((assetIdStr + "_bRate")), Scale16])
182182 let newBRate = (lastBRate + ((height - lastRecalcHeight) * interest))
183183 let lastSRate = max([tryGetInteger((assetIdStr + "_sRate")), Scale16])
184184 let newSRate = (lastSRate + ((((height - lastRecalcHeight) * fraction(interest, ur, Scale8)) * (100 - reserveFund)) / 100))
185185 [IntegerEntry((assetIdStr + "_sRate"), newSRate), IntegerEntry((assetIdStr + "_bRate"), newBRate), IntegerEntry("lastRateHeight", height)]
186186 }
187187
188188
189189 func getActualRate (assetIdStr,rateType) = {
190190 func f (accum,token) = {
191191 let recalc = tokenRatesRecalc(token)
192192 $Tuple2(if ((token != assetIdStr))
193193 then accum._1
194194 else if ((rateType == "sRate"))
195195 then recalc[0].value
196196 else recalc[1].value, (accum._2 ++ recalc))
197197 }
198198
199199 let $l = getMarketAssets()
200200 let $s = size($l)
201201 let $acc0 = $Tuple2(0, nil)
202202 func $f0_1 ($a,$i) = if (($i >= $s))
203203 then $a
204204 else f($a, $l[$i])
205205
206206 func $f0_2 ($a,$i) = if (($i >= $s))
207207 then $a
208208 else throw("List size exceeds 12")
209209
210210 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
211211 }
212212
213213
214214 func getUr (assetIdStr) = {
215215 let rates = tokenRatesRecalc(assetIdStr)
216216 let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[0].value, Scale16)
217217 fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[1].value, Scale16), down)
218218 }
219219
220220
221221 func ratesRecalc () = {
222222 func f (accum,token) = (accum ++ tokenRatesRecalc(token))
223223
224224 let $l = getMarketAssets()
225225 let $s = size($l)
226226 let $acc0 = nil
227227 func $f0_1 ($a,$i) = if (($i >= $s))
228228 then $a
229229 else f($a, $l[$i])
230230
231231 func $f0_2 ($a,$i) = if (($i >= $s))
232232 then $a
233233 else throw("List size exceeds 12")
234234
235235 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
236236 }
237237
238238
239239 func getTokenPriceWithRisk (assetIdStr,riskAversity) = if (if ((assetIdStr == "9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi"))
240240 then true
241241 else (assetIdStr == "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW"))
242242 then $Tuple2(1000000, 1000000)
243243 else {
244244 let price = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_twap5B"))
245245 let riskLevel = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_riskLevel"))
246246 if ((indexOf(axlyLPs, assetIdStr) != unit))
247247 then {
248248 let p = {
249249 let @ = reentrantInvoke(axlyAddress, "getShareAssetPriceREADONLY", [assetIdStr], nil)
250250 if ($isInstanceOf(@, "Int"))
251251 then @
252252 else throw(($getType(@) + " couldn't be cast to Int"))
253253 }
254254 if ((p == p))
255255 then $Tuple2(p, p)
256256 else throw("Strict value is not equal to itself.")
257257 }
258258 else if ((riskAversity >= riskLevel))
259259 then $Tuple2(price, price)
260260 else throw((("oracle prices don't match: " + toString(price)) + " is the price, but risk is too high"))
261261 }
262262
263263
264264 func getTokenPrice (assetIdStr) = getTokenPriceWithRisk(assetIdStr, 1)
265265
266266
267267 func calcAssetScale (assetIdStr) = {
268268 let decimals = if ((assetIdStr == "WAVES"))
269269 then 8
270270 else value(assetInfo(fromBase58String(assetIdStr))).decimals
271271 pow(10, 0, decimals, 0, 0, DOWN)
272272 }
273273
274274
275275 func calcUserCollateral (address) = {
276276 let userCollateralInvoke = invoke(this, "getUserCollateral", [false, address, true, ""], nil)
277277 if ((userCollateralInvoke == userCollateralInvoke))
278278 then {
279279 let userCollateralValue = match userCollateralInvoke {
280280 case x: Int =>
281281 x
282282 case _ =>
283283 throw("issue while doing in-dapp invocation")
284284 }
285285 if ((userCollateralValue == userCollateralValue))
286286 then userCollateralValue
287287 else throw("Strict value is not equal to itself.")
288288 }
289289 else throw("Strict value is not equal to itself.")
290290 }
291291
292292
293293 func supplyInternal (assetIdStr,assetAmount,address) = if (!(tryGetBoolean("setup_active")))
294294 then throw("market is stopped")
295295 else {
296- let $t088398906 = getActualRate(assetIdStr, "sRate")
297- let sRate = $t088398906._1
298- let ratesRecalcResult = $t088398906._2
296+ let $t089138980 = getActualRate(assetIdStr, "sRate")
297+ let sRate = $t089138980._1
298+ let ratesRecalcResult = $t089138980._2
299299 let amount = fraction(assetAmount, Scale16, sRate, DOWN)
300300 let maxSupply = match getString(("setup_maxSupply_" + assetIdStr)) {
301301 case x: String =>
302302 parseIntValue(x)
303303 case _ =>
304304 0
305305 }
306306 let assetPrice = getTokenPrice(assetIdStr)
307307 let newTotalSupplied = (tryGetInteger(((address + "_supplied_") + assetIdStr)) + amount)
308308 let rate = getActualRate(assetIdStr, "sRate")._1
309309 let assetScale = calcAssetScale(assetIdStr)
310310 let newTotalSuppliedUsd = fraction(fraction(newTotalSupplied, rate, Scale16), assetPrice._1, assetScale)
311311 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
312312 then throw("this asset is not supported by the market")
313313 else if (if ((maxSupply != 0))
314314 then (newTotalSuppliedUsd > maxSupply)
315315 else false)
316316 then throw("max total supply for this token reached in the pool")
317317 else $Tuple2(([IntegerEntry(((address + "_supplied_") + assetIdStr), newTotalSupplied), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) + amount))] ++ ratesRecalcResult), assetAmount)
318318 }
319319
320320
321321 func borrowInternal (assetIdStr,assetAmount,address,supplyLater) = {
322- let $t01020710274 = getActualRate(assetIdStr, "bRate")
323- let bRate = $t01020710274._1
324- let ratesRecalcResult = $t01020710274._2
322+ let $t01020210269 = getActualRate(assetIdStr, "bRate")
323+ let bRate = $t01020210269._1
324+ let ratesRecalcResult = $t01020210269._2
325325 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
326326 let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",borrowed,") + toString(amount))], nil)
327327 if ((collateralValueInv == collateralValueInv))
328328 then {
329329 let collateralValue = match collateralValueInv {
330330 case x: Int =>
331331 x
332332 case _ =>
333333 throw("can't get user collateral value")
334334 }
335335 if (!(tryGetBoolean("setup_active")))
336336 then throw("market is stopped")
337337 else if ((indexOf(axlyLPs, assetIdStr) != unit))
338338 then throw("this token cannot be borrowed")
339339 else if (if (!(supplyLater))
340340 then (0 > collateralValue)
341341 else false)
342342 then throw("you have to supply more to borrow")
343343 else {
344344 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
345345 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
346346 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
347347 if ((amount > (assetSupplied - assetBorrowed)))
348348 then throw("this amount is not available")
349349 else $Tuple2(([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed + amount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed + amount))] ++ [IntegerEntry((assetIdStr + "_bRate"), bRate)]), assetAmount)
350350 }
351351 }
352352 else throw("Strict value is not equal to itself.")
353353 }
354354
355355
356356 @Callable(i)
357357 func flashPosition (user,sAssetIdStr,bAssetIdStr,bAmount,callbackAddress,callbackFunction,callbackArgs) = if ((i.caller != axlyAddress))
358358 then throw("available only for Axly protocol")
359359 else {
360360 let borrowRes = borrowInternal(bAssetIdStr, bAmount, user, true)
361361 let bAssetId = getAssetBytes(bAssetIdStr)
362362 let callback = reentrantInvoke(addressFromStringValue(callbackAddress), callbackFunction, split(callbackArgs, ","), [AttachedPayment(bAssetId, bAmount)])
363363 if ((callback == callback))
364364 then {
365365 let sAmount = {
366366 let @ = callback
367367 if ($isInstanceOf(@, "Int"))
368368 then @
369369 else throw(($getType(@) + " couldn't be cast to Int"))
370370 }
371371 if ((sAmount == sAmount))
372372 then {
373373 let supplyRes = supplyInternal(sAssetIdStr, sAmount, user)
374374 let collateralValue = {
375375 let @ = invoke(this, "getUserCollateral", [false, user, true, ((sAssetIdStr + ",supplied,") + toString(sAmount))], nil)
376376 if ($isInstanceOf(@, "Int"))
377377 then @
378378 else throw(($getType(@) + " couldn't be cast to Int"))
379379 }
380380 if ((collateralValue == collateralValue))
381381 then {
382382 let borrowValue = {
383383 let @ = invoke(this, "getUserCollateral", [false, user, true, ((bAssetIdStr + ",borrowed,") + toString(bAmount))], nil)
384384 if ($isInstanceOf(@, "Int"))
385385 then @
386386 else throw(($getType(@) + " couldn't be cast to Int"))
387387 }
388388 if ((borrowValue == borrowValue))
389389 then if ((borrowValue > collateralValue))
390390 then throw("not enough collateral provided")
391391 else (borrowRes._1 ++ supplyRes._1)
392392 else throw("Strict value is not equal to itself.")
393393 }
394394 else throw("Strict value is not equal to itself.")
395395 }
396396 else throw("Strict value is not equal to itself.")
397397 }
398398 else throw("Strict value is not equal to itself.")
399399 }
400400
401401
402402
403403 @Callable(i)
404404 func supply () = if (if ((size(i.payments) != 1))
405405 then true
406406 else (i.payments[0].amount == 0))
407407 then throw("1 payment has to be attached")
408408 else {
409409 let assetIdStr = getAssetString(i.payments[0].assetId)
410410 let assetAmount = i.payments[0].amount
411- supplyInternal(assetIdStr, assetAmount, toString(i.caller))
411+ let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
412+ if ((axlyNotify == axlyNotify))
413+ then supplyInternal(assetIdStr, assetAmount, toString(i.caller))
414+ else throw("Strict value is not equal to itself.")
412415 }
413416
414417
415418
416419 @Callable(i)
417420 func withdraw (assetIdStr,assetAmount) = {
418- let $t01316513232 = getActualRate(assetIdStr, "sRate")
419- let sRate = $t01316513232._1
420- let ratesRecalcResult = $t01316513232._2
421+ let $t01310313170 = getActualRate(assetIdStr, "sRate")
422+ let sRate = $t01310313170._1
423+ let ratesRecalcResult = $t01310313170._2
421424 let amount = fraction(assetAmount, Scale16, sRate, CEILING)
422425 let address = toString(i.caller)
423426 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
424427 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
425428 let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr))
426429 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
427- let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
428- if ((collateralValueInv == collateralValueInv))
430+ let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
431+ if ((axlyNotify == axlyNotify))
429432 then {
430- let collateralValue = match collateralValueInv {
431- case x: Int =>
432- x
433- case _ =>
434- throw("can't get user collateral value")
435- }
436- if (!(tryGetBoolean("setup_active")))
437- then throw("market is stopped")
438- else if ((0 > collateralValue))
439- then throw("you dont have enough collateral for this operation")
440- else if ((amount > (assetSupplied - assetBorrowed)))
441- then throw("this amount is not available on the market")
442- else if ((amount > (userAssetSupplied - userAssetBorrowed)))
443- then throw("this amount is not available for this user")
444- 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)
433+ let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
434+ if ((collateralValueInv == collateralValueInv))
435+ then {
436+ let collateralValue = match collateralValueInv {
437+ case x: Int =>
438+ x
439+ case _ =>
440+ throw("can't get user collateral value")
441+ }
442+ if (!(tryGetBoolean("setup_active")))
443+ then throw("market is stopped")
444+ else if ((0 > collateralValue))
445+ then throw("you dont have enough collateral for this operation")
446+ else if ((amount > (assetSupplied - assetBorrowed)))
447+ then throw("this amount is not available on the market")
448+ else if ((amount > (userAssetSupplied - userAssetBorrowed)))
449+ then throw("this amount is not available for this user")
450+ 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)
451+ }
452+ else throw("Strict value is not equal to itself.")
445453 }
446454 else throw("Strict value is not equal to itself.")
447- }
448-
449-
450-
451-@Callable(i)
452-func borrow (assetIdStr,assetAmount) = {
453- let address = toString(i.caller)
454- let res = borrowInternal(assetIdStr, assetAmount, address, false)
455- let amount = res._2
456- (res._1 ++ [ScriptTransfer(i.caller, amount, getAssetBytes(assetIdStr))])
457455 }
458456
459457
460458
461459 @Callable(i)
462460 func repay () = if (!(tryGetBoolean("setup_active")))
463461 then throw("market is stopped")
464462 else if (if ((size(i.payments) != 1))
465463 then true
466464 else (i.payments[0].amount == 0))
467465 then throw("1 payment has to be attached")
468466 else {
469467 let assetIdStr = getAssetString(i.payments[0].assetId)
470468 let assetAmount = i.payments[0].amount
471- let $t01530615373 = getActualRate(assetIdStr, "bRate")
472- let bRate = $t01530615373._1
473- let ratesRecalcResult = $t01530615373._2
469+ let $t01498715054 = getActualRate(assetIdStr, "bRate")
470+ let bRate = $t01498715054._1
471+ let ratesRecalcResult = $t01498715054._2
474472 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
475473 let address = toString(i.caller)
476474 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
477475 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
478476 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
479477 let amountLeft = (userAssetBorrowed - amount)
480478 let repayAmount = if ((amountLeft >= 0))
481479 then amount
482480 else userAssetBorrowed
483481 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
484482 then throw("this asset is not supported by the market")
485483 else (([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ (if ((amountLeft >= 0))
486484 then nil
487485 else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]))
488486 }
489487
490488
491489
492490 @Callable(i)
493491 func repayFor (address) = if (!(tryGetBoolean("setup_active")))
494492 then throw("market is stopped")
495493 else if (!(verifySentinelRights(i.caller)))
496494 then throw("available only for whitelist")
497495 else if ((address == "global"))
498496 then throw("you can't repay for everyone :_)")
499497 else if (if ((size(i.payments) != 1))
500498 then true
501499 else (i.payments[0].amount == 0))
502500 then throw("1 payment has to be attached")
503501 else {
504502 let assetIdStr = getAssetString(i.payments[0].assetId)
505503 let assetAmount = i.payments[0].amount
506- let $t01696217029 = getActualRate(assetIdStr, "bRate")
507- let bRate = $t01696217029._1
508- let ratesRecalcResult = $t01696217029._2
504+ let $t01651716584 = getActualRate(assetIdStr, "bRate")
505+ let bRate = $t01651716584._1
506+ let ratesRecalcResult = $t01651716584._2
509507 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
510508 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
511509 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
512510 let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
513511 let amountLeft = (userAssetBorrowed - amount)
514512 let repayAmount = if ((amountLeft >= 0))
515513 then amount
516514 else userAssetBorrowed
515+ let extraAmountResult = if ((amountLeft >= 0))
516+ then nil
517+ else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]
518+ let withdrawResult = if ((i.caller == axlyAddress))
519+ then {
520+ func fold (accum,assetIdStr) = {
521+ let userSupplied = tryGetInteger(((address + "_borrowed_") + assetIdStr))
522+ if ((userSupplied > 0))
523+ then (accum ++ [IntegerEntry(((address + "_supplied_") + assetIdStr), 0), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) - userSupplied))])
524+ else accum
525+ }
526+
527+ let $l = axlyLPs
528+ let $s = size($l)
529+ let $acc0 = nil
530+ func $f0_1 ($a,$i) = if (($i >= $s))
531+ then $a
532+ else fold($a, $l[$i])
533+
534+ func $f0_2 ($a,$i) = if (($i >= $s))
535+ then $a
536+ else throw("List size exceeds 8")
537+
538+ $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8)
539+ }
540+ else nil
517541 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
518542 then throw("this asset is not supported by the market")
519- else (([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ (if ((amountLeft >= 0))
520- then nil
521- else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]))
543+ else ((([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ extraAmountResult) ++ withdrawResult)
522544 }
523-
524-
525-
526-@Callable(i)
527-func stakeTokenAll (assetIdStr) = if ((i.caller != this))
528- then throw("only for internal smart contract invocations")
529- else {
530- let amount = getBalance(assetIdStr)
531- let inv = invoke(this, "stakeToken", [assetIdStr, amount], nil)
532- if ((inv == inv))
533- then nil
534- else throw("Strict value is not equal to itself.")
535- }
536-
537-
538-
539-@Callable(i)
540-func stakeToken (assetIdStr,amount) = if ((i.caller != this))
541- then throw("only for internal smart contract invocations")
542- else if ((assetIdStr == "DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p"))
543- then {
544- let amountStaked = tryGetInteger("autostake_amount_DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p")
545- let inv = invoke(Address(base58'3PNikM6yp4NqcSU8guxQtmR5onr2D4e8yTJ'), "stake", nil, [AttachedPayment(base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p', amount)])
546- if ((inv == inv))
547- then [IntegerEntry("autostake_amount_DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p", (amountStaked + amount))]
548- else throw("Strict value is not equal to itself.")
549- }
550- else if ((assetIdStr == "8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91"))
551- then {
552- let amountStaked = tryGetInteger("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91")
553- let inv = invoke(Address(base58'3PQTM38wDmAY9vWonK6ha7QL3PAycLz5oPP'), "stake", nil, [AttachedPayment(base58'8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91', amount)])
554- if ((inv == inv))
555- then [IntegerEntry("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91", (amountStaked + amount))]
556- else throw("Strict value is not equal to itself.")
557- }
558- else if ((assetIdStr == "At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL"))
559- then {
560- let amountStaked = tryGetInteger("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL")
561- let inv = invoke(Address(base58'3PBiotFpqjRMkkeFBccnQNUXUopy7KFez5C'), "stake", nil, [AttachedPayment(base58'At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL', amount)])
562- if ((inv == inv))
563- then [IntegerEntry("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", (amountStaked + amount))]
564- else throw("Strict value is not equal to itself.")
565- }
566- else nil
567-
568-
569-
570-@Callable(i)
571-func unstakeToken (assetIdStr,amount) = if ((i.caller != this))
572- then throw("only for internal smart contract invocations")
573- else if ((assetIdStr == "8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91"))
574- then {
575- let amountStaked = tryGetInteger("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91")
576- let inv = invoke(Address(base58'3PQTM38wDmAY9vWonK6ha7QL3PAycLz5oPP'), "unStake", [amount], nil)
577- if ((inv == inv))
578- then [IntegerEntry("autostake_amount_8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91", (amountStaked - amount))]
579- else throw("Strict value is not equal to itself.")
580- }
581- else if ((assetIdStr == "At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL"))
582- then {
583- let amountStaked = tryGetInteger("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL")
584- let inv = invoke(Address(base58'3PBiotFpqjRMkkeFBccnQNUXUopy7KFez5C'), "unstake", [amount], nil)
585- if ((inv == inv))
586- then {
587- let bal0 = getBalance("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL")
588- if ((bal0 == bal0))
589- then {
590- let inv2 = invoke(Address(base58'3PQrVbTVpqXHqpVKftkNdjy3zZAh4dsRzN6'), "gnsbtRewardsSYSREADONLY", [toString(this)], nil)
591- if ((inv2 == inv2))
592- then {
593- let topupRewards = match inv2 {
594- case x: List[Any] =>
595- let secondEl = x[1]
596- match secondEl {
597- case secondEl: String =>
598- let usdnValue = parseIntValue(split(split(secondEl, "_")[1], ":")[1])
599- let wavesValue = parseIntValue(split(split(secondEl, "_")[0], ":")[1])
600- if (if ((usdnValue != 0))
601- then true
602- else (wavesValue != 0))
603- then {
604- let usdnBal0 = getBalance("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p")
605- if ((usdnBal0 == usdnBal0))
606- then {
607- let wavesBal0 = getBalance("WAVES")
608- if ((wavesBal0 == wavesBal0))
609- then {
610- let inv3 = invoke(Address(base58'3PBiotFpqjRMkkeFBccnQNUXUopy7KFez5C'), "claimRewards", nil, nil)
611- if ((inv3 == inv3))
612- then {
613- let wavesBal1 = getBalance("WAVES")
614- if ((wavesBal1 == wavesBal1))
615- then {
616- let inv4 = invoke(Address(base58'3PLiXyywNThdvf3vVEUxwc7TJTucjZvuegh'), "swap", ["DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p", 0], [AttachedPayment(unit, (wavesBal1 - wavesBal0))])
617- if ((inv4 == inv4))
618- then {
619- let usdnBal1 = getBalance("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p")
620- if ((usdnBal1 == usdnBal1))
621- then {
622- let inv5 = invoke(Address(base58'3P7r93vXHuusageNJVGwzqaz3WMotAu49Yz'), "swap", ["At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", 0], [AttachedPayment(base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p', (usdnBal1 - usdnBal0))])
623- if ((inv5 == inv5))
624- then {
625- let inv6 = invoke(this, "addInterest", ["At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", (getBalance("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL") - bal0)], nil)
626- if ((inv6 == inv6))
627- then 2
628- else throw("Strict value is not equal to itself.")
629- }
630- else throw("Strict value is not equal to itself.")
631- }
632- else throw("Strict value is not equal to itself.")
633- }
634- else throw("Strict value is not equal to itself.")
635- }
636- else throw("Strict value is not equal to itself.")
637- }
638- else throw("Strict value is not equal to itself.")
639- }
640- else throw("Strict value is not equal to itself.")
641- }
642- else throw("Strict value is not equal to itself.")
643- }
644- else 1
645- case _ =>
646- 1
647- }
648- case _ =>
649- 0
650- }
651- if ((topupRewards == topupRewards))
652- then [IntegerEntry("autostake_amount_At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", (amountStaked - amount))]
653- else throw("Strict value is not equal to itself.")
654- }
655- else throw("Strict value is not equal to itself.")
656- }
657- else throw("Strict value is not equal to itself.")
658- }
659- else throw("Strict value is not equal to itself.")
660- }
661- else nil
662-
663-
664-
665-@Callable(i)
666-func addInterest (assetIdStr,amount) = if ((i.caller != this))
667- then throw("only for self invocation")
668- else {
669- let earned = tryGetInteger(("autostake_lastEarned_" + assetIdStr))
670- let lastHeight = tryGetInteger(("autostake_lastBlock_" + assetIdStr))
671- let cleanAmount = fraction(amount, 80, 100)
672- let stateChanges = if (if ((lastHeight == height))
673- then true
674- else (amount == 0))
675- then nil
676- else [IntegerEntry(("autostake_preLastEarned_" + assetIdStr), earned), IntegerEntry(("autostake_preLastBlock_" + assetIdStr), lastHeight), IntegerEntry(("autostake_lastEarned_" + assetIdStr), (earned + cleanAmount)), IntegerEntry(("autostake_lastBlock_" + assetIdStr), height)]
677- (stateChanges ++ [IntegerEntry((assetIdStr + "_sRate"), (tryGetInteger((assetIdStr + "_sRate")) + fraction(Scale16, cleanAmount, tryGetInteger(("total_supplied_" + assetIdStr)))))])
678- }
679545
680546
681547
682548 @Callable(i)
683549 func addInterestEXTERNAL () = {
684550 let amount = fraction(i.payments[0].amount, 80, 100)
685551 let assetId = i.payments[0].assetId
686552 let assetIdStr = getAssetString(assetId)
687553 let earned = tryGetInteger(("autostake_lastEarned_" + assetIdStr))
688554 let lastHeight = tryGetInteger(("autostake_lastBlock_" + assetIdStr))
689555 let stateChanges = if (if ((lastHeight == height))
690556 then true
691557 else (amount == 0))
692558 then nil
693559 else [IntegerEntry(("autostake_preLastEarned_" + assetIdStr), earned), IntegerEntry(("autostake_preLastBlock_" + assetIdStr), lastHeight), IntegerEntry(("autostake_lastEarned_" + assetIdStr), (earned + amount)), IntegerEntry(("autostake_lastBlock_" + assetIdStr), height)]
694560 (stateChanges ++ [IntegerEntry((assetIdStr + "_sRate"), (tryGetInteger((assetIdStr + "_sRate")) + fraction(Scale16, amount, tryGetInteger(("total_supplied_" + assetIdStr)))))])
695561 }
696562
697563
698564
699565 @Callable(i)
700566 func preInit (tokens,ltvs,lts,penalties) = {
701567 func f (accum,token) = (accum ++ [IntegerEntry((token + "_bRate"), Scale16), IntegerEntry((token + "_sRate"), Scale16)])
702568
703569 if ((i.caller != this))
704570 then throw("admin only")
705571 else {
706572 let rates = {
707573 let $l = split(tokens, ",")
708574 let $s = size($l)
709575 let $acc0 = nil
710576 func $f0_1 ($a,$i) = if (($i >= $s))
711577 then $a
712578 else f($a, $l[$i])
713579
714580 func $f0_2 ($a,$i) = if (($i >= $s))
715581 then $a
716582 else throw("List size exceeds 12")
717583
718584 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
719585 }
720586 ([StringEntry("setup_tokens", tokens), StringEntry("setup_ltvs", ltvs), StringEntry("setup_lts", lts), StringEntry("setup_penalties", penalties), BooleanEntry("setup_active", true)] ++ rates)
721587 }
722588 }
723589
724590
725591
726592 @Callable(i)
727593 func initNewToken (token,ltv,lt,penalty) = if ((i.caller != this))
728594 then throw("admin only")
729595 else [StringEntry("setup_tokens", ((tryGetString("setup_tokens") + ",") + token)), StringEntry("setup_ltvs", ((tryGetString("setup_ltvs") + ",") + ltv)), StringEntry("setup_lts", ((tryGetString("setup_lts") + ",") + lt)), StringEntry("setup_penalties", ((tryGetString("setup_penalties") + ",") + penalty)), IntegerEntry((token + "_bRate"), Scale16), IntegerEntry((token + "_sRate"), Scale16)]
730596
731597
732598
733599 @Callable(i)
734-func updateParameter (key,val) = if (if (if ((i.caller != this))
600+func updateParameter (key,val) = if (if ((i.caller != this))
735601 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
736- else false)
737- then (i.caller != Address(base58'3N8ZpuFLVKwxbpVFe1Pcs14hLCrujizpQsa'))
738602 else false)
739603 then throw("admin only")
740604 else [IntegerEntry(key, parseIntValue(val))]
741605
742606
743607
744608 @Callable(i)
745-func updateString (key,val) = if (if (if ((i.caller != this))
609+func updateString (key,val) = if (if ((i.caller != this))
746610 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
747- else false)
748- then (i.caller != Address(base58'3N8ZpuFLVKwxbpVFe1Pcs14hLCrujizpQsa'))
749611 else false)
750612 then throw("admin only")
751613 else [StringEntry(key, val)]
752614
753615
754616
755617 @Callable(i)
756618 func claimToReserveFund (debug) = {
757619 let assets = getMarketAssets()
758620 let rates = getActualRate(assets[0], "sRate")._2
759621 let li = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
760622 func f (accum,n) = if ((n >= size(assets)))
761623 then accum
762624 else {
763625 let assetIdStr = assets[n]
764626 let autostakeAmount = tryGetString(("autostake_amount_" + assetIdStr))
765627 let amount = ((((getBalance(assetIdStr) + tryGetInteger(("autostake_amount_" + assetIdStr))) + (if ((autostakeAmount != ""))
766628 then parseIntValue(autostakeAmount)
767629 else 0)) + fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[((n * 3) + 1)].value, Scale16)) - fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[(n * 3)].value, Scale16))
768630 let inv = if ((tryGetInteger(("autostake_amount_" + assetIdStr)) > 0))
769631 then invoke(this, "unstakeToken", [assetIdStr, max([amount, 0])], nil)
770632 else 0
771633 if ((inv == inv))
772634 then (accum ++ [amount])
773635 else throw("Strict value is not equal to itself.")
774636 }
775637
776638 let parameter = {
777639 let $l = li
778640 let $s = size($l)
779641 let $acc0 = nil
780642 func $f0_1 ($a,$i) = if (($i >= $s))
781643 then $a
782644 else f($a, $l[$i])
783645
784646 func $f0_2 ($a,$i) = if (($i >= $s))
785647 then $a
786648 else throw("List size exceeds 12")
787649
788650 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
789651 }
790652 func f2 (accum,n) = if ((n >= size(assets)))
791653 then accum
792654 else {
793655 let assetIdStr = assets[n]
794656 (accum ++ [ScriptTransfer(reserveFundAddress, max([parameter[n], 0]), getAssetBytes(assetIdStr))])
795657 }
796658
797659 if (debug)
798660 then throw(liIntToStr(parameter))
799661 else $Tuple2({
800662 let $l = li
801663 let $s = size($l)
802664 let $acc0 = nil
803665 func $f1_1 ($a,$i) = if (($i >= $s))
804666 then $a
805667 else f2($a, $l[$i])
806668
807669 func $f1_2 ($a,$i) = if (($i >= $s))
808670 then $a
809671 else throw("List size exceeds 12")
810672
811673 $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
812674 }, parameter)
813675 }
814676
815677
816678
817679 @Callable(i)
818-func reSetup (assetIdStr) = {
819- let lastResetup = tryGetInteger("resetup_lastUpdate")
820- if ((dayBlocks > (height - lastResetup)))
821- then throw("can be updated only once per day")
822- else {
823- let lts = split(tryGetString("setup_lts"), ",")
824- let assets = getMarketAssets()
825- let ur = getUr(assetIdStr)
826- let tempLT = tryGetInteger((("setup_" + assetIdStr) + "_tempLT"))
827- let lt = parseIntValue(assets[value(indexOf(assets, assetIdStr))])
828- if ((ur > 90000000))
829- then [IntegerEntry((("setup_" + assetIdStr) + "_tempLT"), fraction(tempLT, 9975, 10000))]
830- else if (if ((lt > tempLT))
831- then (90000000 > ur)
832- else false)
833- then [IntegerEntry((("setup_" + assetIdStr) + "_tempLT"), fraction(tempLT, 10025, 10000))]
834- else nil
835- }
836- }
837-
838-
839-
840-@Callable(i)
841680 func shutdown (shutdown) = if ((indexOf(shutdownWhitelist, i.caller.bytes) == unit))
842681 then throw("user not in a whitelist")
843682 else [BooleanEntry("setup_active", !(shutdown))]
844683
845684
846685
847686 @Callable(i)
848687 func liquidate (debug,address,assetAmount,sAssetIdStr,bAssetIdStr,routeStr) = if (!(verifyLiquidatorRights(i.caller)))
849- then throw("temporarily available for whitelist only")
688+ then throw("available for whitelist only")
850689 else if (!(tryGetBoolean("setup_active")))
851690 then throw("market is stopped")
852691 else {
853692 let userCollateral = calcUserCollateral(address)
854693 if ((userCollateral == userCollateral))
855694 then {
856- let $t02907829140 = getActualRate(sAssetIdStr, "sRate")
857- let sRate = $t02907829140._1
858- let ratesResult = $t02907829140._2
859- let $t02914529214 = getActualRate(bAssetIdStr, "bRate")
860- let bRate = $t02914529214._1
861- let ratesRecalcResult2 = $t02914529214._2
695+ let $t02265222714 = getActualRate(sAssetIdStr, "sRate")
696+ let sRate = $t02265222714._1
697+ let ratesResult = $t02265222714._2
698+ let $t02271922788 = getActualRate(bAssetIdStr, "bRate")
699+ let bRate = $t02271922788._1
700+ let ratesRecalcResult2 = $t02271922788._2
862701 let sAssetAmount = fraction(assetAmount, Scale16, sRate)
863702 let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
864703 let currentBPositionVal = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
865704 let currentBPosition = if ((currentBPositionVal > 0))
866705 then currentBPositionVal
867706 else throw("user has no borrow in this token")
868707 if ((userCollateral > 0))
869708 then throw("user can't be liquidated")
870709 else if ((sAssetAmount > currentSPosition))
871710 then throw("position to liquidate is bigger than user's supply")
872711 else {
873- let balance0Before = getBalance(sAssetIdStr)
874- if ((balance0Before == balance0Before))
712+ let inv = if ((indexOf(axlyLPs, sAssetIdStr) != unit))
875713 then {
876- let balance1Before = getBalance(bAssetIdStr)
877- if ((balance1Before == balance1Before))
714+ let posId = split(address, "_")[1]
715+ let addressId = split(address, "_")[0]
716+ reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, sAssetAmount], nil)
717+ }
718+ else nil
719+ if ((inv == inv))
720+ then {
721+ let balance0Before = getBalance(sAssetIdStr)
722+ if ((balance0Before == balance0Before))
878723 then {
879- let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
880- if ((exchangeInvoke == exchangeInvoke))
724+ let balance1Before = getBalance(bAssetIdStr)
725+ if ((balance1Before == balance1Before))
881726 then {
882- let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
883- if ((asset0Sold == asset0Sold))
727+ let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
728+ if ((exchangeInvoke == exchangeInvoke))
884729 then {
885- let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
886- if ((asset1Bought == asset1Bought))
730+ let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
731+ if ((asset0Sold == asset0Sold))
887732 then {
888- let asset0Price = getTokenPrice(sAssetIdStr)._1
889- let asset0Scale = calcAssetScale(sAssetIdStr)
890- let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
891- let asset1Price = getTokenPrice(bAssetIdStr)._2
892- let asset1Scale = calcAssetScale(bAssetIdStr)
893- let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
894- let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
895- let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
896- let sAssetChange = fraction(asset0Sold, Scale16, sRate)
897- let bAssetChange = fraction(fraction(asset1Bought, Scale16, bRate), (Scale8 - fraction(liquidationProfit, Scale8, asset1Usd)), Scale8)
898- if ((asset0Sold > assetAmount))
899- then throw("more assets exchanged than expected")
900- else if ((0 > liquidationProfit))
901- then throw("price impact is bigger than liquidation penalty")
902- 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))]
733+ let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
734+ if ((asset1Bought == asset1Bought))
735+ then {
736+ let asset0Price = getTokenPrice(sAssetIdStr)._1
737+ let asset0Scale = calcAssetScale(sAssetIdStr)
738+ let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
739+ let asset1Price = getTokenPrice(bAssetIdStr)._2
740+ let asset1Scale = calcAssetScale(bAssetIdStr)
741+ let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
742+ let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
743+ let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
744+ let sAssetChange = fraction(asset0Sold, Scale16, sRate)
745+ let bAssetChange = fraction(fraction(asset1Bought, Scale16, bRate), (Scale8 - fraction(liquidationProfit, Scale8, asset1Usd)), Scale8)
746+ if ((asset0Sold > assetAmount))
747+ then throw("more assets exchanged than expected")
748+ else if ((0 > liquidationProfit))
749+ then throw("price impact is bigger than liquidation penalty")
750+ 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))]
751+ }
752+ else throw("Strict value is not equal to itself.")
903753 }
904754 else throw("Strict value is not equal to itself.")
905755 }
906756 else throw("Strict value is not equal to itself.")
907757 }
908758 else throw("Strict value is not equal to itself.")
909759 }
910760 else throw("Strict value is not equal to itself.")
911761 }
912762 else throw("Strict value is not equal to itself.")
913763 }
914764 }
915765 else throw("Strict value is not equal to itself.")
916766 }
917767
918768
919769
920770 @Callable(i)
921771 func liquidateV2 (debug,address,sAssetIdStr) = if (!(verifyLiquidatorRights(i.caller)))
922- then throw("temporarily available for whitelist only")
772+ then throw("available for whitelist only")
923773 else if (!(tryGetBoolean("setup_active")))
924774 then throw("market is stopped")
925775 else {
926776 let bAssetId = i.payments[0].assetId
927777 let bAssetIdStr = getAssetString(bAssetId)
928778 let bAssetAmount = i.payments[0].amount
929779 let userCollateral = calcUserCollateral(address)
930780 if ((userCollateral == userCollateral))
931781 then if ((userCollateral > 0))
932782 then throw("user can't be liquidated")
933783 else {
934784 let marketAssets = getMarketAssets()
935785 let asset1Num = value(indexOf(marketAssets, bAssetIdStr))
936786 let asset0Num = value(indexOf(marketAssets, sAssetIdStr))
937- let $t03236932431 = getActualRate(bAssetIdStr, "bRate")
938- let bRate = $t03236932431._1
939- let ratesResult = $t03236932431._2
787+ let $t02610926171 = getActualRate(bAssetIdStr, "bRate")
788+ let bRate = $t02610926171._1
789+ let ratesResult = $t02610926171._2
940790 let asset1Price = getTokenPrice(bAssetIdStr)._2
941791 let asset1Scale = calcAssetScale(bAssetIdStr)
942792 let bAmountUsd = fraction(bAssetAmount, asset1Price, asset1Scale)
943793 let penalty = parseIntValue(value(split(tryGetString("setup_penalties"), ",")[asset1Num]))
944794 let asset0Price = getTokenPrice(sAssetIdStr)._1
945795 let asset0Scale = calcAssetScale(sAssetIdStr)
946796 let sAmountUsd = fraction(bAmountUsd, (Scale8 + penalty), Scale8)
947797 let sAssetAmount = fraction(sAmountUsd, asset0Scale, asset0Price)
948798 let bAmount = fraction(bAssetAmount, Scale16, bRate)
949799 let sAmount = fraction(sAssetAmount, Scale16, ratesResult[((asset0Num * 3) + 1)].value)
950800 let inv = if ((indexOf(axlyLPs, sAssetIdStr) != unit))
951801 then {
952802 let posId = split(address, "_")[1]
953803 let addressId = split(address, "_")[0]
954804 reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, sAmount], nil)
955805 }
956806 else nil
957807 if ((inv == inv))
958808 then {
959809 let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
960810 let currentBPositionVal = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
961811 let currentBPosition = if ((currentBPositionVal > 0))
962812 then currentBPositionVal
963813 else throw("user has no borrow in this token")
964814 if ((sAmount > currentSPosition))
965815 then throw("position to liquidate is bigger than user's supply")
966816 else if (debug)
967817 then throw("liquidation will pass")
968818 else ([ScriptTransfer(i.caller, sAssetAmount, getAssetBytes(sAssetIdStr)), IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAmount)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - bAmount)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAmount)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - bAmount))] ++ ratesResult)
969819 }
970820 else throw("Strict value is not equal to itself.")
971821 }
972822 else throw("Strict value is not equal to itself.")
973823 }
974824
975825
976826
977827 @Callable(i)
978828 func getUserCollateral (debug,address,minusBorrowed,afterChange) = {
979829 let assets = getMarketAssets()
980830 let ltvs = split(tryGetString("setup_ltvs"), ",")
981831 let lts = split(tryGetString("setup_lts"), ",")
982832 let rates = getActualRate(assets[0], "sRate")._2
983833 let changeHandler = split(afterChange, ",")
984834 func f (accum,next) = if ((next >= size(assets)))
985835 then accum
986836 else {
987837 let userSupplied = tryGetInteger(((address + "_supplied_") + assets[next]))
988838 let userBorrowed = tryGetInteger(((address + "_borrowed_") + assets[next]))
989839 let needTokenAccounting = if ((afterChange == ""))
990840 then if (if ((userBorrowed != 0))
991841 then true
992842 else (userSupplied != 0))
993843 then true
994844 else false
995845 else true
996846 if (needTokenAccounting)
997847 then {
998848 let assetScale = calcAssetScale(assets[next])
999849 let assetPrice = getTokenPrice(assets[next])
1000850 ((accum + fraction(fraction(fraction((userSupplied + (if (if (if ((afterChange != ""))
1001851 then (changeHandler[0] == assets[next])
1002852 else false)
1003853 then (changeHandler[1] == "supplied")
1004854 else false)
1005855 then parseIntValue(changeHandler[2])
1006856 else 0)), rates[(next * 3)].value, Scale16), parseIntValue(ltvs[next]), Scale8), assetPrice._1, assetScale)) - (if (minusBorrowed)
1007857 then fraction(fraction(fraction((userBorrowed + (if (if (if ((afterChange != ""))
1008858 then (changeHandler[0] == assets[next])
1009859 else false)
1010860 then (changeHandler[1] == "borrowed")
1011861 else false)
1012862 then parseIntValue(changeHandler[2])
1013863 else 0)), rates[((next * 3) + 1)].value, Scale16), Scale8, parseIntValue(lts[next])), assetPrice._2, assetScale)
1014864 else 0))
1015865 }
1016866 else accum
1017867 }
1018868
1019869 let result = {
1020870 let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
1021871 let $s = size($l)
1022872 let $acc0 = 0
1023873 func $f0_1 ($a,$i) = if (($i >= $s))
1024874 then $a
1025875 else f($a, $l[$i])
1026876
1027877 func $f0_2 ($a,$i) = if (($i >= $s))
1028878 then $a
1029879 else throw("List size exceeds 12")
1030880
1031881 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
1032882 }
1033883 if (debug)
1034884 then throw(toString(result))
1035885 else $Tuple2(nil, result)
1036886 }
1037887
1038888
1039889
1040890 @Callable(i)
1041891 func getAssetDebt (debug,address,assetIdStr) = {
1042892 let userBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
1043893 let assetScale = calcAssetScale(assetIdStr)
1044894 let rate = getActualRate(assetIdStr, "bRate")._1
1045895 let result = fraction(userBorrowed, rate, Scale16)
1046896 if (debug)
1047897 then throw(toString(result))
1048898 else $Tuple2(nil, result)
1049899 }
1050900
1051901
1052902
1053903 @Callable(i)
1054904 func getPrices (debug) = {
1055905 let assets = getMarketAssets()
1056906 func f (accum,next) = if ((next >= size(assets)))
1057907 then accum
1058908 else {
1059909 let assetPrice = getTokenPriceWithRisk(assets[next], 3)
1060910 ((((accum + toString(assetPrice._1)) + ",") + toString(assetPrice._2)) + "|")
1061911 }
1062912
1063913 let result = {
1064914 let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
1065915 let $s = size($l)
1066916 let $acc0 = ""
1067917 func $f0_1 ($a,$i) = if (($i >= $s))
1068918 then $a
1069919 else f($a, $l[$i])
1070920
1071921 func $f0_2 ($a,$i) = if (($i >= $s))
1072922 then $a
1073923 else throw("List size exceeds 12")
1074924
1075925 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
1076926 }
1077927 if (debug)
1078928 then throw(result)
1079929 else $Tuple2(nil, result)
1080930 }
1081931
1082932
1083933
1084934 @Callable(i)
1085935 func calculateUtilizationRatio (assetIdStr,debug) = if (debug)
1086936 then throw(toString(getUr(assetIdStr)))
1087937 else $Tuple2(nil, getUr(assetIdStr))
1088938
1089939
1090940
1091941 @Callable(i)
1092942 func calculateOutdatedUR (assetIdStr,debug) = if (debug)
1093943 then throw(toString(getOutdatedUr(assetIdStr)))
1094944 else $Tuple2(nil, getOutdatedUr(assetIdStr))
1095945
1096946
1097947
1098948 @Callable(i)
1099949 func calculateTokenRates (debug) = {
1100950 func f (accum,assetIdStr) = {
1101951 let rates = tokenRatesRecalc(assetIdStr)
1102952 $Tuple2(((((accum._1 + toString(rates[1].value)) + "|") + toString(rates[0].value)) + ","), (accum._2 ++ rates))
1103953 }
1104954
1105955 let parameter = {
1106956 let $l = getMarketAssets()
1107957 let $s = size($l)
1108958 let $acc0 = $Tuple2("", nil)
1109959 func $f0_1 ($a,$i) = if (($i >= $s))
1110960 then $a
1111961 else f($a, $l[$i])
1112962
1113963 func $f0_2 ($a,$i) = if (($i >= $s))
1114964 then $a
1115965 else throw("List size exceeds 12")
1116966
1117967 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
1118968 }
1119969 if (debug)
1120970 then throw(parameter._1)
1121971 else $Tuple2(parameter._2, parameter._1)
1122972 }
1123973
1124974
1125975
1126976 @Callable(i)
1127977 func calculateTokensInterest (debug) = {
1128978 func f (accum,assetIdStr) = {
1129979 let rate = fraction(getInterest(assetIdStr), dayBlocks, Scale8)
1130980 ((accum + toString(rate)) + ",")
1131981 }
1132982
1133983 let parameter = {
1134984 let $l = getMarketAssets()
1135985 let $s = size($l)
1136986 let $acc0 = ""
1137987 func $f0_1 ($a,$i) = if (($i >= $s))
1138988 then $a
1139989 else f($a, $l[$i])
1140990
1141991 func $f0_2 ($a,$i) = if (($i >= $s))
1142992 then $a
1143993 else throw("List size exceeds 12")
1144994
1145995 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12)
1146996 }
1147997 if (debug)
1148998 then throw(parameter)
1149999 else $Tuple2(nil, parameter)
11501000 }
11511001
11521002
11531003 @Verifier(tx)
1154-func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
1004+func verify () = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'J4QQBsh6FGgZbijQPMDZ7GtNALx2N15K3eHvGhjq43Lt'))
1005+ then true
1006+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS'))
1007+ then true
1008+ else if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'CYCxuKusNeomQkH9wbcEd4rRP1mDYMc7bw8pr82Agi4N'))
1009+ then sigVerify(tx.bodyBytes, tx.proofs[0], base58'3oqguA2LUTZFFjJCJ62UybbXFTLwv88tSJ3oMrNSGhiA')
1010+ else false)
1011+ then true
1012+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'9t2BGHP6EddYu5xvArqDkhHPGbf8ZiYUaWgFcEPtRZhE')
11551013

github/deemru/w8io/3ef1775 
137.20 ms