tx · GXeptUFu1Z2v2tiQG2CLhDiLhBuNaMHpozfgmTtBhqUk

3PAd9Aqg3cQSzu26MFYWQYcjSXPVCJtuf5V:  -0.03800000 Waves

2024.02.21 11:00 [4051519] smart account 3PAd9Aqg3cQSzu26MFYWQYcjSXPVCJtuf5V > SELF 0.00000000 Waves

{ "type": 13, "id": "GXeptUFu1Z2v2tiQG2CLhDiLhBuNaMHpozfgmTtBhqUk", "fee": 3800000, "feeAssetId": null, "timestamp": 1708502233911, "version": 2, "chainId": 87, "sender": "3PAd9Aqg3cQSzu26MFYWQYcjSXPVCJtuf5V", "senderPublicKey": "Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS", "proofs": [ "4rQNNzQHYcUS1KJTKpLfetwjQVdV3hWMfA25Eo6HihpihCJsK78AAWYfJg8HRZhCjof3u71bdtBg8LGeSRSvyZLB" ], "script": "base64:BgKFAQgCEgkKBwgICAEICAgSABIECgIIARIECgIICBIAEgYKBAgICAgSBgoECAgICBIDCgEIEgQKAggIEgQKAggIEgMKAQQSAwoBBBIHCgUECAgIARIGCgQECAQIEggKBgQICAgECBIFCgMECAgSAwoBBBIECgIIBBIECgIIBBIDCgEEEgMKAQQjAAdrQXhseUxwAgdfYXhseUxQAAtheGx5QWRkcmVzcwkBB0FkZHJlc3MBARoBV8/Idd4LdvyJm/d7rwlIRGcJKlBHPw9CLQASYXhseUZhcm1pbmdBZGRyZXNzCQEHQWRkcmVzcwEBGgFXDQFZljUUcJnmXN0+OyddE1iluc01XeXCABJyZXNlcnZlRnVuZEFkZHJlc3MJAQdBZGRyZXNzAQEaAVce4ftAMWuB5uxLVhsFLFGhk27fCbE80sEAC3Jlc2VydmVGdW5kABQAEWFnZ3JlZ2F0b3JBZGRyZXNzCQEHQWRkcmVzcwEBGgFXnQyqxhNRqW7LgPdjfcFkeOLvck2oDLrTABFzaHV0ZG93bldoaXRlbGlzdAkAzAgCARoBV9fhI1GOfKuIWtXXKJd12Ih104+uIwWm4QkAzAgCARoBV2MNA4Gmf1tU0hy55Cgq86eANPDYtY3VZQkAzAgCARoBV7PVklRJCZoqEPvG+J8QdqvvPw9ypYkVjAkAzAgCARoBV/j6+gImwFGL7Go7RjNJ0hxxsHWL+aTKmgkAzAgCARoBV6vdqiLzmLkfRdgs1Kfq+RJir4k2alA7cgkAzAgCARoBVzA5HNGkZhg7fn/j2N6i/bhnmP9XTREmnQUDbmlsARZ2ZXJpZnlMaXF1aWRhdG9yUmlnaHRzAQdhZGRyZXNzCQEBIQEDAwMJAQIhPQIFB2FkZHJlc3MJAQdBZGRyZXNzAQEaAVeSeT23spbzYUQF9BURtBNTlIXBLbXH97MJAQIhPQIFB2FkZHJlc3MJAQdBZGRyZXNzAQEaAVd3qs7ReFboqZ5eq5Xvv+nhcVh1XjKacYEHCQECIT0CBQdhZGRyZXNzCQEHQWRkcmVzcwEBGgFX1+EjUY58q4ha1dcol3XYiHXTj64jBabhBwkBAiE9AgUHYWRkcmVzcwkBB0FkZHJlc3MBARoBV6vdqiLzmLkfRdgs1Kfq+RJir4k2alA7cgcBFHZlcmlmeVNlbnRpbmVsUmlnaHRzAQdhZGRyZXNzCQEBIQEDCQECIT0CBQdhZGRyZXNzBRJyZXNlcnZlRnVuZEFkZHJlc3MJAQIhPQIFB2FkZHJlc3MFC2F4bHlBZGRyZXNzBwEMZ2V0UmF0ZUN1cnZlAQphc3NldElkU3RyBAckbWF0Y2gwBQphc3NldElkU3RyAwkAAAICLDl3YzNMWE5BNFRFQnNYeUt0b0xFOW1yYkREN1dNSFh2WHJDalp2YWJMQXNpBQckbWF0Y2gwCQCWCgQAgIl6AMDw9QsAgOiSJgCAwtcvAwkAAAICLEhHZ2FiVHFVUzhXdFZGVUp6Zm1yVERNZ0VjY0p1WkxCUGhGZ1FGeHZuc29XBQckbWF0Y2gwCQCWCgQAgIl6AMDw9QsAgOiSJgCAwtcvAwkAAAICLDM0TjlZY0VFVExXbjkzcVlRNjRFc1AxeDg5dFNydUpVNDRSckVNU1hYRVBKBQckbWF0Y2gwCQCWCgQAgIl6AMDw9QsAgOiSJgCAwtcvAwkAAAICLDZYdEhqcFhiczlSUkpQMlNyOUdVeVZxekFDY2J5OVRrVGhIWG5qVkM1Q0RKBQckbWF0Y2gwCQCWCgQAgIl6AMDw9QsAgOiSJgCAwtcvAwkAAAICLERHMnhGa1BkRHdLVW9Ca3pHQWhRdExwU0d6ZlhMaUNZUEV6ZUtIMkFkMjRwBQckbWF0Y2gwCQCWCgQAgIl6AMDw9QsAgOiSJgCAwtcvAwkAAAICLEFqc282blRUanB0dTJVSEx4NmhmU1hWdEhGdFJCSkNrS1lkNVNBeWo3emY1BQckbWF0Y2gwCQCWCgQAgIl6AIC0iRMAgOiSJgCAo8NHAwkAAAICLEhFQjhRYXc5eHJXcFdzOHRIc2lBVFlHQldEQnRQMlM3a2NQQUxyTXU0M0FTBQckbWF0Y2gwCQCWCgQAAACA2sQJAIDokiYAgLSJEwMJAAACAgVXQVZFUwUHJG1hdGNoMAkAlgoEAICJegCAh6cOAIDokiYAgOHrFwMJAAACAixBdHF2NTlFWXpqRkd1aXRLVm5NUms2SDhGdWtqb1Yza3RQb3JiRXlzMjVvbgUHJG1hdGNoMAkAlgoEAAAAgNrECQCA6JImAIC0iRMDCQAAAgIsRFNiYmhMc1NUZURnNUxzaXVmazJBbmVoM0RqVnFKdVByMk05dVUxZ3d5NXAFByRtYXRjaDAJAJYKBAAAAIDaxAkAgOiSJgCAwtcvAwkAAAICLDh0NERQV1R3UHpwYXRIQTlBa1R4V0FCNDdUSG5ZekJzRG5vWTdmUXFiRzkxBQckbWF0Y2gwCQCWCgQAAACAh6cOAIDokiYAgLSJEwMJAAACAixBdDhENk5GRnBoZUNidktWbmpWb2VMTDg0RW84TlpuNm92TWFueGZMYUZXTAUHJG1hdGNoMAkAlgoEAAAAwPD1CwCA6JImAIC0iRMDCQAAAgIsOExRVzhmN1A1ZDVQWk03R3RaRUJnYXFSUEdTelMzRGZQdWlYclVSSjRBSlMFByRtYXRjaDAJAJYKBACAiXoAgIenDgCA6JImAIDh6xcDCQAAAgIsNDc0alRlWXgycjJWYTM1Nzk0dENTY0FYV0pHOWhVMkhjZ3h6TW93YVpVbnUFByRtYXRjaDAJAJYKBACAiXoAgIenDgCA6JImAIDh6xcDCQAAAgIsNVVZQlBwcTRXb1U1bjRNd3BGa2dKblczRnE0QjF1M3VrcEszM2lrNFFlclIFByRtYXRjaDAJAJYKBACAiXoAgIenDgCA6JImAIDh6xcDCQAAAgIsMnRoc0FDdUhtekRNdU5lelBNMzJ3ZzlhM0J3VXpCV0RlU0tha2d6M2N3MjEFByRtYXRjaDAJAJYKBACAiXoAgLSJEwCA6JImAIDC1y8DCQAAAgIrWWlOYm9mRnpDMTdqRUhIQ013clJjcHk5TXJyamFiTU1MWnhnOGc1eG1mNwUHJG1hdGNoMAkAlgoEAICJegCAh6cOAIDokiYAgOiSJgkAlgoEAAAAgNrECQCA6JImAIDokiYABlNjYWxlOACAwtcvAAdTY2FsZTEwAIDIr6AlAAdTY2FsZTE2CQBoAgUGU2NhbGU4BQZTY2FsZTgACWRheUJsb2NrcwCgCwEKbGlJbnRUb1N0cgECbGkKAQFmAgVhY2N1bQRuZXh0CQCsAgIJAKwCAgUFYWNjdW0JAKQDAQUEbmV4dAIBLAoAAiRsBQJsaQoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEBZgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIVTGlzdCBzaXplIGV4Y2VlZHMgMTAwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAB8AIAAhACIAIwAkACUAJgAnACgAKQAqACsALAAtAC4ALwAwADEAMgAzADQANQA2ADcAOAA5ADoAOwA8AD0APgA/AEAAQQBCAEMARABFAEYARwBIAEkASgBLAEwATQBOAE8AUABRAFIAUwBUAFUAVgBXAFgAWQBaAFsAXABdAF4AXwBgAGEAYgBjAGQBDXRyeUdldEludGVnZXIBA2tleQQHJG1hdGNoMAkAmggCBQR0aGlzBQNrZXkDCQABAgUHJG1hdGNoMAIDSW50BAFiBQckbWF0Y2gwBQFiAAABDXRyeUdldEJvb2xlYW4BA2tleQQHJG1hdGNoMAkAmwgCBQR0aGlzBQNrZXkDCQABAgUHJG1hdGNoMAIHQm9vbGVhbgQBYgUHJG1hdGNoMAUBYgcBDHRyeUdldFN0cmluZwEDa2V5BAckbWF0Y2gwCQCdCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAWIFByRtYXRjaDAFAWICAAEMdHJ5R2V0QmluYXJ5AQNrZXkEByRtYXRjaDAJAJwIAgUEdGhpcwUDa2V5AwkAAQIFByRtYXRjaDACCkJ5dGVWZWN0b3IEAWIFByRtYXRjaDAFAWIBAAEOZ2V0QXNzZXRTdHJpbmcBB2Fzc2V0SWQEByRtYXRjaDAFB2Fzc2V0SWQDCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQBYgUHJG1hdGNoMAkA2AQBBQFiAgVXQVZFUwENZ2V0QXNzZXRCeXRlcwEKYXNzZXRJZFN0cgMJAAACBQphc3NldElkU3RyAgVXQVZFUwUEdW5pdAkA2QQBBQphc3NldElkU3RyAQpnZXRCYWxhbmNlAQphc3NldElkU3RyAwkAAAIFCmFzc2V0SWRTdHICBVdBVkVTCAkA7wcBBQR0aGlzCWF2YWlsYWJsZQkA8AcCBQR0aGlzCQDZBAEFCmFzc2V0SWRTdHIBD2dldE1hcmtldEFzc2V0cwAJALUJAgkBDHRyeUdldFN0cmluZwECDHNldHVwX3Rva2VucwIBLAESZ2V0QXNzZXRzTWF4U3VwcGx5AAQBcwkBDHRyeUdldFN0cmluZwECD3NldHVwX21heHN1cHBseQMJAAACBQFzAgAJAMwIAgD///////////8BCQDMCAIA////////////AQkAzAgCAP///////////wEJAMwIAgD///////////8BCQDMCAIA////////////AQkAzAgCAP///////////wEJAMwIAgD///////////8BBQNuaWwJALUJAgUBcwIBLAENZ2V0T3V0ZGF0ZWRVcgEKYXNzZXRJZFN0cgQEZG93bgkAawMJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfc1JhdGUFB1NjYWxlMTYDCQAAAgUEZG93bgAAAAAJAGsDBQZTY2FsZTgJAGsDCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgUKYXNzZXRJZFN0cgIGX2JSYXRlBQdTY2FsZTE2BQRkb3duAQtnZXRJbnRlcmVzdAEKYXNzZXRJZFN0cgQCdXIJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgQFY3VydmUJAQxnZXRSYXRlQ3VydmUBBQphc3NldElkU3RyBARyYXRlCQBkAggFBWN1cnZlAl8xAwkAZwIIBQVjdXJ2ZQJfMwUCdXIJAGsDBQJ1cggFBWN1cnZlAl8yCAUFY3VydmUCXzMJAGQCCAUFY3VydmUCXzIJAGsDCQBlAgUCdXIIBQVjdXJ2ZQJfMwgFBWN1cnZlAl80CQBlAgCAwtcvCAUFY3VydmUCXzMJAJYDAQkAzAgCCQBrAwUEcmF0ZQUGU2NhbGU4CQBoAgUJZGF5QmxvY2tzAO0CCQDMCAIAAQUDbmlsARB0b2tlblJhdGVzUmVjYWxjAQphc3NldElkU3RyAwkBDXRyeUdldEJvb2xlYW4BCQCsAgIFCmFzc2V0SWRTdHIFB2tBeGx5THAJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIGX3NSYXRlBQdTY2FsZTE2CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICBl9iUmF0ZQUHU2NhbGUxNgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAg9fbGFzdFJhdGVIZWlnaHQFBmhlaWdodAUDbmlsBAhpbnRlcmVzdAkBC2dldEludGVyZXN0AQUKYXNzZXRJZFN0cgQCdXIJAQ1nZXRPdXRkYXRlZFVyAQUKYXNzZXRJZFN0cgQQbGFzdFJlY2FsY0hlaWdodAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIFCmFzc2V0SWRTdHICD19sYXN0UmF0ZUhlaWdodAkBDXRyeUdldEludGVnZXIBAg5sYXN0UmF0ZUhlaWdodAQJbGFzdEJSYXRlCQCWAwEJAMwIAgkBDXRyeUdldEludGVnZXIBCQCsAgIFCmFzc2V0SWRTdHICBl9iUmF0ZQkAzAgCBQdTY2FsZTE2BQNuaWwECG5ld0JSYXRlCQBkAgUJbGFzdEJSYXRlCQBoAgkAZQIFBmhlaWdodAUQbGFzdFJlY2FsY0hlaWdodAUIaW50ZXJlc3QECWxhc3RTUmF0ZQkAlgMBCQDMCAIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfc1JhdGUJAMwIAgUHU2NhbGUxNgUDbmlsBAhuZXdTUmF0ZQkAZAIFCWxhc3RTUmF0ZQkAaQIJAGgCCQBoAgkAZQIFBmhlaWdodAUQbGFzdFJlY2FsY0hlaWdodAkAawMFCGludGVyZXN0BQJ1cgUGU2NhbGU4CQBlAgBkBQtyZXNlcnZlRnVuZABkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICBl9zUmF0ZQUIbmV3U1JhdGUJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIGX2JSYXRlBQhuZXdCUmF0ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAg9fbGFzdFJhdGVIZWlnaHQFBmhlaWdodAUDbmlsAQ1nZXRBY3R1YWxSYXRlAgphc3NldElkU3RyCHJhdGVUeXBlCgEBZgIFYWNjdW0FdG9rZW4EBnJlY2FsYwkBEHRva2VuUmF0ZXNSZWNhbGMBBQV0b2tlbgkA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AJEDAgUFcmF0ZXMAAQV2YWx1ZQUHU2NhbGUxNgUEZG93bgEVZ2V0VG9rZW5QcmljZVdpdGhSaXNrAgphc3NldElkU3RyDHJpc2tBdmVyc2l0eQMDCQAAAgUKYXNzZXRJZFN0cgIsOXdjM0xYTkE0VEVCc1h5S3RvTEU5bXJiREQ3V01IWHZYckNqWnZhYkxBc2kGCQAAAgUKYXNzZXRJZFN0cgIsSEdnYWJUcVVTOFd0VkZVSnpmbXJURE1nRWNjSnVaTEJQaEZnUUZ4dm5zb1cJAJQKAgDAhD0AwIQ9BAVwcmljZQkBEUBleHRyTmF0aXZlKDEwNTApAgkBB0FkZHJlc3MBARoBV0lm0MvzivVXeq38IW9xxVuJiCvQgbOxfwkArAICBQphc3NldElkU3RyAgdfdHdhcDVCBAlyaXNrTGV2ZWwJARFAZXh0ck5hdGl2ZSgxMDUwKQIJAQdBZGRyZXNzAQEaAVdJZtDL84r1V3qt/CFvccVbiYgr0IGzsX8JAKwCAgUKYXNzZXRJZFN0cgIKX3Jpc2tMZXZlbAMJAQ10cnlHZXRCb29sZWFuAQkArAICBQphc3NldElkU3RyBQdrQXhseUxwBAFwCgABQAkA/QcEBQtheGx5QWRkcmVzcwIaZ2V0U2hhcmVBc3NldFByaWNlUkVBRE9OTFkJAMwIAgUKYXNzZXRJZFN0cgUDbmlsBQNuaWwDCQABAgUBQAIDSW50BQFACQACAQkArAICCQADAQUBQAIYIGNvdWxkbid0IGJlIGNhc3QgdG8gSW50AwkAAAIFAXAFAXAJAJQKAgUBcAUBcAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgMJAGcCBQxyaXNrQXZlcnNpdHkFCXJpc2tMZXZlbAkAlAoCBQVwcmljZQUFcHJpY2UJAAIBCQCsAgIJAKwCAgIbb3JhY2xlIHByaWNlcyBkb24ndCBtYXRjaDogCQCkAwEFBXByaWNlAiMgaXMgdGhlIHByaWNlLCBidXQgcmlzayBpcyB0b28gaGlnaAENZ2V0VG9rZW5QcmljZQEKYXNzZXRJZFN0cgkBFWdldFRva2VuUHJpY2VXaXRoUmlzawIFCmFzc2V0SWRTdHIAAQEOY2FsY0Fzc2V0U2NhbGUBCmFzc2V0SWRTdHIECGRlY2ltYWxzAwkAAAIFCmFzc2V0SWRTdHICBVdBVkVTAAgICQEFdmFsdWUBCQDsBwEJANkEAQUKYXNzZXRJZFN0cghkZWNpbWFscwkAbAYACgAABQhkZWNpbWFscwAAAAAFBERPV04BDnN1cHBseUludGVybmFsAwphc3NldElkU3RyC2Fzc2V0QW1vdW50B2FkZHJlc3MDCQEBIQEJAQ10cnlHZXRCb29sZWFuAQIMc2V0dXBfYWN0aXZlCQACAQIRbWFya2V0IGlzIHN0b3BwZWQECyR0MDg1Mzc4NjEyCQEVZ2V0QWN0dWFsUmF0ZU9uZVRva2VuAgUKYXNzZXRJZFN0cgIFc1JhdGUEBXNSYXRlCAULJHQwODUzNzg2MTICXzEEEXJhdGVzUmVjYWxjUmVzdWx0CAULJHQwODUzNzg2MTICXzIEBmFtb3VudAkAbgQFC2Fzc2V0QW1vdW50BQdTY2FsZTE2BQVzUmF0ZQUERE9XTgQJbWF4U3VwcGx5BAckbWF0Y2gwCQCiCAEJAKwCAgIQc2V0dXBfbWF4U3VwcGx5XwUKYXNzZXRJZFN0cgMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAXgFByRtYXRjaDAJAQ1wYXJzZUludFZhbHVlAQUBeAAABAphc3NldFByaWNlCQENZ2V0VG9rZW5QcmljZQEFCmFzc2V0SWRTdHIEEG5ld1RvdGFsU3VwcGxpZWQJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyBQZhbW91bnQEBHJhdGUFBXNSYXRlBAphc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQphc3NldElkU3RyBBNuZXdUb3RhbFN1cHBsaWVkVXNkCQBrAwkAawMFEG5ld1RvdGFsU3VwcGxpZWQFBHJhdGUFB1NjYWxlMTYIBQphc3NldFByaWNlAl8xBQphc3NldFNjYWxlAwkAAAIJALMJAgkBDHRyeUdldFN0cmluZwECDHNldHVwX3Rva2VucwUKYXNzZXRJZFN0cgUEdW5pdAkAAgECKXRoaXMgYXNzZXQgaXMgbm90IHN1cHBvcnRlZCBieSB0aGUgbWFya2V0AwMJAQIhPQIFCW1heFN1cHBseQAACQBmAgUTbmV3VG90YWxTdXBwbGllZFVzZAUJbWF4U3VwcGx5BwkAAgECM21heCB0b3RhbCBzdXBwbHkgZm9yIHRoaXMgdG9rZW4gcmVhY2hlZCBpbiB0aGUgcG9vbAQIYXNzZXROdW0JAQV2YWx1ZQEJAM8IAgkBD2dldE1hcmtldEFzc2V0cwAFCmFzc2V0SWRTdHIJAJQKAgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgUQbmV3VG90YWxTdXBwbGllZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIJAGQCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBQZhbW91bnQFA25pbAkAzAgCCQCRAwIFEXJhdGVzUmVjYWxjUmVzdWx0AAAJAMwIAgkAkQMCBRFyYXRlc1JlY2FsY1Jlc3VsdAABCQDMCAIJAJEDAgURcmF0ZXNSZWNhbGNSZXN1bHQAAgUDbmlsCQCUCgIFC2Fzc2V0QW1vdW50BRNuZXdUb3RhbFN1cHBsaWVkVXNkAQ5ib3Jyb3dJbnRlcm5hbAMKYXNzZXRJZFN0cgthc3NldEFtb3VudAdhZGRyZXNzBAskdDA5OTIwOTk5NQkBFWdldEFjdHVhbFJhdGVPbmVUb2tlbgIFCmFzc2V0SWRTdHICBWJSYXRlBAViUmF0ZQgFCyR0MDk5MjA5OTk1Al8xBBFyYXRlc1JlY2FsY1Jlc3VsdAgFCyR0MDk5MjA5OTk1Al8yBAZhbW91bnQJAG4EBQthc3NldEFtb3VudAUHU2NhbGUxNgUFYlJhdGUFB0NFSUxJTkcDCQEBIQEJAQ10cnlHZXRCb29sZWFuAQIMc2V0dXBfYWN0aXZlCQACAQIRbWFya2V0IGlzIHN0b3BwZWQDCQENdHJ5R2V0Qm9vbGVhbgEJAKwCAgUKYXNzZXRJZFN0cgUHa0F4bHlMcAkAAgECHXRoaXMgdG9rZW4gY2Fubm90IGJlIGJvcnJvd2VkBA1hc3NldFN1cHBsaWVkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyBA1hc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQphc3NldElkU3RyBBF1c2VyQXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQKYXNzZXRTY2FsZQkBDmNhbGNBc3NldFNjYWxlAQUKYXNzZXRJZFN0cgQKYXNzZXRQcmljZQkBDWdldFRva2VuUHJpY2UBBQphc3NldElkU3RyBA5hc3NldEFtb3VudFVzZAkAawMFBmFtb3VudAgFCmFzc2V0UHJpY2UCXzEFCmFzc2V0U2NhbGUDCQBmAgUGYW1vdW50CQBlAgUNYXNzZXRTdXBwbGllZAUNYXNzZXRCb3Jyb3dlZAkAAgECHHRoaXMgYW1vdW50IGlzIG5vdCBhdmFpbGFibGUECGFzc2V0TnVtCQEFdmFsdWUBCQDPCAIJAQ9nZXRNYXJrZXRBc3NldHMABQphc3NldElkU3RyCQCUCgIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIJAGQCBRF1c2VyQXNzZXRCb3Jyb3dlZAUGYW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgkAZAIFDWFzc2V0Qm9ycm93ZWQFBmFtb3VudAUDbmlsCQDMCAIJAJEDAgURcmF0ZXNSZWNhbGNSZXN1bHQAAAkAzAgCCQCRAwIFEXJhdGVzUmVjYWxjUmVzdWx0AAEJAMwIAgkAkQMCBRFyYXRlc1JlY2FsY1Jlc3VsdAACBQNuaWwJAJQKAgULYXNzZXRBbW91bnQFDmFzc2V0QW1vdW50VXNkFQFpAQ1mbGFzaFBvc2l0aW9uBwR1c2VyC3NBc3NldElkU3RyC2JBc3NldElkU3RyB2JBbW91bnQPY2FsbGJhY2tBZGRyZXNzEGNhbGxiYWNrRnVuY3Rpb24MY2FsbGJhY2tBcmdzAwkBAiE9AggFAWkGY2FsbGVyBQtheGx5QWRkcmVzcwkAAgECIGF2YWlsYWJsZSBvbmx5IGZvciBBeGx5IHByb3RvY29sBAlib3Jyb3dSZXMJAQ5ib3Jyb3dJbnRlcm5hbAMFC2JBc3NldElkU3RyBQdiQW1vdW50BQR1c2VyBAhiQXNzZXRJZAkBDWdldEFzc2V0Qnl0ZXMBBQtiQXNzZXRJZFN0cgQIY2FsbGJhY2sJAP0HBAkBEUBleHRyTmF0aXZlKDEwNjIpAQUPY2FsbGJhY2tBZGRyZXNzBRBjYWxsYmFja0Z1bmN0aW9uCQC1CQIFDGNhbGxiYWNrQXJncwIBLAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgUIYkFzc2V0SWQFB2JBbW91bnQFA25pbAMJAAACBQhjYWxsYmFjawUIY2FsbGJhY2sEB3NBbW91bnQKAAFABQhjYWxsYmFjawMJAAECBQFAAgNJbnQFAUAJAAIBCQCsAgIJAAMBBQFAAhggY291bGRuJ3QgYmUgY2FzdCB0byBJbnQDCQAAAgUHc0Ftb3VudAUHc0Ftb3VudAQJc3VwcGx5UmVzCQEOc3VwcGx5SW50ZXJuYWwDBQtzQXNzZXRJZFN0cgUHc0Ftb3VudAUEdXNlcgMJAGYCCAgFCWJvcnJvd1JlcwJfMgJfMggIBQlzdXBwbHlSZXMCXzICXzIJAAIBAh5ub3QgZW5vdWdoIGNvbGxhdGVyYWwgcHJvdmlkZWQJAM4IAggFCWJvcnJvd1JlcwJfMQgFCXN1cHBseVJlcwJfMQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQZzdXBwbHkAAwMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQYJAAACCAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAAACQACAQIcMSBwYXltZW50IGhhcyB0byBiZSBhdHRhY2hlZAQKYXNzZXRJZFN0cgkBDmdldEFzc2V0U3RyaW5nAQgJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBAthc3NldEFtb3VudAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQDCQENdHJ5R2V0Qm9vbGVhbgEJAKwCAgUKYXNzZXRJZFN0cgUHa0F4bHlMcAkAAgECE1lvdSBjYW4ndCBzdXBwbHkgbHAECmF4bHlOb3RpZnkJAPwHBAUSYXhseUZhcm1pbmdBZGRyZXNzAgpsZW5kQWN0aW9uCQDMCAIJAKUIAQgFAWkGY2FsbGVyCQDMCAIFCmFzc2V0SWRTdHIFA25pbAUDbmlsAwkAAAIFCmF4bHlOb3RpZnkFCmF4bHlOb3RpZnkJAQ5zdXBwbHlJbnRlcm5hbAMFCmFzc2V0SWRTdHIFC2Fzc2V0QW1vdW50CQClCAEIBQFpBmNhbGxlcgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQh3aXRoZHJhdwIKYXNzZXRJZFN0cgthc3NldEFtb3VudAQNJHQwMTI2MDQxMjY3OQkBFWdldEFjdHVhbFJhdGVPbmVUb2tlbgIFCmFzc2V0SWRTdHICBXNSYXRlBAVzUmF0ZQgFDSR0MDEyNjA0MTI2NzkCXzEEEXJhdGVzUmVjYWxjUmVzdWx0CAUNJHQwMTI2MDQxMjY3OQJfMgQGYW1vdW50CQBuBAULYXNzZXRBbW91bnQFB1NjYWxlMTYFBXNSYXRlBQdDRUlMSU5HBAdhZGRyZXNzCQClCAEIBQFpBmNhbGxlcgQNYXNzZXRTdXBwbGllZAkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgQNYXNzZXRCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQRdXNlckFzc2V0U3VwcGxpZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FCmFzc2V0SWRTdHIECmF4bHlOb3RpZnkJAPwHBAUSYXhseUZhcm1pbmdBZGRyZXNzAgpsZW5kQWN0aW9uCQDMCAIJAKUIAQgFAWkGY2FsbGVyCQDMCAIFCmFzc2V0SWRTdHIFA25pbAUDbmlsAwkAAAIFCmF4bHlOb3RpZnkFCmF4bHlOb3RpZnkEEmNvbGxhdGVyYWxWYWx1ZUludgkA/AcEBQR0aGlzAhNnZXRVc2VyQ29sbGF0ZXJhbFYyCQDMCAIHCQDMCAIFB2FkZHJlc3MJAMwIAgUKYXNzZXRJZFN0cgkAzAgCAgAJAMwIAgcJAMwIAgkArAICAglzdXBwbGllZCwJAKQDAQkBAS0BBQZhbW91bnQFA25pbAUDbmlsAwkAAAIFEmNvbGxhdGVyYWxWYWx1ZUludgUSY29sbGF0ZXJhbFZhbHVlSW52BA9jb2xsYXRlcmFsVmFsdWUEByRtYXRjaDAFEmNvbGxhdGVyYWxWYWx1ZUludgMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAh9jYW4ndCBnZXQgdXNlciBjb2xsYXRlcmFsIHZhbHVlAwkBASEBCQENdHJ5R2V0Qm9vbGVhbgECDHNldHVwX2FjdGl2ZQkAAgECEW1hcmtldCBpcyBzdG9wcGVkAwkAZgIAAAUPY29sbGF0ZXJhbFZhbHVlCQACAQIyeW91IGRvbnQgaGF2ZSBlbm91Z2ggY29sbGF0ZXJhbCBmb3IgdGhpcyBvcGVyYXRpb24DCQBmAgUGYW1vdW50CQBlAgUNYXNzZXRTdXBwbGllZAUNYXNzZXRCb3Jyb3dlZAkAAgECKnRoaXMgYW1vdW50IGlzIG5vdCBhdmFpbGFibGUgb24gdGhlIG1hcmtldAMJAGYCBQZhbW91bnQFEXVzZXJBc3NldFN1cHBsaWVkCQACAQIqdGhpcyBhbW91bnQgaXMgbm90IGF2YWlsYWJsZSBvbiB0aGUgbWFya2V0CQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfc3VwcGxpZWRfBQphc3NldElkU3RyCQBlAgkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgUGYW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX3N1cHBsaWVkXwUKYXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIFBmFtb3VudAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIFC2Fzc2V0QW1vdW50CQENZ2V0QXNzZXRCeXRlcwEFCmFzc2V0SWRTdHIFA25pbAURcmF0ZXNSZWNhbGNSZXN1bHQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEIcmVwYXlGb3ICB2FkZHJlc3MEbHBJZAMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAMJAQEhAQkBFHZlcmlmeVNlbnRpbmVsUmlnaHRzAQgFAWkGY2FsbGVyCQACAQIcYXZhaWxhYmxlIG9ubHkgZm9yIHdoaXRlbGlzdAMJAAACBQdhZGRyZXNzAgZnbG9iYWwJAAIBAiB5b3UgY2FuJ3QgcmVwYXkgZm9yIGV2ZXJ5b25lIDpfKQMDCQECIT0CCQCQAwEIBQFpCHBheW1lbnRzAAEGCQAAAggJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQAAAkAAgECHDEgcGF5bWVudCBoYXMgdG8gYmUgYXR0YWNoZWQDCQEBIQEJAQ10cnlHZXRCb29sZWFuAQkArAICBQRscElkBQdrQXhseUxwCQACAQILV3JvbmcgbHAgaWQECmFzc2V0SWRTdHIJAQ5nZXRBc3NldFN0cmluZwEICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQLYXNzZXRBbW91bnQICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BA0kdDAxNDcxOTE0Nzk0CQEVZ2V0QWN0dWFsUmF0ZU9uZVRva2VuAgUKYXNzZXRJZFN0cgIFYlJhdGUEBWJSYXRlCAUNJHQwMTQ3MTkxNDc5NAJfMQQRcmF0ZXNSZWNhbGNSZXN1bHQIBQ0kdDAxNDcxOTE0Nzk0Al8yBAZhbW91bnQJAG4EBQthc3NldEFtb3VudAUHU2NhbGUxNgUFYlJhdGUFB0NFSUxJTkcEDWFzc2V0U3VwcGxpZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIEDWFzc2V0Qm9ycm93ZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9ib3Jyb3dlZF8FCmFzc2V0SWRTdHIEEXVzZXJBc3NldEJvcnJvd2VkCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyBAphbW91bnRMZWZ0CQBlAgURdXNlckFzc2V0Qm9ycm93ZWQFBmFtb3VudAQLcmVwYXlBbW91bnQDCQBnAgUKYW1vdW50TGVmdAAABQZhbW91bnQFEXVzZXJBc3NldEJvcnJvd2VkBBFleHRyYUFtb3VudFJlc3VsdAMJAGcCBQphbW91bnRMZWZ0AAAFA25pbAkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIJAQEtAQUKYW1vdW50TGVmdAgJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBQNuaWwEDndpdGhkcmF3UmVzdWx0AwkAAAIIBQFpBmNhbGxlcgULYXhseUFkZHJlc3MEDHVzZXJTdXBwbGllZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUEbHBJZAMJAGYCBQx1c2VyU3VwcGxpZWQAAAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FBGxwSWQAAAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg90b3RhbF9zdXBwbGllZF8FBGxwSWQJAGUCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQRscElkBQx1c2VyU3VwcGxpZWQFA25pbAUDbmlsBQNuaWwDCQAAAgkAswkCCQEMdHJ5R2V0U3RyaW5nAQIMc2V0dXBfdG9rZW5zBQphc3NldElkU3RyBQR1bml0CQACAQIpdGhpcyBhc3NldCBpcyBub3Qgc3VwcG9ydGVkIGJ5IHRoZSBtYXJrZXQJAM4IAgkAzggCCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQphc3NldElkU3RyCQBlAgURdXNlckFzc2V0Qm9ycm93ZWQFC3JlcGF5QW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgkAZQIFDWFzc2V0Qm9ycm93ZWQFC3JlcGF5QW1vdW50BQNuaWwFEXJhdGVzUmVjYWxjUmVzdWx0BRFleHRyYUFtb3VudFJlc3VsdAUOd2l0aGRyYXdSZXN1bHQBaQETYWRkSW50ZXJlc3RFWFRFUk5BTAAEBmFtb3VudAkAawMICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50AFAAZAQHYXNzZXRJZAgJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBAphc3NldElkU3RyCQEOZ2V0QXNzZXRTdHJpbmcBBQdhc3NldElkBAZlYXJuZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAhVhdXRvc3Rha2VfbGFzdEVhcm5lZF8FCmFzc2V0SWRTdHIECmxhc3RIZWlnaHQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAhRhdXRvc3Rha2VfbGFzdEJsb2NrXwUKYXNzZXRJZFN0cgQMc3RhdGVDaGFuZ2VzAwMJAAACBQpsYXN0SGVpZ2h0BQZoZWlnaHQGCQAAAgUGYW1vdW50AAAFA25pbAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAhhhdXRvc3Rha2VfcHJlTGFzdEVhcm5lZF8FCmFzc2V0SWRTdHIFBmVhcm5lZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAhdhdXRvc3Rha2VfcHJlTGFzdEJsb2NrXwUKYXNzZXRJZFN0cgUKbGFzdEhlaWdodAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAhVhdXRvc3Rha2VfbGFzdEVhcm5lZF8FCmFzc2V0SWRTdHIJAGQCBQZlYXJuZWQFBmFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAhRhdXRvc3Rha2VfbGFzdEJsb2NrXwUKYXNzZXRJZFN0cgUGaGVpZ2h0BQNuaWwJAM4IAgUMc3RhdGVDaGFuZ2VzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICBl9zUmF0ZQkAZAIJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgZfc1JhdGUJAGsDBQdTY2FsZTE2BQZhbW91bnQJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FCmFzc2V0SWRTdHIFA25pbAFpAQdwcmVJbml0BAZ0b2tlbnMEbHR2cwNsdHMJcGVuYWx0aWVzCgEBZgIFYWNjdW0FdG9rZW4JAM4IAgUFYWNjdW0JAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUFdG9rZW4CBl9iUmF0ZQUHU2NhbGUxNgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQV0b2tlbgIGX3NSYXRlBQdTY2FsZTE2BQNuaWwDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAgphZG1pbiBvbmx5BAVyYXRlcwoAAiRsCQC1CQIFBnRva2VucwIBLAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEBZgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTIJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgALAAwJAM4IAgkAzAgCCQELU3RyaW5nRW50cnkCAgxzZXR1cF90b2tlbnMFBnRva2VucwkAzAgCCQELU3RyaW5nRW50cnkCAgpzZXR1cF9sdHZzBQRsdHZzCQDMCAIJAQtTdHJpbmdFbnRyeQICCXNldHVwX2x0cwUDbHRzCQDMCAIJAQtTdHJpbmdFbnRyeQICD3NldHVwX3BlbmFsdGllcwUJcGVuYWx0aWVzCQDMCAIJAQxCb29sZWFuRW50cnkCAgxzZXR1cF9hY3RpdmUGBQNuaWwFBXJhdGVzAWkBDGluaXROZXdUb2tlbgQFdG9rZW4DbHR2Amx0B3BlbmFsdHkDAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQECIT0CCAUBaQZjYWxsZXIJAQdBZGRyZXNzAQEaAVcUeViFwPoaZ+cUNI0M98TwbUXt9hz5PUEHCQACAQIKYWRtaW4gb25seQkAzAgCCQELU3RyaW5nRW50cnkCAgxzZXR1cF90b2tlbnMJAKwCAgkArAICCQEMdHJ5R2V0U3RyaW5nAQIMc2V0dXBfdG9rZW5zAgEsBQV0b2tlbgkAzAgCCQELU3RyaW5nRW50cnkCAgpzZXR1cF9sdHZzCQCsAgIJAKwCAgkBDHRyeUdldFN0cmluZwECCnNldHVwX2x0dnMCASwFA2x0dgkAzAgCCQELU3RyaW5nRW50cnkCAglzZXR1cF9sdHMJAKwCAgkArAICCQEMdHJ5R2V0U3RyaW5nAQIJc2V0dXBfbHRzAgEsBQJsdAkAzAgCCQELU3RyaW5nRW50cnkCAg9zZXR1cF9wZW5hbHRpZXMJAKwCAgkArAICCQEMdHJ5R2V0U3RyaW5nAQIPc2V0dXBfcGVuYWx0aWVzAgEsBQdwZW5hbHR5CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFBXRva2VuAgZfYlJhdGUFB1NjYWxlMTYJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUFdG9rZW4CBl9zUmF0ZQUHU2NhbGUxNgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQV0b2tlbgIPX2xhc3RSYXRlSGVpZ2h0BQZoZWlnaHQFA25pbAFpAQ5pbml0TmV3TFBUb2tlbgEFdG9rZW4DAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQECIT0CCAUBaQZjYWxsZXIJAQdBZGRyZXNzAQEaAVcUeViFwPoaZ+cUNI0M98TwbUXt9hz5PUEHCQACAQIKYWRtaW4gb25seQkAzAgCCQEMQm9vbGVhbkVudHJ5AgkArAICBQV0b2tlbgIHX2F4bHlMcAYFA25pbAFpAQ91cGRhdGVQYXJhbWV0ZXICA2tleQN2YWwDAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQECIT0CCAUBaQZjYWxsZXIJAQdBZGRyZXNzAQEaAVcUeViFwPoaZ+cUNI0M98TwbUXt9hz5PUEHCQACAQIKYWRtaW4gb25seQkAzAgCCQEMSW50ZWdlckVudHJ5AgUDa2V5CQENcGFyc2VJbnRWYWx1ZQEFA3ZhbAUDbmlsAWkBDHVwZGF0ZVN0cmluZwIDa2V5A3ZhbAMDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAQIhPQIIBQFpBmNhbGxlcgkBB0FkZHJlc3MBARoBVxR5WIXA+hpn5xQ0jQz3xPBtRe32HPk9QQcJAAIBAgphZG1pbiBvbmx5CQDMCAIJAQtTdHJpbmdFbnRyeQIFA2tleQUDdmFsBQNuaWwBaQESY2xhaW1Ub1Jlc2VydmVGdW5kAQVkZWJ1ZwQGYXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAQFcmF0ZXMICQENZ2V0QWN0dWFsUmF0ZQIJAJEDAgUGYXNzZXRzAAACBXNSYXRlAl8yCgEBZgIBYQphc3NldElkU3RyBA0kdDAxOTI3MzE5MzIzBQFhBAVhY2N1bQgFDSR0MDE5MjczMTkzMjMCXzEEAW4IBQ0kdDAxOTI3MzE5MzIzAl8yBA9hdXRvc3Rha2VBbW91bnQJAQx0cnlHZXRTdHJpbmcBCQCsAgICEWF1dG9zdGFrZV9hbW91bnRfBQphc3NldElkU3RyBAZhbW91bnQJAGUCCQBkAgkAZAIJAGQCCQEKZ2V0QmFsYW5jZQEFCmFzc2V0SWRTdHIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAhFhdXRvc3Rha2VfYW1vdW50XwUKYXNzZXRJZFN0cgMJAQIhPQIFD2F1dG9zdGFrZUFtb3VudAIACQENcGFyc2VJbnRWYWx1ZQEFD2F1dG9zdGFrZUFtb3VudAAACQBrAwkBDXRyeUdldEludGVnZXIBCQCsAgICD3RvdGFsX2JvcnJvd2VkXwUKYXNzZXRJZFN0cggJAJEDAgUFcmF0ZXMJAGQCCQBoAgUBbgADAAEFdmFsdWUFB1NjYWxlMTYJAGsDCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQphc3NldElkU3RyCAkAkQMCBQVyYXRlcwkAaAIFAW4AAwV2YWx1ZQUHU2NhbGUxNgQDaW52AwkAZgIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAhFhdXRvc3Rha2VfYW1vdW50XwUKYXNzZXRJZFN0cgAACQD8BwQFBHRoaXMCDHVuc3Rha2VUb2tlbgkAzAgCBQphc3NldElkU3RyCQDMCAIJAJYDAQkAzAgCBQZhbW91bnQJAMwIAgAABQNuaWwFA25pbAUDbmlsAAADCQAAAgUDaW52BQNpbnYJAJQKAgkAzggCBQVhY2N1bQkAzAgCBQZhbW91bnQFA25p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ZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgALAAwADQAOAA8AEAARABIAEwAUABUAFgAXABgAGQAaABsAHAAdAB4AHwAgACEAIgAjACQAJQAmACcAKAApACoAKwAsAC0ALgAvADAAMQAyADMANAA1ADYANwA4ADkAOgA7ADwAPQA+AD8AQABBAEIAQwBEAEUARgBHAEgASQBKAEsATABNAE4ATwBQAFEAUgBTAFQAVQBWAFcAWABZAFoAWwBcAF0AXgBfAGAAYQBiAGMAZAUJcGFyYW1ldGVyAl8xAWkBCHNodXRkb3duAQhzaHV0ZG93bgMJAAACCQDPCAIFEXNodXRkb3duV2hpdGVsaXN0CAgFAWkGY2FsbGVyBWJ5dGVzBQR1bml0CQACAQIXdXNlciBub3QgaW4gYSB3aGl0ZWxpc3QJAMwIAgkBDEJvb2xlYW5FbnRyeQICDHNldHVwX2FjdGl2ZQkBASEBBQhzaHV0ZG93bgUDbmlsAWkBC2xpcXVpZGF0ZVYyBQVkZWJ1ZwdhZGRyZXNzC3NBc3NldElkU3RyC2JBc3NldElkU3RyEGxpcXVpZGF0ZVBlcmNlbnQDCQEBIQEJARZ2ZXJpZnlMaXF1aWRhdG9yUmlnaHRzAQgFAWkGY2FsbGVyCQACAQIcYXZhaWxhYmxlIGZvciB3aGl0ZWxpc3Qgb25seQMJAQEhAQkBDXRyeUdldEJvb2xlYW4BAgxzZXR1cF9hY3RpdmUJAAIBAhFtYXJrZXQgaXMgc3RvcHBlZAQSY29sbGF0ZXJhbFZhbHVlSW52CQD8BwQFBHRoaXMCE2dldFVzZXJDb2xsYXRlcmFsVjIJAMwIAgcJAMwIAgUHYWRkcmVzcwkAzAgCBQtzQXNzZXRJZFN0cgkAzAgCBQtiQXNzZXRJZFN0cgkAzAgCBgkAzAgCAgAFA25pbAUDbmlsAwkAAAIFEmNvbGxhdGVyYWxWYWx1ZUludgUSY29sbGF0ZXJhbFZhbHVlSW52BA51c2VyQ29sbGF0ZXJhbAQHJG1hdGNoMAUSY29sbGF0ZXJhbFZhbHVlSW52AwkAAQIFByRtYXRjaDACA0ludAQBeAUHJG1hdGNoMAUBeAkAAgECH2Nhbid0IGdldCB1c2VyIGNvbGxhdGVyYWwgdmFsdWUEEGN1cnJlbnRTUG9zaXRpb24JAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FC3NBc3NldElkU3RyBBBjdXJyZW50QlBvc2l0aW9uCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgkArAICBQdhZGRyZXNzAgpfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgMJAQEhAQkBDXRyeUdldEJvb2xlYW4BCQCsAgIFC3NBc3NldElkU3RyBQdrQXhseUxwCQACAQIOd3JvbmcgbHAgYXNzZXQDCQBnAgAABRBjdXJyZW50QlBvc2l0aW9uCQACAQIgdXNlciBoYXMgbm8gYm9ycm93IGluIHRoaXMgdG9rZW4DAwkAZgIFDnVzZXJDb2xsYXRlcmFsAAAGCQAAAgUQY3VycmVudEJQb3NpdGlvbgAACQACAQIYdXNlciBjYW4ndCBiZSBsaXF1aWRhdGVkBAhiQXNzZXRJZAkBDWdldEFzc2V0Qnl0ZXMBBQtiQXNzZXRJZFN0cgQMbWFya2V0QXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAQJYXNzZXQxTnVtCQEFdmFsdWUBCQDPCAIFDG1hcmtldEFzc2V0cwULYkFzc2V0SWRTdHIECWFzc2V0ME51bQkBBXZhbHVlAQkAzwgCBQxtYXJrZXRBc3NldHMFC3NBc3NldElkU3RyBAdwZW5hbHR5CQENcGFyc2VJbnRWYWx1ZQEJAQV2YWx1ZQEJAJEDAgkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIPc2V0dXBfcGVuYWx0aWVzAgEsBQlhc3NldDFOdW0EDSR0MDIxODkxMjE5NjEJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQtiQXNzZXRJZFN0cgIFYlJhdGUEBWJSYXRlCAUNJHQwMjE4OTEyMTk2MQJfMQQLcmF0ZXNSZXN1bHQIBQ0kdDAyMTg5MTIxOTYxAl8yBAdiQW1vdW50CQBrAwUQY3VycmVudEJQb3NpdGlvbgUFYlJhdGUFB1NjYWxlMTYEB2xBbW91bnQJAGsDBQdiQW1vdW50BRBsaXF1aWRhdGVQZXJjZW50AGQEEmxBbW91bnRXaXRoUGVuYWx0eQkAawMFB2xBbW91bnQJAGQCBQZTY2FsZTgFB3BlbmFsdHkFBlNjYWxlOAQIbGJBbW91bnQJAGsDBQdsQW1vdW50BQdTY2FsZTE2BQViUmF0ZQQFcG9zSWQJAJEDAgkAtQkCBQdhZGRyZXNzAgFfAAEECWFkZHJlc3NJZAkAkQMCCQC1CQIFB2FkZHJlc3MCAV8AAAQJc0Ftb3VudExwCgABQAkA/QcEBQtheGx5QWRkcmVzcwIJbGlxdWlkYXRlCQDMCAIFCWFkZHJlc3NJZAkAzAgCBQVwb3NJZAkAzAgCBRJsQW1vdW50V2l0aFBlbmFsdHkFA25pbAUDbmlsAwkAAQIFAUACA0ludAUBQAkAAgEJAKwCAgkAAwEFAUACGCBjb3VsZG4ndCBiZSBjYXN0IHRvIEludAMJAAACBQlzQW1vdW50THAFCXNBbW91bnRMcAQHc0Ftb3VudAkAawMFCXNBbW91bnRMcAUHU2NhbGUxNggJAJEDAgULcmF0ZXNSZXN1bHQAAQV2YWx1ZQMFBWRlYnVnCQACAQIVbGlxdWlkYXRpb24gd2lsbCBwYXNzCQDOCAIJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCQBlAgUSbEFtb3VudFdpdGhQZW5hbHR5BQdsQW1vdW50BQhiQXNzZXRJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FC3NBc3NldElkU3RyCQBlAgUQY3VycmVudFNQb3NpdGlvbgUHc0Ftb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FC2JBc3NldElkU3RyCQBlAgUQY3VycmVudEJQb3NpdGlvbgUIbGJBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQtzQXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FC3NBc3NldElkU3RyBQdzQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX2JvcnJvd2VkXwULYkFzc2V0SWRTdHIJAGUCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgUIbGJBbW91bnQFA25pbAULcmF0ZXNSZXN1bHQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQERZ2V0VXNlckNvbGxhdGVyYWwEBWRlYnVnB2FkZHJlc3MNbWludXNCb3Jyb3dlZAthZnRlckNoYW5nZQQGYXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAQEbHR2cwkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIKc2V0dXBfbHR2cwIBLAQDbHRzCQC1CQIJAQx0cnlHZXRTdHJpbmcBAglzZXR1cF9sdHMCASwEBXJhdGVzCAkBDWdldEFjdHVhbFJhdGUCCQCRAwIFBmFzc2V0cwAAAgVzUmF0ZQJfMgQNY2hhbmdlSGFuZGxlcgkAtQkCBQthZnRlckNoYW5nZQIBLAoBAWYCBWFjY3VtBG5leHQDCQBnAgUEbmV4dAkAkAMBBQZhc3NldHMFBWFjY3VtBAx1c2VyU3VwcGxpZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8JAJEDAgUGYXNzZXRzBQRuZXh0BAx1c2VyQm9ycm93ZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8JAJEDAgUGYXNzZXRzBQRuZXh0BBNuZWVkVG9rZW5BY2NvdW50aW5nAwkAAAIFC2FmdGVyQ2hhbmdlAgADAwkBAiE9AgUMdXNlckJvcnJvd2VkAAAGCQECIT0CBQx1c2VyU3VwcGxpZWQAAAYHBgMFE25lZWRUb2tlbkFjY291bnRpbmcECmFzc2V0U2NhbGUJAQ5jYWxjQXNzZXRTY2FsZQEJAJEDAgUGYXNzZXRzBQRuZXh0BAphc3NldFByaWNlCQENZ2V0VG9rZW5QcmljZQEJAJEDAgUGYXNzZXRzBQRuZXh0CQBlAgkAZAIFBWFjY3VtCQBrAwkAawMJAGsDCQBkAgUMdXNlclN1cHBsaWVkAwMDCQECIT0CBQthZnRlckNoYW5nZQIACQAAAgkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAAJAJEDAgUGYXNzZXRzBQRuZXh0BwkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgABAghzdXBwbGllZAcJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAIAAAgJAJEDAgUFcmF0ZXMJAGgCBQRuZXh0AAMFdmFsdWUFB1NjYWxlMTYJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQRsdHZzBQRuZXh0BQZTY2FsZTgIBQphc3NldFByaWNlAl8xBQphc3NldFNjYWxlAwUNbWludXNCb3Jyb3dlZAkAawMJAGsDCQBrAwkAZAIFDHVzZXJCb3Jyb3dlZAMDAwkBAiE9AgULYWZ0ZXJDaGFuZ2UCAAkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgAACQCRAwIFBmFzc2V0cwUEbmV4dAcJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAQIIYm9ycm93ZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgACAAAICQCRAwIFBXJhdGVzCQBkAgkAaAIFBG5leHQAAwABBXZhbHVlBQdTY2FsZTE2BQZTY2FsZTgJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQNsdHMFBG5leHQIBQphc3NldFByaWNlAl8yBQphc3NldFNjYWxlAAAFBWFjY3VtBAZyZXN1bHQKAAIkbAkAzAgCAAAJAMwIAgABCQDMCAIAAgkAzAgCAAMJAMwIAgAECQDMCAIABQkAzAgCAAYJAMwIAgAHCQDMCAIACAkAzAgCAAkJAMwIAgAKCQDMCAIACwUDbmlsCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAMFBWRlYnVnCQACAQkApAMBBQZyZXN1bHQJAJQKAgUDbmlsBQZyZXN1bHQBaQETZ2V0VXNlckNvbGxhdGVyYWxWMgYFZGVidWcHYWRkcmVzcwhzQXNzZXRJZAhiQXNzZXRJZA1taW51c0JvcnJvd2VkC2FmdGVyQ2hhbmdlBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMABAZzSW5kZXgJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAzwgCBQZhc3NldHMFCHNBc3NldElkAg5Xcm9uZyBzQXNzZXRJZAQEbHR2cwkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIKc2V0dXBfbHR2cwIBLAQDbHRzCQC1CQIJAQx0cnlHZXRTdHJpbmcBAglzZXR1cF9sdHMCASwECnNBc3NldFJhdGUJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQhzQXNzZXRJZAIFc1JhdGUEDWNoYW5nZUhhbmRsZXIJALUJAgULYWZ0ZXJDaGFuZ2UCASwEDHVzZXJTdXBwbGllZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUIc0Fzc2V0SWQEDHVzZXJCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUIYkFzc2V0SWQEDnN1cHBsaWVkQ2hhbmdlAwMJAQIhPQIFC2FmdGVyQ2hhbmdlAgAJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAAIIc3VwcGxpZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgABAAAEC3NBc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQhzQXNzZXRJZAQLc0Fzc2V0UHJpY2UJAQ1nZXRUb2tlblByaWNlAQUIc0Fzc2V0SWQEBnJlc3VsdAkAZQIJAGsDCQBrAwkAawMJAGQCBQx1c2VyU3VwcGxpZWQFDnN1cHBsaWVkQ2hhbmdlCAUKc0Fzc2V0UmF0ZQJfMQUHU2NhbGUxNgkBDXBhcnNlSW50VmFsdWUBCQCRAwIFBGx0dnMFBnNJbmRleAUGU2NhbGU4CAULc0Fzc2V0UHJpY2UCXzEFC3NBc3NldFNjYWxlAwUNbWludXNCb3Jyb3dlZAQGYkluZGV4CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAM8IAgUGYXNzZXRzBQhiQXNzZXRJZAIOV3JvbmcgYkFzc2V0SWQECmJBc3NldFJhdGUJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQhiQXNzZXRJZAIFYlJhdGUEDmJvcnJvd2VkQ2hhbmdlAwMJAQIhPQIFC2FmdGVyQ2hhbmdlAgAJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAAIIYm9ycm93ZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgABAAAEC2JBc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQhiQXNzZXRJZAQLYkFzc2V0UHJpY2UJAQ1nZXRUb2tlblByaWNlAQUIYkFzc2V0SWQJAGsDCQBrAwkAawMJAGQCBQx1c2VyQm9ycm93ZWQFDmJvcnJvd2VkQ2hhbmdlCAUKYkFzc2V0UmF0ZQJfMQUHU2NhbGUxNgUGU2NhbGU4CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUDbHRzBQZiSW5kZXgIBQtiQXNzZXRQcmljZQJfMgULYkFzc2V0U2NhbGUAAAMFBWRlYnVnCQACAQkApAMBBQZyZXN1bHQJAJQKAgUDbmlsBQZyZXN1bHQBaQEMZ2V0QXNzZXREZWJ0AwVkZWJ1ZwdhZGRyZXNzCmFzc2V0SWRTdHIEDHVzZXJCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQKYXNzZXRTY2FsZQkBDmNhbGNBc3NldFNjYWxlAQUKYXNzZXRJZFN0cgQEcmF0ZQgJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQphc3NldElkU3RyAgViUmF0ZQJfMQQGcmVzdWx0CQBrAwUMdXNlckJvcnJvd2VkBQRyYXRlBQdTY2FsZTE2AwUFZGVidWcJAAIBCQCkAwEFBnJlc3VsdAkAlAoCBQNuaWwFBnJlc3VsdAFpAQlnZXRQcmljZXMBBWRlYnVnBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMACgEBZgIFYWNjdW0KYXNzZXRJZFN0cgQKYXNzZXRQcmljZQkBFWdldFRva2VuUHJpY2VXaXRoUmlzawIFCmFz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BSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQAFQAWABcAGAAZABoAGwAcAB0AHgAfACAAIQAiACMAJAAlACYAJwAoACkAKgArACwALQAuAC8AMAAxADIAMwA0ADUANgA3ADgAOQA6ADsAPAA9AD4APwBAAEEAQgBDAEQARQBGAEcASABJAEoASwBMAE0ATgBPAFAAUQBSAFMAVABVAFYAVwBYAFkAWgBbAFwAXQBeAF8AYABhAGIAYwBkAwUFZGVidWcJAAIBBQlwYXJhbWV0ZXIJAJQKAgUDbmlsBQlwYXJhbWV0ZXIBAnR4AQZ2ZXJpZnkAAwMDAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAEg/XXg3oez6DuanjNsQ3pWRVALIo9nVy/USV6BgJV9XQEGCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAAASCQpp9/Me/Ta3E4MktVjg3Z8DeDRSAIRX8xeT7HVuKEewYJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAABIKtvFry/bTtkfOGq1hidoH1VvCftr7DwxV9WUchHp74PBgkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAEgsqjd/2RuCiE8iQa/ePj4wRFXp96pg+zd+fUDwp6q02UDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwABASAptySGTmbKD+EhOKgxR3AX8vMsoOhmHOMDi6IUwkAuOwYJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAEBIIPvcoztDFRvh/+rabZ7oQ4UQnF2/9sodDexlJRAcugNB9HO5JY=", "height": 4051519, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 3P1kaoW1vZHheFai94Xa5CfRaQe3iAFRFpw443svSAqE Next: 2BUa59N1xymvWugFTNK63gGktUHJTXA2R8eyakw4THF9 Diff:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4+let kAxlyLp = "_axlyLP"
5+
46 let axlyAddress = Address(base58'3PLsYkBw7taejV1J3qWPCN2yeyVRu31d5HW')
5-
6-let axlyLPs = ["7KZbJrVopwJhkdwbe1eFDBbex4dkY63MxjTNjqXtrzj1", "Btw3G1j4wQgdp49PTxaFkNvn75dQtqGDM7ejQppHnWC1", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "F2AKkA513k5yHEJkLsU6vWxCYYk811GpjLhwEv2WGwZ9", "4CQ5CPGLXLbWBUs2JBjKUaRqF49CmKHkwzvPgSvQpAQV", "6iMB6LKSrgv9waEvEnN6Ydyx7dfxPnGcTw8318WVm5bR"]
77
88 let axlyFarmingAddress = Address(base58'3P37f8Y3rNpKzSkHNVZMAStXdpmNBgftHN9')
99
8989
9090 func $f0_2 ($a,$i) = if (($i >= $s))
9191 then $a
92- else throw("List size exceeds 12")
92+ else throw("List size exceeds 100")
9393
94- $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)
94+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
9595 }
9696
9797
174174 }
175175
176176
177-func tokenRatesRecalc (assetIdStr) = if ((indexOf(axlyLPs, assetIdStr) != unit))
178- then [IntegerEntry((assetIdStr + "_sRate"), Scale16), IntegerEntry((assetIdStr + "_bRate"), Scale16), IntegerEntry("lastRateHeight", height)]
177+func tokenRatesRecalc (assetIdStr) = if (tryGetBoolean((assetIdStr + kAxlyLp)))
178+ then [IntegerEntry((assetIdStr + "_sRate"), Scale16), IntegerEntry((assetIdStr + "_bRate"), Scale16), IntegerEntry((assetIdStr + "_lastRateHeight"), height)]
179179 else {
180180 let interest = getInterest(assetIdStr)
181181 let ur = getOutdatedUr(assetIdStr)
182- let lastRecalcHeight = tryGetInteger("lastRateHeight")
182+ let lastRecalcHeight = valueOrElse(getInteger(this, (assetIdStr + "_lastRateHeight")), tryGetInteger("lastRateHeight"))
183183 let lastBRate = max([tryGetInteger((assetIdStr + "_bRate")), Scale16])
184184 let newBRate = (lastBRate + ((height - lastRecalcHeight) * interest))
185185 let lastSRate = max([tryGetInteger((assetIdStr + "_sRate")), Scale16])
186186 let newSRate = (lastSRate + ((((height - lastRecalcHeight) * fraction(interest, ur, Scale8)) * (100 - reserveFund)) / 100))
187-[IntegerEntry((assetIdStr + "_sRate"), newSRate), IntegerEntry((assetIdStr + "_bRate"), newBRate), IntegerEntry("lastRateHeight", height)]
187+[IntegerEntry((assetIdStr + "_sRate"), newSRate), IntegerEntry((assetIdStr + "_bRate"), newBRate), IntegerEntry((assetIdStr + "_lastRateHeight"), height)]
188188 }
189189
190190
207207
208208 func $f0_2 ($a,$i) = if (($i >= $s))
209209 then $a
210- else throw("List size exceeds 12")
210+ else throw("List size exceeds 100")
211211
212- $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)
212+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
213+ }
214+
215+
216+func getActualRateOneToken (assetIdStr,rateType) = {
217+ let recalc = tokenRatesRecalc(assetIdStr)
218+ $Tuple2(if ((rateType == "sRate"))
219+ then recalc[0].value
220+ else recalc[1].value, recalc)
213221 }
214222
215223
220228 }
221229
222230
223-func ratesRecalc () = {
224- func f (accum,token) = (accum ++ tokenRatesRecalc(token))
225-
226- let $l = getMarketAssets()
227- let $s = size($l)
228- let $acc0 = nil
229- func $f0_1 ($a,$i) = if (($i >= $s))
230- then $a
231- else f($a, $l[$i])
232-
233- func $f0_2 ($a,$i) = if (($i >= $s))
234- then $a
235- else throw("List size exceeds 12")
236-
237- $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)
238- }
239-
240-
241231 func getTokenPriceWithRisk (assetIdStr,riskAversity) = if (if ((assetIdStr == "9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi"))
242232 then true
243233 else (assetIdStr == "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW"))
245235 else {
246236 let price = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_twap5B"))
247237 let riskLevel = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_riskLevel"))
248- if ((indexOf(axlyLPs, assetIdStr) != unit))
238+ if (tryGetBoolean((assetIdStr + kAxlyLp)))
249239 then {
250240 let p = {
251241 let @ = reentrantInvoke(axlyAddress, "getShareAssetPriceREADONLY", [assetIdStr], nil)
274264 }
275265
276266
277-func calcUserCollateral (address) = {
278- let userCollateralInvoke = invoke(this, "getUserCollateral", [false, address, true, ""], nil)
279- if ((userCollateralInvoke == userCollateralInvoke))
280- then {
281- let userCollateralValue = match userCollateralInvoke {
282- case x: Int =>
283- x
284- case _ =>
285- throw("issue while doing in-dapp invocation")
286- }
287- if ((userCollateralValue == userCollateralValue))
288- then userCollateralValue
289- else throw("Strict value is not equal to itself.")
290- }
291- else throw("Strict value is not equal to itself.")
292- }
293-
294-
295267 func supplyInternal (assetIdStr,assetAmount,address) = if (!(tryGetBoolean("setup_active")))
296268 then throw("market is stopped")
297269 else {
298- let $t090489115 = getActualRate(assetIdStr, "sRate")
299- let sRate = $t090489115._1
300- let ratesRecalcResult = $t090489115._2
270+ let $t085378612 = getActualRateOneToken(assetIdStr, "sRate")
271+ let sRate = $t085378612._1
272+ let ratesRecalcResult = $t085378612._2
301273 let amount = fraction(assetAmount, Scale16, sRate, DOWN)
302274 let maxSupply = match getString(("setup_maxSupply_" + assetIdStr)) {
303275 case x: String =>
307279 }
308280 let assetPrice = getTokenPrice(assetIdStr)
309281 let newTotalSupplied = (tryGetInteger(((address + "_supplied_") + assetIdStr)) + amount)
310- let rate = getActualRate(assetIdStr, "sRate")._1
282+ let rate = sRate
311283 let assetScale = calcAssetScale(assetIdStr)
312284 let newTotalSuppliedUsd = fraction(fraction(newTotalSupplied, rate, Scale16), assetPrice._1, assetScale)
313285 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
316288 then (newTotalSuppliedUsd > maxSupply)
317289 else false)
318290 then throw("max total supply for this token reached in the pool")
319- else $Tuple2(([IntegerEntry(((address + "_supplied_") + assetIdStr), newTotalSupplied), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) + amount))] ++ ratesRecalcResult), assetAmount)
291+ else {
292+ let assetNum = value(indexOf(getMarketAssets(), assetIdStr))
293+ $Tuple2(([IntegerEntry(((address + "_supplied_") + assetIdStr), newTotalSupplied), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) + amount))] ++ [ratesRecalcResult[0], ratesRecalcResult[1], ratesRecalcResult[2]]), $Tuple2(assetAmount, newTotalSuppliedUsd))
294+ }
320295 }
321296
322297
323-func borrowInternal (assetIdStr,assetAmount,address,supplyLater) = {
324- let $t01033710404 = getActualRate(assetIdStr, "bRate")
325- let bRate = $t01033710404._1
326- let ratesRecalcResult = $t01033710404._2
298+func borrowInternal (assetIdStr,assetAmount,address) = {
299+ let $t099209995 = getActualRateOneToken(assetIdStr, "bRate")
300+ let bRate = $t099209995._1
301+ let ratesRecalcResult = $t099209995._2
327302 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
328- let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",borrowed,") + toString(amount))], nil)
329- if ((collateralValueInv == collateralValueInv))
330- then {
331- let collateralValue = match collateralValueInv {
332- case x: Int =>
333- x
334- case _ =>
335- throw("can't get user collateral value")
336- }
337- if (!(tryGetBoolean("setup_active")))
338- then throw("market is stopped")
339- else if ((indexOf(axlyLPs, assetIdStr) != unit))
340- then throw("this token cannot be borrowed")
341- else if (if (!(supplyLater))
342- then (0 > collateralValue)
343- else false)
344- then throw("you have to supply more to borrow")
345- else {
346- let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
347- let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
348- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
349- if ((amount > (assetSupplied - assetBorrowed)))
350- then throw("this amount is not available")
351- else $Tuple2(([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed + amount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed + amount))] ++ [IntegerEntry((assetIdStr + "_bRate"), bRate)]), assetAmount)
352- }
353- }
354- else throw("Strict value is not equal to itself.")
303+ if (!(tryGetBoolean("setup_active")))
304+ then throw("market is stopped")
305+ else if (tryGetBoolean((assetIdStr + kAxlyLp)))
306+ then throw("this token cannot be borrowed")
307+ else {
308+ let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
309+ let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
310+ let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
311+ let assetScale = calcAssetScale(assetIdStr)
312+ let assetPrice = getTokenPrice(assetIdStr)
313+ let assetAmountUsd = fraction(amount, assetPrice._1, assetScale)
314+ if ((amount > (assetSupplied - assetBorrowed)))
315+ then throw("this amount is not available")
316+ else {
317+ let assetNum = value(indexOf(getMarketAssets(), assetIdStr))
318+ $Tuple2(([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed + amount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed + amount))] ++ [ratesRecalcResult[0], ratesRecalcResult[1], ratesRecalcResult[2]]), $Tuple2(assetAmount, assetAmountUsd))
319+ }
320+ }
355321 }
356322
357323
359325 func flashPosition (user,sAssetIdStr,bAssetIdStr,bAmount,callbackAddress,callbackFunction,callbackArgs) = if ((i.caller != axlyAddress))
360326 then throw("available only for Axly protocol")
361327 else {
362- let borrowRes = borrowInternal(bAssetIdStr, bAmount, user, true)
328+ let borrowRes = borrowInternal(bAssetIdStr, bAmount, user)
363329 let bAssetId = getAssetBytes(bAssetIdStr)
364330 let callback = reentrantInvoke(addressFromStringValue(callbackAddress), callbackFunction, split(callbackArgs, ","), [AttachedPayment(bAssetId, bAmount)])
365331 if ((callback == callback))
373339 if ((sAmount == sAmount))
374340 then {
375341 let supplyRes = supplyInternal(sAssetIdStr, sAmount, user)
376- let collateralValue = {
377- let @ = invoke(this, "getUserCollateral", [false, user, true, ((sAssetIdStr + ",supplied,") + toString(sAmount))], nil)
378- if ($isInstanceOf(@, "Int"))
379- then @
380- else throw(($getType(@) + " couldn't be cast to Int"))
381- }
382- if ((collateralValue == collateralValue))
383- then {
384- let borrowValue = {
385- let @ = invoke(this, "getUserCollateral", [false, user, true, ((bAssetIdStr + ",borrowed,") + toString(bAmount))], nil)
386- if ($isInstanceOf(@, "Int"))
387- then @
388- else throw(($getType(@) + " couldn't be cast to Int"))
389- }
390- if ((borrowValue == borrowValue))
391- then if ((borrowValue > collateralValue))
392- then throw("not enough collateral provided")
393- else (borrowRes._1 ++ supplyRes._1)
394- else throw("Strict value is not equal to itself.")
395- }
396- else throw("Strict value is not equal to itself.")
342+ if ((borrowRes._2._2 > supplyRes._2._2))
343+ then throw("not enough collateral provided")
344+ else (borrowRes._1 ++ supplyRes._1)
397345 }
398346 else throw("Strict value is not equal to itself.")
399347 }
410358 else {
411359 let assetIdStr = getAssetString(i.payments[0].assetId)
412360 let assetAmount = i.payments[0].amount
413- let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
414- if ((axlyNotify == axlyNotify))
415- then supplyInternal(assetIdStr, assetAmount, toString(i.caller))
416- else throw("Strict value is not equal to itself.")
361+ if (tryGetBoolean((assetIdStr + kAxlyLp)))
362+ then throw("You can't supply lp")
363+ else {
364+ let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
365+ if ((axlyNotify == axlyNotify))
366+ then supplyInternal(assetIdStr, assetAmount, toString(i.caller))
367+ else throw("Strict value is not equal to itself.")
368+ }
417369 }
418370
419371
420372
421373 @Callable(i)
422374 func withdraw (assetIdStr,assetAmount) = {
423- let $t01323813305 = getActualRate(assetIdStr, "sRate")
424- let sRate = $t01323813305._1
425- let ratesRecalcResult = $t01323813305._2
375+ let $t01260412679 = getActualRateOneToken(assetIdStr, "sRate")
376+ let sRate = $t01260412679._1
377+ let ratesRecalcResult = $t01260412679._2
426378 let amount = fraction(assetAmount, Scale16, sRate, CEILING)
427379 let address = toString(i.caller)
428380 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
429381 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
430382 let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr))
431- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
432383 let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
433384 if ((axlyNotify == axlyNotify))
434385 then {
435- let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
386+ let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, assetIdStr, "", false, ("supplied," + toString(-(amount)))], nil)
436387 if ((collateralValueInv == collateralValueInv))
437388 then {
438389 let collateralValue = match collateralValueInv {
447398 then throw("you dont have enough collateral for this operation")
448399 else if ((amount > (assetSupplied - assetBorrowed)))
449400 then throw("this amount is not available on the market")
450- else if ((amount > (userAssetSupplied - userAssetBorrowed)))
451- then throw("this amount is not available for this user")
401+ else if ((amount > userAssetSupplied))
402+ then throw("this amount is not available on the market")
452403 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)
453404 }
454405 else throw("Strict value is not equal to itself.")
459410
460411
461412 @Callable(i)
462-func repay () = if (!(tryGetBoolean("setup_active")))
463- then throw("market is stopped")
464- else if (if ((size(i.payments) != 1))
465- then true
466- else (i.payments[0].amount == 0))
467- then throw("1 payment has to be attached")
468- else {
469- let assetIdStr = getAssetString(i.payments[0].assetId)
470- let assetAmount = i.payments[0].amount
471- let $t01512215189 = getActualRate(assetIdStr, "bRate")
472- let bRate = $t01512215189._1
473- let ratesRecalcResult = $t01512215189._2
474- let amount = fraction(assetAmount, Scale16, bRate, CEILING)
475- let address = toString(i.caller)
476- let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
477- let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
478- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
479- let amountLeft = (userAssetBorrowed - amount)
480- let repayAmount = if ((amountLeft >= 0))
481- then amount
482- else userAssetBorrowed
483- if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
484- then throw("this asset is not supported by the market")
485- else (([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ (if ((amountLeft >= 0))
486- then nil
487- else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]))
488- }
489-
490-
491-
492-@Callable(i)
493-func repayFor (address) = if (!(tryGetBoolean("setup_active")))
413+func repayFor (address,lpId) = if (!(tryGetBoolean("setup_active")))
494414 then throw("market is stopped")
495415 else if (!(verifySentinelRights(i.caller)))
496416 then throw("available only for whitelist")
500420 then true
501421 else (i.payments[0].amount == 0))
502422 then throw("1 payment has to be attached")
503- else {
504- let assetIdStr = getAssetString(i.payments[0].assetId)
505- let assetAmount = i.payments[0].amount
506- let $t01665216719 = getActualRate(assetIdStr, "bRate")
507- let bRate = $t01665216719._1
508- let ratesRecalcResult = $t01665216719._2
509- let amount = fraction(assetAmount, Scale16, bRate, CEILING)
510- let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
511- let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
512- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
513- let amountLeft = (userAssetBorrowed - amount)
514- let repayAmount = if ((amountLeft >= 0))
515- then amount
516- else userAssetBorrowed
517- let extraAmountResult = if ((amountLeft >= 0))
518- then nil
519- else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]
520- let withdrawResult = if ((i.caller == axlyAddress))
521- then {
522- func fold (accum,assetIdStr) = {
523- let userSupplied = tryGetInteger(((address + "_borrowed_") + assetIdStr))
423+ else if (!(tryGetBoolean((lpId + kAxlyLp))))
424+ then throw("Wrong lp id")
425+ else {
426+ let assetIdStr = getAssetString(i.payments[0].assetId)
427+ let assetAmount = i.payments[0].amount
428+ let $t01471914794 = getActualRateOneToken(assetIdStr, "bRate")
429+ let bRate = $t01471914794._1
430+ let ratesRecalcResult = $t01471914794._2
431+ let amount = fraction(assetAmount, Scale16, bRate, CEILING)
432+ let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
433+ let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
434+ let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
435+ let amountLeft = (userAssetBorrowed - amount)
436+ let repayAmount = if ((amountLeft >= 0))
437+ then amount
438+ else userAssetBorrowed
439+ let extraAmountResult = if ((amountLeft >= 0))
440+ then nil
441+ else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]
442+ let withdrawResult = if ((i.caller == axlyAddress))
443+ then {
444+ let userSupplied = tryGetInteger(((address + "_supplied_") + lpId))
524445 if ((userSupplied > 0))
525- then (accum ++ [IntegerEntry(((address + "_supplied_") + assetIdStr), 0), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) - userSupplied))])
526- else accum
446+ then [IntegerEntry(((address + "_supplied_") + lpId), 0), IntegerEntry(("total_supplied_" + lpId), (tryGetInteger(("total_supplied_" + lpId)) - userSupplied))]
447+ else nil
527448 }
528-
529- let $l = axlyLPs
530- let $s = size($l)
531- let $acc0 = nil
532- func $f0_1 ($a,$i) = if (($i >= $s))
533- then $a
534- else fold($a, $l[$i])
535-
536- func $f0_2 ($a,$i) = if (($i >= $s))
537- then $a
538- else throw("List size exceeds 8")
539-
540- $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)
541- }
542- else nil
543- if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
544- then throw("this asset is not supported by the market")
545- else ((([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ extraAmountResult) ++ withdrawResult)
546- }
449+ else nil
450+ if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
451+ then throw("this asset is not supported by the market")
452+ else ((([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ extraAmountResult) ++ withdrawResult)
453+ }
547454
548455
549456
596503 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
597504 else false)
598505 then throw("admin only")
599- 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)]
506+ 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), IntegerEntry((token + "_lastRateHeight"), height)]
507+
508+
509+
510+@Callable(i)
511+func initNewLPToken (token) = if (if ((i.caller != this))
512+ then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
513+ else false)
514+ then throw("admin only")
515+ else [BooleanEntry((token + "_axlyLp"), true)]
600516
601517
602518
622538 func claimToReserveFund (debug) = {
623539 let assets = getMarketAssets()
624540 let rates = getActualRate(assets[0], "sRate")._2
625- let li = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
626- func f (accum,n) = if ((n >= size(assets)))
627- then accum
628- else {
629- let assetIdStr = assets[n]
630- let autostakeAmount = tryGetString(("autostake_amount_" + assetIdStr))
631- let amount = ((((getBalance(assetIdStr) + tryGetInteger(("autostake_amount_" + assetIdStr))) + (if ((autostakeAmount != ""))
632- then parseIntValue(autostakeAmount)
633- else 0)) + fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[((n * 3) + 1)].value, Scale16)) - fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[(n * 3)].value, Scale16))
634- let inv = if ((tryGetInteger(("autostake_amount_" + assetIdStr)) > 0))
635- then invoke(this, "unstakeToken", [assetIdStr, max([amount, 0])], nil)
636- else 0
637- if ((inv == inv))
638- then (accum ++ [amount])
639- else throw("Strict value is not equal to itself.")
640- }
541+ func f (a,assetIdStr) = {
542+ let $t01927319323 = a
543+ let accum = $t01927319323._1
544+ let n = $t01927319323._2
545+ let autostakeAmount = tryGetString(("autostake_amount_" + assetIdStr))
546+ let amount = ((((getBalance(assetIdStr) + tryGetInteger(("autostake_amount_" + assetIdStr))) + (if ((autostakeAmount != ""))
547+ then parseIntValue(autostakeAmount)
548+ else 0)) + fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[((n * 3) + 1)].value, Scale16)) - fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[(n * 3)].value, Scale16))
549+ let inv = if ((tryGetInteger(("autostake_amount_" + assetIdStr)) > 0))
550+ then invoke(this, "unstakeToken", [assetIdStr, max([amount, 0])], nil)
551+ else 0
552+ if ((inv == inv))
553+ then $Tuple2((accum ++ [amount]), (n + 1))
554+ else throw("Strict value is not equal to itself.")
555+ }
641556
642- let parameter = {
643- let $l = li
644- let $s = size($l)
645- let $acc0 = nil
646- func $f0_1 ($a,$i) = if (($i >= $s))
647- then $a
648- else f($a, $l[$i])
557+ let parameter = ( let $l = assets
558+ let $s = size($l)
559+ let $acc0 = $Tuple2(nil, 0)
560+ func $f0_1 ($a,$i) = if (($i >= $s))
561+ then $a
562+ else f($a, $l[$i])
649563
650- func $f0_2 ($a,$i) = if (($i >= $s))
651- then $a
652- else throw("List size exceeds 12")
564+ func $f0_2 ($a,$i) = if (($i >= $s))
565+ then $a
566+ else throw("List size exceeds 100")
653567
654- $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)
568+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100))._1
569+ func f2 (a,assetIdStr) = {
570+ let $t02007520093 = a
571+ let accum = $t02007520093._1
572+ let n = $t02007520093._2
573+ $Tuple2((accum ++ [ScriptTransfer(reserveFundAddress, max([parameter[n], 0]), getAssetBytes(assetIdStr))]), (n + 1))
655574 }
656- func f2 (accum,n) = if ((n >= size(assets)))
657- then accum
658- else {
659- let assetIdStr = assets[n]
660- (accum ++ [ScriptTransfer(reserveFundAddress, max([parameter[n], 0]), getAssetBytes(assetIdStr))])
661- }
662575
663576 if (debug)
664577 then throw(liIntToStr(parameter))
665578 else $Tuple2({
666- let $l = li
579+ let $l = assets
667580 let $s = size($l)
668- let $acc0 = nil
581+ let $acc0 = $Tuple2(nil, 0)
669582 func $f1_1 ($a,$i) = if (($i >= $s))
670583 then $a
671584 else f2($a, $l[$i])
672585
673586 func $f1_2 ($a,$i) = if (($i >= $s))
674587 then $a
675- else throw("List size exceeds 12")
588+ else throw("List size exceeds 100")
676589
677- $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)
678- }, parameter)
590+ $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($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
591+ }, parameter)._1
679592 }
680593
681594
688601
689602
690603 @Callable(i)
691-func liquidate (debug,address,assetAmount,sAssetIdStr,bAssetIdStr,routeStr) = if (!(verifyLiquidatorRights(i.caller)))
692- then throw("available for whitelist only")
693- else if (!(tryGetBoolean("setup_active")))
694- then throw("market is stopped")
695- else {
696- let userCollateral = calcUserCollateral(address)
697- if ((userCollateral == userCollateral))
698- then {
699- let $t02285522917 = getActualRate(sAssetIdStr, "sRate")
700- let sRate = $t02285522917._1
701- let ratesResult = $t02285522917._2
702- let $t02292222991 = getActualRate(bAssetIdStr, "bRate")
703- let bRate = $t02292222991._1
704- let ratesRecalcResult2 = $t02292222991._2
705- let sAssetAmount = fraction(assetAmount, Scale16, sRate)
706- let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
707- let currentBPositionVal = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
708- let currentBPosition = if ((currentBPositionVal > 0))
709- then currentBPositionVal
710- else throw("user has no borrow in this token")
711- if ((userCollateral > 0))
712- then throw("user can't be liquidated")
713- else if ((sAssetAmount > currentSPosition))
714- then throw("position to liquidate is bigger than user's supply")
715- else {
716- let inv = if ((indexOf(axlyLPs, sAssetIdStr) != unit))
717- then {
718- let posId = split(address, "_")[1]
719- let addressId = split(address, "_")[0]
720- reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, sAssetAmount], nil)
721- }
722- else nil
723- if ((inv == inv))
724- then {
725- let balance0Before = getBalance(sAssetIdStr)
726- if ((balance0Before == balance0Before))
727- then {
728- let balance1Before = getBalance(bAssetIdStr)
729- if ((balance1Before == balance1Before))
730- then {
731- let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
732- if ((exchangeInvoke == exchangeInvoke))
733- then {
734- let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
735- if ((asset0Sold == asset0Sold))
736- then {
737- let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
738- if ((asset1Bought == asset1Bought))
739- then {
740- let asset0Price = getTokenPrice(sAssetIdStr)._1
741- let asset0Scale = calcAssetScale(sAssetIdStr)
742- let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
743- let asset1Price = getTokenPrice(bAssetIdStr)._2
744- let asset1Scale = calcAssetScale(bAssetIdStr)
745- let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
746- let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
747- let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
748- let sAssetChange = fraction(asset0Sold, Scale16, sRate)
749- let bAssetChange = fraction(fraction(asset1Bought, Scale16, bRate), (Scale8 - fraction(liquidationProfit, Scale8, asset1Usd)), Scale8)
750- if ((asset0Sold > assetAmount))
751- then throw("more assets exchanged than expected")
752- else if ((0 > liquidationProfit))
753- then throw("price impact is bigger than liquidation penalty")
754- 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))]
755- }
756- else throw("Strict value is not equal to itself.")
757- }
758- else throw("Strict value is not equal to itself.")
759- }
760- else throw("Strict value is not equal to itself.")
761- }
762- else throw("Strict value is not equal to itself.")
763- }
764- else throw("Strict value is not equal to itself.")
765- }
766- else throw("Strict value is not equal to itself.")
767- }
768- }
769- else throw("Strict value is not equal to itself.")
770- }
771-
772-
773-
774-@Callable(i)
775604 func liquidateV2 (debug,address,sAssetIdStr,bAssetIdStr,liquidatePercent) = if (!(verifyLiquidatorRights(i.caller)))
776605 then throw("available for whitelist only")
777606 else if (!(tryGetBoolean("setup_active")))
778607 then throw("market is stopped")
779608 else {
780- let userCollateral = calcUserCollateral(address)
781- if ((userCollateral == userCollateral))
609+ let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, sAssetIdStr, bAssetIdStr, true, ""], nil)
610+ if ((collateralValueInv == collateralValueInv))
782611 then {
612+ let userCollateral = match collateralValueInv {
613+ case x: Int =>
614+ x
615+ case _ =>
616+ throw("can't get user collateral value")
617+ }
783618 let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
784619 let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
785- if ((indexOf(axlyLPs, sAssetIdStr) == unit))
620+ if (!(tryGetBoolean((sAssetIdStr + kAxlyLp))))
786621 then throw("wrong lp asset")
787622 else if ((0 >= currentBPosition))
788623 then throw("user has no borrow in this token")
796631 let asset1Num = value(indexOf(marketAssets, bAssetIdStr))
797632 let asset0Num = value(indexOf(marketAssets, sAssetIdStr))
798633 let penalty = parseIntValue(value(split(tryGetString("setup_penalties"), ",")[asset1Num]))
799- let $t02670626768 = getActualRate(bAssetIdStr, "bRate")
800- let bRate = $t02670626768._1
801- let ratesResult = $t02670626768._2
634+ let $t02189121961 = getActualRateOneToken(bAssetIdStr, "bRate")
635+ let bRate = $t02189121961._1
636+ let ratesResult = $t02189121961._2
802637 let bAmount = fraction(currentBPosition, bRate, Scale16)
803638 let lAmount = fraction(bAmount, liquidatePercent, 100)
804639 let lAmountWithPenalty = fraction(lAmount, (Scale8 + penalty), Scale8)
813648 }
814649 if ((sAmountLp == sAmountLp))
815650 then {
816- let sAmount = fraction(sAmountLp, Scale16, ratesResult[((asset0Num * 3) + 1)].value)
651+ let sAmount = fraction(sAmountLp, Scale16, ratesResult[1].value)
817652 if (debug)
818653 then throw("liquidation will pass")
819654 else ([ScriptTransfer(i.caller, (lAmountWithPenalty - lAmount), bAssetId), IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAmount)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - lbAmount)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAmount)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - lbAmount))] ++ ratesResult)
890725
891726
892727 @Callable(i)
728+func getUserCollateralV2 (debug,address,sAssetId,bAssetId,minusBorrowed,afterChange) = {
729+ let assets = getMarketAssets()
730+ let sIndex = valueOrErrorMessage(indexOf(assets, sAssetId), "Wrong sAssetId")
731+ let ltvs = split(tryGetString("setup_ltvs"), ",")
732+ let lts = split(tryGetString("setup_lts"), ",")
733+ let sAssetRate = getActualRateOneToken(sAssetId, "sRate")
734+ let changeHandler = split(afterChange, ",")
735+ let userSupplied = tryGetInteger(((address + "_supplied_") + sAssetId))
736+ let userBorrowed = tryGetInteger(((address + "_borrowed_") + bAssetId))
737+ let suppliedChange = if (if ((afterChange != ""))
738+ then (changeHandler[0] == "supplied")
739+ else false)
740+ then parseIntValue(changeHandler[1])
741+ else 0
742+ let sAssetScale = calcAssetScale(sAssetId)
743+ let sAssetPrice = getTokenPrice(sAssetId)
744+ let result = (fraction(fraction(fraction((userSupplied + suppliedChange), sAssetRate._1, Scale16), parseIntValue(ltvs[sIndex]), Scale8), sAssetPrice._1, sAssetScale) - (if (minusBorrowed)
745+ then {
746+ let bIndex = valueOrErrorMessage(indexOf(assets, bAssetId), "Wrong bAssetId")
747+ let bAssetRate = getActualRateOneToken(bAssetId, "bRate")
748+ let borrowedChange = if (if ((afterChange != ""))
749+ then (changeHandler[0] == "borrowed")
750+ else false)
751+ then parseIntValue(changeHandler[1])
752+ else 0
753+ let bAssetScale = calcAssetScale(bAssetId)
754+ let bAssetPrice = getTokenPrice(bAssetId)
755+ fraction(fraction(fraction((userBorrowed + borrowedChange), bAssetRate._1, Scale16), Scale8, parseIntValue(lts[bIndex])), bAssetPrice._2, bAssetScale)
756+ }
757+ else 0))
758+ if (debug)
759+ then throw(toString(result))
760+ else $Tuple2(nil, result)
761+ }
762+
763+
764+
765+@Callable(i)
893766 func getAssetDebt (debug,address,assetIdStr) = {
894767 let userBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
895768 let assetScale = calcAssetScale(assetIdStr)
896- let rate = getActualRate(assetIdStr, "bRate")._1
769+ let rate = getActualRateOneToken(assetIdStr, "bRate")._1
897770 let result = fraction(userBorrowed, rate, Scale16)
898771 if (debug)
899772 then throw(toString(result))
905778 @Callable(i)
906779 func getPrices (debug) = {
907780 let assets = getMarketAssets()
908- func f (accum,next) = if ((next >= size(assets)))
909- then accum
910- else {
911- let assetPrice = getTokenPriceWithRisk(assets[next], 3)
912- ((((accum + toString(assetPrice._1)) + ",") + toString(assetPrice._2)) + "|")
913- }
781+ func f (accum,assetIdStr) = {
782+ let assetPrice = getTokenPriceWithRisk(assetIdStr, 3)
783+ ((((accum + toString(assetPrice._1)) + ",") + toString(assetPrice._2)) + "|")
784+ }
914785
915786 let result = {
916- let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
787+ let $l = assets
917788 let $s = size($l)
918789 let $acc0 = ""
919790 func $f0_1 ($a,$i) = if (($i >= $s))
922793
923794 func $f0_2 ($a,$i) = if (($i >= $s))
924795 then $a
925- else throw("List size exceeds 12")
796+ else throw("List size exceeds 100")
926797
927- $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)
798+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
928799 }
929800 if (debug)
930801 then throw(result)
964835
965836 func $f0_2 ($a,$i) = if (($i >= $s))
966837 then $a
967- else throw("List size exceeds 12")
838+ else throw("List size exceeds 100")
968839
969- $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)
840+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
970841 }
971842 if (debug)
972843 then throw(parameter._1)
992863
993864 func $f0_2 ($a,$i) = if (($i >= $s))
994865 then $a
995- else throw("List size exceeds 12")
866+ else throw("List size exceeds 100")
996867
997- $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)
868+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
998869 }
999870 if (debug)
1000871 then throw(parameter)
1003874
1004875
1005876 @Verifier(tx)
1006-func verify () = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'J4QQBsh6FGgZbijQPMDZ7GtNALx2N15K3eHvGhjq43Lt'))
877+func verify () = if (if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'J4QQBsh6FGgZbijQPMDZ7GtNALx2N15K3eHvGhjq43Lt'))
1007878 then true
1008879 else sigVerify(tx.bodyBytes, tx.proofs[0], base58'Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS'))
1009880 then true
1010- else if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'CYCxuKusNeomQkH9wbcEd4rRP1mDYMc7bw8pr82Agi4N'))
1011- then sigVerify(tx.bodyBytes, tx.proofs[0], base58'3oqguA2LUTZFFjJCJ62UybbXFTLwv88tSJ3oMrNSGhiA')
1012- else false)
881+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'CYCxuKusNeomQkH9wbcEd4rRP1mDYMc7bw8pr82Agi4N'))
1013882 then true
1014- else sigVerify(tx.bodyBytes, tx.proofs[0], base58'9t2BGHP6EddYu5xvArqDkhHPGbf8ZiYUaWgFcEPtRZhE')
883+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'D2QvCRmindBLJfAb2GWftPgjkLUZFsoCiYxJFDisaKqN'))
884+ then if (sigVerify(tx.bodyBytes, tx.proofs[1], base58'3oqguA2LUTZFFjJCJ62UybbXFTLwv88tSJ3oMrNSGhiA'))
885+ then true
886+ else sigVerify(tx.bodyBytes, tx.proofs[1], base58'9t2BGHP6EddYu5xvArqDkhHPGbf8ZiYUaWgFcEPtRZhE')
887+ else false
1015888
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
4+let kAxlyLp = "_axlyLP"
5+
46 let axlyAddress = Address(base58'3PLsYkBw7taejV1J3qWPCN2yeyVRu31d5HW')
5-
6-let axlyLPs = ["7KZbJrVopwJhkdwbe1eFDBbex4dkY63MxjTNjqXtrzj1", "Btw3G1j4wQgdp49PTxaFkNvn75dQtqGDM7ejQppHnWC1", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "F2AKkA513k5yHEJkLsU6vWxCYYk811GpjLhwEv2WGwZ9", "4CQ5CPGLXLbWBUs2JBjKUaRqF49CmKHkwzvPgSvQpAQV", "6iMB6LKSrgv9waEvEnN6Ydyx7dfxPnGcTw8318WVm5bR"]
77
88 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 (if ((address != Address(base58'3PFHNdEPu71QwBAnc2HZPcrbxoseNfUvHEA')))
1919 then (address != Address(base58'3PCqdm1mAoQqR46oZotFanmqb5CLUvrKEo2'))
2020 else false)
2121 then (address != Address(base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL'))
2222 else false)
2323 then (address != Address(base58'3PHbdpaKzz8EiAngGHaFu2hVuNCdsC67qh3'))
2424 else false)
2525
2626
2727 func verifySentinelRights (address) = !(if ((address != reserveFundAddress))
2828 then (address != axlyAddress)
2929 else false)
3030
3131
3232 func getRateCurve (assetIdStr) = match assetIdStr {
3333 case _ =>
3434 if (("9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi" == $match0))
3535 then $Tuple4(2000000, 25000000, 80000000, 100000000)
3636 else if (("HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW" == $match0))
3737 then $Tuple4(2000000, 25000000, 80000000, 100000000)
3838 else if (("34N9YcEETLWn93qYQ64EsP1x89tSruJU44RrEMSXXEPJ" == $match0))
3939 then $Tuple4(2000000, 25000000, 80000000, 100000000)
4040 else if (("6XtHjpXbs9RRJP2Sr9GUyVqzACcby9TkThHXnjVC5CDJ" == $match0))
4141 then $Tuple4(2000000, 25000000, 80000000, 100000000)
4242 else if (("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p" == $match0))
4343 then $Tuple4(2000000, 25000000, 80000000, 100000000)
4444 else if (("Ajso6nTTjptu2UHLx6hfSXVtHFtRBJCkKYd5SAyj7zf5" == $match0))
4545 then $Tuple4(2000000, 40000000, 80000000, 150000000)
4646 else if (("HEB8Qaw9xrWpWs8tHsiATYGBWDBtP2S7kcPALrMu43AS" == $match0))
4747 then $Tuple4(0, 20000000, 80000000, 40000000)
4848 else if (("WAVES" == $match0))
4949 then $Tuple4(2000000, 30000000, 80000000, 50000000)
5050 else if (("Atqv59EYzjFGuitKVnMRk6H8FukjoV3ktPorbEys25on" == $match0))
5151 then $Tuple4(0, 20000000, 80000000, 40000000)
5252 else if (("DSbbhLsSTeDg5Lsiufk2Aneh3DjVqJuPr2M9uU1gwy5p" == $match0))
5353 then $Tuple4(0, 20000000, 80000000, 100000000)
5454 else if (("8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91" == $match0))
5555 then $Tuple4(0, 30000000, 80000000, 40000000)
5656 else if (("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL" == $match0))
5757 then $Tuple4(0, 25000000, 80000000, 40000000)
5858 else if (("8LQW8f7P5d5PZM7GtZEBgaqRPGSzS3DfPuiXrURJ4AJS" == $match0))
5959 then $Tuple4(2000000, 30000000, 80000000, 50000000)
6060 else if (("474jTeYx2r2Va35794tCScAXWJG9hU2HcgxzMowaZUnu" == $match0))
6161 then $Tuple4(2000000, 30000000, 80000000, 50000000)
6262 else if (("5UYBPpq4WoU5n4MwpFkgJnW3Fq4B1u3ukpK33ik4QerR" == $match0))
6363 then $Tuple4(2000000, 30000000, 80000000, 50000000)
6464 else if (("2thsACuHmzDMuNezPM32wg9a3BwUzBWDeSKakgz3cw21" == $match0))
6565 then $Tuple4(2000000, 40000000, 80000000, 100000000)
6666 else if (("YiNbofFzC17jEHHCMwrRcpy9MrrjabMMLZxg8g5xmf7" == $match0))
6767 then $Tuple4(2000000, 30000000, 80000000, 80000000)
6868 else $Tuple4(0, 20000000, 80000000, 80000000)
6969 }
7070
7171
7272 let Scale8 = 100000000
7373
7474 let Scale10 = 10000000000
7575
7676 let Scale16 = (Scale8 * Scale8)
7777
7878 let dayBlocks = 1440
7979
8080 func liIntToStr (li) = {
8181 func f (accum,next) = ((accum + toString(next)) + ",")
8282
8383 let $l = li
8484 let $s = size($l)
8585 let $acc0 = ""
8686 func $f0_1 ($a,$i) = if (($i >= $s))
8787 then $a
8888 else f($a, $l[$i])
8989
9090 func $f0_2 ($a,$i) = if (($i >= $s))
9191 then $a
92- else throw("List size exceeds 12")
92+ else throw("List size exceeds 100")
9393
94- $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)
94+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
9595 }
9696
9797
9898 func tryGetInteger (key) = match getInteger(this, key) {
9999 case b: Int =>
100100 b
101101 case _ =>
102102 0
103103 }
104104
105105
106106 func tryGetBoolean (key) = match getBoolean(this, key) {
107107 case b: Boolean =>
108108 b
109109 case _ =>
110110 false
111111 }
112112
113113
114114 func tryGetString (key) = match getString(this, key) {
115115 case b: String =>
116116 b
117117 case _ =>
118118 ""
119119 }
120120
121121
122122 func tryGetBinary (key) = match getBinary(this, key) {
123123 case b: ByteVector =>
124124 b
125125 case _ =>
126126 base58''
127127 }
128128
129129
130130 func getAssetString (assetId) = match assetId {
131131 case b: ByteVector =>
132132 toBase58String(b)
133133 case _ =>
134134 "WAVES"
135135 }
136136
137137
138138 func getAssetBytes (assetIdStr) = if ((assetIdStr == "WAVES"))
139139 then unit
140140 else fromBase58String(assetIdStr)
141141
142142
143143 func getBalance (assetIdStr) = if ((assetIdStr == "WAVES"))
144144 then wavesBalance(this).available
145145 else assetBalance(this, fromBase58String(assetIdStr))
146146
147147
148148 func getMarketAssets () = split(tryGetString("setup_tokens"), ",")
149149
150150
151151 func getAssetsMaxSupply () = {
152152 let s = tryGetString("setup_maxsupply")
153153 if ((s == ""))
154154 then [-1, -1, -1, -1, -1, -1, -1]
155155 else split(s, ",")
156156 }
157157
158158
159159 func getOutdatedUr (assetIdStr) = {
160160 let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), tryGetInteger((assetIdStr + "_sRate")), Scale16)
161161 if ((down == 0))
162162 then 0
163163 else fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), tryGetInteger((assetIdStr + "_bRate")), Scale16), down)
164164 }
165165
166166
167167 func getInterest (assetIdStr) = {
168168 let ur = getOutdatedUr(assetIdStr)
169169 let curve = getRateCurve(assetIdStr)
170170 let rate = (curve._1 + (if ((curve._3 >= ur))
171171 then fraction(ur, curve._2, curve._3)
172172 else (curve._2 + fraction((ur - curve._3), curve._4, (100000000 - curve._3)))))
173173 max([fraction(rate, Scale8, (dayBlocks * 365)), 1])
174174 }
175175
176176
177-func tokenRatesRecalc (assetIdStr) = if ((indexOf(axlyLPs, assetIdStr) != unit))
178- then [IntegerEntry((assetIdStr + "_sRate"), Scale16), IntegerEntry((assetIdStr + "_bRate"), Scale16), IntegerEntry("lastRateHeight", height)]
177+func tokenRatesRecalc (assetIdStr) = if (tryGetBoolean((assetIdStr + kAxlyLp)))
178+ then [IntegerEntry((assetIdStr + "_sRate"), Scale16), IntegerEntry((assetIdStr + "_bRate"), Scale16), IntegerEntry((assetIdStr + "_lastRateHeight"), height)]
179179 else {
180180 let interest = getInterest(assetIdStr)
181181 let ur = getOutdatedUr(assetIdStr)
182- let lastRecalcHeight = tryGetInteger("lastRateHeight")
182+ let lastRecalcHeight = valueOrElse(getInteger(this, (assetIdStr + "_lastRateHeight")), tryGetInteger("lastRateHeight"))
183183 let lastBRate = max([tryGetInteger((assetIdStr + "_bRate")), Scale16])
184184 let newBRate = (lastBRate + ((height - lastRecalcHeight) * interest))
185185 let lastSRate = max([tryGetInteger((assetIdStr + "_sRate")), Scale16])
186186 let newSRate = (lastSRate + ((((height - lastRecalcHeight) * fraction(interest, ur, Scale8)) * (100 - reserveFund)) / 100))
187-[IntegerEntry((assetIdStr + "_sRate"), newSRate), IntegerEntry((assetIdStr + "_bRate"), newBRate), IntegerEntry("lastRateHeight", height)]
187+[IntegerEntry((assetIdStr + "_sRate"), newSRate), IntegerEntry((assetIdStr + "_bRate"), newBRate), IntegerEntry((assetIdStr + "_lastRateHeight"), height)]
188188 }
189189
190190
191191 func getActualRate (assetIdStr,rateType) = {
192192 func f (accum,token) = {
193193 let recalc = tokenRatesRecalc(token)
194194 $Tuple2(if ((token != assetIdStr))
195195 then accum._1
196196 else if ((rateType == "sRate"))
197197 then recalc[0].value
198198 else recalc[1].value, (accum._2 ++ recalc))
199199 }
200200
201201 let $l = getMarketAssets()
202202 let $s = size($l)
203203 let $acc0 = $Tuple2(0, nil)
204204 func $f0_1 ($a,$i) = if (($i >= $s))
205205 then $a
206206 else f($a, $l[$i])
207207
208208 func $f0_2 ($a,$i) = if (($i >= $s))
209209 then $a
210- else throw("List size exceeds 12")
210+ else throw("List size exceeds 100")
211211
212- $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)
212+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
213+ }
214+
215+
216+func getActualRateOneToken (assetIdStr,rateType) = {
217+ let recalc = tokenRatesRecalc(assetIdStr)
218+ $Tuple2(if ((rateType == "sRate"))
219+ then recalc[0].value
220+ else recalc[1].value, recalc)
213221 }
214222
215223
216224 func getUr (assetIdStr) = {
217225 let rates = tokenRatesRecalc(assetIdStr)
218226 let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[0].value, Scale16)
219227 fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[1].value, Scale16), down)
220228 }
221229
222230
223-func ratesRecalc () = {
224- func f (accum,token) = (accum ++ tokenRatesRecalc(token))
225-
226- let $l = getMarketAssets()
227- let $s = size($l)
228- let $acc0 = nil
229- func $f0_1 ($a,$i) = if (($i >= $s))
230- then $a
231- else f($a, $l[$i])
232-
233- func $f0_2 ($a,$i) = if (($i >= $s))
234- then $a
235- else throw("List size exceeds 12")
236-
237- $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)
238- }
239-
240-
241231 func getTokenPriceWithRisk (assetIdStr,riskAversity) = if (if ((assetIdStr == "9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi"))
242232 then true
243233 else (assetIdStr == "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW"))
244234 then $Tuple2(1000000, 1000000)
245235 else {
246236 let price = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_twap5B"))
247237 let riskLevel = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_riskLevel"))
248- if ((indexOf(axlyLPs, assetIdStr) != unit))
238+ if (tryGetBoolean((assetIdStr + kAxlyLp)))
249239 then {
250240 let p = {
251241 let @ = reentrantInvoke(axlyAddress, "getShareAssetPriceREADONLY", [assetIdStr], nil)
252242 if ($isInstanceOf(@, "Int"))
253243 then @
254244 else throw(($getType(@) + " couldn't be cast to Int"))
255245 }
256246 if ((p == p))
257247 then $Tuple2(p, p)
258248 else throw("Strict value is not equal to itself.")
259249 }
260250 else if ((riskAversity >= riskLevel))
261251 then $Tuple2(price, price)
262252 else throw((("oracle prices don't match: " + toString(price)) + " is the price, but risk is too high"))
263253 }
264254
265255
266256 func getTokenPrice (assetIdStr) = getTokenPriceWithRisk(assetIdStr, 1)
267257
268258
269259 func calcAssetScale (assetIdStr) = {
270260 let decimals = if ((assetIdStr == "WAVES"))
271261 then 8
272262 else value(assetInfo(fromBase58String(assetIdStr))).decimals
273263 pow(10, 0, decimals, 0, 0, DOWN)
274264 }
275265
276266
277-func calcUserCollateral (address) = {
278- let userCollateralInvoke = invoke(this, "getUserCollateral", [false, address, true, ""], nil)
279- if ((userCollateralInvoke == userCollateralInvoke))
280- then {
281- let userCollateralValue = match userCollateralInvoke {
282- case x: Int =>
283- x
284- case _ =>
285- throw("issue while doing in-dapp invocation")
286- }
287- if ((userCollateralValue == userCollateralValue))
288- then userCollateralValue
289- else throw("Strict value is not equal to itself.")
290- }
291- else throw("Strict value is not equal to itself.")
292- }
293-
294-
295267 func supplyInternal (assetIdStr,assetAmount,address) = if (!(tryGetBoolean("setup_active")))
296268 then throw("market is stopped")
297269 else {
298- let $t090489115 = getActualRate(assetIdStr, "sRate")
299- let sRate = $t090489115._1
300- let ratesRecalcResult = $t090489115._2
270+ let $t085378612 = getActualRateOneToken(assetIdStr, "sRate")
271+ let sRate = $t085378612._1
272+ let ratesRecalcResult = $t085378612._2
301273 let amount = fraction(assetAmount, Scale16, sRate, DOWN)
302274 let maxSupply = match getString(("setup_maxSupply_" + assetIdStr)) {
303275 case x: String =>
304276 parseIntValue(x)
305277 case _ =>
306278 0
307279 }
308280 let assetPrice = getTokenPrice(assetIdStr)
309281 let newTotalSupplied = (tryGetInteger(((address + "_supplied_") + assetIdStr)) + amount)
310- let rate = getActualRate(assetIdStr, "sRate")._1
282+ let rate = sRate
311283 let assetScale = calcAssetScale(assetIdStr)
312284 let newTotalSuppliedUsd = fraction(fraction(newTotalSupplied, rate, Scale16), assetPrice._1, assetScale)
313285 if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
314286 then throw("this asset is not supported by the market")
315287 else if (if ((maxSupply != 0))
316288 then (newTotalSuppliedUsd > maxSupply)
317289 else false)
318290 then throw("max total supply for this token reached in the pool")
319- else $Tuple2(([IntegerEntry(((address + "_supplied_") + assetIdStr), newTotalSupplied), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) + amount))] ++ ratesRecalcResult), assetAmount)
291+ else {
292+ let assetNum = value(indexOf(getMarketAssets(), assetIdStr))
293+ $Tuple2(([IntegerEntry(((address + "_supplied_") + assetIdStr), newTotalSupplied), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) + amount))] ++ [ratesRecalcResult[0], ratesRecalcResult[1], ratesRecalcResult[2]]), $Tuple2(assetAmount, newTotalSuppliedUsd))
294+ }
320295 }
321296
322297
323-func borrowInternal (assetIdStr,assetAmount,address,supplyLater) = {
324- let $t01033710404 = getActualRate(assetIdStr, "bRate")
325- let bRate = $t01033710404._1
326- let ratesRecalcResult = $t01033710404._2
298+func borrowInternal (assetIdStr,assetAmount,address) = {
299+ let $t099209995 = getActualRateOneToken(assetIdStr, "bRate")
300+ let bRate = $t099209995._1
301+ let ratesRecalcResult = $t099209995._2
327302 let amount = fraction(assetAmount, Scale16, bRate, CEILING)
328- let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",borrowed,") + toString(amount))], nil)
329- if ((collateralValueInv == collateralValueInv))
330- then {
331- let collateralValue = match collateralValueInv {
332- case x: Int =>
333- x
334- case _ =>
335- throw("can't get user collateral value")
336- }
337- if (!(tryGetBoolean("setup_active")))
338- then throw("market is stopped")
339- else if ((indexOf(axlyLPs, assetIdStr) != unit))
340- then throw("this token cannot be borrowed")
341- else if (if (!(supplyLater))
342- then (0 > collateralValue)
343- else false)
344- then throw("you have to supply more to borrow")
345- else {
346- let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
347- let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
348- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
349- if ((amount > (assetSupplied - assetBorrowed)))
350- then throw("this amount is not available")
351- else $Tuple2(([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed + amount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed + amount))] ++ [IntegerEntry((assetIdStr + "_bRate"), bRate)]), assetAmount)
352- }
353- }
354- else throw("Strict value is not equal to itself.")
303+ if (!(tryGetBoolean("setup_active")))
304+ then throw("market is stopped")
305+ else if (tryGetBoolean((assetIdStr + kAxlyLp)))
306+ then throw("this token cannot be borrowed")
307+ else {
308+ let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
309+ let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
310+ let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
311+ let assetScale = calcAssetScale(assetIdStr)
312+ let assetPrice = getTokenPrice(assetIdStr)
313+ let assetAmountUsd = fraction(amount, assetPrice._1, assetScale)
314+ if ((amount > (assetSupplied - assetBorrowed)))
315+ then throw("this amount is not available")
316+ else {
317+ let assetNum = value(indexOf(getMarketAssets(), assetIdStr))
318+ $Tuple2(([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed + amount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed + amount))] ++ [ratesRecalcResult[0], ratesRecalcResult[1], ratesRecalcResult[2]]), $Tuple2(assetAmount, assetAmountUsd))
319+ }
320+ }
355321 }
356322
357323
358324 @Callable(i)
359325 func flashPosition (user,sAssetIdStr,bAssetIdStr,bAmount,callbackAddress,callbackFunction,callbackArgs) = if ((i.caller != axlyAddress))
360326 then throw("available only for Axly protocol")
361327 else {
362- let borrowRes = borrowInternal(bAssetIdStr, bAmount, user, true)
328+ let borrowRes = borrowInternal(bAssetIdStr, bAmount, user)
363329 let bAssetId = getAssetBytes(bAssetIdStr)
364330 let callback = reentrantInvoke(addressFromStringValue(callbackAddress), callbackFunction, split(callbackArgs, ","), [AttachedPayment(bAssetId, bAmount)])
365331 if ((callback == callback))
366332 then {
367333 let sAmount = {
368334 let @ = callback
369335 if ($isInstanceOf(@, "Int"))
370336 then @
371337 else throw(($getType(@) + " couldn't be cast to Int"))
372338 }
373339 if ((sAmount == sAmount))
374340 then {
375341 let supplyRes = supplyInternal(sAssetIdStr, sAmount, user)
376- let collateralValue = {
377- let @ = invoke(this, "getUserCollateral", [false, user, true, ((sAssetIdStr + ",supplied,") + toString(sAmount))], nil)
378- if ($isInstanceOf(@, "Int"))
379- then @
380- else throw(($getType(@) + " couldn't be cast to Int"))
381- }
382- if ((collateralValue == collateralValue))
383- then {
384- let borrowValue = {
385- let @ = invoke(this, "getUserCollateral", [false, user, true, ((bAssetIdStr + ",borrowed,") + toString(bAmount))], nil)
386- if ($isInstanceOf(@, "Int"))
387- then @
388- else throw(($getType(@) + " couldn't be cast to Int"))
389- }
390- if ((borrowValue == borrowValue))
391- then if ((borrowValue > collateralValue))
392- then throw("not enough collateral provided")
393- else (borrowRes._1 ++ supplyRes._1)
394- else throw("Strict value is not equal to itself.")
395- }
396- else throw("Strict value is not equal to itself.")
342+ if ((borrowRes._2._2 > supplyRes._2._2))
343+ then throw("not enough collateral provided")
344+ else (borrowRes._1 ++ supplyRes._1)
397345 }
398346 else throw("Strict value is not equal to itself.")
399347 }
400348 else throw("Strict value is not equal to itself.")
401349 }
402350
403351
404352
405353 @Callable(i)
406354 func supply () = if (if ((size(i.payments) != 1))
407355 then true
408356 else (i.payments[0].amount == 0))
409357 then throw("1 payment has to be attached")
410358 else {
411359 let assetIdStr = getAssetString(i.payments[0].assetId)
412360 let assetAmount = i.payments[0].amount
413- let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
414- if ((axlyNotify == axlyNotify))
415- then supplyInternal(assetIdStr, assetAmount, toString(i.caller))
416- else throw("Strict value is not equal to itself.")
361+ if (tryGetBoolean((assetIdStr + kAxlyLp)))
362+ then throw("You can't supply lp")
363+ else {
364+ let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
365+ if ((axlyNotify == axlyNotify))
366+ then supplyInternal(assetIdStr, assetAmount, toString(i.caller))
367+ else throw("Strict value is not equal to itself.")
368+ }
417369 }
418370
419371
420372
421373 @Callable(i)
422374 func withdraw (assetIdStr,assetAmount) = {
423- let $t01323813305 = getActualRate(assetIdStr, "sRate")
424- let sRate = $t01323813305._1
425- let ratesRecalcResult = $t01323813305._2
375+ let $t01260412679 = getActualRateOneToken(assetIdStr, "sRate")
376+ let sRate = $t01260412679._1
377+ let ratesRecalcResult = $t01260412679._2
426378 let amount = fraction(assetAmount, Scale16, sRate, CEILING)
427379 let address = toString(i.caller)
428380 let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
429381 let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
430382 let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr))
431- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
432383 let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil)
433384 if ((axlyNotify == axlyNotify))
434385 then {
435- let collateralValueInv = invoke(this, "getUserCollateral", [false, address, true, ((assetIdStr + ",supplied,") + toString(-(amount)))], nil)
386+ let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, assetIdStr, "", false, ("supplied," + toString(-(amount)))], nil)
436387 if ((collateralValueInv == collateralValueInv))
437388 then {
438389 let collateralValue = match collateralValueInv {
439390 case x: Int =>
440391 x
441392 case _ =>
442393 throw("can't get user collateral value")
443394 }
444395 if (!(tryGetBoolean("setup_active")))
445396 then throw("market is stopped")
446397 else if ((0 > collateralValue))
447398 then throw("you dont have enough collateral for this operation")
448399 else if ((amount > (assetSupplied - assetBorrowed)))
449400 then throw("this amount is not available on the market")
450- else if ((amount > (userAssetSupplied - userAssetBorrowed)))
451- then throw("this amount is not available for this user")
401+ else if ((amount > userAssetSupplied))
402+ then throw("this amount is not available on the market")
452403 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)
453404 }
454405 else throw("Strict value is not equal to itself.")
455406 }
456407 else throw("Strict value is not equal to itself.")
457408 }
458409
459410
460411
461412 @Callable(i)
462-func repay () = if (!(tryGetBoolean("setup_active")))
463- then throw("market is stopped")
464- else if (if ((size(i.payments) != 1))
465- then true
466- else (i.payments[0].amount == 0))
467- then throw("1 payment has to be attached")
468- else {
469- let assetIdStr = getAssetString(i.payments[0].assetId)
470- let assetAmount = i.payments[0].amount
471- let $t01512215189 = getActualRate(assetIdStr, "bRate")
472- let bRate = $t01512215189._1
473- let ratesRecalcResult = $t01512215189._2
474- let amount = fraction(assetAmount, Scale16, bRate, CEILING)
475- let address = toString(i.caller)
476- let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
477- let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
478- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
479- let amountLeft = (userAssetBorrowed - amount)
480- let repayAmount = if ((amountLeft >= 0))
481- then amount
482- else userAssetBorrowed
483- if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
484- then throw("this asset is not supported by the market")
485- else (([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ (if ((amountLeft >= 0))
486- then nil
487- else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]))
488- }
489-
490-
491-
492-@Callable(i)
493-func repayFor (address) = if (!(tryGetBoolean("setup_active")))
413+func repayFor (address,lpId) = if (!(tryGetBoolean("setup_active")))
494414 then throw("market is stopped")
495415 else if (!(verifySentinelRights(i.caller)))
496416 then throw("available only for whitelist")
497417 else if ((address == "global"))
498418 then throw("you can't repay for everyone :_)")
499419 else if (if ((size(i.payments) != 1))
500420 then true
501421 else (i.payments[0].amount == 0))
502422 then throw("1 payment has to be attached")
503- else {
504- let assetIdStr = getAssetString(i.payments[0].assetId)
505- let assetAmount = i.payments[0].amount
506- let $t01665216719 = getActualRate(assetIdStr, "bRate")
507- let bRate = $t01665216719._1
508- let ratesRecalcResult = $t01665216719._2
509- let amount = fraction(assetAmount, Scale16, bRate, CEILING)
510- let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
511- let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
512- let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
513- let amountLeft = (userAssetBorrowed - amount)
514- let repayAmount = if ((amountLeft >= 0))
515- then amount
516- else userAssetBorrowed
517- let extraAmountResult = if ((amountLeft >= 0))
518- then nil
519- else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]
520- let withdrawResult = if ((i.caller == axlyAddress))
521- then {
522- func fold (accum,assetIdStr) = {
523- let userSupplied = tryGetInteger(((address + "_borrowed_") + assetIdStr))
423+ else if (!(tryGetBoolean((lpId + kAxlyLp))))
424+ then throw("Wrong lp id")
425+ else {
426+ let assetIdStr = getAssetString(i.payments[0].assetId)
427+ let assetAmount = i.payments[0].amount
428+ let $t01471914794 = getActualRateOneToken(assetIdStr, "bRate")
429+ let bRate = $t01471914794._1
430+ let ratesRecalcResult = $t01471914794._2
431+ let amount = fraction(assetAmount, Scale16, bRate, CEILING)
432+ let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr))
433+ let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr))
434+ let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
435+ let amountLeft = (userAssetBorrowed - amount)
436+ let repayAmount = if ((amountLeft >= 0))
437+ then amount
438+ else userAssetBorrowed
439+ let extraAmountResult = if ((amountLeft >= 0))
440+ then nil
441+ else [ScriptTransfer(i.caller, -(amountLeft), i.payments[0].assetId)]
442+ let withdrawResult = if ((i.caller == axlyAddress))
443+ then {
444+ let userSupplied = tryGetInteger(((address + "_supplied_") + lpId))
524445 if ((userSupplied > 0))
525- then (accum ++ [IntegerEntry(((address + "_supplied_") + assetIdStr), 0), IntegerEntry(("total_supplied_" + assetIdStr), (tryGetInteger(("total_supplied_" + assetIdStr)) - userSupplied))])
526- else accum
446+ then [IntegerEntry(((address + "_supplied_") + lpId), 0), IntegerEntry(("total_supplied_" + lpId), (tryGetInteger(("total_supplied_" + lpId)) - userSupplied))]
447+ else nil
527448 }
528-
529- let $l = axlyLPs
530- let $s = size($l)
531- let $acc0 = nil
532- func $f0_1 ($a,$i) = if (($i >= $s))
533- then $a
534- else fold($a, $l[$i])
535-
536- func $f0_2 ($a,$i) = if (($i >= $s))
537- then $a
538- else throw("List size exceeds 8")
539-
540- $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)
541- }
542- else nil
543- if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
544- then throw("this asset is not supported by the market")
545- else ((([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ extraAmountResult) ++ withdrawResult)
546- }
449+ else nil
450+ if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit))
451+ then throw("this asset is not supported by the market")
452+ else ((([IntegerEntry(((address + "_borrowed_") + assetIdStr), (userAssetBorrowed - repayAmount)), IntegerEntry(("total_borrowed_" + assetIdStr), (assetBorrowed - repayAmount))] ++ ratesRecalcResult) ++ extraAmountResult) ++ withdrawResult)
453+ }
547454
548455
549456
550457 @Callable(i)
551458 func addInterestEXTERNAL () = {
552459 let amount = fraction(i.payments[0].amount, 80, 100)
553460 let assetId = i.payments[0].assetId
554461 let assetIdStr = getAssetString(assetId)
555462 let earned = tryGetInteger(("autostake_lastEarned_" + assetIdStr))
556463 let lastHeight = tryGetInteger(("autostake_lastBlock_" + assetIdStr))
557464 let stateChanges = if (if ((lastHeight == height))
558465 then true
559466 else (amount == 0))
560467 then nil
561468 else [IntegerEntry(("autostake_preLastEarned_" + assetIdStr), earned), IntegerEntry(("autostake_preLastBlock_" + assetIdStr), lastHeight), IntegerEntry(("autostake_lastEarned_" + assetIdStr), (earned + amount)), IntegerEntry(("autostake_lastBlock_" + assetIdStr), height)]
562469 (stateChanges ++ [IntegerEntry((assetIdStr + "_sRate"), (tryGetInteger((assetIdStr + "_sRate")) + fraction(Scale16, amount, tryGetInteger(("total_supplied_" + assetIdStr)))))])
563470 }
564471
565472
566473
567474 @Callable(i)
568475 func preInit (tokens,ltvs,lts,penalties) = {
569476 func f (accum,token) = (accum ++ [IntegerEntry((token + "_bRate"), Scale16), IntegerEntry((token + "_sRate"), Scale16)])
570477
571478 if ((i.caller != this))
572479 then throw("admin only")
573480 else {
574481 let rates = {
575482 let $l = split(tokens, ",")
576483 let $s = size($l)
577484 let $acc0 = nil
578485 func $f0_1 ($a,$i) = if (($i >= $s))
579486 then $a
580487 else f($a, $l[$i])
581488
582489 func $f0_2 ($a,$i) = if (($i >= $s))
583490 then $a
584491 else throw("List size exceeds 12")
585492
586493 $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)
587494 }
588495 ([StringEntry("setup_tokens", tokens), StringEntry("setup_ltvs", ltvs), StringEntry("setup_lts", lts), StringEntry("setup_penalties", penalties), BooleanEntry("setup_active", true)] ++ rates)
589496 }
590497 }
591498
592499
593500
594501 @Callable(i)
595502 func initNewToken (token,ltv,lt,penalty) = if (if ((i.caller != this))
596503 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
597504 else false)
598505 then throw("admin only")
599- 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)]
506+ 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), IntegerEntry((token + "_lastRateHeight"), height)]
507+
508+
509+
510+@Callable(i)
511+func initNewLPToken (token) = if (if ((i.caller != this))
512+ then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
513+ else false)
514+ then throw("admin only")
515+ else [BooleanEntry((token + "_axlyLp"), true)]
600516
601517
602518
603519 @Callable(i)
604520 func updateParameter (key,val) = if (if ((i.caller != this))
605521 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
606522 else false)
607523 then throw("admin only")
608524 else [IntegerEntry(key, parseIntValue(val))]
609525
610526
611527
612528 @Callable(i)
613529 func updateString (key,val) = if (if ((i.caller != this))
614530 then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8'))
615531 else false)
616532 then throw("admin only")
617533 else [StringEntry(key, val)]
618534
619535
620536
621537 @Callable(i)
622538 func claimToReserveFund (debug) = {
623539 let assets = getMarketAssets()
624540 let rates = getActualRate(assets[0], "sRate")._2
625- let li = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
626- func f (accum,n) = if ((n >= size(assets)))
627- then accum
628- else {
629- let assetIdStr = assets[n]
630- let autostakeAmount = tryGetString(("autostake_amount_" + assetIdStr))
631- let amount = ((((getBalance(assetIdStr) + tryGetInteger(("autostake_amount_" + assetIdStr))) + (if ((autostakeAmount != ""))
632- then parseIntValue(autostakeAmount)
633- else 0)) + fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[((n * 3) + 1)].value, Scale16)) - fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[(n * 3)].value, Scale16))
634- let inv = if ((tryGetInteger(("autostake_amount_" + assetIdStr)) > 0))
635- then invoke(this, "unstakeToken", [assetIdStr, max([amount, 0])], nil)
636- else 0
637- if ((inv == inv))
638- then (accum ++ [amount])
639- else throw("Strict value is not equal to itself.")
640- }
541+ func f (a,assetIdStr) = {
542+ let $t01927319323 = a
543+ let accum = $t01927319323._1
544+ let n = $t01927319323._2
545+ let autostakeAmount = tryGetString(("autostake_amount_" + assetIdStr))
546+ let amount = ((((getBalance(assetIdStr) + tryGetInteger(("autostake_amount_" + assetIdStr))) + (if ((autostakeAmount != ""))
547+ then parseIntValue(autostakeAmount)
548+ else 0)) + fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[((n * 3) + 1)].value, Scale16)) - fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[(n * 3)].value, Scale16))
549+ let inv = if ((tryGetInteger(("autostake_amount_" + assetIdStr)) > 0))
550+ then invoke(this, "unstakeToken", [assetIdStr, max([amount, 0])], nil)
551+ else 0
552+ if ((inv == inv))
553+ then $Tuple2((accum ++ [amount]), (n + 1))
554+ else throw("Strict value is not equal to itself.")
555+ }
641556
642- let parameter = {
643- let $l = li
644- let $s = size($l)
645- let $acc0 = nil
646- func $f0_1 ($a,$i) = if (($i >= $s))
647- then $a
648- else f($a, $l[$i])
557+ let parameter = ( let $l = assets
558+ let $s = size($l)
559+ let $acc0 = $Tuple2(nil, 0)
560+ func $f0_1 ($a,$i) = if (($i >= $s))
561+ then $a
562+ else f($a, $l[$i])
649563
650- func $f0_2 ($a,$i) = if (($i >= $s))
651- then $a
652- else throw("List size exceeds 12")
564+ func $f0_2 ($a,$i) = if (($i >= $s))
565+ then $a
566+ else throw("List size exceeds 100")
653567
654- $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)
568+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100))._1
569+ func f2 (a,assetIdStr) = {
570+ let $t02007520093 = a
571+ let accum = $t02007520093._1
572+ let n = $t02007520093._2
573+ $Tuple2((accum ++ [ScriptTransfer(reserveFundAddress, max([parameter[n], 0]), getAssetBytes(assetIdStr))]), (n + 1))
655574 }
656- func f2 (accum,n) = if ((n >= size(assets)))
657- then accum
658- else {
659- let assetIdStr = assets[n]
660- (accum ++ [ScriptTransfer(reserveFundAddress, max([parameter[n], 0]), getAssetBytes(assetIdStr))])
661- }
662575
663576 if (debug)
664577 then throw(liIntToStr(parameter))
665578 else $Tuple2({
666- let $l = li
579+ let $l = assets
667580 let $s = size($l)
668- let $acc0 = nil
581+ let $acc0 = $Tuple2(nil, 0)
669582 func $f1_1 ($a,$i) = if (($i >= $s))
670583 then $a
671584 else f2($a, $l[$i])
672585
673586 func $f1_2 ($a,$i) = if (($i >= $s))
674587 then $a
675- else throw("List size exceeds 12")
588+ else throw("List size exceeds 100")
676589
677- $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)
678- }, parameter)
590+ $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($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
591+ }, parameter)._1
679592 }
680593
681594
682595
683596 @Callable(i)
684597 func shutdown (shutdown) = if ((indexOf(shutdownWhitelist, i.caller.bytes) == unit))
685598 then throw("user not in a whitelist")
686599 else [BooleanEntry("setup_active", !(shutdown))]
687600
688601
689602
690603 @Callable(i)
691-func liquidate (debug,address,assetAmount,sAssetIdStr,bAssetIdStr,routeStr) = if (!(verifyLiquidatorRights(i.caller)))
692- then throw("available for whitelist only")
693- else if (!(tryGetBoolean("setup_active")))
694- then throw("market is stopped")
695- else {
696- let userCollateral = calcUserCollateral(address)
697- if ((userCollateral == userCollateral))
698- then {
699- let $t02285522917 = getActualRate(sAssetIdStr, "sRate")
700- let sRate = $t02285522917._1
701- let ratesResult = $t02285522917._2
702- let $t02292222991 = getActualRate(bAssetIdStr, "bRate")
703- let bRate = $t02292222991._1
704- let ratesRecalcResult2 = $t02292222991._2
705- let sAssetAmount = fraction(assetAmount, Scale16, sRate)
706- let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
707- let currentBPositionVal = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
708- let currentBPosition = if ((currentBPositionVal > 0))
709- then currentBPositionVal
710- else throw("user has no borrow in this token")
711- if ((userCollateral > 0))
712- then throw("user can't be liquidated")
713- else if ((sAssetAmount > currentSPosition))
714- then throw("position to liquidate is bigger than user's supply")
715- else {
716- let inv = if ((indexOf(axlyLPs, sAssetIdStr) != unit))
717- then {
718- let posId = split(address, "_")[1]
719- let addressId = split(address, "_")[0]
720- reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, sAssetAmount], nil)
721- }
722- else nil
723- if ((inv == inv))
724- then {
725- let balance0Before = getBalance(sAssetIdStr)
726- if ((balance0Before == balance0Before))
727- then {
728- let balance1Before = getBalance(bAssetIdStr)
729- if ((balance1Before == balance1Before))
730- then {
731- let exchangeInvoke = invoke(aggregatorAddress, "swap", [routeStr, 0], [AttachedPayment(getAssetBytes(sAssetIdStr), assetAmount)])
732- if ((exchangeInvoke == exchangeInvoke))
733- then {
734- let asset0Sold = (balance0Before - getBalance(sAssetIdStr))
735- if ((asset0Sold == asset0Sold))
736- then {
737- let asset1Bought = (getBalance(bAssetIdStr) - balance1Before)
738- if ((asset1Bought == asset1Bought))
739- then {
740- let asset0Price = getTokenPrice(sAssetIdStr)._1
741- let asset0Scale = calcAssetScale(sAssetIdStr)
742- let asset0Usd = fraction(asset0Sold, asset0Price, asset0Scale)
743- let asset1Price = getTokenPrice(bAssetIdStr)._2
744- let asset1Scale = calcAssetScale(bAssetIdStr)
745- let asset1Usd = fraction(asset1Bought, asset1Price, asset1Scale)
746- let penalty = parseIntValue(split(tryGetString("setup_penalties"), ",")[value(indexOf(getMarketAssets(), bAssetIdStr))])
747- let liquidationProfit = (asset1Usd - fraction(asset0Usd, (Scale8 - penalty), Scale8))
748- let sAssetChange = fraction(asset0Sold, Scale16, sRate)
749- let bAssetChange = fraction(fraction(asset1Bought, Scale16, bRate), (Scale8 - fraction(liquidationProfit, Scale8, asset1Usd)), Scale8)
750- if ((asset0Sold > assetAmount))
751- then throw("more assets exchanged than expected")
752- else if ((0 > liquidationProfit))
753- then throw("price impact is bigger than liquidation penalty")
754- 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))]
755- }
756- else throw("Strict value is not equal to itself.")
757- }
758- else throw("Strict value is not equal to itself.")
759- }
760- else throw("Strict value is not equal to itself.")
761- }
762- else throw("Strict value is not equal to itself.")
763- }
764- else throw("Strict value is not equal to itself.")
765- }
766- else throw("Strict value is not equal to itself.")
767- }
768- }
769- else throw("Strict value is not equal to itself.")
770- }
771-
772-
773-
774-@Callable(i)
775604 func liquidateV2 (debug,address,sAssetIdStr,bAssetIdStr,liquidatePercent) = if (!(verifyLiquidatorRights(i.caller)))
776605 then throw("available for whitelist only")
777606 else if (!(tryGetBoolean("setup_active")))
778607 then throw("market is stopped")
779608 else {
780- let userCollateral = calcUserCollateral(address)
781- if ((userCollateral == userCollateral))
609+ let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, sAssetIdStr, bAssetIdStr, true, ""], nil)
610+ if ((collateralValueInv == collateralValueInv))
782611 then {
612+ let userCollateral = match collateralValueInv {
613+ case x: Int =>
614+ x
615+ case _ =>
616+ throw("can't get user collateral value")
617+ }
783618 let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr))
784619 let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr))
785- if ((indexOf(axlyLPs, sAssetIdStr) == unit))
620+ if (!(tryGetBoolean((sAssetIdStr + kAxlyLp))))
786621 then throw("wrong lp asset")
787622 else if ((0 >= currentBPosition))
788623 then throw("user has no borrow in this token")
789624 else if (if ((userCollateral > 0))
790625 then true
791626 else (currentBPosition == 0))
792627 then throw("user can't be liquidated")
793628 else {
794629 let bAssetId = getAssetBytes(bAssetIdStr)
795630 let marketAssets = getMarketAssets()
796631 let asset1Num = value(indexOf(marketAssets, bAssetIdStr))
797632 let asset0Num = value(indexOf(marketAssets, sAssetIdStr))
798633 let penalty = parseIntValue(value(split(tryGetString("setup_penalties"), ",")[asset1Num]))
799- let $t02670626768 = getActualRate(bAssetIdStr, "bRate")
800- let bRate = $t02670626768._1
801- let ratesResult = $t02670626768._2
634+ let $t02189121961 = getActualRateOneToken(bAssetIdStr, "bRate")
635+ let bRate = $t02189121961._1
636+ let ratesResult = $t02189121961._2
802637 let bAmount = fraction(currentBPosition, bRate, Scale16)
803638 let lAmount = fraction(bAmount, liquidatePercent, 100)
804639 let lAmountWithPenalty = fraction(lAmount, (Scale8 + penalty), Scale8)
805640 let lbAmount = fraction(lAmount, Scale16, bRate)
806641 let posId = split(address, "_")[1]
807642 let addressId = split(address, "_")[0]
808643 let sAmountLp = {
809644 let @ = reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, lAmountWithPenalty], nil)
810645 if ($isInstanceOf(@, "Int"))
811646 then @
812647 else throw(($getType(@) + " couldn't be cast to Int"))
813648 }
814649 if ((sAmountLp == sAmountLp))
815650 then {
816- let sAmount = fraction(sAmountLp, Scale16, ratesResult[((asset0Num * 3) + 1)].value)
651+ let sAmount = fraction(sAmountLp, Scale16, ratesResult[1].value)
817652 if (debug)
818653 then throw("liquidation will pass")
819654 else ([ScriptTransfer(i.caller, (lAmountWithPenalty - lAmount), bAssetId), IntegerEntry(((address + "_supplied_") + sAssetIdStr), (currentSPosition - sAmount)), IntegerEntry(((address + "_borrowed_") + bAssetIdStr), (currentBPosition - lbAmount)), IntegerEntry(("total_supplied_" + sAssetIdStr), (tryGetInteger(("total_supplied_" + sAssetIdStr)) - sAmount)), IntegerEntry(("total_borrowed_" + bAssetIdStr), (tryGetInteger(("total_borrowed_" + bAssetIdStr)) - lbAmount))] ++ ratesResult)
820655 }
821656 else throw("Strict value is not equal to itself.")
822657 }
823658 }
824659 else throw("Strict value is not equal to itself.")
825660 }
826661
827662
828663
829664 @Callable(i)
830665 func getUserCollateral (debug,address,minusBorrowed,afterChange) = {
831666 let assets = getMarketAssets()
832667 let ltvs = split(tryGetString("setup_ltvs"), ",")
833668 let lts = split(tryGetString("setup_lts"), ",")
834669 let rates = getActualRate(assets[0], "sRate")._2
835670 let changeHandler = split(afterChange, ",")
836671 func f (accum,next) = if ((next >= size(assets)))
837672 then accum
838673 else {
839674 let userSupplied = tryGetInteger(((address + "_supplied_") + assets[next]))
840675 let userBorrowed = tryGetInteger(((address + "_borrowed_") + assets[next]))
841676 let needTokenAccounting = if ((afterChange == ""))
842677 then if (if ((userBorrowed != 0))
843678 then true
844679 else (userSupplied != 0))
845680 then true
846681 else false
847682 else true
848683 if (needTokenAccounting)
849684 then {
850685 let assetScale = calcAssetScale(assets[next])
851686 let assetPrice = getTokenPrice(assets[next])
852687 ((accum + fraction(fraction(fraction((userSupplied + (if (if (if ((afterChange != ""))
853688 then (changeHandler[0] == assets[next])
854689 else false)
855690 then (changeHandler[1] == "supplied")
856691 else false)
857692 then parseIntValue(changeHandler[2])
858693 else 0)), rates[(next * 3)].value, Scale16), parseIntValue(ltvs[next]), Scale8), assetPrice._1, assetScale)) - (if (minusBorrowed)
859694 then fraction(fraction(fraction((userBorrowed + (if (if (if ((afterChange != ""))
860695 then (changeHandler[0] == assets[next])
861696 else false)
862697 then (changeHandler[1] == "borrowed")
863698 else false)
864699 then parseIntValue(changeHandler[2])
865700 else 0)), rates[((next * 3) + 1)].value, Scale16), Scale8, parseIntValue(lts[next])), assetPrice._2, assetScale)
866701 else 0))
867702 }
868703 else accum
869704 }
870705
871706 let result = {
872707 let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
873708 let $s = size($l)
874709 let $acc0 = 0
875710 func $f0_1 ($a,$i) = if (($i >= $s))
876711 then $a
877712 else f($a, $l[$i])
878713
879714 func $f0_2 ($a,$i) = if (($i >= $s))
880715 then $a
881716 else throw("List size exceeds 12")
882717
883718 $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)
884719 }
885720 if (debug)
886721 then throw(toString(result))
887722 else $Tuple2(nil, result)
888723 }
889724
890725
891726
892727 @Callable(i)
728+func getUserCollateralV2 (debug,address,sAssetId,bAssetId,minusBorrowed,afterChange) = {
729+ let assets = getMarketAssets()
730+ let sIndex = valueOrErrorMessage(indexOf(assets, sAssetId), "Wrong sAssetId")
731+ let ltvs = split(tryGetString("setup_ltvs"), ",")
732+ let lts = split(tryGetString("setup_lts"), ",")
733+ let sAssetRate = getActualRateOneToken(sAssetId, "sRate")
734+ let changeHandler = split(afterChange, ",")
735+ let userSupplied = tryGetInteger(((address + "_supplied_") + sAssetId))
736+ let userBorrowed = tryGetInteger(((address + "_borrowed_") + bAssetId))
737+ let suppliedChange = if (if ((afterChange != ""))
738+ then (changeHandler[0] == "supplied")
739+ else false)
740+ then parseIntValue(changeHandler[1])
741+ else 0
742+ let sAssetScale = calcAssetScale(sAssetId)
743+ let sAssetPrice = getTokenPrice(sAssetId)
744+ let result = (fraction(fraction(fraction((userSupplied + suppliedChange), sAssetRate._1, Scale16), parseIntValue(ltvs[sIndex]), Scale8), sAssetPrice._1, sAssetScale) - (if (minusBorrowed)
745+ then {
746+ let bIndex = valueOrErrorMessage(indexOf(assets, bAssetId), "Wrong bAssetId")
747+ let bAssetRate = getActualRateOneToken(bAssetId, "bRate")
748+ let borrowedChange = if (if ((afterChange != ""))
749+ then (changeHandler[0] == "borrowed")
750+ else false)
751+ then parseIntValue(changeHandler[1])
752+ else 0
753+ let bAssetScale = calcAssetScale(bAssetId)
754+ let bAssetPrice = getTokenPrice(bAssetId)
755+ fraction(fraction(fraction((userBorrowed + borrowedChange), bAssetRate._1, Scale16), Scale8, parseIntValue(lts[bIndex])), bAssetPrice._2, bAssetScale)
756+ }
757+ else 0))
758+ if (debug)
759+ then throw(toString(result))
760+ else $Tuple2(nil, result)
761+ }
762+
763+
764+
765+@Callable(i)
893766 func getAssetDebt (debug,address,assetIdStr) = {
894767 let userBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr))
895768 let assetScale = calcAssetScale(assetIdStr)
896- let rate = getActualRate(assetIdStr, "bRate")._1
769+ let rate = getActualRateOneToken(assetIdStr, "bRate")._1
897770 let result = fraction(userBorrowed, rate, Scale16)
898771 if (debug)
899772 then throw(toString(result))
900773 else $Tuple2(nil, result)
901774 }
902775
903776
904777
905778 @Callable(i)
906779 func getPrices (debug) = {
907780 let assets = getMarketAssets()
908- func f (accum,next) = if ((next >= size(assets)))
909- then accum
910- else {
911- let assetPrice = getTokenPriceWithRisk(assets[next], 3)
912- ((((accum + toString(assetPrice._1)) + ",") + toString(assetPrice._2)) + "|")
913- }
781+ func f (accum,assetIdStr) = {
782+ let assetPrice = getTokenPriceWithRisk(assetIdStr, 3)
783+ ((((accum + toString(assetPrice._1)) + ",") + toString(assetPrice._2)) + "|")
784+ }
914785
915786 let result = {
916- let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
787+ let $l = assets
917788 let $s = size($l)
918789 let $acc0 = ""
919790 func $f0_1 ($a,$i) = if (($i >= $s))
920791 then $a
921792 else f($a, $l[$i])
922793
923794 func $f0_2 ($a,$i) = if (($i >= $s))
924795 then $a
925- else throw("List size exceeds 12")
796+ else throw("List size exceeds 100")
926797
927- $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)
798+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
928799 }
929800 if (debug)
930801 then throw(result)
931802 else $Tuple2(nil, result)
932803 }
933804
934805
935806
936807 @Callable(i)
937808 func calculateUtilizationRatio (assetIdStr,debug) = if (debug)
938809 then throw(toString(getUr(assetIdStr)))
939810 else $Tuple2(nil, getUr(assetIdStr))
940811
941812
942813
943814 @Callable(i)
944815 func calculateOutdatedUR (assetIdStr,debug) = if (debug)
945816 then throw(toString(getOutdatedUr(assetIdStr)))
946817 else $Tuple2(nil, getOutdatedUr(assetIdStr))
947818
948819
949820
950821 @Callable(i)
951822 func calculateTokenRates (debug) = {
952823 func f (accum,assetIdStr) = {
953824 let rates = tokenRatesRecalc(assetIdStr)
954825 $Tuple2(((((accum._1 + toString(rates[1].value)) + "|") + toString(rates[0].value)) + ","), (accum._2 ++ rates))
955826 }
956827
957828 let parameter = {
958829 let $l = getMarketAssets()
959830 let $s = size($l)
960831 let $acc0 = $Tuple2("", nil)
961832 func $f0_1 ($a,$i) = if (($i >= $s))
962833 then $a
963834 else f($a, $l[$i])
964835
965836 func $f0_2 ($a,$i) = if (($i >= $s))
966837 then $a
967- else throw("List size exceeds 12")
838+ else throw("List size exceeds 100")
968839
969- $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)
840+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
970841 }
971842 if (debug)
972843 then throw(parameter._1)
973844 else $Tuple2(parameter._2, parameter._1)
974845 }
975846
976847
977848
978849 @Callable(i)
979850 func calculateTokensInterest (debug) = {
980851 func f (accum,assetIdStr) = {
981852 let rate = fraction(getInterest(assetIdStr), dayBlocks, Scale8)
982853 ((accum + toString(rate)) + ",")
983854 }
984855
985856 let parameter = {
986857 let $l = getMarketAssets()
987858 let $s = size($l)
988859 let $acc0 = ""
989860 func $f0_1 ($a,$i) = if (($i >= $s))
990861 then $a
991862 else f($a, $l[$i])
992863
993864 func $f0_2 ($a,$i) = if (($i >= $s))
994865 then $a
995- else throw("List size exceeds 12")
866+ else throw("List size exceeds 100")
996867
997- $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)
868+ $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($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90), 91), 92), 93), 94), 95), 96), 97), 98), 99), 100)
998869 }
999870 if (debug)
1000871 then throw(parameter)
1001872 else $Tuple2(nil, parameter)
1002873 }
1003874
1004875
1005876 @Verifier(tx)
1006-func verify () = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'J4QQBsh6FGgZbijQPMDZ7GtNALx2N15K3eHvGhjq43Lt'))
877+func verify () = if (if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'J4QQBsh6FGgZbijQPMDZ7GtNALx2N15K3eHvGhjq43Lt'))
1007878 then true
1008879 else sigVerify(tx.bodyBytes, tx.proofs[0], base58'Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS'))
1009880 then true
1010- else if (sigVerify(tx.bodyBytes, tx.proofs[0], base58'CYCxuKusNeomQkH9wbcEd4rRP1mDYMc7bw8pr82Agi4N'))
1011- then sigVerify(tx.bodyBytes, tx.proofs[0], base58'3oqguA2LUTZFFjJCJ62UybbXFTLwv88tSJ3oMrNSGhiA')
1012- else false)
881+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'CYCxuKusNeomQkH9wbcEd4rRP1mDYMc7bw8pr82Agi4N'))
1013882 then true
1014- else sigVerify(tx.bodyBytes, tx.proofs[0], base58'9t2BGHP6EddYu5xvArqDkhHPGbf8ZiYUaWgFcEPtRZhE')
883+ else sigVerify(tx.bodyBytes, tx.proofs[0], base58'D2QvCRmindBLJfAb2GWftPgjkLUZFsoCiYxJFDisaKqN'))
884+ then if (sigVerify(tx.bodyBytes, tx.proofs[1], base58'3oqguA2LUTZFFjJCJ62UybbXFTLwv88tSJ3oMrNSGhiA'))
885+ then true
886+ else sigVerify(tx.bodyBytes, tx.proofs[1], base58'9t2BGHP6EddYu5xvArqDkhHPGbf8ZiYUaWgFcEPtRZhE')
887+ else false
1015888

github/deemru/w8io/3ef1775 
170.75 ms