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/QcEBQtheGx5QWRkcmVzcwIJbGlxdWlkYXRlCQDMCAIFCWFkZHJlc3NJZAkAzAgCBQVwb3NJZAkAzAgCBRJsQW1vdW50V2l0aFBlbmFsdHkFA25pbAUDbmlsAwkAAQIFAUACA0ludAUBQAkAAgEJAKwCAgkAAwEFAUACGCBjb3VsZG4ndCBiZSBjYXN0IHRvIEludAMJAAACBQlzQW1vdW50THAFCXNBbW91bnRMcAQHc0Ftb3VudAkAawMFCXNBbW91bnRMcAUHU2NhbGUxNggJAJEDAgULcmF0ZXNSZXN1bHQAAQV2YWx1ZQMFBWRlYnVnCQACAQIVbGlxdWlkYXRpb24gd2lsbCBwYXNzCQDOCAIJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFAWkGY2FsbGVyCQBlAgUSbEFtb3VudFdpdGhQZW5hbHR5BQdsQW1vdW50BQhiQXNzZXRJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8FC3NBc3NldElkU3RyCQBlAgUQY3VycmVudFNQb3NpdGlvbgUHc0Ftb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8FC2JBc3NldElkU3RyCQBlAgUQY3VycmVudEJQb3NpdGlvbgUIbGJBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgIPdG90YWxfc3VwcGxpZWRfBQtzQXNzZXRJZFN0cgkAZQIJAQ10cnlHZXRJbnRlZ2VyAQkArAICAg90b3RhbF9zdXBwbGllZF8FC3NBc3NldElkU3RyBQdzQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICD3RvdGFsX2JvcnJvd2VkXwULYkFzc2V0SWRTdHIJAGUCCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgIPdG90YWxfYm9ycm93ZWRfBQtiQXNzZXRJZFN0cgUIbGJBbW91bnQFA25pbAULcmF0ZXNSZXN1bHQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQERZ2V0VXNlckNvbGxhdGVyYWwEBWRlYnVnB2FkZHJlc3MNbWludXNCb3Jyb3dlZAthZnRlckNoYW5nZQQGYXNzZXRzCQEPZ2V0TWFya2V0QXNzZXRzAAQEbHR2cwkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIKc2V0dXBfbHR2cwIBLAQDbHRzCQC1CQIJAQx0cnlHZXRTdHJpbmcBAglzZXR1cF9sdHMCASwEBXJhdGVzCAkBDWdldEFjdHVhbFJhdGUCCQCRAwIFBmFzc2V0cwAAAgVzUmF0ZQJfMgQNY2hhbmdlSGFuZGxlcgkAtQkCBQthZnRlckNoYW5nZQIBLAoBAWYCBWFjY3VtBG5leHQDCQBnAgUEbmV4dAkAkAMBBQZhc3NldHMFBWFjY3VtBAx1c2VyU3VwcGxpZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9zdXBwbGllZF8JAJEDAgUGYXNzZXRzBQRuZXh0BAx1c2VyQm9ycm93ZWQJAQ10cnlHZXRJbnRlZ2VyAQkArAICCQCsAgIFB2FkZHJlc3MCCl9ib3Jyb3dlZF8JAJEDAgUGYXNzZXRzBQRuZXh0BBNuZWVkVG9rZW5BY2NvdW50aW5nAwkAAAIFC2FmdGVyQ2hhbmdlAgADAwkBAiE9AgUMdXNlckJvcnJvd2VkAAAGCQECIT0CBQx1c2VyU3VwcGxpZWQAAAYHBgMFE25lZWRUb2tlbkFjY291bnRpbmcECmFzc2V0U2NhbGUJAQ5jYWxjQXNzZXRTY2FsZQEJAJEDAgUGYXNzZXRzBQRuZXh0BAphc3NldFByaWNlCQENZ2V0VG9rZW5QcmljZQEJAJEDAgUGYXNzZXRzBQRuZXh0CQBlAgkAZAIFBWFjY3VtCQBrAwkAawMJAGsDCQBkAgUMdXNlclN1cHBsaWVkAwMDCQECIT0CBQthZnRlckNoYW5nZQIACQAAAgkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAAJAJEDAgUGYXNzZXRzBQRuZXh0BwkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgABAghzdXBwbGllZAcJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQ1jaGFuZ2VIYW5kbGVyAAIAAAgJAJEDAgUFcmF0ZXMJAGgCBQRuZXh0AAMFdmFsdWUFB1NjYWxlMTYJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQRsdHZzBQRuZXh0BQZTY2FsZTgIBQphc3NldFByaWNlAl8xBQphc3NldFNjYWxlAwUNbWludXNCb3Jyb3dlZAkAawMJAGsDCQBrAwkAZAIFDHVzZXJCb3Jyb3dlZAMDAwkBAiE9AgULYWZ0ZXJDaGFuZ2UCAAkAAAIJAJEDAgUNY2hhbmdlSGFuZGxlcgAACQCRAwIFBmFzc2V0cwUEbmV4dAcJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAQIIYm9ycm93ZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgACAAAICQCRAwIFBXJhdGVzCQBkAgkAaAIFBG5leHQAAwABBXZhbHVlBQdTY2FsZTE2BQZTY2FsZTgJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQNsdHMFBG5leHQIBQphc3NldFByaWNlAl8yBQphc3NldFNjYWxlAAAFBWFjY3VtBAZyZXN1bHQKAAIkbAkAzAgCAAAJAMwIAgABCQDMCAIAAgkAzAgCAAMJAMwIAgAECQDMCAIABQkAzAgCAAYJAMwIAgAHCQDMCAIACAkAzAgCAAkJAMwIAgAKCQDMCAIACwUDbmlsCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQFmAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAMFBWRlYnVnCQACAQkApAMBBQZyZXN1bHQJAJQKAgUDbmlsBQZyZXN1bHQBaQETZ2V0VXNlckNvbGxhdGVyYWxWMgYFZGVidWcHYWRkcmVzcwhzQXNzZXRJZAhiQXNzZXRJZA1taW51c0JvcnJvd2VkC2FmdGVyQ2hhbmdlBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMABAZzSW5kZXgJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAzwgCBQZhc3NldHMFCHNBc3NldElkAg5Xcm9uZyBzQXNzZXRJZAQEbHR2cwkAtQkCCQEMdHJ5R2V0U3RyaW5nAQIKc2V0dXBfbHR2cwIBLAQDbHRzCQC1CQIJAQx0cnlHZXRTdHJpbmcBAglzZXR1cF9sdHMCASwECnNBc3NldFJhdGUJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQhzQXNzZXRJZAIFc1JhdGUEDWNoYW5nZUhhbmRsZXIJALUJAgULYWZ0ZXJDaGFuZ2UCASwEDHVzZXJTdXBwbGllZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX3N1cHBsaWVkXwUIc0Fzc2V0SWQEDHVzZXJCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUIYkFzc2V0SWQEDnN1cHBsaWVkQ2hhbmdlAwMJAQIhPQIFC2FmdGVyQ2hhbmdlAgAJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAAIIc3VwcGxpZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgABAAAEC3NBc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQhzQXNzZXRJZAQLc0Fzc2V0UHJpY2UJAQ1nZXRUb2tlblByaWNlAQUIc0Fzc2V0SWQEBnJlc3VsdAkAZQIJAGsDCQBrAwkAawMJAGQCBQx1c2VyU3VwcGxpZWQFDnN1cHBsaWVkQ2hhbmdlCAUKc0Fzc2V0UmF0ZQJfMQUHU2NhbGUxNgkBDXBhcnNlSW50VmFsdWUBCQCRAwIFBGx0dnMFBnNJbmRleAUGU2NhbGU4CAULc0Fzc2V0UHJpY2UCXzEFC3NBc3NldFNjYWxlAwUNbWludXNCb3Jyb3dlZAQGYkluZGV4CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAM8IAgUGYXNzZXRzBQhiQXNzZXRJZAIOV3JvbmcgYkFzc2V0SWQECmJBc3NldFJhdGUJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQhiQXNzZXRJZAIFYlJhdGUEDmJvcnJvd2VkQ2hhbmdlAwMJAQIhPQIFC2FmdGVyQ2hhbmdlAgAJAAACCQCRAwIFDWNoYW5nZUhhbmRsZXIAAAIIYm9ycm93ZWQHCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUNY2hhbmdlSGFuZGxlcgABAAAEC2JBc3NldFNjYWxlCQEOY2FsY0Fzc2V0U2NhbGUBBQhiQXNzZXRJZAQLYkFzc2V0UHJpY2UJAQ1nZXRUb2tlblByaWNlAQUIYkFzc2V0SWQJAGsDCQBrAwkAawMJAGQCBQx1c2VyQm9ycm93ZWQFDmJvcnJvd2VkQ2hhbmdlCAUKYkFzc2V0UmF0ZQJfMQUHU2NhbGUxNgUGU2NhbGU4CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUDbHRzBQZiSW5kZXgIBQtiQXNzZXRQcmljZQJfMgULYkFzc2V0U2NhbGUAAAMFBWRlYnVnCQACAQkApAMBBQZyZXN1bHQJAJQKAgUDbmlsBQZyZXN1bHQBaQEMZ2V0QXNzZXREZWJ0AwVkZWJ1ZwdhZGRyZXNzCmFzc2V0SWRTdHIEDHVzZXJCb3Jyb3dlZAkBDXRyeUdldEludGVnZXIBCQCsAgIJAKwCAgUHYWRkcmVzcwIKX2JvcnJvd2VkXwUKYXNzZXRJZFN0cgQKYXNzZXRTY2FsZQkBDmNhbGNBc3NldFNjYWxlAQUKYXNzZXRJZFN0cgQEcmF0ZQgJARVnZXRBY3R1YWxSYXRlT25lVG9rZW4CBQphc3NldElkU3RyAgViUmF0ZQJfMQQGcmVzdWx0CQBrAwUMdXNlckJvcnJvd2VkBQRyYXRlBQdTY2FsZTE2AwUFZGVidWcJAAIBCQCkAwEFBnJlc3VsdAkAlAoCBQNuaWwFBnJlc3VsdAFpAQlnZXRQcmljZXMBBWRlYnVnBAZhc3NldHMJAQ9nZXRNYXJrZXRBc3NldHMACgEBZgIFYWNjdW0KYXNzZXRJZFN0cgQKYXNzZXRQcmljZQkBFWdldFRva2VuUHJpY2VXaXRoUmlzawIFCmFzc2V0SWRTdHIAAwkArAICCQCsAgIJAKwCAgkArAICBQVhY2N1bQkApAMBCAUKYXNzZXRQcmljZQJfMQIBLAkApAMBCAUKYXNzZXRQcmljZQJfMgIBfAQGcmVzdWx0CgACJGwFBmFzc2V0cwoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEBZgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIVTGlzdCBzaXplIGV4Y2VlZHMgMTAwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAB8AIAAhACIAIwAkACUAJgAnACgAKQAqACsALAAtAC4ALwAwADEAMgAzADQANQA2ADcAOAA5ADoAOwA8AD0APgA/AEAAQQBCAEMARABFAEYARwBIAEkASgBLAEwATQBOAE8AUABRAFIAUwBUAFUAVgBXAFgAWQBaAFsAXABdAF4AXwBgAGEAYgBjAGQDBQVkZWJ1ZwkAAgEFBnJlc3VsdAkAlAoCBQNuaWwFBnJlc3VsdAFpARljYWxjdWxhdGVVdGlsaXphdGlvblJhdGlvAgphc3NldElkU3RyBWRlYnVnAwUFZGVidWcJAAIBCQCkAwEJAQVnZXRVcgEFCmFzc2V0SWRTdHIJAJQKAgUDbmlsCQEFZ2V0VXIBBQphc3NldElkU3RyAWkBE2NhbGN1bGF0ZU91dGRhdGVkVVICCmFzc2V0SWRTdHIFZGVidWcDBQVkZWJ1ZwkAAgEJAKQDAQkBDWdldE91dGRhdGVkVXIBBQphc3NldElkU3RyCQCUCgIFA25pbAkBDWdldE91dGRhdGVkVXIBBQphc3NldElkU3RyAWkBE2NhbGN1bGF0ZVRva2VuUmF0ZXMBBWRlYnVnCgEBZgIFYWNjdW0KYXNzZXRJZFN0cgQFcmF0ZXMJARB0b2tlblJhdGVzUmVjYWxjAQUKYXNzZXRJZFN0cgkAlAoCCQCsAgIJAKwCAgkArAICCQCsAgIIBQVhY2N1bQJfMQkApAMBCAkAkQMCBQVyYXRlcwABBXZhbHVlAgF8CQCkAwEICQCRAwIFBXJhdGVzAAAFdmFsdWUCASwJAM4IAggFBWFjY3VtAl8yBQVyYXRlcwQJcGFyYW1ldGVyCgACJGwJAQ9nZXRNYXJrZXRBc3NldHMACgACJHMJAJADAQUCJGwKAAUkYWNjMAkAlAoCAgAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEBZgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIVTGlzdCBzaXplIGV4Y2VlZHMgMTAwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAB8AIAAhACIAIwAkACUAJgAnACgAKQAqACsALAAtAC4ALwAwADEAMgAzADQANQA2ADcAOAA5ADoAOwA8AD0APgA/AEAAQQBCAEMARABFAEYARwBIAEkASgBLAEwATQBOAE8AUABRAFIAUwBUAFUAVgBXAFgAWQBaAFsAXABdAF4AXwBgAGEAYgBjAGQDBQVkZWJ1ZwkAAgEIBQlwYXJhbWV0ZXICXzEJAJQKAggFCXBhcmFtZXRlcgJfMggFCXBhcmFtZXRlcgJfMQFpARdjYWxjdWxhdGVUb2tlbnNJbnRlcmVzdAEFZGVidWcKAQFmAgVhY2N1bQphc3NldElkU3RyBARyYXRlCQBrAwkBC2dldEludGVyZXN0AQUKYXNzZXRJZFN0cgUJZGF5QmxvY2tzBQZTY2FsZTgJAKwCAgkArAICBQVhY2N1bQkApAMBBQRyYXRlAgEsBAlwYXJhbWV0ZXIKAAIkbAkBD2dldE1hcmtldEFzc2V0cwAKAAIkcwkAkAMBBQIkbAoABSRhY2MwAgAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBAWYCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFUxpc3Qgc2l6ZSBleGNlZWRzIDEwMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQAFQAWABcAGAAZABoAGwAcAB0AHgAfACAAIQAiACMAJAAlACYAJwAoACkAKgArACwALQAuAC8AMAAxADIAMwA0ADUANgA3ADgAOQA6ADsAPAA9AD4APwBAAEEAQgBDAEQARQBGAEcASABJAEoASwBMAE0ATgBPAFAAUQBSAFMAVABVAFYAVwBYAFkAWgBbAFwAXQBeAF8AYABhAGIAYwBkAwUFZGVidWcJAAIBBQlwYXJhbWV0ZXIJAJQKAgUDbmlsBQlwYXJhbWV0ZXIBAnR4AQZ2ZXJpZnkAAwMDAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAEg/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:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let kAxlyLp = "_axlyLP" | |
5 | + | ||
4 | 6 | let axlyAddress = Address(base58'3PLsYkBw7taejV1J3qWPCN2yeyVRu31d5HW') | |
5 | - | ||
6 | - | let axlyLPs = ["7KZbJrVopwJhkdwbe1eFDBbex4dkY63MxjTNjqXtrzj1", "Btw3G1j4wQgdp49PTxaFkNvn75dQtqGDM7ejQppHnWC1", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "F2AKkA513k5yHEJkLsU6vWxCYYk811GpjLhwEv2WGwZ9", "4CQ5CPGLXLbWBUs2JBjKUaRqF49CmKHkwzvPgSvQpAQV", "6iMB6LKSrgv9waEvEnN6Ydyx7dfxPnGcTw8318WVm5bR"] | |
7 | 7 | ||
8 | 8 | let axlyFarmingAddress = Address(base58'3P37f8Y3rNpKzSkHNVZMAStXdpmNBgftHN9') | |
9 | 9 | ||
89 | 89 | ||
90 | 90 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
91 | 91 | then $a | |
92 | - | else throw("List size exceeds | |
92 | + | else throw("List size exceeds 100") | |
93 | 93 | ||
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) | |
95 | 95 | } | |
96 | 96 | ||
97 | 97 | ||
174 | 174 | } | |
175 | 175 | ||
176 | 176 | ||
177 | - | func tokenRatesRecalc (assetIdStr) = if (( | |
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)] | |
179 | 179 | else { | |
180 | 180 | let interest = getInterest(assetIdStr) | |
181 | 181 | let ur = getOutdatedUr(assetIdStr) | |
182 | - | let lastRecalcHeight = tryGetInteger("lastRateHeight") | |
182 | + | let lastRecalcHeight = valueOrElse(getInteger(this, (assetIdStr + "_lastRateHeight")), tryGetInteger("lastRateHeight")) | |
183 | 183 | let lastBRate = max([tryGetInteger((assetIdStr + "_bRate")), Scale16]) | |
184 | 184 | let newBRate = (lastBRate + ((height - lastRecalcHeight) * interest)) | |
185 | 185 | let lastSRate = max([tryGetInteger((assetIdStr + "_sRate")), Scale16]) | |
186 | 186 | 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)] | |
188 | 188 | } | |
189 | 189 | ||
190 | 190 | ||
207 | 207 | ||
208 | 208 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
209 | 209 | then $a | |
210 | - | else throw("List size exceeds | |
210 | + | else throw("List size exceeds 100") | |
211 | 211 | ||
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) | |
213 | 221 | } | |
214 | 222 | ||
215 | 223 | ||
220 | 228 | } | |
221 | 229 | ||
222 | 230 | ||
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 | - | ||
241 | 231 | func getTokenPriceWithRisk (assetIdStr,riskAversity) = if (if ((assetIdStr == "9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi")) | |
242 | 232 | then true | |
243 | 233 | else (assetIdStr == "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW")) | |
245 | 235 | else { | |
246 | 236 | let price = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_twap5B")) | |
247 | 237 | let riskLevel = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_riskLevel")) | |
248 | - | if (( | |
238 | + | if (tryGetBoolean((assetIdStr + kAxlyLp))) | |
249 | 239 | then { | |
250 | 240 | let p = { | |
251 | 241 | let @ = reentrantInvoke(axlyAddress, "getShareAssetPriceREADONLY", [assetIdStr], nil) | |
274 | 264 | } | |
275 | 265 | ||
276 | 266 | ||
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 | - | ||
295 | 267 | func supplyInternal (assetIdStr,assetAmount,address) = if (!(tryGetBoolean("setup_active"))) | |
296 | 268 | then throw("market is stopped") | |
297 | 269 | else { | |
298 | - | let $ | |
299 | - | let sRate = $ | |
300 | - | let ratesRecalcResult = $ | |
270 | + | let $t085378612 = getActualRateOneToken(assetIdStr, "sRate") | |
271 | + | let sRate = $t085378612._1 | |
272 | + | let ratesRecalcResult = $t085378612._2 | |
301 | 273 | let amount = fraction(assetAmount, Scale16, sRate, DOWN) | |
302 | 274 | let maxSupply = match getString(("setup_maxSupply_" + assetIdStr)) { | |
303 | 275 | case x: String => | |
307 | 279 | } | |
308 | 280 | let assetPrice = getTokenPrice(assetIdStr) | |
309 | 281 | let newTotalSupplied = (tryGetInteger(((address + "_supplied_") + assetIdStr)) + amount) | |
310 | - | let rate = | |
282 | + | let rate = sRate | |
311 | 283 | let assetScale = calcAssetScale(assetIdStr) | |
312 | 284 | let newTotalSuppliedUsd = fraction(fraction(newTotalSupplied, rate, Scale16), assetPrice._1, assetScale) | |
313 | 285 | if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit)) | |
316 | 288 | then (newTotalSuppliedUsd > maxSupply) | |
317 | 289 | else false) | |
318 | 290 | 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 | + | } | |
320 | 295 | } | |
321 | 296 | ||
322 | 297 | ||
323 | - | func borrowInternal (assetIdStr,assetAmount,address | |
324 | - | let $ | |
325 | - | let bRate = $ | |
326 | - | let ratesRecalcResult = $ | |
298 | + | func borrowInternal (assetIdStr,assetAmount,address) = { | |
299 | + | let $t099209995 = getActualRateOneToken(assetIdStr, "bRate") | |
300 | + | let bRate = $t099209995._1 | |
301 | + | let ratesRecalcResult = $t099209995._2 | |
327 | 302 | 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 | + | } | |
355 | 321 | } | |
356 | 322 | ||
357 | 323 | ||
359 | 325 | func flashPosition (user,sAssetIdStr,bAssetIdStr,bAmount,callbackAddress,callbackFunction,callbackArgs) = if ((i.caller != axlyAddress)) | |
360 | 326 | then throw("available only for Axly protocol") | |
361 | 327 | else { | |
362 | - | let borrowRes = borrowInternal(bAssetIdStr, bAmount, user | |
328 | + | let borrowRes = borrowInternal(bAssetIdStr, bAmount, user) | |
363 | 329 | let bAssetId = getAssetBytes(bAssetIdStr) | |
364 | 330 | let callback = reentrantInvoke(addressFromStringValue(callbackAddress), callbackFunction, split(callbackArgs, ","), [AttachedPayment(bAssetId, bAmount)]) | |
365 | 331 | if ((callback == callback)) | |
373 | 339 | if ((sAmount == sAmount)) | |
374 | 340 | then { | |
375 | 341 | 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) | |
397 | 345 | } | |
398 | 346 | else throw("Strict value is not equal to itself.") | |
399 | 347 | } | |
410 | 358 | else { | |
411 | 359 | let assetIdStr = getAssetString(i.payments[0].assetId) | |
412 | 360 | 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 | + | } | |
417 | 369 | } | |
418 | 370 | ||
419 | 371 | ||
420 | 372 | ||
421 | 373 | @Callable(i) | |
422 | 374 | func withdraw (assetIdStr,assetAmount) = { | |
423 | - | let $ | |
424 | - | let sRate = $ | |
425 | - | let ratesRecalcResult = $ | |
375 | + | let $t01260412679 = getActualRateOneToken(assetIdStr, "sRate") | |
376 | + | let sRate = $t01260412679._1 | |
377 | + | let ratesRecalcResult = $t01260412679._2 | |
426 | 378 | let amount = fraction(assetAmount, Scale16, sRate, CEILING) | |
427 | 379 | let address = toString(i.caller) | |
428 | 380 | let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr)) | |
429 | 381 | let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr)) | |
430 | 382 | let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr)) | |
431 | - | let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr)) | |
432 | 383 | let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil) | |
433 | 384 | if ((axlyNotify == axlyNotify)) | |
434 | 385 | then { | |
435 | - | let collateralValueInv = invoke(this, " | |
386 | + | let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, assetIdStr, "", false, ("supplied," + toString(-(amount)))], nil) | |
436 | 387 | if ((collateralValueInv == collateralValueInv)) | |
437 | 388 | then { | |
438 | 389 | let collateralValue = match collateralValueInv { | |
447 | 398 | then throw("you dont have enough collateral for this operation") | |
448 | 399 | else if ((amount > (assetSupplied - assetBorrowed))) | |
449 | 400 | then throw("this amount is not available on the market") | |
450 | - | else if ((amount > | |
451 | - | then throw("this amount is not available | |
401 | + | else if ((amount > userAssetSupplied)) | |
402 | + | then throw("this amount is not available on the market") | |
452 | 403 | 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) | |
453 | 404 | } | |
454 | 405 | else throw("Strict value is not equal to itself.") | |
459 | 410 | ||
460 | 411 | ||
461 | 412 | @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"))) | |
494 | 414 | then throw("market is stopped") | |
495 | 415 | else if (!(verifySentinelRights(i.caller))) | |
496 | 416 | then throw("available only for whitelist") | |
500 | 420 | then true | |
501 | 421 | else (i.payments[0].amount == 0)) | |
502 | 422 | 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)) | |
524 | 445 | if ((userSupplied > 0)) | |
525 | - | then | |
526 | - | else | |
446 | + | then [IntegerEntry(((address + "_supplied_") + lpId), 0), IntegerEntry(("total_supplied_" + lpId), (tryGetInteger(("total_supplied_" + lpId)) - userSupplied))] | |
447 | + | else nil | |
527 | 448 | } | |
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 | + | } | |
547 | 454 | ||
548 | 455 | ||
549 | 456 | ||
596 | 503 | then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8')) | |
597 | 504 | else false) | |
598 | 505 | 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)] | |
600 | 516 | ||
601 | 517 | ||
602 | 518 | ||
622 | 538 | func claimToReserveFund (debug) = { | |
623 | 539 | let assets = getMarketAssets() | |
624 | 540 | 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 | + | } | |
641 | 556 | ||
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]) | |
649 | 563 | ||
650 | - | ||
651 | - | ||
652 | - | ||
564 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
565 | + | then $a | |
566 | + | else throw("List size exceeds 100") | |
653 | 567 | ||
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)) | |
655 | 574 | } | |
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 | - | } | |
662 | 575 | ||
663 | 576 | if (debug) | |
664 | 577 | then throw(liIntToStr(parameter)) | |
665 | 578 | else $Tuple2({ | |
666 | - | let $l = | |
579 | + | let $l = assets | |
667 | 580 | let $s = size($l) | |
668 | - | let $acc0 = nil | |
581 | + | let $acc0 = $Tuple2(nil, 0) | |
669 | 582 | func $f1_1 ($a,$i) = if (($i >= $s)) | |
670 | 583 | then $a | |
671 | 584 | else f2($a, $l[$i]) | |
672 | 585 | ||
673 | 586 | func $f1_2 ($a,$i) = if (($i >= $s)) | |
674 | 587 | then $a | |
675 | - | else throw("List size exceeds | |
588 | + | else throw("List size exceeds 100") | |
676 | 589 | ||
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 | |
679 | 592 | } | |
680 | 593 | ||
681 | 594 | ||
688 | 601 | ||
689 | 602 | ||
690 | 603 | @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) | |
775 | 604 | func liquidateV2 (debug,address,sAssetIdStr,bAssetIdStr,liquidatePercent) = if (!(verifyLiquidatorRights(i.caller))) | |
776 | 605 | then throw("available for whitelist only") | |
777 | 606 | else if (!(tryGetBoolean("setup_active"))) | |
778 | 607 | then throw("market is stopped") | |
779 | 608 | else { | |
780 | - | let | |
781 | - | if (( | |
609 | + | let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, sAssetIdStr, bAssetIdStr, true, ""], nil) | |
610 | + | if ((collateralValueInv == collateralValueInv)) | |
782 | 611 | then { | |
612 | + | let userCollateral = match collateralValueInv { | |
613 | + | case x: Int => | |
614 | + | x | |
615 | + | case _ => | |
616 | + | throw("can't get user collateral value") | |
617 | + | } | |
783 | 618 | let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr)) | |
784 | 619 | let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr)) | |
785 | - | if (( | |
620 | + | if (!(tryGetBoolean((sAssetIdStr + kAxlyLp)))) | |
786 | 621 | then throw("wrong lp asset") | |
787 | 622 | else if ((0 >= currentBPosition)) | |
788 | 623 | then throw("user has no borrow in this token") | |
796 | 631 | let asset1Num = value(indexOf(marketAssets, bAssetIdStr)) | |
797 | 632 | let asset0Num = value(indexOf(marketAssets, sAssetIdStr)) | |
798 | 633 | let penalty = parseIntValue(value(split(tryGetString("setup_penalties"), ",")[asset1Num])) | |
799 | - | let $ | |
800 | - | let bRate = $ | |
801 | - | let ratesResult = $ | |
634 | + | let $t02189121961 = getActualRateOneToken(bAssetIdStr, "bRate") | |
635 | + | let bRate = $t02189121961._1 | |
636 | + | let ratesResult = $t02189121961._2 | |
802 | 637 | let bAmount = fraction(currentBPosition, bRate, Scale16) | |
803 | 638 | let lAmount = fraction(bAmount, liquidatePercent, 100) | |
804 | 639 | let lAmountWithPenalty = fraction(lAmount, (Scale8 + penalty), Scale8) | |
813 | 648 | } | |
814 | 649 | if ((sAmountLp == sAmountLp)) | |
815 | 650 | then { | |
816 | - | let sAmount = fraction(sAmountLp, Scale16, ratesResult[ | |
651 | + | let sAmount = fraction(sAmountLp, Scale16, ratesResult[1].value) | |
817 | 652 | if (debug) | |
818 | 653 | then throw("liquidation will pass") | |
819 | 654 | 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) | |
890 | 725 | ||
891 | 726 | ||
892 | 727 | @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) | |
893 | 766 | func getAssetDebt (debug,address,assetIdStr) = { | |
894 | 767 | let userBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr)) | |
895 | 768 | let assetScale = calcAssetScale(assetIdStr) | |
896 | - | let rate = | |
769 | + | let rate = getActualRateOneToken(assetIdStr, "bRate")._1 | |
897 | 770 | let result = fraction(userBorrowed, rate, Scale16) | |
898 | 771 | if (debug) | |
899 | 772 | then throw(toString(result)) | |
905 | 778 | @Callable(i) | |
906 | 779 | func getPrices (debug) = { | |
907 | 780 | 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 | + | } | |
914 | 785 | ||
915 | 786 | let result = { | |
916 | - | let $l = | |
787 | + | let $l = assets | |
917 | 788 | let $s = size($l) | |
918 | 789 | let $acc0 = "" | |
919 | 790 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
922 | 793 | ||
923 | 794 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
924 | 795 | then $a | |
925 | - | else throw("List size exceeds | |
796 | + | else throw("List size exceeds 100") | |
926 | 797 | ||
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) | |
928 | 799 | } | |
929 | 800 | if (debug) | |
930 | 801 | then throw(result) | |
964 | 835 | ||
965 | 836 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
966 | 837 | then $a | |
967 | - | else throw("List size exceeds | |
838 | + | else throw("List size exceeds 100") | |
968 | 839 | ||
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) | |
970 | 841 | } | |
971 | 842 | if (debug) | |
972 | 843 | then throw(parameter._1) | |
992 | 863 | ||
993 | 864 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
994 | 865 | then $a | |
995 | - | else throw("List size exceeds | |
866 | + | else throw("List size exceeds 100") | |
996 | 867 | ||
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) | |
998 | 869 | } | |
999 | 870 | if (debug) | |
1000 | 871 | then throw(parameter) | |
1003 | 874 | ||
1004 | 875 | ||
1005 | 876 | @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')) | |
1007 | 878 | then true | |
1008 | 879 | else sigVerify(tx.bodyBytes, tx.proofs[0], base58'Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS')) | |
1009 | 880 | 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')) | |
1013 | 882 | 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 | |
1015 | 888 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let kAxlyLp = "_axlyLP" | |
5 | + | ||
4 | 6 | let axlyAddress = Address(base58'3PLsYkBw7taejV1J3qWPCN2yeyVRu31d5HW') | |
5 | - | ||
6 | - | let axlyLPs = ["7KZbJrVopwJhkdwbe1eFDBbex4dkY63MxjTNjqXtrzj1", "Btw3G1j4wQgdp49PTxaFkNvn75dQtqGDM7ejQppHnWC1", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "F2AKkA513k5yHEJkLsU6vWxCYYk811GpjLhwEv2WGwZ9", "4CQ5CPGLXLbWBUs2JBjKUaRqF49CmKHkwzvPgSvQpAQV", "6iMB6LKSrgv9waEvEnN6Ydyx7dfxPnGcTw8318WVm5bR"] | |
7 | 7 | ||
8 | 8 | let axlyFarmingAddress = Address(base58'3P37f8Y3rNpKzSkHNVZMAStXdpmNBgftHN9') | |
9 | 9 | ||
10 | 10 | let reserveFundAddress = Address(base58'3P4kBiU4wr2yV1S5gMfu3MdkVvy7kxXHsKe') | |
11 | 11 | ||
12 | 12 | let reserveFund = 20 | |
13 | 13 | ||
14 | 14 | let aggregatorAddress = Address(base58'3PGFHzVGT4NTigwCKP1NcwoXkodVZwvBuuU') | |
15 | 15 | ||
16 | 16 | let shutdownWhitelist = [base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL', base58'3PAxdDSmN758L5SHSGRC5apEtQE2aApZotG', base58'3PJKmXoHJvVeQXjSJdhtkUcFDtdiQqMbUTD', base58'3PQdNxynJy5mche2kxMVc5shXWzK8Gstq3o', base58'3PHbdpaKzz8EiAngGHaFu2hVuNCdsC67qh3', base58'3P6Ksahs71SiKQgQ4qaZuFAVhqncdi2nvJQ'] | |
17 | 17 | ||
18 | 18 | func verifyLiquidatorRights (address) = !(if (if (if ((address != Address(base58'3PFHNdEPu71QwBAnc2HZPcrbxoseNfUvHEA'))) | |
19 | 19 | then (address != Address(base58'3PCqdm1mAoQqR46oZotFanmqb5CLUvrKEo2')) | |
20 | 20 | else false) | |
21 | 21 | then (address != Address(base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL')) | |
22 | 22 | else false) | |
23 | 23 | then (address != Address(base58'3PHbdpaKzz8EiAngGHaFu2hVuNCdsC67qh3')) | |
24 | 24 | else false) | |
25 | 25 | ||
26 | 26 | ||
27 | 27 | func verifySentinelRights (address) = !(if ((address != reserveFundAddress)) | |
28 | 28 | then (address != axlyAddress) | |
29 | 29 | else false) | |
30 | 30 | ||
31 | 31 | ||
32 | 32 | func getRateCurve (assetIdStr) = match assetIdStr { | |
33 | 33 | case _ => | |
34 | 34 | if (("9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi" == $match0)) | |
35 | 35 | then $Tuple4(2000000, 25000000, 80000000, 100000000) | |
36 | 36 | else if (("HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW" == $match0)) | |
37 | 37 | then $Tuple4(2000000, 25000000, 80000000, 100000000) | |
38 | 38 | else if (("34N9YcEETLWn93qYQ64EsP1x89tSruJU44RrEMSXXEPJ" == $match0)) | |
39 | 39 | then $Tuple4(2000000, 25000000, 80000000, 100000000) | |
40 | 40 | else if (("6XtHjpXbs9RRJP2Sr9GUyVqzACcby9TkThHXnjVC5CDJ" == $match0)) | |
41 | 41 | then $Tuple4(2000000, 25000000, 80000000, 100000000) | |
42 | 42 | else if (("DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p" == $match0)) | |
43 | 43 | then $Tuple4(2000000, 25000000, 80000000, 100000000) | |
44 | 44 | else if (("Ajso6nTTjptu2UHLx6hfSXVtHFtRBJCkKYd5SAyj7zf5" == $match0)) | |
45 | 45 | then $Tuple4(2000000, 40000000, 80000000, 150000000) | |
46 | 46 | else if (("HEB8Qaw9xrWpWs8tHsiATYGBWDBtP2S7kcPALrMu43AS" == $match0)) | |
47 | 47 | then $Tuple4(0, 20000000, 80000000, 40000000) | |
48 | 48 | else if (("WAVES" == $match0)) | |
49 | 49 | then $Tuple4(2000000, 30000000, 80000000, 50000000) | |
50 | 50 | else if (("Atqv59EYzjFGuitKVnMRk6H8FukjoV3ktPorbEys25on" == $match0)) | |
51 | 51 | then $Tuple4(0, 20000000, 80000000, 40000000) | |
52 | 52 | else if (("DSbbhLsSTeDg5Lsiufk2Aneh3DjVqJuPr2M9uU1gwy5p" == $match0)) | |
53 | 53 | then $Tuple4(0, 20000000, 80000000, 100000000) | |
54 | 54 | else if (("8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91" == $match0)) | |
55 | 55 | then $Tuple4(0, 30000000, 80000000, 40000000) | |
56 | 56 | else if (("At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL" == $match0)) | |
57 | 57 | then $Tuple4(0, 25000000, 80000000, 40000000) | |
58 | 58 | else if (("8LQW8f7P5d5PZM7GtZEBgaqRPGSzS3DfPuiXrURJ4AJS" == $match0)) | |
59 | 59 | then $Tuple4(2000000, 30000000, 80000000, 50000000) | |
60 | 60 | else if (("474jTeYx2r2Va35794tCScAXWJG9hU2HcgxzMowaZUnu" == $match0)) | |
61 | 61 | then $Tuple4(2000000, 30000000, 80000000, 50000000) | |
62 | 62 | else if (("5UYBPpq4WoU5n4MwpFkgJnW3Fq4B1u3ukpK33ik4QerR" == $match0)) | |
63 | 63 | then $Tuple4(2000000, 30000000, 80000000, 50000000) | |
64 | 64 | else if (("2thsACuHmzDMuNezPM32wg9a3BwUzBWDeSKakgz3cw21" == $match0)) | |
65 | 65 | then $Tuple4(2000000, 40000000, 80000000, 100000000) | |
66 | 66 | else if (("YiNbofFzC17jEHHCMwrRcpy9MrrjabMMLZxg8g5xmf7" == $match0)) | |
67 | 67 | then $Tuple4(2000000, 30000000, 80000000, 80000000) | |
68 | 68 | else $Tuple4(0, 20000000, 80000000, 80000000) | |
69 | 69 | } | |
70 | 70 | ||
71 | 71 | ||
72 | 72 | let Scale8 = 100000000 | |
73 | 73 | ||
74 | 74 | let Scale10 = 10000000000 | |
75 | 75 | ||
76 | 76 | let Scale16 = (Scale8 * Scale8) | |
77 | 77 | ||
78 | 78 | let dayBlocks = 1440 | |
79 | 79 | ||
80 | 80 | func liIntToStr (li) = { | |
81 | 81 | func f (accum,next) = ((accum + toString(next)) + ",") | |
82 | 82 | ||
83 | 83 | let $l = li | |
84 | 84 | let $s = size($l) | |
85 | 85 | let $acc0 = "" | |
86 | 86 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
87 | 87 | then $a | |
88 | 88 | else f($a, $l[$i]) | |
89 | 89 | ||
90 | 90 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
91 | 91 | then $a | |
92 | - | else throw("List size exceeds | |
92 | + | else throw("List size exceeds 100") | |
93 | 93 | ||
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) | |
95 | 95 | } | |
96 | 96 | ||
97 | 97 | ||
98 | 98 | func tryGetInteger (key) = match getInteger(this, key) { | |
99 | 99 | case b: Int => | |
100 | 100 | b | |
101 | 101 | case _ => | |
102 | 102 | 0 | |
103 | 103 | } | |
104 | 104 | ||
105 | 105 | ||
106 | 106 | func tryGetBoolean (key) = match getBoolean(this, key) { | |
107 | 107 | case b: Boolean => | |
108 | 108 | b | |
109 | 109 | case _ => | |
110 | 110 | false | |
111 | 111 | } | |
112 | 112 | ||
113 | 113 | ||
114 | 114 | func tryGetString (key) = match getString(this, key) { | |
115 | 115 | case b: String => | |
116 | 116 | b | |
117 | 117 | case _ => | |
118 | 118 | "" | |
119 | 119 | } | |
120 | 120 | ||
121 | 121 | ||
122 | 122 | func tryGetBinary (key) = match getBinary(this, key) { | |
123 | 123 | case b: ByteVector => | |
124 | 124 | b | |
125 | 125 | case _ => | |
126 | 126 | base58'' | |
127 | 127 | } | |
128 | 128 | ||
129 | 129 | ||
130 | 130 | func getAssetString (assetId) = match assetId { | |
131 | 131 | case b: ByteVector => | |
132 | 132 | toBase58String(b) | |
133 | 133 | case _ => | |
134 | 134 | "WAVES" | |
135 | 135 | } | |
136 | 136 | ||
137 | 137 | ||
138 | 138 | func getAssetBytes (assetIdStr) = if ((assetIdStr == "WAVES")) | |
139 | 139 | then unit | |
140 | 140 | else fromBase58String(assetIdStr) | |
141 | 141 | ||
142 | 142 | ||
143 | 143 | func getBalance (assetIdStr) = if ((assetIdStr == "WAVES")) | |
144 | 144 | then wavesBalance(this).available | |
145 | 145 | else assetBalance(this, fromBase58String(assetIdStr)) | |
146 | 146 | ||
147 | 147 | ||
148 | 148 | func getMarketAssets () = split(tryGetString("setup_tokens"), ",") | |
149 | 149 | ||
150 | 150 | ||
151 | 151 | func getAssetsMaxSupply () = { | |
152 | 152 | let s = tryGetString("setup_maxsupply") | |
153 | 153 | if ((s == "")) | |
154 | 154 | then [-1, -1, -1, -1, -1, -1, -1] | |
155 | 155 | else split(s, ",") | |
156 | 156 | } | |
157 | 157 | ||
158 | 158 | ||
159 | 159 | func getOutdatedUr (assetIdStr) = { | |
160 | 160 | let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), tryGetInteger((assetIdStr + "_sRate")), Scale16) | |
161 | 161 | if ((down == 0)) | |
162 | 162 | then 0 | |
163 | 163 | else fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), tryGetInteger((assetIdStr + "_bRate")), Scale16), down) | |
164 | 164 | } | |
165 | 165 | ||
166 | 166 | ||
167 | 167 | func getInterest (assetIdStr) = { | |
168 | 168 | let ur = getOutdatedUr(assetIdStr) | |
169 | 169 | let curve = getRateCurve(assetIdStr) | |
170 | 170 | let rate = (curve._1 + (if ((curve._3 >= ur)) | |
171 | 171 | then fraction(ur, curve._2, curve._3) | |
172 | 172 | else (curve._2 + fraction((ur - curve._3), curve._4, (100000000 - curve._3))))) | |
173 | 173 | max([fraction(rate, Scale8, (dayBlocks * 365)), 1]) | |
174 | 174 | } | |
175 | 175 | ||
176 | 176 | ||
177 | - | func tokenRatesRecalc (assetIdStr) = if (( | |
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)] | |
179 | 179 | else { | |
180 | 180 | let interest = getInterest(assetIdStr) | |
181 | 181 | let ur = getOutdatedUr(assetIdStr) | |
182 | - | let lastRecalcHeight = tryGetInteger("lastRateHeight") | |
182 | + | let lastRecalcHeight = valueOrElse(getInteger(this, (assetIdStr + "_lastRateHeight")), tryGetInteger("lastRateHeight")) | |
183 | 183 | let lastBRate = max([tryGetInteger((assetIdStr + "_bRate")), Scale16]) | |
184 | 184 | let newBRate = (lastBRate + ((height - lastRecalcHeight) * interest)) | |
185 | 185 | let lastSRate = max([tryGetInteger((assetIdStr + "_sRate")), Scale16]) | |
186 | 186 | 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)] | |
188 | 188 | } | |
189 | 189 | ||
190 | 190 | ||
191 | 191 | func getActualRate (assetIdStr,rateType) = { | |
192 | 192 | func f (accum,token) = { | |
193 | 193 | let recalc = tokenRatesRecalc(token) | |
194 | 194 | $Tuple2(if ((token != assetIdStr)) | |
195 | 195 | then accum._1 | |
196 | 196 | else if ((rateType == "sRate")) | |
197 | 197 | then recalc[0].value | |
198 | 198 | else recalc[1].value, (accum._2 ++ recalc)) | |
199 | 199 | } | |
200 | 200 | ||
201 | 201 | let $l = getMarketAssets() | |
202 | 202 | let $s = size($l) | |
203 | 203 | let $acc0 = $Tuple2(0, nil) | |
204 | 204 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
205 | 205 | then $a | |
206 | 206 | else f($a, $l[$i]) | |
207 | 207 | ||
208 | 208 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
209 | 209 | then $a | |
210 | - | else throw("List size exceeds | |
210 | + | else throw("List size exceeds 100") | |
211 | 211 | ||
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) | |
213 | 221 | } | |
214 | 222 | ||
215 | 223 | ||
216 | 224 | func getUr (assetIdStr) = { | |
217 | 225 | let rates = tokenRatesRecalc(assetIdStr) | |
218 | 226 | let down = fraction(tryGetInteger(("total_supplied_" + assetIdStr)), rates[0].value, Scale16) | |
219 | 227 | fraction(Scale8, fraction(tryGetInteger(("total_borrowed_" + assetIdStr)), rates[1].value, Scale16), down) | |
220 | 228 | } | |
221 | 229 | ||
222 | 230 | ||
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 | - | ||
241 | 231 | func getTokenPriceWithRisk (assetIdStr,riskAversity) = if (if ((assetIdStr == "9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi")) | |
242 | 232 | then true | |
243 | 233 | else (assetIdStr == "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW")) | |
244 | 234 | then $Tuple2(1000000, 1000000) | |
245 | 235 | else { | |
246 | 236 | let price = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_twap5B")) | |
247 | 237 | let riskLevel = getIntegerValue(Address(base58'3P8d1E1BLKoD52y3bQJ1bDTd2TD1gpaLn9t'), (assetIdStr + "_riskLevel")) | |
248 | - | if (( | |
238 | + | if (tryGetBoolean((assetIdStr + kAxlyLp))) | |
249 | 239 | then { | |
250 | 240 | let p = { | |
251 | 241 | let @ = reentrantInvoke(axlyAddress, "getShareAssetPriceREADONLY", [assetIdStr], nil) | |
252 | 242 | if ($isInstanceOf(@, "Int")) | |
253 | 243 | then @ | |
254 | 244 | else throw(($getType(@) + " couldn't be cast to Int")) | |
255 | 245 | } | |
256 | 246 | if ((p == p)) | |
257 | 247 | then $Tuple2(p, p) | |
258 | 248 | else throw("Strict value is not equal to itself.") | |
259 | 249 | } | |
260 | 250 | else if ((riskAversity >= riskLevel)) | |
261 | 251 | then $Tuple2(price, price) | |
262 | 252 | else throw((("oracle prices don't match: " + toString(price)) + " is the price, but risk is too high")) | |
263 | 253 | } | |
264 | 254 | ||
265 | 255 | ||
266 | 256 | func getTokenPrice (assetIdStr) = getTokenPriceWithRisk(assetIdStr, 1) | |
267 | 257 | ||
268 | 258 | ||
269 | 259 | func calcAssetScale (assetIdStr) = { | |
270 | 260 | let decimals = if ((assetIdStr == "WAVES")) | |
271 | 261 | then 8 | |
272 | 262 | else value(assetInfo(fromBase58String(assetIdStr))).decimals | |
273 | 263 | pow(10, 0, decimals, 0, 0, DOWN) | |
274 | 264 | } | |
275 | 265 | ||
276 | 266 | ||
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 | - | ||
295 | 267 | func supplyInternal (assetIdStr,assetAmount,address) = if (!(tryGetBoolean("setup_active"))) | |
296 | 268 | then throw("market is stopped") | |
297 | 269 | else { | |
298 | - | let $ | |
299 | - | let sRate = $ | |
300 | - | let ratesRecalcResult = $ | |
270 | + | let $t085378612 = getActualRateOneToken(assetIdStr, "sRate") | |
271 | + | let sRate = $t085378612._1 | |
272 | + | let ratesRecalcResult = $t085378612._2 | |
301 | 273 | let amount = fraction(assetAmount, Scale16, sRate, DOWN) | |
302 | 274 | let maxSupply = match getString(("setup_maxSupply_" + assetIdStr)) { | |
303 | 275 | case x: String => | |
304 | 276 | parseIntValue(x) | |
305 | 277 | case _ => | |
306 | 278 | 0 | |
307 | 279 | } | |
308 | 280 | let assetPrice = getTokenPrice(assetIdStr) | |
309 | 281 | let newTotalSupplied = (tryGetInteger(((address + "_supplied_") + assetIdStr)) + amount) | |
310 | - | let rate = | |
282 | + | let rate = sRate | |
311 | 283 | let assetScale = calcAssetScale(assetIdStr) | |
312 | 284 | let newTotalSuppliedUsd = fraction(fraction(newTotalSupplied, rate, Scale16), assetPrice._1, assetScale) | |
313 | 285 | if ((indexOf(tryGetString("setup_tokens"), assetIdStr) == unit)) | |
314 | 286 | then throw("this asset is not supported by the market") | |
315 | 287 | else if (if ((maxSupply != 0)) | |
316 | 288 | then (newTotalSuppliedUsd > maxSupply) | |
317 | 289 | else false) | |
318 | 290 | 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 | + | } | |
320 | 295 | } | |
321 | 296 | ||
322 | 297 | ||
323 | - | func borrowInternal (assetIdStr,assetAmount,address | |
324 | - | let $ | |
325 | - | let bRate = $ | |
326 | - | let ratesRecalcResult = $ | |
298 | + | func borrowInternal (assetIdStr,assetAmount,address) = { | |
299 | + | let $t099209995 = getActualRateOneToken(assetIdStr, "bRate") | |
300 | + | let bRate = $t099209995._1 | |
301 | + | let ratesRecalcResult = $t099209995._2 | |
327 | 302 | 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 | + | } | |
355 | 321 | } | |
356 | 322 | ||
357 | 323 | ||
358 | 324 | @Callable(i) | |
359 | 325 | func flashPosition (user,sAssetIdStr,bAssetIdStr,bAmount,callbackAddress,callbackFunction,callbackArgs) = if ((i.caller != axlyAddress)) | |
360 | 326 | then throw("available only for Axly protocol") | |
361 | 327 | else { | |
362 | - | let borrowRes = borrowInternal(bAssetIdStr, bAmount, user | |
328 | + | let borrowRes = borrowInternal(bAssetIdStr, bAmount, user) | |
363 | 329 | let bAssetId = getAssetBytes(bAssetIdStr) | |
364 | 330 | let callback = reentrantInvoke(addressFromStringValue(callbackAddress), callbackFunction, split(callbackArgs, ","), [AttachedPayment(bAssetId, bAmount)]) | |
365 | 331 | if ((callback == callback)) | |
366 | 332 | then { | |
367 | 333 | let sAmount = { | |
368 | 334 | let @ = callback | |
369 | 335 | if ($isInstanceOf(@, "Int")) | |
370 | 336 | then @ | |
371 | 337 | else throw(($getType(@) + " couldn't be cast to Int")) | |
372 | 338 | } | |
373 | 339 | if ((sAmount == sAmount)) | |
374 | 340 | then { | |
375 | 341 | 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) | |
397 | 345 | } | |
398 | 346 | else throw("Strict value is not equal to itself.") | |
399 | 347 | } | |
400 | 348 | else throw("Strict value is not equal to itself.") | |
401 | 349 | } | |
402 | 350 | ||
403 | 351 | ||
404 | 352 | ||
405 | 353 | @Callable(i) | |
406 | 354 | func supply () = if (if ((size(i.payments) != 1)) | |
407 | 355 | then true | |
408 | 356 | else (i.payments[0].amount == 0)) | |
409 | 357 | then throw("1 payment has to be attached") | |
410 | 358 | else { | |
411 | 359 | let assetIdStr = getAssetString(i.payments[0].assetId) | |
412 | 360 | 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 | + | } | |
417 | 369 | } | |
418 | 370 | ||
419 | 371 | ||
420 | 372 | ||
421 | 373 | @Callable(i) | |
422 | 374 | func withdraw (assetIdStr,assetAmount) = { | |
423 | - | let $ | |
424 | - | let sRate = $ | |
425 | - | let ratesRecalcResult = $ | |
375 | + | let $t01260412679 = getActualRateOneToken(assetIdStr, "sRate") | |
376 | + | let sRate = $t01260412679._1 | |
377 | + | let ratesRecalcResult = $t01260412679._2 | |
426 | 378 | let amount = fraction(assetAmount, Scale16, sRate, CEILING) | |
427 | 379 | let address = toString(i.caller) | |
428 | 380 | let assetSupplied = tryGetInteger(("total_supplied_" + assetIdStr)) | |
429 | 381 | let assetBorrowed = tryGetInteger(("total_borrowed_" + assetIdStr)) | |
430 | 382 | let userAssetSupplied = tryGetInteger(((address + "_supplied_") + assetIdStr)) | |
431 | - | let userAssetBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr)) | |
432 | 383 | let axlyNotify = invoke(axlyFarmingAddress, "lendAction", [toString(i.caller), assetIdStr], nil) | |
433 | 384 | if ((axlyNotify == axlyNotify)) | |
434 | 385 | then { | |
435 | - | let collateralValueInv = invoke(this, " | |
386 | + | let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, assetIdStr, "", false, ("supplied," + toString(-(amount)))], nil) | |
436 | 387 | if ((collateralValueInv == collateralValueInv)) | |
437 | 388 | then { | |
438 | 389 | let collateralValue = match collateralValueInv { | |
439 | 390 | case x: Int => | |
440 | 391 | x | |
441 | 392 | case _ => | |
442 | 393 | throw("can't get user collateral value") | |
443 | 394 | } | |
444 | 395 | if (!(tryGetBoolean("setup_active"))) | |
445 | 396 | then throw("market is stopped") | |
446 | 397 | else if ((0 > collateralValue)) | |
447 | 398 | then throw("you dont have enough collateral for this operation") | |
448 | 399 | else if ((amount > (assetSupplied - assetBorrowed))) | |
449 | 400 | then throw("this amount is not available on the market") | |
450 | - | else if ((amount > | |
451 | - | then throw("this amount is not available | |
401 | + | else if ((amount > userAssetSupplied)) | |
402 | + | then throw("this amount is not available on the market") | |
452 | 403 | 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) | |
453 | 404 | } | |
454 | 405 | else throw("Strict value is not equal to itself.") | |
455 | 406 | } | |
456 | 407 | else throw("Strict value is not equal to itself.") | |
457 | 408 | } | |
458 | 409 | ||
459 | 410 | ||
460 | 411 | ||
461 | 412 | @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"))) | |
494 | 414 | then throw("market is stopped") | |
495 | 415 | else if (!(verifySentinelRights(i.caller))) | |
496 | 416 | then throw("available only for whitelist") | |
497 | 417 | else if ((address == "global")) | |
498 | 418 | then throw("you can't repay for everyone :_)") | |
499 | 419 | else if (if ((size(i.payments) != 1)) | |
500 | 420 | then true | |
501 | 421 | else (i.payments[0].amount == 0)) | |
502 | 422 | 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)) | |
524 | 445 | if ((userSupplied > 0)) | |
525 | - | then | |
526 | - | else | |
446 | + | then [IntegerEntry(((address + "_supplied_") + lpId), 0), IntegerEntry(("total_supplied_" + lpId), (tryGetInteger(("total_supplied_" + lpId)) - userSupplied))] | |
447 | + | else nil | |
527 | 448 | } | |
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 | + | } | |
547 | 454 | ||
548 | 455 | ||
549 | 456 | ||
550 | 457 | @Callable(i) | |
551 | 458 | func addInterestEXTERNAL () = { | |
552 | 459 | let amount = fraction(i.payments[0].amount, 80, 100) | |
553 | 460 | let assetId = i.payments[0].assetId | |
554 | 461 | let assetIdStr = getAssetString(assetId) | |
555 | 462 | let earned = tryGetInteger(("autostake_lastEarned_" + assetIdStr)) | |
556 | 463 | let lastHeight = tryGetInteger(("autostake_lastBlock_" + assetIdStr)) | |
557 | 464 | let stateChanges = if (if ((lastHeight == height)) | |
558 | 465 | then true | |
559 | 466 | else (amount == 0)) | |
560 | 467 | then nil | |
561 | 468 | else [IntegerEntry(("autostake_preLastEarned_" + assetIdStr), earned), IntegerEntry(("autostake_preLastBlock_" + assetIdStr), lastHeight), IntegerEntry(("autostake_lastEarned_" + assetIdStr), (earned + amount)), IntegerEntry(("autostake_lastBlock_" + assetIdStr), height)] | |
562 | 469 | (stateChanges ++ [IntegerEntry((assetIdStr + "_sRate"), (tryGetInteger((assetIdStr + "_sRate")) + fraction(Scale16, amount, tryGetInteger(("total_supplied_" + assetIdStr)))))]) | |
563 | 470 | } | |
564 | 471 | ||
565 | 472 | ||
566 | 473 | ||
567 | 474 | @Callable(i) | |
568 | 475 | func preInit (tokens,ltvs,lts,penalties) = { | |
569 | 476 | func f (accum,token) = (accum ++ [IntegerEntry((token + "_bRate"), Scale16), IntegerEntry((token + "_sRate"), Scale16)]) | |
570 | 477 | ||
571 | 478 | if ((i.caller != this)) | |
572 | 479 | then throw("admin only") | |
573 | 480 | else { | |
574 | 481 | let rates = { | |
575 | 482 | let $l = split(tokens, ",") | |
576 | 483 | let $s = size($l) | |
577 | 484 | let $acc0 = nil | |
578 | 485 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
579 | 486 | then $a | |
580 | 487 | else f($a, $l[$i]) | |
581 | 488 | ||
582 | 489 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
583 | 490 | then $a | |
584 | 491 | else throw("List size exceeds 12") | |
585 | 492 | ||
586 | 493 | $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) | |
587 | 494 | } | |
588 | 495 | ([StringEntry("setup_tokens", tokens), StringEntry("setup_ltvs", ltvs), StringEntry("setup_lts", lts), StringEntry("setup_penalties", penalties), BooleanEntry("setup_active", true)] ++ rates) | |
589 | 496 | } | |
590 | 497 | } | |
591 | 498 | ||
592 | 499 | ||
593 | 500 | ||
594 | 501 | @Callable(i) | |
595 | 502 | func initNewToken (token,ltv,lt,penalty) = if (if ((i.caller != this)) | |
596 | 503 | then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8')) | |
597 | 504 | else false) | |
598 | 505 | 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)] | |
600 | 516 | ||
601 | 517 | ||
602 | 518 | ||
603 | 519 | @Callable(i) | |
604 | 520 | func updateParameter (key,val) = if (if ((i.caller != this)) | |
605 | 521 | then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8')) | |
606 | 522 | else false) | |
607 | 523 | then throw("admin only") | |
608 | 524 | else [IntegerEntry(key, parseIntValue(val))] | |
609 | 525 | ||
610 | 526 | ||
611 | 527 | ||
612 | 528 | @Callable(i) | |
613 | 529 | func updateString (key,val) = if (if ((i.caller != this)) | |
614 | 530 | then (i.caller != Address(base58'3P3o9cLTV2u9N4nYNKRYL6gy6cUEU9DwXW8')) | |
615 | 531 | else false) | |
616 | 532 | then throw("admin only") | |
617 | 533 | else [StringEntry(key, val)] | |
618 | 534 | ||
619 | 535 | ||
620 | 536 | ||
621 | 537 | @Callable(i) | |
622 | 538 | func claimToReserveFund (debug) = { | |
623 | 539 | let assets = getMarketAssets() | |
624 | 540 | 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 | + | } | |
641 | 556 | ||
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]) | |
649 | 563 | ||
650 | - | ||
651 | - | ||
652 | - | ||
564 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
565 | + | then $a | |
566 | + | else throw("List size exceeds 100") | |
653 | 567 | ||
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)) | |
655 | 574 | } | |
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 | - | } | |
662 | 575 | ||
663 | 576 | if (debug) | |
664 | 577 | then throw(liIntToStr(parameter)) | |
665 | 578 | else $Tuple2({ | |
666 | - | let $l = | |
579 | + | let $l = assets | |
667 | 580 | let $s = size($l) | |
668 | - | let $acc0 = nil | |
581 | + | let $acc0 = $Tuple2(nil, 0) | |
669 | 582 | func $f1_1 ($a,$i) = if (($i >= $s)) | |
670 | 583 | then $a | |
671 | 584 | else f2($a, $l[$i]) | |
672 | 585 | ||
673 | 586 | func $f1_2 ($a,$i) = if (($i >= $s)) | |
674 | 587 | then $a | |
675 | - | else throw("List size exceeds | |
588 | + | else throw("List size exceeds 100") | |
676 | 589 | ||
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 | |
679 | 592 | } | |
680 | 593 | ||
681 | 594 | ||
682 | 595 | ||
683 | 596 | @Callable(i) | |
684 | 597 | func shutdown (shutdown) = if ((indexOf(shutdownWhitelist, i.caller.bytes) == unit)) | |
685 | 598 | then throw("user not in a whitelist") | |
686 | 599 | else [BooleanEntry("setup_active", !(shutdown))] | |
687 | 600 | ||
688 | 601 | ||
689 | 602 | ||
690 | 603 | @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) | |
775 | 604 | func liquidateV2 (debug,address,sAssetIdStr,bAssetIdStr,liquidatePercent) = if (!(verifyLiquidatorRights(i.caller))) | |
776 | 605 | then throw("available for whitelist only") | |
777 | 606 | else if (!(tryGetBoolean("setup_active"))) | |
778 | 607 | then throw("market is stopped") | |
779 | 608 | else { | |
780 | - | let | |
781 | - | if (( | |
609 | + | let collateralValueInv = invoke(this, "getUserCollateralV2", [false, address, sAssetIdStr, bAssetIdStr, true, ""], nil) | |
610 | + | if ((collateralValueInv == collateralValueInv)) | |
782 | 611 | then { | |
612 | + | let userCollateral = match collateralValueInv { | |
613 | + | case x: Int => | |
614 | + | x | |
615 | + | case _ => | |
616 | + | throw("can't get user collateral value") | |
617 | + | } | |
783 | 618 | let currentSPosition = tryGetInteger(((address + "_supplied_") + sAssetIdStr)) | |
784 | 619 | let currentBPosition = tryGetInteger(((address + "_borrowed_") + bAssetIdStr)) | |
785 | - | if (( | |
620 | + | if (!(tryGetBoolean((sAssetIdStr + kAxlyLp)))) | |
786 | 621 | then throw("wrong lp asset") | |
787 | 622 | else if ((0 >= currentBPosition)) | |
788 | 623 | then throw("user has no borrow in this token") | |
789 | 624 | else if (if ((userCollateral > 0)) | |
790 | 625 | then true | |
791 | 626 | else (currentBPosition == 0)) | |
792 | 627 | then throw("user can't be liquidated") | |
793 | 628 | else { | |
794 | 629 | let bAssetId = getAssetBytes(bAssetIdStr) | |
795 | 630 | let marketAssets = getMarketAssets() | |
796 | 631 | let asset1Num = value(indexOf(marketAssets, bAssetIdStr)) | |
797 | 632 | let asset0Num = value(indexOf(marketAssets, sAssetIdStr)) | |
798 | 633 | let penalty = parseIntValue(value(split(tryGetString("setup_penalties"), ",")[asset1Num])) | |
799 | - | let $ | |
800 | - | let bRate = $ | |
801 | - | let ratesResult = $ | |
634 | + | let $t02189121961 = getActualRateOneToken(bAssetIdStr, "bRate") | |
635 | + | let bRate = $t02189121961._1 | |
636 | + | let ratesResult = $t02189121961._2 | |
802 | 637 | let bAmount = fraction(currentBPosition, bRate, Scale16) | |
803 | 638 | let lAmount = fraction(bAmount, liquidatePercent, 100) | |
804 | 639 | let lAmountWithPenalty = fraction(lAmount, (Scale8 + penalty), Scale8) | |
805 | 640 | let lbAmount = fraction(lAmount, Scale16, bRate) | |
806 | 641 | let posId = split(address, "_")[1] | |
807 | 642 | let addressId = split(address, "_")[0] | |
808 | 643 | let sAmountLp = { | |
809 | 644 | let @ = reentrantInvoke(axlyAddress, "liquidate", [addressId, posId, lAmountWithPenalty], nil) | |
810 | 645 | if ($isInstanceOf(@, "Int")) | |
811 | 646 | then @ | |
812 | 647 | else throw(($getType(@) + " couldn't be cast to Int")) | |
813 | 648 | } | |
814 | 649 | if ((sAmountLp == sAmountLp)) | |
815 | 650 | then { | |
816 | - | let sAmount = fraction(sAmountLp, Scale16, ratesResult[ | |
651 | + | let sAmount = fraction(sAmountLp, Scale16, ratesResult[1].value) | |
817 | 652 | if (debug) | |
818 | 653 | then throw("liquidation will pass") | |
819 | 654 | 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) | |
820 | 655 | } | |
821 | 656 | else throw("Strict value is not equal to itself.") | |
822 | 657 | } | |
823 | 658 | } | |
824 | 659 | else throw("Strict value is not equal to itself.") | |
825 | 660 | } | |
826 | 661 | ||
827 | 662 | ||
828 | 663 | ||
829 | 664 | @Callable(i) | |
830 | 665 | func getUserCollateral (debug,address,minusBorrowed,afterChange) = { | |
831 | 666 | let assets = getMarketAssets() | |
832 | 667 | let ltvs = split(tryGetString("setup_ltvs"), ",") | |
833 | 668 | let lts = split(tryGetString("setup_lts"), ",") | |
834 | 669 | let rates = getActualRate(assets[0], "sRate")._2 | |
835 | 670 | let changeHandler = split(afterChange, ",") | |
836 | 671 | func f (accum,next) = if ((next >= size(assets))) | |
837 | 672 | then accum | |
838 | 673 | else { | |
839 | 674 | let userSupplied = tryGetInteger(((address + "_supplied_") + assets[next])) | |
840 | 675 | let userBorrowed = tryGetInteger(((address + "_borrowed_") + assets[next])) | |
841 | 676 | let needTokenAccounting = if ((afterChange == "")) | |
842 | 677 | then if (if ((userBorrowed != 0)) | |
843 | 678 | then true | |
844 | 679 | else (userSupplied != 0)) | |
845 | 680 | then true | |
846 | 681 | else false | |
847 | 682 | else true | |
848 | 683 | if (needTokenAccounting) | |
849 | 684 | then { | |
850 | 685 | let assetScale = calcAssetScale(assets[next]) | |
851 | 686 | let assetPrice = getTokenPrice(assets[next]) | |
852 | 687 | ((accum + fraction(fraction(fraction((userSupplied + (if (if (if ((afterChange != "")) | |
853 | 688 | then (changeHandler[0] == assets[next]) | |
854 | 689 | else false) | |
855 | 690 | then (changeHandler[1] == "supplied") | |
856 | 691 | else false) | |
857 | 692 | then parseIntValue(changeHandler[2]) | |
858 | 693 | else 0)), rates[(next * 3)].value, Scale16), parseIntValue(ltvs[next]), Scale8), assetPrice._1, assetScale)) - (if (minusBorrowed) | |
859 | 694 | then fraction(fraction(fraction((userBorrowed + (if (if (if ((afterChange != "")) | |
860 | 695 | then (changeHandler[0] == assets[next]) | |
861 | 696 | else false) | |
862 | 697 | then (changeHandler[1] == "borrowed") | |
863 | 698 | else false) | |
864 | 699 | then parseIntValue(changeHandler[2]) | |
865 | 700 | else 0)), rates[((next * 3) + 1)].value, Scale16), Scale8, parseIntValue(lts[next])), assetPrice._2, assetScale) | |
866 | 701 | else 0)) | |
867 | 702 | } | |
868 | 703 | else accum | |
869 | 704 | } | |
870 | 705 | ||
871 | 706 | let result = { | |
872 | 707 | let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11] | |
873 | 708 | let $s = size($l) | |
874 | 709 | let $acc0 = 0 | |
875 | 710 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
876 | 711 | then $a | |
877 | 712 | else f($a, $l[$i]) | |
878 | 713 | ||
879 | 714 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
880 | 715 | then $a | |
881 | 716 | else throw("List size exceeds 12") | |
882 | 717 | ||
883 | 718 | $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) | |
884 | 719 | } | |
885 | 720 | if (debug) | |
886 | 721 | then throw(toString(result)) | |
887 | 722 | else $Tuple2(nil, result) | |
888 | 723 | } | |
889 | 724 | ||
890 | 725 | ||
891 | 726 | ||
892 | 727 | @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) | |
893 | 766 | func getAssetDebt (debug,address,assetIdStr) = { | |
894 | 767 | let userBorrowed = tryGetInteger(((address + "_borrowed_") + assetIdStr)) | |
895 | 768 | let assetScale = calcAssetScale(assetIdStr) | |
896 | - | let rate = | |
769 | + | let rate = getActualRateOneToken(assetIdStr, "bRate")._1 | |
897 | 770 | let result = fraction(userBorrowed, rate, Scale16) | |
898 | 771 | if (debug) | |
899 | 772 | then throw(toString(result)) | |
900 | 773 | else $Tuple2(nil, result) | |
901 | 774 | } | |
902 | 775 | ||
903 | 776 | ||
904 | 777 | ||
905 | 778 | @Callable(i) | |
906 | 779 | func getPrices (debug) = { | |
907 | 780 | 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 | + | } | |
914 | 785 | ||
915 | 786 | let result = { | |
916 | - | let $l = | |
787 | + | let $l = assets | |
917 | 788 | let $s = size($l) | |
918 | 789 | let $acc0 = "" | |
919 | 790 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
920 | 791 | then $a | |
921 | 792 | else f($a, $l[$i]) | |
922 | 793 | ||
923 | 794 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
924 | 795 | then $a | |
925 | - | else throw("List size exceeds | |
796 | + | else throw("List size exceeds 100") | |
926 | 797 | ||
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) | |
928 | 799 | } | |
929 | 800 | if (debug) | |
930 | 801 | then throw(result) | |
931 | 802 | else $Tuple2(nil, result) | |
932 | 803 | } | |
933 | 804 | ||
934 | 805 | ||
935 | 806 | ||
936 | 807 | @Callable(i) | |
937 | 808 | func calculateUtilizationRatio (assetIdStr,debug) = if (debug) | |
938 | 809 | then throw(toString(getUr(assetIdStr))) | |
939 | 810 | else $Tuple2(nil, getUr(assetIdStr)) | |
940 | 811 | ||
941 | 812 | ||
942 | 813 | ||
943 | 814 | @Callable(i) | |
944 | 815 | func calculateOutdatedUR (assetIdStr,debug) = if (debug) | |
945 | 816 | then throw(toString(getOutdatedUr(assetIdStr))) | |
946 | 817 | else $Tuple2(nil, getOutdatedUr(assetIdStr)) | |
947 | 818 | ||
948 | 819 | ||
949 | 820 | ||
950 | 821 | @Callable(i) | |
951 | 822 | func calculateTokenRates (debug) = { | |
952 | 823 | func f (accum,assetIdStr) = { | |
953 | 824 | let rates = tokenRatesRecalc(assetIdStr) | |
954 | 825 | $Tuple2(((((accum._1 + toString(rates[1].value)) + "|") + toString(rates[0].value)) + ","), (accum._2 ++ rates)) | |
955 | 826 | } | |
956 | 827 | ||
957 | 828 | let parameter = { | |
958 | 829 | let $l = getMarketAssets() | |
959 | 830 | let $s = size($l) | |
960 | 831 | let $acc0 = $Tuple2("", nil) | |
961 | 832 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
962 | 833 | then $a | |
963 | 834 | else f($a, $l[$i]) | |
964 | 835 | ||
965 | 836 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
966 | 837 | then $a | |
967 | - | else throw("List size exceeds | |
838 | + | else throw("List size exceeds 100") | |
968 | 839 | ||
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) | |
970 | 841 | } | |
971 | 842 | if (debug) | |
972 | 843 | then throw(parameter._1) | |
973 | 844 | else $Tuple2(parameter._2, parameter._1) | |
974 | 845 | } | |
975 | 846 | ||
976 | 847 | ||
977 | 848 | ||
978 | 849 | @Callable(i) | |
979 | 850 | func calculateTokensInterest (debug) = { | |
980 | 851 | func f (accum,assetIdStr) = { | |
981 | 852 | let rate = fraction(getInterest(assetIdStr), dayBlocks, Scale8) | |
982 | 853 | ((accum + toString(rate)) + ",") | |
983 | 854 | } | |
984 | 855 | ||
985 | 856 | let parameter = { | |
986 | 857 | let $l = getMarketAssets() | |
987 | 858 | let $s = size($l) | |
988 | 859 | let $acc0 = "" | |
989 | 860 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
990 | 861 | then $a | |
991 | 862 | else f($a, $l[$i]) | |
992 | 863 | ||
993 | 864 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
994 | 865 | then $a | |
995 | - | else throw("List size exceeds | |
866 | + | else throw("List size exceeds 100") | |
996 | 867 | ||
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) | |
998 | 869 | } | |
999 | 870 | if (debug) | |
1000 | 871 | then throw(parameter) | |
1001 | 872 | else $Tuple2(nil, parameter) | |
1002 | 873 | } | |
1003 | 874 | ||
1004 | 875 | ||
1005 | 876 | @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')) | |
1007 | 878 | then true | |
1008 | 879 | else sigVerify(tx.bodyBytes, tx.proofs[0], base58'Ajf56x532JLzVoourPv9FW75kYsvScNxztPnt1enhNkS')) | |
1009 | 880 | 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')) | |
1013 | 882 | 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 | |
1015 | 888 |
github/deemru/w8io/3ef1775 170.75 ms ◑