tx · BaNz1e9wsvwNELjxB2YyhBHgcJvx8X28vXroJjB3JjYX 3PHMz1nGiPAHCFHfPQcW18ovsb5VgCCGzfJ: -0.02700000 Waves 2024.02.05 15:14 [4028727] smart account 3PHMz1nGiPAHCFHfPQcW18ovsb5VgCCGzfJ > SELF 0.00000000 Waves
{ "type": 13, "id": "BaNz1e9wsvwNELjxB2YyhBHgcJvx8X28vXroJjB3JjYX", "fee": 2700000, "feeAssetId": null, "timestamp": 1707135354592, "version": 2, "chainId": 87, "sender": "3PHMz1nGiPAHCFHfPQcW18ovsb5VgCCGzfJ", "senderPublicKey": "75XcdocCZDdHda5bUegEwkcDk2Tt4i8yZcUzZmwS1ooA", "proofs": [ "52u1rTWMnZbiwsfs799U7z8vrf3irJYT3A2amPYPSFAMHn4vaiut5tAfUF7h3autd6aNdnCcrU7HMQiz58qjvowg" ], "script": "base64:BgInCAISBAoCCAQSAwoBCBIDCgEIEgMKAQgSABIAEgMKAQgSAwoBCBIALwAGU2NhbGU4AIDC1y8ACXNlbnRpbmVscwkAzAgCCQEHQWRkcmVzcwEBGgFX1+EjUY58q4ha1dcol3XYiHXTj64jBabhCQDMCAIJAQdBZGRyZXNzAQEaAVf4+voCJsBRi+xqO0YzSdIccbB1i/mkypoJAMwIAgkBB0FkZHJlc3MBARoBV0kBGPZDfz9MbOFndeW35Jcv3lAtsoPd8QkAzAgCCQEHQWRkcmVzcwEBGgFXS8NszvszWENYQ1EQ7u/uIRAXS/J4y3wyBQNuaWwADWJhY2tlbmRDYWxsZXIJAQdBZGRyZXNzAQEaAVfJHNfeTF8LA4Ccr7yVO4hazH+crESuo2cAD29yZGVyYm9va09yYWNsZQkBB0FkZHJlc3MBARoBV+zlqYlG/aOqpXgS89UamjWD3Wewn8WidgAFeHRuSWQBILYmKcME9c5TkaQOS3UkL2SMUbH6369UKb1I0h0qsqrRAAh4dG5JZFN0cgIsREcyeEZrUGREd0tVb0JrekdBaFF0THBTR3pmWExpQ1lQRXplS0gyQWQyNHAABmV0aFN0cgIsM1Z1VjVXVG1EejQ3RG1kbjNRcGNZanpiU2RpcGpRRTRKTWROZTF4WnBYMTMAB3N0YWJsZXMJAMwIAgIsOXdjM0xYTkE0VEVCc1h5S3RvTEU5bXJiREQ3V01IWHZYckNqWnZhYkxBc2kJAMwIAgIsSEdnYWJUcVVTOFd0VkZVSnpmbXJURE1nRWNjSnVaTEJQaEZnUUZ4dm5zb1cFA25pbAAIdG9rZW5JZHMJAMwIAgIsNm5TcFZ5Tkg3eU02OWVnNDQ2d3JRUjk0aXBiYmNtWk1VMUVOUHdhbkM5N2cJAMwIAgIsRWhpZTV4WXBlTjhvcDFDY3RjNmFHVXJxeDhqcTNqdGYxRFNqWERiZm03YVQJAMwIAgIsNExISHZZR05LSlVnNWhqNjVhR0Q1dmdTY3ZDQm1McGRSRnRqb2t2Q2pTTDgJAMwIAgIsQzFpV3NLR3FMd2pIVW5kaVE3aVhwZG1QdW05UGVDREZmeVhCZEpKb3NEUlMJAMwIAgIsMzROOVljRUVUTFduOTNxWVE2NEVzUDF4ODl0U3J1SlU0NFJyRU1TWFhFUEoJAMwIAgIsNlh0SGpwWGJzOVJSSlAyU3I5R1V5VnF6QUNjYnk5VGtUaEhYbmpWQzVDREoJAMwIAgIsSEVCOFFhdzl4cldwV3M4dEhzaUFUWUdCV0RCdFAyUzdrY1BBTHJNdTQzQVMJAMwIAgIsQWpzbzZuVFRqcHR1MlVITHg2aGZTWFZ0SEZ0UkJKQ2tLWWQ1U0F5ajd6ZjUJAMwIAgIsRFNiYmhMc1NUZURnNUxzaXVmazJBbmVoM0RqVnFKdVByMk05dVUxZ3d5NXAJAMwIAgIsQXRxdjU5RVl6akZHdWl0S1ZuTVJrNkg4RnVram9WM2t0UG9yYkV5czI1b24JAMwIAgIsMnRoc0FDdUhtekRNdU5lelBNMzJ3ZzlhM0J3VXpCV0RlU0tha2d6M2N3MjEJAMwIAgIrWWlOYm9mRnpDMTdqRUhIQ013clJjcHk5TXJyamFiTU1MWnhnOGc1eG1mNwkAzAgCAiw4dDREUFdUd1B6cGF0SEE5QWtUeFdBQjQ3VEhuWXpCc0Rub1k3ZlFxYkc5MQkAzAgCAixBdDhENk5GRnBoZUNidktWbmpWb2VMTDg0RW84TlpuNm92TWFueGZMYUZXTAkAzAgCAiw2cGhLMjJ6dEdCVzEyN2dVRm1kTUVIS0IzQ1ZkNlpoV294Mld0d0prYnFUcQkAzAgCAiwzVnVWNVdUbUR6NDdEbWRuM1FwY1lqemJTZGlwalFFNEpNZE5lMXhacFgxMwkAzAgCAiw5d2MzTFhOQTRURUJzWHlLdG9MRTltcmJERDdXTUhYdlhyQ2padmFiTEFzaQUDbmlsABB0b2tlbklzQmFzZUFzc2V0CQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIGCQDMCAIGCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIHCQDMCAIGCQDMCAIHCQDMCAIGCQDMCAIGBQNuaWwADXRva2VuRGVjaW1hbHMJAMwIAgDAhD0JAMwIAgCAwtcvCQDMCAIAgMLXLwkAzAgCAIDC1y8JAMwIAgDAhD0JAMwIAgDAhD0JAMwIAgCAwtcvCQDMCAIAgMLXLwkAzAgCAIDC1y8JAMwIAgCAwtcvCQDMCAIAgMLXLwkAzAgCAIDC1y8JAMwIAgCAwtcvCQDMCAIAwIQ9CQDMCAIAwIQ9CQDMCAIAgMLXLwkAzAgCAMCEPQkAzAgCAMCEPQUDbmlsAA90b2tlbldYV2F2ZXNMUHMJAMwIAgIsRFplQTdSYlZhYjk4cjYxaFZINXpjRlpYd05BTFBKR2dxMWV5SGFuZHNLS1cJAMwIAgIsNzJFNE1NRkhuZ3NpeHFBa05aQ1JKTUxUbWdBUDludWZkc1hleG5TNXF4VngJAMwIAgIsRlRUSm5CNnhHNENLWkVKTWdzZ3FqbWdiemN5SDh5YTN0VjhMUVAyR0NEczcJAMwIAgIsM1VueXRObmtkNDhmSG9mSFBCcDU5QmtwWkExdU1TSjhQZGhXVUN1ZXJpY3MJAMwIAgIsQnlKdFZ1WWk1OEN0ZGp0YXNEdVRxdGdldjhxV2tkSnliRjdyWTF4b3NtaEUJAMwIAgIsQm9YdWY5ZXdKU0UxVU1yS2liWm1pZ3pDNnRCenY4UkZUZVptb2ZFb3MyWkoJAMwIAgIsN0c2ODd1aTNtNTlvQWQ2cWt5bmkydnQ3N2JIRjRiYkE2eGNQZkRvM3VpVHUJAMwIAgIsRU1zWnBxY1Q4TnVNWVpDRVBIenZyaFFDUUtoZlhqZGVla1p2Y1dwSHNOdmgJAMwIAgIsRTNZaFVSTnNDbWpVYVlOaWdINkFheUx4Ymg5dTgxdVZHVTNadkhLem5ZNXYJAMwIAgIsQmlTekZlOG5TTDc4b1phZWJmb2luNXZCWjVQemU2ZDdrYWVpakxxcjV4WmUJAMwIAgIsQUtRc0VRb2VpbktSRnRkeDZyaEtXY3BrQU11NmNiRExkdFNXblI4dHBCQ3EJAMwIAgIsRGpnd1c0Q01KRWZwemo3U0xGRlBpQ3FmMXpmbUFnZjFNRDVnQVM4U2pLdDQJAMwIAgIACQDMCAICLEYzczkyVlJkcHBFdExWQVA2OTdDVE02QXpaSkVDUmNvTXg1a2oyRmtHdjRNCQDMCAICLDZpTUI2TEtTcmd2OXdhRXZFbk42WWR5eDdkZnhQbkdjVHc4MzE4V1ZtNWJSCQDMCAICLEU2TVVHU1VpZVNzcTZRaUpjYnA2UkVkUVh5anA2dVBXTGl0bnA5R2R0c3VoCQDMCAICLDZLV3BucExrNkdOaDVIRzljdDlwTkRDWUZXSkJvQUFzaFpzcDREOGpjRnBWCQDMCAICAAUDbmlsAA90b2tlbldYWFROUG9vbHMJAMwIAgIACQDMCAICAAkAzAgCAgAJAMwIAgIjM1AzWTM4enltQ0hvVE02cWNFOUpjRUE4cmhhR1RRYlZNZ3UJAMwIAgIACQDMCAICAAkAzAgCAiMzUDYxNXlYZVE5UXU0cUJNMVFHaW1Heml4eU1TNVc0S3R1ZwkAzAgCAiMzUEVrRDVMdUhHV2hVZmdCVzFvd1pGeFZ5ZFlvV2pWNkppYQkAzAgCAiMzUDVIalBvd2dpcGlWQjNVdlhwaERpUHZLczQyNFdDOXhRdwkAzAgCAiMzUENFTnBFS2U4YXR3RUxaN29DU21jZEVmY1J1S1RyVXg5OQkAzAgCAgAJAMwIAgIACQDMCAICIzNQSDgzMmVLbnc2N1BGdkpmVWc5S256NkJITTg3UXZqTWVMCQDMCAICAAkAzAgCAiMzUEdjWXpvVWZRRVFraFJnckJxbUc1V3BjN0ExbmNGdlJ0TQkAzAgCAgAJAMwIAgIACQDMCAICAAUDbmlsAA50b2tlblN3b3BQb29scwkAzAgCAiMzUDJWNjNYZDZCdmlEa2VNenhoVXcyU0p5b2pCeVJ6OGE4bQkAzAgCAiMzUDI3UzlWMzZrdzJNY2pXUlozN0F4VHg4aXdrZDdIWHc2VwkAzAgCAiMzUDZETGRKVFAyRXlTcTlNRmRKdTZiZVVldnJRZDJzVlZCaAkAzAgCAgAJAMwIAgIACQDMCAICAAkAzAgCAiMzUEJIeUV3bUVSUjFDRWtyVE5iUGoyYmd5aXNUZlBScWZlZQkAzAgCAiMzUERTWHd3QTJ4QXRteGVvMmJlYkRQM0FBNXluWjJLcmdudwkAzAgCAiMzUEo0OFAzcDJ3dldVamdRYVFpWjJjRmJyOHFteE1va0JHZAkAzAgCAiMzUEtpNEczVlgyazQyWlNtTk5ybXZnZERIN0p6UmFVaFk3UgkAzAgCAgAJAMwIAgIACQDMCAICIzNQQ3d2bnMyZG5tb2JENlo0Y1I4NnY5OHM3TGdNWll5Z0V5CQDMCAICIzNQTHA5S3JkcmJXQXZTRkhHSlZFYXN6VXVicXV6M002bWZlCQDMCAICAAkAzAgCAgAJAMwIAgIACQDMCAICAAUDbmlsAApscFRva2VuSWRzCQDMCAICLDlkYnBTcjhkMThxV1F4bjVmSkpTUzFMTFE4Q21TWjZnWW1qdVBSemczUkJNCQDMCAICLDlNS2l4UnQ5ck5SeWFKQ1QycGV4YlhrdXZwWkJkSlJFZFRVMzZiR2l0OGl3CQDMCAICLDZiWmJSbW91N003d1hCdW5NWFFuWjRSbTY2SHhaRjNLZk1FaUZ3azN3bW5BCQDMCAICK1hqZEpLV3RQWUN6NTg1UUI3TG54RFA3NlVHUnVrYXplZER1YlV4OURIUUgJAMwIAgIsRUE3c2lHTVNUeHo2RXRkcGtDaVZXUUh1cEZUNU43VWJ2UXJXOWt2eENFNDIFA25pbAALbHBUb2tlbkJhc2UJAMwIAgIsQXRxdjU5RVl6akZHdWl0S1ZuTVJrNkg4RnVram9WM2t0UG9yYkV5czI1b24JAMwIAgIFV0FWRVMJAMwIAgIFV0FWRVMJAMwIAgIFV0FWRVMJAMwIAgIsM1Z1VjVXVG1EejQ3RG1kbjNRcGNZanpiU2RpcGpRRTRKTWROZTF4WnBYMTMFA25pbAALbHBUb2tlblBvb2wJAMwIAgkBB0FkZHJlc3MBARoBV76mSii9AJleh5A76ohLq7pjgLF/UUsEDQkAzAgCCQEHQWRkcmVzcwEBGgFXGuxwbi4sl64A1TNuR20jEf9SoAJwPdWrCQDMCAIJAQdBZGRyZXNzAQEaAVd3fwtdLh7A+SLuBE0CCrsD63aXOyCB6yYJAMwIAgkBB0FkZHJlc3MBARoBVw5aceg7GvwyONbfJykDPJw63Iy1Pmy34wkAzAgCCQEHQWRkcmVzcwEBGgFXG9cdIxxZJ2N+N0sZx83U3Oa4c/QIdsyVBQNuaWwAEGxwVG9rZW5CYXNlU2hhcmUJAMwIAgCAh6cOCQDMCAIAgNrECQkAzAgCAIDaxAkJAMwIAgCAmZsQCQDMCAIAgJChDwUDbmlsABJscFRva2VuUmVzZXJ2ZUJhc2UJAMwIAgIACQDMCAICAAkAzAgCAgAJAMwIAgIsM1Z1VjVXVG1EejQ3RG1kbjNRcGNZanpiU2RpcGpRRTRKTWROZTF4WnBYMTMJAMwIAgIABQNuaWwAE2xwVG9rZW5SZXNlcnZlU2hhcmUJAMwIAgAACQDMCAIAAAkAzAgCAAAJAMwIAgDAlN4PCQDMCAIAAAUDbmlsAAx3eExwVG9rZW5JZHMJAMwIAgIrclpNUTZnMzFMcjdzUEFhQW9ZYzRVMlBIQ1ZhdVR1S1VTelViSnRVeVBaTgUDbmlsAA53eExwVW5kZXJseWluZwkAzAgCCQDMCAICLDl3YzNMWE5BNFRFQnNYeUt0b0xFOW1yYkREN1dNSFh2WHJDalp2YWJMQXNpCQDMCAICLEhHZ2FiVHFVUzhXdFZGVUp6Zm1yVERNZ0VjY0p1WkxCUGhGZ1FGeHZuc29XBQNuaWwFA25pbAARd3hfcmVzdEFkZHJlc3NTdHICIzNQOE1vUG5zYXVyb2ZrMVZ5aHNkQUZrZVE2aWpwSllYQ3BXAAt3eF9yZXN0RGFwcAkBEUBleHRyTmF0aXZlKDEwNjIpAQURd3hfcmVzdEFkZHJlc3NTdHIADHN3b3BSZXN0RGFwcAkBEUBleHRyTmF0aXZlKDEwNjIpAQIjM1A1NmpOUXpFQ1hucldwbmJiU0pLdzdFb29vNmZrVWFNUHAAD29yYWNsZVRvbGVyYW5jZQDoBwAPc2h1dGRvd25UcmlnZ2VyALgXAAt1cGRhdGVEZWxheQAAAAtyZXZpc2lvbk51bQIAAANTRVACAl9fAQhhc1N0cmluZwEBdgQHJG1hdGNoMAUBdgMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAXMFByRtYXRjaDAFAXMJAAIBAhhmYWlsIHRvIGNhc3QgaW50byBTdHJpbmcBBWFzSW50AQF2BAckbWF0Y2gwBQF2AwkAAQIFByRtYXRjaDACA0ludAQBaQUHJG1hdGNoMAUBaQkAAgECFWZhaWwgdG8gY2FzdCBpbnRvIEludAENdHJ5R2V0SW50ZWdlcgEDa2V5BAckbWF0Y2gwCQCaCAIFBHRoaXMFA2tleQMJAAECBQckbWF0Y2gwAgNJbnQEAWIFByRtYXRjaDAFAWIAAAENZ2V0QXNzZXRCeXRlcwEKYXNzZXRJZFN0cgMJAAACBQphc3NldElkU3RyAgVXQVZFUwUEdW5pdAkA2QQBBQphc3NldElkU3RyAA5kb3JhQWRkcmVzc1N0cgIjM1BLa29qS2RkNkJCelRmMVJYYlFWZlVEcmFORlhYSEt6UUYAC2RvcmFBZGRyZXNzCQERQGV4dHJOYXRpdmUoMTA2MikBBQ5kb3JhQWRkcmVzc1N0cgANeHRuVXNkUHJpY2VYNgkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFC2RvcmFBZGRyZXNzAhYlcyVzX19wcmljZV9fVVNETi1VU0RUAiRET1JBIGRvZXNuJ3QgY29udGFpbiBVU0ROL1VTRFQgcHJpY2UACndhdmVzUHJpY2UJARFAZXh0ck5hdGl2ZSgxMDUwKQIFC2RvcmFBZGRyZXNzAhclcyVzX19wcmljZV9fV0FWRVMtVVNEVAAMZXRoRG9yYVByaWNlCQERQGV4dHJOYXRpdmUoMTA1MCkCBQtkb3JhQWRkcmVzcwIVJXMlc19fcHJpY2VfX0VUSC1VU0RUAQ1nZXRUb2tlblByaWNlAgh0b2tlbk51bQ93YXZlc1VzZFByaWNlWDYECnRva2VuSWRTdHIJAJEDAgUIdG9rZW5JZHMFCHRva2VuTnVtBAd0b2tlbklkCQDZBAEFCnRva2VuSWRTdHIDCQECIT0CCQDPCAIFB3N0YWJsZXMFCnRva2VuSWRTdHIFBHVuaXQJAJQKAgDAhD0CEnN0YWJsZSB0b2tlbiBwcmljZQQNd3hfVDFXYXZlc19scAkAkQMCBQ90b2tlbldYV2F2ZXNMUHMFCHRva2VuTnVtBAskdDA2NDU5NzE3NQMJAAACBQ13eF9UMVdhdmVzX2xwAgAJAJQKAgAAAAAEDHd4UmVzdFJlc3VsdAkAtQkCCQEIYXNTdHJpbmcBCQD8BwQFC3d4X3Jlc3REYXBwAhFwb29sU3RhdHNSRUFET05MWQkAzAgCBQ13eF9UMVdhdmVzX2xwBQNuaWwFA25pbAICX18ECyR0MDY2ODM2Nzc1AwkAkQMCBRB0b2tlbklzQmFzZUFzc2V0BQh0b2tlbk51bQkAlAoCAAEAAgkAlAoCAAIAAQQOd2F2ZXNJblBhaXJOdW0IBQskdDA2NjgzNjc3NQJfMQQOdG9rZW5JblBhaXJOdW0IBQskdDA2NjgzNjc3NQJfMgQSd3hfVDFXYXZlc19XQVZFU3g4CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUMd3hSZXN0UmVzdWx0BQ53YXZlc0luUGFpck51bQQPd3hfVDFXYXZlc19UMXg4CQBrAwkBDXBhcnNlSW50VmFsdWUBCQCRAwIFDHd4UmVzdFJlc3VsdAUOdG9rZW5JblBhaXJOdW0AgMLXLwkAkQMCBQ10b2tlbkRlY2ltYWxzBQh0b2tlbk51bQQXd3hfVDFXYXZlc19UMVVzZFByaWNlWDYJAGsDBRJ3eF9UMVdhdmVzX1dBVkVTeDgFD3dhdmVzVXNkUHJpY2VYNgUPd3hfVDFXYXZlc19UMXg4BBF3eF9UMVdhdmVzX1dlaWdodAUPd3hfVDFXYXZlc19UMXg4CQCUCgIFF3d4X1QxV2F2ZXNfVDFVc2RQcmljZVg2BRF3eF9UMVdhdmVzX1dlaWdodAQXd3hfVDFXYXZlc19UMVVzZFByaWNlWDYIBQskdDA2NDU5NzE3NQJfMQQRd3hfVDFXYXZlc19XZWlnaHQIBQskdDA2NDU5NzE3NQJfMgQTd3hfVDFYdG5fQWRkcmVzc1N0cgkAkQMCBQ90b2tlbldYWFROUG9vbHMFCHRva2VuTnVtBAskdDA3MjM3NzY2OAMJAAACBRN3eF9UMVh0bl9BZGRyZXNzU3RyAgAJAJQKAgAAAAAEEHd4X1QxWHRuX0FkZHJlc3MJARFAZXh0ck5hdGl2ZSgxMDYyKQEFE3d4X1QxWHRuX0FkZHJlc3NTdHIEDnd4X1QxWHRuX1hUTng2CQDwBwIFEHd4X1QxWHRuX0FkZHJlc3MFBXh0bklkBA13eF9UMVh0bl9UMXg4CQDwBwIFEHd4X1QxWHRuX0FkZHJlc3MFB3Rva2VuSWQJAJQKAgkAawMJAGgCBQ53eF9UMVh0bl9YVE54NgkAaQIJAJEDAgUNdG9rZW5EZWNpbWFscwUIdG9rZW5OdW0AwIQ9BQ14dG5Vc2RQcmljZVg2BQ13eF9UMVh0bl9UMXg4BQ13eF9UMVh0bl9UMXg4BBV3eF9UMVh0bl9UMVVzZFByaWNlWDYIBQskdDA3MjM3NzY2OAJfMQQPd3hfVDFYdG5fV2VpZ2h0CAULJHQwNzIzNzc2NjgCXzIEFXN3b3BfVDFYdG5fQWRkcmVzc1N0cgkAkQMCBQ50b2tlblN3b3BQb29scwUIdG9rZW5OdW0ECyR0MDc3MzE4MzQ5AwkAAAIFFXN3b3BfVDFYdG5fQWRkcmVzc1N0cgIACQCUCgIAAAAABBJzd29wX1QxWHRuX0FkZHJlc3MJARFAZXh0ck5hdGl2ZSgxMDYyKQEFFXN3b3BfVDFYdG5fQWRkcmVzc1N0cgQXc3dvcF9UMVh0bl9UMVh0blByaWNlWDYJAQVhc0ludAEJAPwHBAUMc3dvcFJlc3REYXBwAhFjYWxjR2V0QW1vdW50Q1BNTQkAzAgCBRVzd29wX1QxWHRuX0FkZHJlc3NTdHIJAMwIAgIjM1A4OHFrMUt6RjFCS2pEN2ZDN0xqTlZBS000ZXpmZjVXRTYJAMwIAgUKdG9rZW5JZFN0cgkAzAgCCQCRAwIFDXRva2VuRGVjaW1hbHMFCHRva2VuTnVtBQNuaWwFA25pbAQXc3dvcF9UMVh0bl9UMVVzZFByaWNlWDYJAGsDBRdzd29wX1QxWHRuX1QxWHRuUHJpY2VYNgUNeHRuVXNkUHJpY2VYNgDAhD0EEXN3b3BfVDFYdG5fV2VpZ2h0CQERQGV4dHJOYXRpdmUoMTA1MCkCBRJzd29wX1QxWHRuX0FkZHJlc3MCD0FfYXNzZXRfYmFsYW5jZQkAlAoCBRdzd29wX1QxWHRuX1QxVXNkUHJpY2VYNgURc3dvcF9UMVh0bl9XZWlnaHQEF3N3b3BfVDFYdG5fVDFVc2RQcmljZVg2CAULJHQwNzczMTgzNDkCXzEEEXN3b3BfVDFYdG5fV2VpZ2h0CAULJHQwNzczMTgzNDkCXzIEAVcJAGQCCQBkAgURd3hfVDFXYXZlc19XZWlnaHQFD3d4X1QxWHRuX1dlaWdodAURc3dvcF9UMVh0bl9XZWlnaHQEDFQxVXNkUHJpY2VYNgkAZAIJAGQCCQBrAwUXd3hfVDFXYXZlc19UMVVzZFByaWNlWDYFEXd4X1QxV2F2ZXNfV2VpZ2h0BQFXCQBrAwUVd3hfVDFYdG5fVDFVc2RQcmljZVg2BQ93eF9UMVh0bl9XZWlnaHQFAVcJAGsDBRdzd29wX1QxWHRuX1QxVXNkUHJpY2VYNgURc3dvcF9UMVh0bl9XZWlnaHQFAVcEBWRlYnVnCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICDVQxVXNkUHJpY2VYNj0JAKQDAQUMVDFVc2RQcmljZVg2Ahkgd3hfVDFXYXZlc19UMVVzZFByaWNlWDY9CQCkAwEFF3d4X1QxV2F2ZXNfVDFVc2RQcmljZVg2Ahcgd3hfVDFYdG5fVDFVc2RQcmljZVg2PQkApAMBBRV3eF9UMVh0bl9UMVVzZFByaWNlWDYCGSBzd29wX1QxWHRuX1QxVXNkUHJpY2VYNj0JAKQDAQUXc3dvcF9UMVh0bl9UMVVzZFByaWNlWDYJAJQKAgUMVDFVc2RQcmljZVg2BQVkZWJ1ZwATbGFzdFVwZGF0ZWRCbG9ja0tleQIQbGFzdFVwZGF0ZWRCbG9jawAHbjE1TGlzdAkAzAgCAAEJAMwIAgABCQDMCAIAAQkAzAgCAAEJAMwIAgABCQDMCAIAAQkAzAgCAAEJAMwIAgABCQDMCAIAAQkAzAgCAAEJAMwIAgABCQDMCAIAAQkAzAgCAAEJAMwIAgABCQDMCAIAAQUDbmlsAQ9nZXRQcmljZURhdGFLZXkCBWJsb2NrB3Rva2VuSWQJAKwCAgkArAICCQCsAgICBmJsb2NrXwkApAMBBQVibG9jawIBXwUHdG9rZW5JZAENZ2V0UHJpY2VWYWx1ZQIHdG9rZW5JZAVibG9jawkBC3ZhbHVlT3JFbHNlAgkAmggCBQ9vcmRlcmJvb2tPcmFjbGUJAQ9nZXRQcmljZURhdGFLZXkCBQVibG9jawUHdG9rZW5JZAAAARNnZXRMYXN0VXBkYXRlZEJsb2NrAAQQbGFzdFVwZGF0ZWRCbG9jawkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFD29yZGVyYm9va09yYWNsZQUTbGFzdFVwZGF0ZWRCbG9ja0tleQIgTGFzdCB1cGRhdGVkIGJsb2NrIGtleSBub3QgZm91bmQDCQBnAgAABRBsYXN0VXBkYXRlZEJsb2NrCQACAQITSW52YWxpZCBibG9jayB2YWx1ZQMJAGYCCQBlAgUGaGVpZ2h0BRBsYXN0VXBkYXRlZEJsb2NrAAoJAAIBAi5UaGUgcHJpY2UgaGFzIG5vdCBiZWVuIHVwZGF0ZWQgZm9yIGEgbG9uZyB0aW1lBRBsYXN0VXBkYXRlZEJsb2NrAQxnZXRQcmljZUxpc3QCBWFjY3VtBG5leHQECyR0MDk3OTA5ODMxBQVhY2N1bQQGcmVzdWx0CAULJHQwOTc5MDk4MzECXzEEBWJsb2NrCAULJHQwOTc5MDk4MzECXzIEB3Rva2VuSWQIBQskdDA5NzkwOTgzMQJfMwQDc3VtCAULJHQwOTc5MDk4MzECXzQECnByaWNlVmFsdWUJAQ1nZXRQcmljZVZhbHVlAgUHdG9rZW5JZAUFYmxvY2sECW5leHRCbG9jawkAZQIFBWJsb2NrBQRuZXh0AwkAZgIFCnByaWNlVmFsdWUAAAkAlgoECQDNCAIFBnJlc3VsdAUKcHJpY2VWYWx1ZQUJbmV4dEJsb2NrBQd0b2tlbklkCQBkAgUDc3VtBQpwcmljZVZhbHVlCQCWCgQFBnJlc3VsdAUJbmV4dEJsb2NrBQd0b2tlbklkBQNzdW0BEGdldE9yZGVyYm9va1R3YXABCnRva2VuSWRTdHIEA2ludgkA/AcEBQR0aGlzAhJnZXRPcmRlcmJvb2tUd2FwMTUJAMwIAgUKdG9rZW5JZFN0cgkAzAgCBwUDbmlsBQNuaWwDCQAAAgUDaW52BQNpbnYEDm9yZGVyYm9va1ByaWNlCgABQAUDaW52AwkAAQIFAUACDihJbnQsIEJvb2xlYW4pBQFACQACAQkArAICCQADAQUBQAIjIGNvdWxkbid0IGJlIGNhc3QgdG8gKEludCwgQm9vbGVhbikDCQAAAgUOb3JkZXJib29rUHJpY2UFDm9yZGVyYm9va1ByaWNlBQ5vcmRlcmJvb2tQcmljZQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkBaQESZ2V0T3JkZXJib29rVHdhcDE1Agd0b2tlbklkBWRlYnVnBBBsYXN0VXBkYXRlZEJsb2NrCQETZ2V0TGFzdFVwZGF0ZWRCbG9jawAEBXR1cGxlCgACJGwFB24xNUxpc3QKAAIkcwkAkAMBBQIkbAoABSRhY2MwCQCWCgQFA25pbAUQbGFzdFVwZGF0ZWRCbG9jawUHdG9rZW5JZAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQxnZXRQcmljZUxpc3QCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDE1CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPBApwcmljZXNMaXN0CAUFdHVwbGUCXzEEA3N1bQgFBXR1cGxlAl80BAdhdmVyYWdlCQBrAwUDc3VtAMCEPQkAawMJAJADAQUKcHJpY2VzTGlzdADAhD0AAQQIaXNMb2NrZWQJAQt2YWx1ZU9yRWxzZQIJAJsIAgUPb3JkZXJib29rT3JhY2xlCQCsAgIFB3Rva2VuSWQCB19sb2NrZWQHAwUFZGVidWcJAAIBCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgIJYXZlcmFnZTogCQCkAwEFB2F2ZXJhZ2UCDCwgaXNMb2NrZWQ6IAkApQMBBQhpc0xvY2tlZAICLCACD3ByaWNlTGlzdFNpemU6IAkApAMBCQCQAwEFCnByaWNlc0xpc3QCBywgc3VtOiAJAKQDAQUDc3VtCQCUCgIFA25pbAkAlAoCBQdhdmVyYWdlBQhpc0xvY2tlZAFpARR1cGRhdGVXeExwVG9rZW5QcmljZQEKYXNzZXRJZFN0cgQIdG9rZW5OdW0JAQV2YWx1ZQEJAM8IAgUMd3hMcFRva2VuSWRzBQphc3NldElkU3RyBAp1bmRlcmx5aW5nCQCRAwIFDnd4THBVbmRlcmx5aW5nBQh0b2tlbk51bQQQdW5kZXJseWluZzFQcmljZQMJAAACCQCRAwIFCnVuZGVybHlpbmcAAAIFV0FWRVMFCndhdmVzUHJpY2UDCQECIT0CCQDPCAIFB3N0YWJsZXMJAJEDAgUKdW5kZXJseWluZwAABQR1bml0AMCEPQgJAQ1nZXRUb2tlblByaWNlAgkBBXZhbHVlAQkAzwgCBQh0b2tlbklkcwkAkQMCBQp1bmRlcmx5aW5nAAAFCndhdmVzUHJpY2UCXzEEE3VuZGVybHlpbmcxRGVjaW1hbHMDCQAAAgkAkQMCBQp1bmRlcmx5aW5nAAACBVdBVkVTAAgJAQV2YWx1ZQEICQEFdmFsdWUBCQDsBwEJANkEAQkAkQMCBQp1bmRlcmx5aW5nAAAIZGVjaW1hbHMEEHVuZGVybHlpbmcyUHJpY2UDCQAAAgkAkQMCBQp1bmRlcmx5aW5nAAECBVdBVkVTBQp3YXZlc1ByaWNlAwkBAiE9AgkAzwgCBQdzdGFibGVzCQCRAwIFCnVuZGVybHlpbmcAAQUEdW5pdADAhD0ICQENZ2V0VG9rZW5QcmljZQIJAQV2YWx1ZQEJAM8IAgUIdG9rZW5JZHMJAJEDAgUKdW5kZXJseWluZwABBQp3YXZlc1ByaWNlAl8xBBN1bmRlcmx5aW5nMkRlY2ltYWxzAwkAAAIJAJEDAgUKdW5kZXJseWluZwABAgVXQVZFUwAICQEFdmFsdWUBCAkBBXZhbHVlAQkA7AcBCQDZBAEJAJEDAgUKdW5kZXJseWluZwABCGRlY2ltYWxzBBB1bmRlcmx5aW5nMVNjYWxlAwkAAAIFE3VuZGVybHlpbmcxRGVjaW1hbHMACAUGU2NhbGU4AMCEPQQQdW5kZXJseWluZzJTY2FsZQMJAAACBRN1bmRlcmx5aW5nMkRlY2ltYWxzAAgFBlNjYWxlOADAhD0EA2ludgoAAUAJAPwHBAULd3hfcmVzdERhcHACF3Bvb2xFdmFsdWF0ZUdldFJFQURPTkxZCQDMCAIFCmFzc2V0SWRTdHIJAMwIAgCAwtcvBQNuaWwFA25pbAMJAAECBQFAAgZTdHJpbmcFAUAJAAIBCQCsAgIJAAMBBQFAAhsgY291bGRuJ3QgYmUgY2FzdCB0byBTdHJpbmcDCQAAAgUDaW52BQNpbnYEBHZhbHMJALUJAgUDaW52AgJfXwQRdW5kZXJseWluZzFBbW91bnQJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQR2YWxzAAEEEXVuZGVybHlpbmcyQW1vdW50CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUEdmFscwACBAp0b2tlblByaWNlCQBkAgkAawMFEHVuZGVybHlpbmcxUHJpY2UFEXVuZGVybHlpbmcxQW1vdW50BRB1bmRlcmx5aW5nMVNjYWxlCQBrAwUQdW5kZXJseWluZzJQcmljZQURdW5kZXJseWluZzJBbW91bnQFEHVuZGVybHlpbmcyU2NhbGUECGxhc3RUd2FwCQENdHJ5R2V0SW50ZWdlcgEJAKwCAgUKYXNzZXRJZFN0cgIHX3R3YXA1QgQKbGFzdEhlaWdodAkBDXRyeUdldEludGVnZXIBCQCsAgIFCmFzc2V0SWRTdHICCl9sYXN0QmxvY2sDCQECIT0CCAUBaQxvcmlnaW5DYWxsZXIFDWJhY2tlbmRDYWxsZXIJAAIBAhphdmFpbGFibGUgZm9yIGJhY2tlbmQgb25seQMJAGYCBQt1cGRhdGVEZWxheQkAZQIFBmhlaWdodAUKbGFzdEhlaWdodAUDbmlsAwMJAAACBQpsYXN0SGVpZ2h0AAAGCQBmAgkAZQIFBmhlaWdodAUKbGFzdEhlaWdodABkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICB190d2FwNUIFCnRva2VuUHJpY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIKX2xhc3RCbG9jawUGaGVpZ2h0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgILX2xhc3RQcmljZXMJAKQDAQUKdG9rZW5QcmljZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfcmlza0xldmVsAAEFA25pbAQKbGFzdFByaWNlcwkAtQkCCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFCmFzc2V0SWRTdHICC19sYXN0UHJpY2VzAgEsBAluZXdQcmljZXMJAMwIAgkApAMBBQp0b2tlblByaWNlBQpsYXN0UHJpY2VzCgEKc3VtSGFuZGxlcgIFYWNjdW0EbmV4dAkAZAIFBWFjY3VtCQENcGFyc2VJbnRWYWx1ZQEFBG5leHQEB25ld1R3YXAJAGkCCgACJGwFCW5ld1ByaWNlcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAAAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEKc3VtSGFuZGxlcgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQITTGlzdCBzaXplIGV4Y2VlZHMgNgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgkAkAMBBQluZXdQcmljZXMECXByaWNlRGlmZgkAawMFCnRva2VuUHJpY2UAkE4JAQ1wYXJzZUludFZhbHVlAQkAkQMCBQpsYXN0UHJpY2VzCQBlAgkAkAMBBQpsYXN0UHJpY2VzAAEEC3ByaWNlRGlmZk9rAwkAZgIJAGQCAJBOBQ9zaHV0ZG93blRyaWdnZXIFCXByaWNlRGlmZgkAZgIFCXByaWNlRGlmZgkAZQIAkE4FD3NodXRkb3duVHJpZ2dlcgcECXJpc2tMZXZlbAMFC3ByaWNlRGlmZk9rAAEAAgMJAGcCAAUJAJADAQUJbmV3UHJpY2VzBAl1cGRQcmljZXMJALkJAgUJbmV3UHJpY2VzAgEsCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICB190d2FwNUIFB25ld1R3YXAJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIKX2xhc3RCbG9jawUGaGVpZ2h0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgILX2xhc3RQcmljZXMFCXVwZFByaWNlcwkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfcmlza0xldmVsBQlyaXNrTGV2ZWwFA25pbAQJdXBkUHJpY2VzCQC5CQIJANEIAgUJbmV3UHJpY2VzCQBlAgkAkAMBBQluZXdQcmljZXMAAQIBLAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgdfdHdhcDVCBQduZXdUd2FwCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICCl9sYXN0QmxvY2sFBmhlaWdodAkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICC19sYXN0UHJpY2VzBQl1cGRQcmljZXMJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIKX3Jpc2tMZXZlbAUJcmlza0xldmVsBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQESdXBkYXRlTHBUb2tlblByaWNlAQphc3NldElkU3RyBAh0b2tlbk51bQkBBXZhbHVlAQkAzwgCBQpscFRva2VuSWRzBQphc3NldElkU3RyBAliYXNlSWRTdHIJAJEDAgULbHBUb2tlbkJhc2UFCHRva2VuTnVtBAliYXNlUHJpY2UDCQAAAgUJYmFzZUlkU3RyAgVXQVZFUwUKd2F2ZXNQcmljZQgJAQ1nZXRUb2tlblByaWNlAgkBBXZhbHVlAQkAzwgCBQh0b2tlbklkcwUJYmFzZUlkU3RyBQp3YXZlc1ByaWNlAl8xBAxiYXNlUXVhbnRpdHkJARFAZXh0ck5hdGl2ZSgxMDUwKQIJAJEDAgULbHBUb2tlblBvb2wFCHRva2VuTnVtCQCsAgIJAKwCAgIHZ2xvYmFsXwUJYmFzZUlkU3RyAghfYmFsYW5jZQQNdG9rZW5RdWFudGl0eQgJAQV2YWx1ZQEJAOwHAQkBBXZhbHVlAQkBDWdldEFzc2V0Qnl0ZXMBBQphc3NldElkU3RyCHF1YW50aXR5BAp0b2tlblByaWNlCQBrAwkAawMJAGsDBQliYXNlUHJpY2UFDGJhc2VRdWFudGl0eQDAhD0FBlNjYWxlOAkAkQMCBRBscFRva2VuQmFzZVNoYXJlBQh0b2tlbk51bQDAhD0FDXRva2VuUXVhbnRpdHkEEXRva2VuUmVzZXJ2ZVByaWNlAwkBAiE9AgkAkQMCBRJscFRva2VuUmVzZXJ2ZUJhc2UFCHRva2VuTnVtAgAECmJhc2UySWRTdHIJAJEDAgUSbHBUb2tlblJlc2VydmVCYXNlBQh0b2tlbk51bQQKYmFzZTJQcmljZQMJAAACBQpiYXNlMklkU3RyAgVXQVZFUwUKd2F2ZXNQcmljZQMJAAACBQpiYXNlMklkU3RyAiwzVnVWNVdUbUR6NDdEbWRuM1FwY1lqemJTZGlwalFFNEpNZE5lMXhacFgxMwUMZXRoRG9yYVByaWNlAwkAAAIFCmJhc2UySWRTdHIFCHh0bklkU3RyCQBoAgUNeHRuVXNkUHJpY2VYNgBkCAkBDWdldFRva2VuUHJpY2UCCQEFdmFsdWUBCQDPCAIFCHRva2VuSWRzBQpiYXNlMklkU3RyBQp3YXZlc1ByaWNlAl8xBA1iYXNlMlF1YW50aXR5CQERQGV4dHJOYXRpdmUoMTA1MCkCCQCRAwIFC2xwVG9rZW5Qb29sBQh0b2tlbk51bQkArAICCQCsAgICB2dsb2JhbF8FCmJhc2UySWRTdHICCF9iYWxhbmNlCQBrAwkAawMJAGsDBQpiYXNlMlByaWNlBQ1iYXNlMlF1YW50aXR5AMCEPQUGU2NhbGU4CQCRAwIFE2xwVG9rZW5SZXNlcnZlU2hhcmUFCHRva2VuTnVtAMCEPQUNdG9rZW5RdWFudGl0eQUKdG9rZW5QcmljZQQIbGFzdFR3YXAJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgdfdHdhcDVCBApsYXN0SGVpZ2h0CQENdHJ5R2V0SW50ZWdlcgEJAKwCAgUKYXNzZXRJZFN0cgIKX2xhc3RCbG9jawMJAQIhPQIIBQFpDG9yaWdpbkNhbGxlcgUNYmFja2VuZENhbGxlcgkAAgECGmF2YWlsYWJsZSBmb3IgYmFja2VuZCBvbmx5AwkAZgIAAgkAZQIFBmhlaWdodAUKbGFzdEhlaWdodAUDbmlsAwMJAAACBQpsYXN0SGVpZ2h0AAAGCQBmAgkAZQIFBmhlaWdodAUKbGFzdEhlaWdodABkCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICB190d2FwNUIFCnRva2VuUHJpY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIKX2xhc3RCbG9jawUGaGVpZ2h0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgILX2xhc3RQcmljZXMJAKQDAQUKdG9rZW5QcmljZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfcmlza0xldmVsAAEFA25pbAQKbGFzdFByaWNlcwkAtQkCCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFCmFzc2V0SWRTdHICC19sYXN0UHJpY2VzAgEsBAluZXdQcmljZXMJAMwIAgkApAMBBQp0b2tlblByaWNlBQpsYXN0UHJpY2VzCgEKc3VtSGFuZGxlcgIFYWNjdW0EbmV4dAkAZAIFBWFjY3VtCQENcGFyc2VJbnRWYWx1ZQEFBG5leHQEB25ld1R3YXAJAGkCCgACJGwFCW5ld1ByaWNlcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAAAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEKc3VtSGFuZGxlcgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQITTGlzdCBzaXplIGV4Y2VlZHMgNgkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgkAkAMBBQluZXdQcmljZXMECXByaWNlRGlmZgkAawMFCnRva2VuUHJpY2UAkE4JAQ1wYXJzZUludFZhbHVlAQkAkQMCBQpsYXN0UHJpY2VzCQBlAgkAkAMBBQpsYXN0UHJpY2VzAAEECnByaWNlRGlmZjIJAGsDBQp0b2tlblByaWNlAJBOBRF0b2tlblJlc2VydmVQcmljZQMDCQEBIQEJAGYCAIRSBQpwcmljZURpZmYyCQBmAgUKcHJpY2VEaWZmMgCcSgcFA25pbAQLcHJpY2VEaWZmT2sDCQBmAgkAZAIAkE4FD3NodXRkb3duVHJpZ2dlcgUJcHJpY2VEaWZmCQBmAgUJcHJpY2VEaWZmCQBlAgCQTgUPc2h1dGRvd25UcmlnZ2VyBwQJcmlza0xldmVsAwULcHJpY2VEaWZmT2sAAQACBA1uZXdQcmljZXNTaXplCQCQAwEFCW5ld1ByaWNlcwMJAGcCAAUFDW5ld1ByaWNlc1NpemUECXVwZFByaWNlcwkAuQkCBQluZXdQcmljZXMCASwJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIHX3R3YXA1QgUHbmV3VHdhcAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfbGFzdEJsb2NrBQZoZWlnaHQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQphc3NldElkU3RyAgtfbGFzdFByaWNlcwUJdXBkUHJpY2VzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICCl9yaXNrTGV2ZWwFCXJpc2tMZXZlbAUDbmlsBAl1cGRQcmljZXMJALkJAgkA0QgCBQluZXdQcmljZXMJAGUCBQ1uZXdQcmljZXNTaXplAAECASwJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIHX3R3YXA1QgUHbmV3VHdhcAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfbGFzdEJsb2NrBQZoZWlnaHQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQphc3NldElkU3RyAgtfbGFzdFByaWNlcwUJdXBkUHJpY2VzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICCl9yaXNrTGV2ZWwFCXJpc2tMZXZlbAUDbmlsAWkBEHVwZGF0ZVRva2VuUHJpY2UBCmFzc2V0SWRTdHIECnRva2VuUHJpY2UDCQAAAgUKYXNzZXRJZFN0cgIFV0FWRVMFCndhdmVzUHJpY2UDCQAAAgUKYXNzZXRJZFN0cgUIeHRuSWRTdHIFDXh0blVzZFByaWNlWDYEDHByaWNlSW5Qb29scwgJAQ1nZXRUb2tlblByaWNlAgkBBXZhbHVlAQkAzwgCBQh0b2tlbklkcwUKYXNzZXRJZFN0cgUKd2F2ZXNQcmljZQJfMQMJAAACBQphc3NldElkU3RyBQZldGhTdHIECGV0aFByaWNlCQERQGV4dHJOYXRpdmUoMTA1MCkCBQtkb3JhQWRkcmVzcwIVJXMlc19fcHJpY2VfX0VUSC1VU0RUBARjb21wCQBrAwUMcHJpY2VJblBvb2xzAJBOBQhldGhQcmljZQMDCQBmAgUEY29tcAkAZQIAkE4FD29yYWNsZVRvbGVyYW5jZQkAZgIJAGQCAJBOBQ9vcmFjbGVUb2xlcmFuY2UFBGNvbXAHBQxwcmljZUluUG9vbHMJAAIBAjJpbnRlcm5hbCBFVEggcHJpY2UgZG9lc24ndCBtYXRjaCB3aXRoIGdsb2JhbCBwcmljZQMJAQIhPQIJAM8IAgUHc3RhYmxlcwUKYXNzZXRJZFN0cgUEdW5pdADAhD0FDHByaWNlSW5Qb29scwQIbGFzdFR3YXAJAQ10cnlHZXRJbnRlZ2VyAQkArAICBQphc3NldElkU3RyAgdfdHdhcDVCBApsYXN0SGVpZ2h0CQENdHJ5R2V0SW50ZWdlcgEJAKwCAgUKYXNzZXRJZFN0cgIKX2xhc3RCbG9jawMJAQIhPQIIBQFpDG9yaWdpbkNhbGxlcgUNYmFja2VuZENhbGxlcgkAAgECGmF2YWlsYWJsZSBmb3IgYmFja2VuZCBvbmx5AwkAZgIAAgkAZQIFBmhlaWdodAUKbGFzdEhlaWdodAUDbmlsAwMJAAACBQpsYXN0SGVpZ2h0AAAGCQBmAgkAZQIFBmhlaWdodAUKbGFzdEhlaWdodAAeCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICB190d2FwNUIFCnRva2VuUHJpY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIKX2xhc3RCbG9jawUGaGVpZ2h0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgILX2xhc3RQcmljZXMJAKQDAQUKdG9rZW5QcmljZQUDbmlsBApsYXN0UHJpY2VzCQC1CQIJARFAZXh0ck5hdGl2ZSgxMDU4KQEJAKwCAgUKYXNzZXRJZFN0cgILX2xhc3RQcmljZXMCASwECW5ld1ByaWNlcwkAzAgCCQCkAwEFCnRva2VuUHJpY2UFCmxhc3RQcmljZXMKAQpzdW1IYW5kbGVyAgVhY2N1bQRuZXh0CQBkAgUFYWNjdW0JAQ1wYXJzZUludFZhbHVlAQUEbmV4dAQHbmV3VHdhcAkAaQIKAAIkbAUJbmV3UHJpY2VzCgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQpzdW1IYW5kbGVyAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhNMaXN0IHNpemUgZXhjZWVkcyA2CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGCQCQAwEFCW5ld1ByaWNlcwQTb3JkZXJib29rVHdhcFJlc3VsdAkBEGdldE9yZGVyYm9va1R3YXABBQphc3NldElkU3RyAwkAAAIFE29yZGVyYm9va1R3YXBSZXN1bHQFE29yZGVyYm9va1R3YXBSZXN1bHQEDSR0MDE5NjEyMTk2NzAFE29yZGVyYm9va1R3YXBSZXN1bHQEDW9yZGVyYm9va1R3YXAIBQ0kdDAxOTYxMjE5NjcwAl8xBA9vcmRlcmJvb2tMb2NrZWQIBQ0kdDAxOTYxMjE5NjcwAl8yBA90d2Fwc0NvbXBhcmlzb24DCQECIT0CBQ1vcmRlcmJvb2tUd2FwAAAJAGsDBQduZXdUd2FwAJBOBQ1vcmRlcmJvb2tUd2FwCQACAQkArAICAhpvcmRlcmJvb2sgcmVzdWx0IHplcm8gZm9yIAUKYXNzZXRJZFN0cgQMdHdhcHNTaW1pbGFyAwkAZgIFD3R3YXBzQ29tcGFyaXNvbgkAZQIAkE4FD29yYWNsZVRvbGVyYW5jZQkAZgIJAGQCAJBOBQ9vcmFjbGVUb2xlcmFuY2UFD3R3YXBzQ29tcGFyaXNvbgcECXByaWNlRGlmZgkAawMFCnRva2VuUHJpY2UAkE4JAQ1wYXJzZUludFZhbHVlAQkAkQMCBQpsYXN0UHJpY2VzCQBlAgkAkAMBBQpsYXN0UHJpY2VzAAEEC3ByaWNlRGlmZk9rAwkAZgIJAGQCAJBOBQ9zaHV0ZG93blRyaWdnZXIFCXByaWNlRGlmZgkAZgIFCXByaWNlRGlmZgkAZQIAkE4FD3NodXRkb3duVHJpZ2dlcgcECXJpc2tMZXZlbAkAZAIJAGQCAwUMdHdhcHNTaW1pbGFyAAEAAgMFD29yZGVyYm9va0xvY2tlZAABAAADCQEBIQEFC3ByaWNlRGlmZk9rAAEAAAQNbmV3UHJpY2VzU2l6ZQkAkAMBBQluZXdQcmljZXMDCQEBIQEFDHR3YXBzU2ltaWxhcgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfcmlza0xldmVsBQlyaXNrTGV2ZWwFA25pbAMJAGcCAAUFDW5ld1ByaWNlc1NpemUECXVwZFByaWNlcwkAuQkCBQluZXdQcmljZXMCASwJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIHX3R3YXA1QgUHbmV3VHdhcAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfbGFzdEJsb2NrBQZoZWlnaHQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQphc3NldElkU3RyAgtfbGFzdFByaWNlcwUJdXBkUHJpY2VzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICCl9yaXNrTGV2ZWwFCXJpc2tMZXZlbAUDbmlsBAl1cGRQcmljZXMJALkJAgkA0QgCBQluZXdQcmljZXMJAGUCBQ1uZXdQcmljZXNTaXplAAECASwJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgUKYXNzZXRJZFN0cgIHX3R3YXA1QgUHbmV3VHdhcAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQphc3NldElkU3RyAgpfbGFzdEJsb2NrBQZoZWlnaHQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQphc3NldElkU3RyAgtfbGFzdFByaWNlcwUJdXBkUHJpY2VzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIFCmFzc2V0SWRTdHICCl9yaXNrTGV2ZWwFCXJpc2tMZXZlbAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBDHVwZGF0ZVRva2VucwADCQECIT0CCAUBaQxvcmlnaW5DYWxsZXIFDWJhY2tlbmRDYWxsZXIJAAIBAhphdmFpbGFibGUgZm9yIGJhY2tlbmQgb25seQQCaTEJAPwHBAUEdGhpcwIQdXBkYXRlVG9rZW5QcmljZQkAzAgCAgVXQVZFUwUDbmlsBQNuaWwDCQAAAgUCaTEFAmkxBAJpMgkA/AcEBQR0aGlzAhB1cGRhdGVUb2tlblByaWNlCQDMCAIFCHh0bklkU3RyBQNuaWwFA25pbAMJAAACBQJpMgUCaTIKAQVjYWxsdQIFYWNjdW0EbmV4dAQCaTMJAPwHBAUEdGhpcwIQdXBkYXRlVG9rZW5QcmljZQkAzAgCBQRuZXh0BQNuaWwFA25pbAMJAAACBQJpMwUCaTMFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgQDcmVzCgACJGwFCHRva2VuSWRzCgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQVjYWxsdQIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMjAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgALAAwADQAOAA8AEAARABIAEwAUAwkAAAIFA3JlcwUDcmVzBQNyZXMJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEOdXBkYXRlTFBUb2tlbnMAAwkBAiE9AggFAWkMb3JpZ2luQ2FsbGVyBQ1iYWNrZW5kQ2FsbGVyCQACAQIaYXZhaWxhYmxlIGZvciBiYWNrZW5kIG9ubHkKAQVjYWxsdQIFYWNjdW0EbmV4dAQCaTEJAPwHBAUEdGhpcwISdXBkYXRlTHBUb2tlblByaWNlCQDMCAIFBG5leHQFA25pbAUDbmlsAwkAAAIFAmkxBQJpMQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBANyZXMKAAIkbAUKbHBUb2tlbklkcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEFY2FsbHUCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDIwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAMJAAACBQNyZXMFA3JlcwQCaTIJAPwHBAUEdGhpcwIUdXBkYXRlV3hMcFRva2VuUHJpY2UJAMwIAgkAkQMCBQx3eExwVG9rZW5JZHMAAAUDbmlsBQNuaWwDCQAAAgUCaTIFAmkyBQNyZXMJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQENaW5pdE5ld01hc3RlcgEGbWFzdGVyBAlhZGRyZXNzT0sEByRtYXRjaDAJAKYIAQUGbWFzdGVyAwkAAQIFByRtYXRjaDACB0FkZHJlc3MEAWEFByRtYXRjaDAGBwMJAQEhAQUJYWRkcmVzc09LCQACAQkArAICAhJpbmNvcnJlY3QgYWRkcmVzcyAFBm1hc3RlcgMJAQIhPQIIBQFpBmNhbGxlcgUEdGhpcwkAAgECHGF2YWlsYWJsZSBmb3Igc2VsZiBjYWxsIG9ubHkJAMwIAgkBC1N0cmluZ0VudHJ5AgIPdmVyaWZpZXJfbWFzdGVyBQZtYXN0ZXIFA25pbAFpARByZXF1ZXN0TmV3TWFzdGVyAQluZXdNYXN0ZXIECWFkZHJlc3NPSwQHJG1hdGNoMAkApggBBQluZXdNYXN0ZXIDCQABAgUHJG1hdGNoMAIHQWRkcmVzcwQBYQUHJG1hdGNoMAYHAwkBASEBBQlhZGRyZXNzT0sJAAIBCQCsAgICEmluY29ycmVjdCBhZGRyZXNzIAUJbmV3TWFzdGVyAwkAAAIJAM8IAgUJc2VudGluZWxzCAUBaQZjYWxsZXIFBHVuaXQJAAIBAg53aGl0ZWxpc3Qgb25seQkAzAgCCQELU3RyaW5nRW50cnkCAhJ2ZXJpZmllcl9uZXdNYXN0ZXIFCW5ld01hc3RlcgkAzAgCCQEMSW50ZWdlckVudHJ5AgIfdmVyaWZpZXJfbmV3TWFzdGVyRWxpZ2libGVBZnRlcgkAZAIFBmhlaWdodACQTgUDbmlsAWkBEWFjdGl2YXRlTmV3TWFzdGVyAAQOZWxpZ2libGVIZWlnaHQJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMCH3ZlcmlmaWVyX25ld01hc3RlckVsaWdpYmxlQWZ0ZXIECW5ld01hc3RlcgkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhpcwISdmVyaWZpZXJfbmV3TWFzdGVyAwkAAAIJAM8IAgUJc2VudGluZWxzCAUBaQZjYWxsZXIFBHVuaXQJAAIBAg53aGl0ZWxpc3Qgb25seQMJAGYCBQ5lbGlnaWJsZUhlaWdodAUGaGVpZ2h0CQACAQIibmV3IG1hc3RlciBjYW5ub3QgYmUgYWN0aXZhdGVkIHlldAkAzAgCCQELRGVsZXRlRW50cnkBAhJ2ZXJpZmllcl9uZXdNYXN0ZXIJAMwIAgkBC0RlbGV0ZUVudHJ5AQIfdmVyaWZpZXJfbmV3TWFzdGVyRWxpZ2libGVBZnRlcgkAzAgCCQELU3RyaW5nRW50cnkCAg92ZXJpZmllcl9tYXN0ZXIFCW5ld01hc3RlcgUDbmlsAQJ0eAEGdmVyaWZ5AAQNbWFzdGVyQWRkcmVzcwkAoggBAg92ZXJpZmllcl9tYXN0ZXIEEGFwcHJvdmVkQnlNYXN0ZXIEByRtYXRjaDAFDW1hc3RlckFkZHJlc3MDCQABAgUHJG1hdGNoMAIGU3RyaW5nBAF4BQckbWF0Y2gwBAckbWF0Y2gxCQCaCAIJARFAZXh0ck5hdGl2ZSgxMDYyKQEFAXgJAKwCAgILYXBwcm92ZWRUeF8JANgEAQgFAnR4AmlkAwkAAQIFByRtYXRjaDECA0ludAQBeQUHJG1hdGNoMQUBeQAAAAEDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAACAUCdHgPc2VuZGVyUHVibGljS2V5CQBmAgUQYXBwcm92ZWRCeU1hc3RlcgAAB1wubqM=", "height": 4028727, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: Bg27BHi8SiTRRTa6Y5E6iv5WJNRoBPYijMpcKi6HGhHy Next: H2mMg1oAJNHS4hyu7QFAerzJcpBb6WAB4YubmoRs4C6t Full:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let Scale8 = 100000000 | |
5 | 5 | ||
6 | 6 | let sentinels = [Address(base58'3PMcMiMEs6w56NRGacksXtFG5zS7doE9fpL'), Address(base58'3PQdNxynJy5mche2kxMVc5shXWzK8Gstq3o'), Address(base58'3P8auNWJkxxByyJtwErFXaxiXcGM45qQ1hA'), Address(base58'3P8qVX189qpoTJZQQQdKS9endHK5sxWsvrd')] | |
7 | 7 | ||
8 | 8 | let backendCaller = Address(base58'3PLGH6sG6ND59GU6gFXHKQRuL5bdpWFs6U6') | |
9 | 9 | ||
10 | 10 | let orderbookOracle = Address(base58'3PPXVKjN6nRMzXeegcYhfiic96pd2c98Ekm') | |
11 | 11 | ||
12 | 12 | let xtnId = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p' | |
13 | 13 | ||
14 | 14 | let xtnIdStr = "DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p" | |
15 | 15 | ||
16 | 16 | let ethStr = "3VuV5WTmDz47Dmdn3QpcYjzbSdipjQE4JMdNe1xZpX13" | |
17 | 17 | ||
18 | 18 | let stables = ["9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi", "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW"] | |
19 | 19 | ||
20 | 20 | let tokenIds = ["6nSpVyNH7yM69eg446wrQR94ipbbcmZMU1ENPwanC97g", "Ehie5xYpeN8op1Cctc6aGUrqx8jq3jtf1DSjXDbfm7aT", "4LHHvYGNKJUg5hj65aGD5vgScvCBmLpdRFtjokvCjSL8", "C1iWsKGqLwjHUndiQ7iXpdmPum9PeCDFfyXBdJJosDRS", "34N9YcEETLWn93qYQ64EsP1x89tSruJU44RrEMSXXEPJ", "6XtHjpXbs9RRJP2Sr9GUyVqzACcby9TkThHXnjVC5CDJ", "HEB8Qaw9xrWpWs8tHsiATYGBWDBtP2S7kcPALrMu43AS", "Ajso6nTTjptu2UHLx6hfSXVtHFtRBJCkKYd5SAyj7zf5", "DSbbhLsSTeDg5Lsiufk2Aneh3DjVqJuPr2M9uU1gwy5p", "Atqv59EYzjFGuitKVnMRk6H8FukjoV3ktPorbEys25on", "2thsACuHmzDMuNezPM32wg9a3BwUzBWDeSKakgz3cw21", "YiNbofFzC17jEHHCMwrRcpy9MrrjabMMLZxg8g5xmf7", "8t4DPWTwPzpatHA9AkTxWAB47THnYzBsDnoY7fQqbG91", "At8D6NFFpheCbvKVnjVoeLL84Eo8NZn6ovManxfLaFWL", "6phK22ztGBW127gUFmdMEHKB3CVd6ZhWox2WtwJkbqTq", "3VuV5WTmDz47Dmdn3QpcYjzbSdipjQE4JMdNe1xZpX13", "9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi"] | |
21 | 21 | ||
22 | 22 | let tokenIsBaseAsset = [false, false, false, false, true, true, false, false, false, false, false, false, false, false, true, false, true, true] | |
23 | 23 | ||
24 | 24 | let tokenDecimals = [1000000, 100000000, 100000000, 100000000, 1000000, 1000000, 100000000, 100000000, 100000000, 100000000, 100000000, 100000000, 100000000, 1000000, 1000000, 100000000, 1000000, 1000000] | |
25 | 25 | ||
26 | 26 | let tokenWXWavesLPs = ["DZeA7RbVab98r61hVH5zcFZXwNALPJGgq1eyHandsKKW", "72E4MMFHngsixqAkNZCRJMLTmgAP9nufdsXexnS5qxVx", "FTTJnB6xG4CKZEJMgsgqjmgbzcyH8ya3tV8LQP2GCDs7", "3UnytNnkd48fHofHPBp59BkpZA1uMSJ8PdhWUCuerics", "ByJtVuYi58CtdjtasDuTqtgev8qWkdJybF7rY1xosmhE", "BoXuf9ewJSE1UMrKibZmigzC6tBzv8RFTeZmofEos2ZJ", "7G687ui3m59oAd6qkyni2vt77bHF4bbA6xcPfDo3uiTu", "EMsZpqcT8NuMYZCEPHzvrhQCQKhfXjdeekZvcWpHsNvh", "E3YhURNsCmjUaYNigH6AayLxbh9u81uVGU3ZvHKznY5v", "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe", "AKQsEQoeinKRFtdx6rhKWcpkAMu6cbDLdtSWnR8tpBCq", "DjgwW4CMJEfpzj7SLFFPiCqf1zfmAgf1MD5gAS8SjKt4", "", "F3s92VRdppEtLVAP697CTM6AzZJECRcoMx5kj2FkGv4M", "6iMB6LKSrgv9waEvEnN6Ydyx7dfxPnGcTw8318WVm5bR", "E6MUGSUieSsq6QiJcbp6REdQXyjp6uPWLitnp9Gdtsuh", "6KWpnpLk6GNh5HG9ct9pNDCYFWJBoAAshZsp4D8jcFpV", ""] | |
27 | 27 | ||
28 | 28 | let tokenWXXTNPools = ["", "", "", "3P3Y38zymCHoTM6qcE9JcEA8rhaGTQbVMgu", "", "", "3P615yXeQ9Qu4qBM1QGimGzixyMS5W4Ktug", "3PEkD5LuHGWhUfgBW1owZFxVydYoWjV6Jia", "3P5HjPowgipiVB3UvXphDiPvKs424WC9xQw", "3PCENpEKe8atwELZ7oCSmcdEfcRuKTrUx99", "", "", "3PH832eKnw67PFvJfUg9Knz6BHM87QvjMeL", "", "3PGcYzoUfQEQkhRgrBqmG5Wpc7A1ncFvRtM", "", "", ""] | |
29 | 29 | ||
30 | 30 | let tokenSwopPools = ["3P2V63Xd6BviDkeMzxhUw2SJyojByRz8a8m", "3P27S9V36kw2McjWRZ37AxTx8iwkd7HXw6W", "3P6DLdJTP2EySq9MFdJu6beUevrQd2sVVBh", "", "", "", "3PBHyEwmERR1CEkrTNbPj2bgyisTfPRqfee", "3PDSXwwA2xAtmxeo2bebDP3AA5ynZ2Krgnw", "3PJ48P3p2wvWUjgQaQiZ2cFbr8qmxMokBGd", "3PKi4G3VX2k42ZSmNNrmvgdDH7JzRaUhY7R", "", "", "3PCwvns2dnmobD6Z4cR86v98s7LgMZYygEy", "3PLp9KrdrbWAvSFHGJVEaszUubquz3M6mfe", "", "", "", ""] | |
31 | 31 | ||
32 | 32 | let lpTokenIds = ["9dbpSr8d18qWQxn5fJJSS1LLQ8CmSZ6gYmjuPRzg3RBM", "9MKixRt9rNRyaJCT2pexbXkuvpZBdJREdTU36bGit8iw", "6bZbRmou7M7wXBunMXQnZ4Rm66HxZF3KfMEiFwk3wmnA", "XjdJKWtPYCz585QB7LnxDP76UGRukazedDubUx9DHQH", "EA7siGMSTxz6EtdpkCiVWQHupFT5N7UbvQrW9kvxCE42"] | |
33 | 33 | ||
34 | 34 | let lpTokenBase = ["Atqv59EYzjFGuitKVnMRk6H8FukjoV3ktPorbEys25on", "WAVES", "WAVES", "WAVES", "3VuV5WTmDz47Dmdn3QpcYjzbSdipjQE4JMdNe1xZpX13"] | |
35 | 35 | ||
36 | 36 | let lpTokenPool = [Address(base58'3PKJxKhn9MrzcvZv689bUpZgh4rurEgoKC8'), Address(base58'3P4PFYKHyxitgwVb4WVWEcdH7ZsuR4GgcXC'), Address(base58'3PCpjLGo4wDuv23kjmGF6mHs86Bf3soYvAq'), Address(base58'3P3EnYA57kMZ8kXVVThi1ZZApZeXUQHqtEe'), Address(base58'3P4U6fo7BZHDymQK82G5pveGPH7CA6PbKit')] | |
37 | 37 | ||
38 | 38 | let lpTokenBaseShare = [30000000, 20000000, 20000000, 34000000, 32000000] | |
39 | 39 | ||
40 | 40 | let lpTokenReserveBase = ["", "", "", "3VuV5WTmDz47Dmdn3QpcYjzbSdipjQE4JMdNe1xZpX13", ""] | |
41 | 41 | ||
42 | 42 | let lpTokenReserveShare = [0, 0, 0, 33000000, 0] | |
43 | 43 | ||
44 | 44 | let wxLpTokenIds = ["rZMQ6g31Lr7sPAaAoYc4U2PHCVauTuKUSzUbJtUyPZN"] | |
45 | 45 | ||
46 | 46 | let wxLpUnderlying = [["9wc3LXNA4TEBsXyKtoLE9mrbDD7WMHXvXrCjZvabLAsi", "HGgabTqUS8WtVFUJzfmrTDMgEccJuZLBPhFgQFxvnsoW"]] | |
47 | 47 | ||
48 | 48 | let wx_restAddressStr = "3P8MoPnsaurofk1VyhsdAFkeQ6ijpJYXCpW" | |
49 | 49 | ||
50 | 50 | let wx_restDapp = addressFromStringValue(wx_restAddressStr) | |
51 | 51 | ||
52 | 52 | let swopRestDapp = addressFromStringValue("3P56jNQzECXnrWpnbbSJKw7Eooo6fkUaMPp") | |
53 | 53 | ||
54 | 54 | let oracleTolerance = 1000 | |
55 | 55 | ||
56 | 56 | let shutdownTrigger = 3000 | |
57 | 57 | ||
58 | 58 | let updateDelay = 0 | |
59 | 59 | ||
60 | 60 | let revisionNum = "" | |
61 | 61 | ||
62 | 62 | let SEP = "__" | |
63 | 63 | ||
64 | 64 | func asString (v) = match v { | |
65 | 65 | case s: String => | |
66 | 66 | s | |
67 | 67 | case _ => | |
68 | 68 | throw("fail to cast into String") | |
69 | 69 | } | |
70 | 70 | ||
71 | 71 | ||
72 | 72 | func asInt (v) = match v { | |
73 | 73 | case i: Int => | |
74 | 74 | i | |
75 | 75 | case _ => | |
76 | 76 | throw("fail to cast into Int") | |
77 | 77 | } | |
78 | 78 | ||
79 | 79 | ||
80 | 80 | func tryGetInteger (key) = match getInteger(this, key) { | |
81 | 81 | case b: Int => | |
82 | 82 | b | |
83 | 83 | case _ => | |
84 | 84 | 0 | |
85 | 85 | } | |
86 | 86 | ||
87 | 87 | ||
88 | 88 | func getAssetBytes (assetIdStr) = if ((assetIdStr == "WAVES")) | |
89 | 89 | then unit | |
90 | 90 | else fromBase58String(assetIdStr) | |
91 | 91 | ||
92 | 92 | ||
93 | 93 | let doraAddressStr = "3PKkojKdd6BBzTf1RXbQVfUDraNFXXHKzQF" | |
94 | 94 | ||
95 | 95 | let doraAddress = addressFromStringValue(doraAddressStr) | |
96 | 96 | ||
97 | 97 | let xtnUsdPriceX6 = valueOrErrorMessage(getInteger(doraAddress, "%s%s__price__USDN-USDT"), "DORA doesn't contain USDN/USDT price") | |
98 | 98 | ||
99 | 99 | let wavesPrice = getIntegerValue(doraAddress, "%s%s__price__WAVES-USDT") | |
100 | 100 | ||
101 | 101 | let ethDoraPrice = getIntegerValue(doraAddress, "%s%s__price__ETH-USDT") | |
102 | 102 | ||
103 | 103 | func getTokenPrice (tokenNum,wavesUsdPriceX6) = { | |
104 | 104 | let tokenIdStr = tokenIds[tokenNum] | |
105 | 105 | let tokenId = fromBase58String(tokenIdStr) | |
106 | 106 | if ((indexOf(stables, tokenIdStr) != unit)) | |
107 | 107 | then $Tuple2(1000000, "stable token price") | |
108 | 108 | else { | |
109 | 109 | let wx_T1Waves_lp = tokenWXWavesLPs[tokenNum] | |
110 | 110 | let $t064597175 = if ((wx_T1Waves_lp == "")) | |
111 | 111 | then $Tuple2(0, 0) | |
112 | 112 | else { | |
113 | 113 | let wxRestResult = split(asString(invoke(wx_restDapp, "poolStatsREADONLY", [wx_T1Waves_lp], nil)), "__") | |
114 | 114 | let $t066836775 = if (tokenIsBaseAsset[tokenNum]) | |
115 | 115 | then $Tuple2(1, 2) | |
116 | 116 | else $Tuple2(2, 1) | |
117 | 117 | let wavesInPairNum = $t066836775._1 | |
118 | 118 | let tokenInPairNum = $t066836775._2 | |
119 | 119 | let wx_T1Waves_WAVESx8 = parseIntValue(wxRestResult[wavesInPairNum]) | |
120 | 120 | let wx_T1Waves_T1x8 = fraction(parseIntValue(wxRestResult[tokenInPairNum]), 100000000, tokenDecimals[tokenNum]) | |
121 | 121 | let wx_T1Waves_T1UsdPriceX6 = fraction(wx_T1Waves_WAVESx8, wavesUsdPriceX6, wx_T1Waves_T1x8) | |
122 | 122 | let wx_T1Waves_Weight = wx_T1Waves_T1x8 | |
123 | 123 | $Tuple2(wx_T1Waves_T1UsdPriceX6, wx_T1Waves_Weight) | |
124 | 124 | } | |
125 | 125 | let wx_T1Waves_T1UsdPriceX6 = $t064597175._1 | |
126 | 126 | let wx_T1Waves_Weight = $t064597175._2 | |
127 | 127 | let wx_T1Xtn_AddressStr = tokenWXXTNPools[tokenNum] | |
128 | 128 | let $t072377668 = if ((wx_T1Xtn_AddressStr == "")) | |
129 | 129 | then $Tuple2(0, 0) | |
130 | 130 | else { | |
131 | 131 | let wx_T1Xtn_Address = addressFromStringValue(wx_T1Xtn_AddressStr) | |
132 | 132 | let wx_T1Xtn_XTNx6 = assetBalance(wx_T1Xtn_Address, xtnId) | |
133 | 133 | let wx_T1Xtn_T1x8 = assetBalance(wx_T1Xtn_Address, tokenId) | |
134 | 134 | $Tuple2(fraction((wx_T1Xtn_XTNx6 * (tokenDecimals[tokenNum] / 1000000)), xtnUsdPriceX6, wx_T1Xtn_T1x8), wx_T1Xtn_T1x8) | |
135 | 135 | } | |
136 | 136 | let wx_T1Xtn_T1UsdPriceX6 = $t072377668._1 | |
137 | 137 | let wx_T1Xtn_Weight = $t072377668._2 | |
138 | 138 | let swop_T1Xtn_AddressStr = tokenSwopPools[tokenNum] | |
139 | 139 | let $t077318349 = if ((swop_T1Xtn_AddressStr == "")) | |
140 | 140 | then $Tuple2(0, 0) | |
141 | 141 | else { | |
142 | 142 | let swop_T1Xtn_Address = addressFromStringValue(swop_T1Xtn_AddressStr) | |
143 | 143 | let swop_T1Xtn_T1XtnPriceX6 = asInt(invoke(swopRestDapp, "calcGetAmountCPMM", [swop_T1Xtn_AddressStr, "3P88qk1KzF1BKjD7fC7LjNVAKM4ezff5WE6", tokenIdStr, tokenDecimals[tokenNum]], nil)) | |
144 | 144 | let swop_T1Xtn_T1UsdPriceX6 = fraction(swop_T1Xtn_T1XtnPriceX6, xtnUsdPriceX6, 1000000) | |
145 | 145 | let swop_T1Xtn_Weight = getIntegerValue(swop_T1Xtn_Address, "A_asset_balance") | |
146 | 146 | $Tuple2(swop_T1Xtn_T1UsdPriceX6, swop_T1Xtn_Weight) | |
147 | 147 | } | |
148 | 148 | let swop_T1Xtn_T1UsdPriceX6 = $t077318349._1 | |
149 | 149 | let swop_T1Xtn_Weight = $t077318349._2 | |
150 | 150 | let W = ((wx_T1Waves_Weight + wx_T1Xtn_Weight) + swop_T1Xtn_Weight) | |
151 | 151 | let T1UsdPriceX6 = ((fraction(wx_T1Waves_T1UsdPriceX6, wx_T1Waves_Weight, W) + fraction(wx_T1Xtn_T1UsdPriceX6, wx_T1Xtn_Weight, W)) + fraction(swop_T1Xtn_T1UsdPriceX6, swop_T1Xtn_Weight, W)) | |
152 | 152 | let debug = ((((((("T1UsdPriceX6=" + toString(T1UsdPriceX6)) + " wx_T1Waves_T1UsdPriceX6=") + toString(wx_T1Waves_T1UsdPriceX6)) + " wx_T1Xtn_T1UsdPriceX6=") + toString(wx_T1Xtn_T1UsdPriceX6)) + " swop_T1Xtn_T1UsdPriceX6=") + toString(swop_T1Xtn_T1UsdPriceX6)) | |
153 | 153 | $Tuple2(T1UsdPriceX6, debug) | |
154 | 154 | } | |
155 | 155 | } | |
156 | 156 | ||
157 | 157 | ||
158 | 158 | let lastUpdatedBlockKey = "lastUpdatedBlock" | |
159 | 159 | ||
160 | 160 | let n15List = [1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1] | |
161 | 161 | ||
162 | 162 | func getPriceDataKey (block,tokenId) = ((("block_" + toString(block)) + "_") + tokenId) | |
163 | 163 | ||
164 | 164 | ||
165 | 165 | func getPriceValue (tokenId,block) = valueOrElse(getInteger(orderbookOracle, getPriceDataKey(block, tokenId)), 0) | |
166 | 166 | ||
167 | 167 | ||
168 | 168 | func getLastUpdatedBlock () = { | |
169 | 169 | let lastUpdatedBlock = valueOrErrorMessage(getInteger(orderbookOracle, lastUpdatedBlockKey), "Last updated block key not found") | |
170 | 170 | if ((0 >= lastUpdatedBlock)) | |
171 | 171 | then throw("Invalid block value") | |
172 | 172 | else if (((height - lastUpdatedBlock) > 10)) | |
173 | 173 | then throw("The price has not been updated for a long time") | |
174 | 174 | else lastUpdatedBlock | |
175 | 175 | } | |
176 | 176 | ||
177 | 177 | ||
178 | 178 | func getPriceList (accum,next) = { | |
179 | 179 | let $t097909831 = accum | |
180 | 180 | let result = $t097909831._1 | |
181 | 181 | let block = $t097909831._2 | |
182 | 182 | let tokenId = $t097909831._3 | |
183 | 183 | let sum = $t097909831._4 | |
184 | 184 | let priceValue = getPriceValue(tokenId, block) | |
185 | 185 | let nextBlock = (block - next) | |
186 | 186 | if ((priceValue > 0)) | |
187 | 187 | then $Tuple4((result :+ priceValue), nextBlock, tokenId, (sum + priceValue)) | |
188 | 188 | else $Tuple4(result, nextBlock, tokenId, sum) | |
189 | 189 | } | |
190 | 190 | ||
191 | 191 | ||
192 | 192 | func getOrderbookTwap (tokenIdStr) = { | |
193 | 193 | let inv = invoke(this, "getOrderbookTwap15", [tokenIdStr, false], nil) | |
194 | 194 | if ((inv == inv)) | |
195 | 195 | then { | |
196 | 196 | let orderbookPrice = { | |
197 | 197 | let @ = inv | |
198 | 198 | if ($isInstanceOf(@, "(Int, Boolean)")) | |
199 | 199 | then @ | |
200 | 200 | else throw(($getType(@) + " couldn't be cast to (Int, Boolean)")) | |
201 | 201 | } | |
202 | 202 | if ((orderbookPrice == orderbookPrice)) | |
203 | 203 | then orderbookPrice | |
204 | 204 | else throw("Strict value is not equal to itself.") | |
205 | 205 | } | |
206 | 206 | else throw("Strict value is not equal to itself.") | |
207 | 207 | } | |
208 | 208 | ||
209 | 209 | ||
210 | 210 | @Callable(i) | |
211 | 211 | func getOrderbookTwap15 (tokenId,debug) = { | |
212 | 212 | let lastUpdatedBlock = getLastUpdatedBlock() | |
213 | 213 | let tuple = { | |
214 | 214 | let $l = n15List | |
215 | 215 | let $s = size($l) | |
216 | 216 | let $acc0 = $Tuple4(nil, lastUpdatedBlock, tokenId, 0) | |
217 | 217 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
218 | 218 | then $a | |
219 | 219 | else getPriceList($a, $l[$i]) | |
220 | 220 | ||
221 | 221 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
222 | 222 | then $a | |
223 | 223 | else throw("List size exceeds 15") | |
224 | 224 | ||
225 | 225 | $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($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15) | |
226 | 226 | } | |
227 | 227 | let pricesList = tuple._1 | |
228 | 228 | let sum = tuple._4 | |
229 | 229 | let average = fraction(sum, 1000000, fraction(size(pricesList), 1000000, 1)) | |
230 | 230 | let isLocked = valueOrElse(getBoolean(orderbookOracle, (tokenId + "_locked")), false) | |
231 | 231 | if (debug) | |
232 | 232 | then throw((((((((("average: " + toString(average)) + ", isLocked: ") + toString(isLocked)) + ", ") + "priceListSize: ") + toString(size(pricesList))) + ", sum: ") + toString(sum))) | |
233 | 233 | else $Tuple2(nil, $Tuple2(average, isLocked)) | |
234 | 234 | } | |
235 | 235 | ||
236 | 236 | ||
237 | 237 | ||
238 | 238 | @Callable(i) | |
239 | 239 | func updateWxLpTokenPrice (assetIdStr) = { | |
240 | 240 | let tokenNum = value(indexOf(wxLpTokenIds, assetIdStr)) | |
241 | 241 | let underlying = wxLpUnderlying[tokenNum] | |
242 | 242 | let underlying1Price = if ((underlying[0] == "WAVES")) | |
243 | 243 | then wavesPrice | |
244 | 244 | else if ((indexOf(stables, underlying[0]) != unit)) | |
245 | 245 | then 1000000 | |
246 | 246 | else getTokenPrice(value(indexOf(tokenIds, underlying[0])), wavesPrice)._1 | |
247 | 247 | let underlying1Decimals = if ((underlying[0] == "WAVES")) | |
248 | 248 | then 8 | |
249 | 249 | else value(value(assetInfo(fromBase58String(underlying[0]))).decimals) | |
250 | 250 | let underlying2Price = if ((underlying[1] == "WAVES")) | |
251 | 251 | then wavesPrice | |
252 | 252 | else if ((indexOf(stables, underlying[1]) != unit)) | |
253 | 253 | then 1000000 | |
254 | 254 | else getTokenPrice(value(indexOf(tokenIds, underlying[1])), wavesPrice)._1 | |
255 | 255 | let underlying2Decimals = if ((underlying[1] == "WAVES")) | |
256 | 256 | then 8 | |
257 | 257 | else value(value(assetInfo(fromBase58String(underlying[1]))).decimals) | |
258 | 258 | let underlying1Scale = if ((underlying1Decimals == 8)) | |
259 | 259 | then Scale8 | |
260 | 260 | else 1000000 | |
261 | 261 | let underlying2Scale = if ((underlying2Decimals == 8)) | |
262 | 262 | then Scale8 | |
263 | 263 | else 1000000 | |
264 | 264 | let inv = { | |
265 | 265 | let @ = invoke(wx_restDapp, "poolEvaluateGetREADONLY", [assetIdStr, 100000000], nil) | |
266 | 266 | if ($isInstanceOf(@, "String")) | |
267 | 267 | then @ | |
268 | 268 | else throw(($getType(@) + " couldn't be cast to String")) | |
269 | 269 | } | |
270 | 270 | if ((inv == inv)) | |
271 | 271 | then { | |
272 | 272 | let vals = split(inv, "__") | |
273 | 273 | let underlying1Amount = parseIntValue(vals[1]) | |
274 | 274 | let underlying2Amount = parseIntValue(vals[2]) | |
275 | 275 | let tokenPrice = (fraction(underlying1Price, underlying1Amount, underlying1Scale) + fraction(underlying2Price, underlying2Amount, underlying2Scale)) | |
276 | 276 | let lastTwap = tryGetInteger((assetIdStr + "_twap5B")) | |
277 | 277 | let lastHeight = tryGetInteger((assetIdStr + "_lastBlock")) | |
278 | 278 | if ((i.originCaller != backendCaller)) | |
279 | 279 | then throw("available for backend only") | |
280 | 280 | else if ((updateDelay > (height - lastHeight))) | |
281 | 281 | then nil | |
282 | 282 | else if (if ((lastHeight == 0)) | |
283 | 283 | then true | |
284 | 284 | else ((height - lastHeight) > 100)) | |
285 | 285 | then [IntegerEntry((assetIdStr + "_twap5B"), tokenPrice), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), toString(tokenPrice)), IntegerEntry((assetIdStr + "_riskLevel"), 1)] | |
286 | 286 | else { | |
287 | 287 | let lastPrices = split(getStringValue((assetIdStr + "_lastPrices")), ",") | |
288 | 288 | let newPrices = toString(tokenPrice) :: lastPrices | |
289 | 289 | func sumHandler (accum,next) = (accum + parseIntValue(next)) | |
290 | 290 | ||
291 | 291 | let newTwap = ({ | |
292 | 292 | let $l = newPrices | |
293 | 293 | let $s = size($l) | |
294 | 294 | let $acc0 = 0 | |
295 | 295 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
296 | 296 | then $a | |
297 | 297 | else sumHandler($a, $l[$i]) | |
298 | 298 | ||
299 | 299 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
300 | 300 | then $a | |
301 | 301 | else throw("List size exceeds 6") | |
302 | 302 | ||
303 | 303 | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
304 | 304 | } / size(newPrices)) | |
305 | 305 | let priceDiff = fraction(tokenPrice, 10000, parseIntValue(lastPrices[(size(lastPrices) - 1)])) | |
306 | 306 | let priceDiffOk = if (((10000 + shutdownTrigger) > priceDiff)) | |
307 | 307 | then (priceDiff > (10000 - shutdownTrigger)) | |
308 | 308 | else false | |
309 | 309 | let riskLevel = if (priceDiffOk) | |
310 | 310 | then 1 | |
311 | 311 | else 2 | |
312 | 312 | if ((5 >= size(newPrices))) | |
313 | 313 | then { | |
314 | 314 | let updPrices = makeString(newPrices, ",") | |
315 | 315 | [IntegerEntry((assetIdStr + "_twap5B"), newTwap), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), updPrices), IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
316 | 316 | } | |
317 | 317 | else { | |
318 | 318 | let updPrices = makeString(removeByIndex(newPrices, (size(newPrices) - 1)), ",") | |
319 | 319 | [IntegerEntry((assetIdStr + "_twap5B"), newTwap), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), updPrices), IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
320 | 320 | } | |
321 | 321 | } | |
322 | 322 | } | |
323 | 323 | else throw("Strict value is not equal to itself.") | |
324 | 324 | } | |
325 | 325 | ||
326 | 326 | ||
327 | 327 | ||
328 | 328 | @Callable(i) | |
329 | 329 | func updateLpTokenPrice (assetIdStr) = { | |
330 | 330 | let tokenNum = value(indexOf(lpTokenIds, assetIdStr)) | |
331 | 331 | let baseIdStr = lpTokenBase[tokenNum] | |
332 | 332 | let basePrice = if ((baseIdStr == "WAVES")) | |
333 | 333 | then wavesPrice | |
334 | 334 | else getTokenPrice(value(indexOf(tokenIds, baseIdStr)), wavesPrice)._1 | |
335 | 335 | let baseQuantity = getIntegerValue(lpTokenPool[tokenNum], (("global_" + baseIdStr) + "_balance")) | |
336 | 336 | let tokenQuantity = value(assetInfo(value(getAssetBytes(assetIdStr)))).quantity | |
337 | 337 | let tokenPrice = fraction(fraction(fraction(basePrice, baseQuantity, 1000000), Scale8, lpTokenBaseShare[tokenNum]), 1000000, tokenQuantity) | |
338 | 338 | let tokenReservePrice = if ((lpTokenReserveBase[tokenNum] != "")) | |
339 | 339 | then { | |
340 | 340 | let base2IdStr = lpTokenReserveBase[tokenNum] | |
341 | 341 | let base2Price = if ((base2IdStr == "WAVES")) | |
342 | 342 | then wavesPrice | |
343 | 343 | else if ((base2IdStr == "3VuV5WTmDz47Dmdn3QpcYjzbSdipjQE4JMdNe1xZpX13")) | |
344 | 344 | then ethDoraPrice | |
345 | 345 | else if ((base2IdStr == xtnIdStr)) | |
346 | 346 | then (xtnUsdPriceX6 * 100) | |
347 | 347 | else getTokenPrice(value(indexOf(tokenIds, base2IdStr)), wavesPrice)._1 | |
348 | 348 | let base2Quantity = getIntegerValue(lpTokenPool[tokenNum], (("global_" + base2IdStr) + "_balance")) | |
349 | 349 | fraction(fraction(fraction(base2Price, base2Quantity, 1000000), Scale8, lpTokenReserveShare[tokenNum]), 1000000, tokenQuantity) | |
350 | 350 | } | |
351 | 351 | else tokenPrice | |
352 | 352 | let lastTwap = tryGetInteger((assetIdStr + "_twap5B")) | |
353 | 353 | let lastHeight = tryGetInteger((assetIdStr + "_lastBlock")) | |
354 | 354 | if ((i.originCaller != backendCaller)) | |
355 | 355 | then throw("available for backend only") | |
356 | 356 | else if ((2 > (height - lastHeight))) | |
357 | 357 | then nil | |
358 | 358 | else if (if ((lastHeight == 0)) | |
359 | 359 | then true | |
360 | 360 | else ((height - lastHeight) > 100)) | |
361 | 361 | then [IntegerEntry((assetIdStr + "_twap5B"), tokenPrice), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), toString(tokenPrice)), IntegerEntry((assetIdStr + "_riskLevel"), 1)] | |
362 | 362 | else { | |
363 | 363 | let lastPrices = split(getStringValue((assetIdStr + "_lastPrices")), ",") | |
364 | 364 | let newPrices = toString(tokenPrice) :: lastPrices | |
365 | 365 | func sumHandler (accum,next) = (accum + parseIntValue(next)) | |
366 | 366 | ||
367 | 367 | let newTwap = ({ | |
368 | 368 | let $l = newPrices | |
369 | 369 | let $s = size($l) | |
370 | 370 | let $acc0 = 0 | |
371 | 371 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
372 | 372 | then $a | |
373 | 373 | else sumHandler($a, $l[$i]) | |
374 | 374 | ||
375 | 375 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
376 | 376 | then $a | |
377 | 377 | else throw("List size exceeds 6") | |
378 | 378 | ||
379 | 379 | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
380 | 380 | } / size(newPrices)) | |
381 | 381 | let priceDiff = fraction(tokenPrice, 10000, parseIntValue(lastPrices[(size(lastPrices) - 1)])) | |
382 | 382 | let priceDiff2 = fraction(tokenPrice, 10000, tokenReservePrice) | |
383 | 383 | if (if (!((10500 > priceDiff2))) | |
384 | 384 | then (priceDiff2 > 9500) | |
385 | 385 | else false) | |
386 | 386 | then nil | |
387 | 387 | else { | |
388 | 388 | let priceDiffOk = if (((10000 + shutdownTrigger) > priceDiff)) | |
389 | 389 | then (priceDiff > (10000 - shutdownTrigger)) | |
390 | 390 | else false | |
391 | 391 | let riskLevel = if (priceDiffOk) | |
392 | 392 | then 1 | |
393 | 393 | else 2 | |
394 | 394 | let newPricesSize = size(newPrices) | |
395 | 395 | if ((5 >= newPricesSize)) | |
396 | 396 | then { | |
397 | 397 | let updPrices = makeString(newPrices, ",") | |
398 | 398 | [IntegerEntry((assetIdStr + "_twap5B"), newTwap), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), updPrices), IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
399 | 399 | } | |
400 | 400 | else { | |
401 | 401 | let updPrices = makeString(removeByIndex(newPrices, (newPricesSize - 1)), ",") | |
402 | 402 | [IntegerEntry((assetIdStr + "_twap5B"), newTwap), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), updPrices), IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
403 | 403 | } | |
404 | 404 | } | |
405 | 405 | } | |
406 | 406 | } | |
407 | 407 | ||
408 | 408 | ||
409 | 409 | ||
410 | 410 | @Callable(i) | |
411 | 411 | func updateTokenPrice (assetIdStr) = { | |
412 | 412 | let tokenPrice = if ((assetIdStr == "WAVES")) | |
413 | 413 | then wavesPrice | |
414 | 414 | else if ((assetIdStr == xtnIdStr)) | |
415 | 415 | then xtnUsdPriceX6 | |
416 | 416 | else { | |
417 | 417 | let priceInPools = getTokenPrice(value(indexOf(tokenIds, assetIdStr)), wavesPrice)._1 | |
418 | 418 | if ((assetIdStr == ethStr)) | |
419 | 419 | then { | |
420 | 420 | let ethPrice = getIntegerValue(doraAddress, "%s%s__price__ETH-USDT") | |
421 | 421 | let comp = fraction(priceInPools, 10000, ethPrice) | |
422 | 422 | if (if ((comp > (10000 - oracleTolerance))) | |
423 | 423 | then ((10000 + oracleTolerance) > comp) | |
424 | 424 | else false) | |
425 | 425 | then priceInPools | |
426 | 426 | else throw("internal ETH price doesn't match with global price") | |
427 | 427 | } | |
428 | 428 | else if ((indexOf(stables, assetIdStr) != unit)) | |
429 | 429 | then 1000000 | |
430 | 430 | else priceInPools | |
431 | 431 | } | |
432 | 432 | let lastTwap = tryGetInteger((assetIdStr + "_twap5B")) | |
433 | 433 | let lastHeight = tryGetInteger((assetIdStr + "_lastBlock")) | |
434 | 434 | if ((i.originCaller != backendCaller)) | |
435 | 435 | then throw("available for backend only") | |
436 | 436 | else if ((2 > (height - lastHeight))) | |
437 | 437 | then nil | |
438 | 438 | else if (if ((lastHeight == 0)) | |
439 | 439 | then true | |
440 | 440 | else ((height - lastHeight) > 30)) | |
441 | 441 | then [IntegerEntry((assetIdStr + "_twap5B"), tokenPrice), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), toString(tokenPrice))] | |
442 | 442 | else { | |
443 | 443 | let lastPrices = split(getStringValue((assetIdStr + "_lastPrices")), ",") | |
444 | 444 | let newPrices = toString(tokenPrice) :: lastPrices | |
445 | 445 | func sumHandler (accum,next) = (accum + parseIntValue(next)) | |
446 | 446 | ||
447 | 447 | let newTwap = ({ | |
448 | 448 | let $l = newPrices | |
449 | 449 | let $s = size($l) | |
450 | 450 | let $acc0 = 0 | |
451 | 451 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
452 | 452 | then $a | |
453 | 453 | else sumHandler($a, $l[$i]) | |
454 | 454 | ||
455 | 455 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
456 | 456 | then $a | |
457 | 457 | else throw("List size exceeds 6") | |
458 | 458 | ||
459 | 459 | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6) | |
460 | 460 | } / size(newPrices)) | |
461 | 461 | let orderbookTwapResult = getOrderbookTwap(assetIdStr) | |
462 | 462 | if ((orderbookTwapResult == orderbookTwapResult)) | |
463 | 463 | then { | |
464 | 464 | let $t01961219670 = orderbookTwapResult | |
465 | 465 | let orderbookTwap = $t01961219670._1 | |
466 | 466 | let orderbookLocked = $t01961219670._2 | |
467 | 467 | let twapsComparison = if ((orderbookTwap != 0)) | |
468 | 468 | then fraction(newTwap, 10000, orderbookTwap) | |
469 | 469 | else throw(("orderbook result zero for " + assetIdStr)) | |
470 | 470 | let twapsSimilar = if ((twapsComparison > (10000 - oracleTolerance))) | |
471 | 471 | then ((10000 + oracleTolerance) > twapsComparison) | |
472 | 472 | else false | |
473 | 473 | let priceDiff = fraction(tokenPrice, 10000, parseIntValue(lastPrices[(size(lastPrices) - 1)])) | |
474 | 474 | let priceDiffOk = if (((10000 + shutdownTrigger) > priceDiff)) | |
475 | 475 | then (priceDiff > (10000 - shutdownTrigger)) | |
476 | 476 | else false | |
477 | 477 | let riskLevel = (((if (twapsSimilar) | |
478 | 478 | then 1 | |
479 | 479 | else 2) + (if (orderbookLocked) | |
480 | 480 | then 1 | |
481 | 481 | else 0)) + (if (!(priceDiffOk)) | |
482 | 482 | then 1 | |
483 | 483 | else 0)) | |
484 | 484 | let newPricesSize = size(newPrices) | |
485 | 485 | if (!(twapsSimilar)) | |
486 | 486 | then [IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
487 | 487 | else if ((5 >= newPricesSize)) | |
488 | 488 | then { | |
489 | 489 | let updPrices = makeString(newPrices, ",") | |
490 | 490 | [IntegerEntry((assetIdStr + "_twap5B"), newTwap), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), updPrices), IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
491 | 491 | } | |
492 | 492 | else { | |
493 | 493 | let updPrices = makeString(removeByIndex(newPrices, (newPricesSize - 1)), ",") | |
494 | 494 | [IntegerEntry((assetIdStr + "_twap5B"), newTwap), IntegerEntry((assetIdStr + "_lastBlock"), height), StringEntry((assetIdStr + "_lastPrices"), updPrices), IntegerEntry((assetIdStr + "_riskLevel"), riskLevel)] | |
495 | 495 | } | |
496 | 496 | } | |
497 | 497 | else throw("Strict value is not equal to itself.") | |
498 | 498 | } | |
499 | 499 | } | |
500 | 500 | ||
501 | 501 | ||
502 | 502 | ||
503 | 503 | @Callable(i) | |
504 | 504 | func updateTokens () = if ((i.originCaller != backendCaller)) | |
505 | 505 | then throw("available for backend only") | |
506 | 506 | else { | |
507 | 507 | let i1 = invoke(this, "updateTokenPrice", ["WAVES"], nil) | |
508 | 508 | if ((i1 == i1)) | |
509 | 509 | then { | |
510 | 510 | let i2 = invoke(this, "updateTokenPrice", [xtnIdStr], nil) | |
511 | 511 | if ((i2 == i2)) | |
512 | 512 | then { | |
513 | 513 | func callu (accum,next) = { | |
514 | 514 | let i3 = invoke(this, "updateTokenPrice", [next], nil) | |
515 | 515 | if ((i3 == i3)) | |
516 | 516 | then nil | |
517 | 517 | else throw("Strict value is not equal to itself.") | |
518 | 518 | } | |
519 | 519 | ||
520 | 520 | let res = { | |
521 | 521 | let $l = tokenIds | |
522 | 522 | let $s = size($l) | |
523 | 523 | let $acc0 = nil | |
524 | 524 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
525 | 525 | then $a | |
526 | 526 | else callu($a, $l[$i]) | |
527 | 527 | ||
528 | 528 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
529 | 529 | then $a | |
530 | 530 | else throw("List size exceeds 20") | |
531 | 531 | ||
532 | 532 | $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($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20) | |
533 | 533 | } | |
534 | 534 | if ((res == res)) | |
535 | 535 | then res | |
536 | 536 | else throw("Strict value is not equal to itself.") | |
537 | 537 | } | |
538 | 538 | else throw("Strict value is not equal to itself.") | |
539 | 539 | } | |
540 | 540 | else throw("Strict value is not equal to itself.") | |
541 | 541 | } | |
542 | 542 | ||
543 | 543 | ||
544 | 544 | ||
545 | 545 | @Callable(i) | |
546 | 546 | func updateLPTokens () = if ((i.originCaller != backendCaller)) | |
547 | 547 | then throw("available for backend only") | |
548 | 548 | else { | |
549 | 549 | func callu (accum,next) = { | |
550 | 550 | let i1 = invoke(this, "updateLpTokenPrice", [next], nil) | |
551 | 551 | if ((i1 == i1)) | |
552 | 552 | then nil | |
553 | 553 | else throw("Strict value is not equal to itself.") | |
554 | 554 | } | |
555 | 555 | ||
556 | 556 | let res = { | |
557 | 557 | let $l = lpTokenIds | |
558 | 558 | let $s = size($l) | |
559 | 559 | let $acc0 = nil | |
560 | 560 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
561 | 561 | then $a | |
562 | 562 | else callu($a, $l[$i]) | |
563 | 563 | ||
564 | 564 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
565 | 565 | then $a | |
566 | 566 | else throw("List size exceeds 20") | |
567 | 567 | ||
568 | 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($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20) | |
569 | 569 | } | |
570 | 570 | if ((res == res)) | |
571 | 571 | then { | |
572 | 572 | let i2 = invoke(this, "updateWxLpTokenPrice", [wxLpTokenIds[0]], nil) | |
573 | 573 | if ((i2 == i2)) | |
574 | 574 | then res | |
575 | 575 | else throw("Strict value is not equal to itself.") | |
576 | 576 | } | |
577 | 577 | else throw("Strict value is not equal to itself.") | |
578 | 578 | } | |
579 | 579 | ||
580 | 580 | ||
581 | 581 | ||
582 | 582 | @Callable(i) | |
583 | 583 | func initNewMaster (master) = { | |
584 | 584 | let addressOK = match addressFromString(master) { | |
585 | 585 | case a: Address => | |
586 | 586 | true | |
587 | 587 | case _ => | |
588 | 588 | false | |
589 | 589 | } | |
590 | 590 | if (!(addressOK)) | |
591 | 591 | then throw(("incorrect address " + master)) | |
592 | 592 | else if ((i.caller != this)) | |
593 | 593 | then throw("available for self call only") | |
594 | 594 | else [StringEntry("verifier_master", master)] | |
595 | 595 | } | |
596 | 596 | ||
597 | 597 | ||
598 | 598 | ||
599 | 599 | @Callable(i) | |
600 | 600 | func requestNewMaster (newMaster) = { | |
601 | 601 | let addressOK = match addressFromString(newMaster) { | |
602 | 602 | case a: Address => | |
603 | 603 | true | |
604 | 604 | case _ => | |
605 | 605 | false | |
606 | 606 | } | |
607 | 607 | if (!(addressOK)) | |
608 | 608 | then throw(("incorrect address " + newMaster)) | |
609 | 609 | else if ((indexOf(sentinels, i.caller) == unit)) | |
610 | 610 | then throw("whitelist only") | |
611 | 611 | else [StringEntry("verifier_newMaster", newMaster), IntegerEntry("verifier_newMasterEligibleAfter", (height + 10000))] | |
612 | 612 | } | |
613 | 613 | ||
614 | 614 | ||
615 | 615 | ||
616 | 616 | @Callable(i) | |
617 | 617 | func activateNewMaster () = { | |
618 | 618 | let eligibleHeight = getIntegerValue(this, "verifier_newMasterEligibleAfter") | |
619 | 619 | let newMaster = getStringValue(this, "verifier_newMaster") | |
620 | 620 | if ((indexOf(sentinels, i.caller) == unit)) | |
621 | 621 | then throw("whitelist only") | |
622 | 622 | else if ((eligibleHeight > height)) | |
623 | 623 | then throw("new master cannot be activated yet") | |
624 | 624 | else [DeleteEntry("verifier_newMaster"), DeleteEntry("verifier_newMasterEligibleAfter"), StringEntry("verifier_master", newMaster)] | |
625 | 625 | } | |
626 | 626 | ||
627 | 627 | ||
628 | 628 | @Verifier(tx) | |
629 | 629 | func verify () = { | |
630 | 630 | let masterAddress = getString("verifier_master") | |
631 | 631 | let approvedByMaster = match masterAddress { | |
632 | 632 | case x: String => | |
633 | 633 | match getInteger(addressFromStringValue(x), ("approvedTx_" + toBase58String(tx.id))) { | |
634 | 634 | case y: Int => | |
635 | 635 | y | |
636 | 636 | case _ => | |
637 | 637 | 0 | |
638 | 638 | } | |
639 | 639 | case _ => | |
640 | 640 | 1 | |
641 | 641 | } | |
642 | 642 | if (sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)) | |
643 | 643 | then (approvedByMaster > 0) | |
644 | 644 | else false | |
645 | 645 | } | |
646 | 646 |
github/deemru/w8io/3ef1775 58.36 ms ◑