tx · dCQGSCQgvfacdRZPqQe1MeGfJEUNWXJr13Yp4Vtfa3N 3NBTXksVZPU2vfJsdejxtSnXzL19bTjhLPu: -0.10000000 Waves 2022.09.19 19:53 [2236265] smart account 3NBTXksVZPU2vfJsdejxtSnXzL19bTjhLPu > SELF 0.00000000 Waves
{ "type": 13, "id": "dCQGSCQgvfacdRZPqQe1MeGfJEUNWXJr13Yp4Vtfa3N", "fee": 10000000, "feeAssetId": null, "timestamp": 1663606406757, "version": 1, "sender": "3NBTXksVZPU2vfJsdejxtSnXzL19bTjhLPu", "senderPublicKey": "7h6998AE3NwUR7WvQwSUjPFkuTLHUFJWDuoHHSBf891h", "proofs": [ "3gPdke1oFicN9cDiWhk91r2gfEd52yKY4i7qYB7XBvSN8NTS9zGxEw3cAmBtVmQkGc7SiQzeCfbswcf3i24KNawb" ], "script": "base64:BgJ8CAISBgoEAQEBARIICgYICAgBAQESAwoBARIFCgMBAQESBAoCAQESBQoDAQEBEgQKAgEBEgQKAgEBEgASABIGCgQBAQEEEgQKAgEBEgASBAoCAQESABIAEgMKAQgSABIAEgMKAQESABIAEgMKAQQSABIAEgMKAQgSAwoBCEAAB05fQ09JTlMAAwAPRkVFX0RFTk9NSU5BVE9SAIDIr6AlAAlQUkVDSVNJT04AwIQ9AAhERUNJTUFMUwAGAA1NQVhfQURNSU5fRkVFAIDIr6AlAAdNQVhfRkVFAIDkl9ASAAVNQVhfQQDAhD0ADE1BWF9BX0NIQU5HRQAKABNBRE1JTl9BQ1RJT05TX0RFTEFZCQBpAgkAaAIAAwCAowUAPAANTUlOX1JBTVBfVElNRQkAaQIAgKMFADwADVZQX0xPR19QRVJJT0QAgLiZKQAMVlBfUFJFQ0lTSU9OAICglKWNHQAHY29pbnNfMAkA2QQBCQERQGV4dHJOYXRpdmUoMTA1MykCBQR0aGlzAgdjb2luc18wAAdjb2luc18xCQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMCB2NvaW5zXzEAB2NvaW5zXzIJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhpcwIHY29pbnNfMgAKYmFsYW5jZXNfMAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzAgpiYWxhbmNlc18wAAAACmJhbGFuY2VzXzEJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwIKYmFsYW5jZXNfMQAAAApiYWxhbmNlc18yCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMCCmJhbGFuY2VzXzIAAAAIYmFsYW5jZXMJAJUKAwUKYmFsYW5jZXNfMAUKYmFsYW5jZXNfMQUKYmFsYW5jZXNfMgADZmVlCQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzAgNmZWUACWFkbWluX2ZlZQkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwIJYWRtaW5fZmVlAAxpc19hdXRvX2ZlZXMJARFAZXh0ck5hdGl2ZSgxMDUxKQIFBHRoaXMCDGlzX2F1dG9fZmVlcwAFb3duZXIJARFAZXh0ck5hdGl2ZSgxMDYyKQEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMCBW93bmVyAAV0b2tlbgkA2QQBCQERQGV4dHJOYXRpdmUoMTA1MykCBQR0aGlzAgV0b2tlbgAOdG9rZW5fcXVhbnRpdHkICQEFdmFsdWUBCQDsBwEFBXRva2VuCHF1YW50aXR5AAlpbml0aWFsX0EJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMCCWluaXRpYWxfQQAIZnV0dXJlX0EJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMCCGZ1dHVyZV9BAA5pbml0aWFsX0FfdGltZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzAg5pbml0aWFsX0FfdGltZQAAAA1mdXR1cmVfQV90aW1lCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMCDWZ1dHVyZV9BX3RpbWUAAAAWYWRtaW5fYWN0aW9uc19kZWFkbGluZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzAhZhZG1pbl9hY3Rpb25zX2RlYWRsaW5lAAAAG3RyYW5zZmVyX293bmVyc2hpcF9kZWFkbGluZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzAht0cmFuc2Zlcl9vd25lcnNoaXBfZGVhZGxpbmUAAAAKZnV0dXJlX2ZlZQkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwIKZnV0dXJlX2ZlZQAQZnV0dXJlX2FkbWluX2ZlZQkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwIQZnV0dXJlX2FkbWluX2ZlZQAMZnV0dXJlX293bmVyCQERQGV4dHJOYXRpdmUoMTA1MykCBQR0aGlzAgxmdXR1cmVfb3duZXIACWlzX2tpbGxlZAkBEUBleHRyTmF0aXZlKDEwNTEpAgUEdGhpcwIJaXNfa2lsbGVkAA1raWxsX2RlYWRsaW5lCQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzAg1raWxsX2RlYWRsaW5lABBLSUxMX0RFQURMSU5FX0RUCQBpAgkAaAIJAGgCAAIAHgCAowUAPAAEYmlnMAkAtgIBAAAABGJpZzEJALYCAQABAARiaWcyCQC2AgEAAgAEYmlnMwkAtgIBAAMABGJpZzQJALYCAQAEAAViaWcyNwkAtgIBABsADWhlaWdodEFkZHJlc3MJARN2YWx1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CQBrAwUNX3Rva2VuX2Ftb3VudAUCRDAFDHRvdGFsX3N1cHBseQQFbmV3X3kJAQdnZXRfeV9EBAUDYW1wBQFpBQJ4cAUCRDEEBGR5XzAJAGUCCQEDeF9pAgUCeHAFAWkFBW5ld195CgEPeHBfcmVkdWNlZF9wcm9jAgJqaQR4cF9qBAtkeF9leHBlY3RlZAMFAmppCQBlAgkAawMFBHhwX2oFAkQxBQJEMAUFbmV3X3kJAGUCBQR4cF9qCQBrAwUEeHBfagUCRDEFAkQwCQBlAgUEeHBfagkAawMFBF9mZWUFC2R4X2V4cGVjdGVkBQ9GRUVfREVOT01JTkFUT1IECnhwX3JlZHVjZWQJAJUKAwkBD3hwX3JlZHVjZWRfcHJvYwIJAAACBQFpAAAIBQJ4cAJfMQkBD3hwX3JlZHVjZWRfcHJvYwIJAAACBQFpAAEIBQJ4cAJfMgkBD3hwX3JlZHVjZWRfcHJvYwIJAAACBQFpAAIIBQJ4cAJfMwQMeHBfcmVkdWNlZF9pCQEDeF9pAgUKeHBfcmVkdWNlZAUBaQQCZHkJAGUCCQBlAgUMeHBfcmVkdWNlZF9pCQEHZ2V0X3lfRAQFA2FtcAUBaQUKeHBfcmVkdWNlZAUCRDEAAQkAlQoDBQJkeQkAZQIFBGR5XzAFAmR5BQJEMAENdmlydHVhbF9wcmljZQEBRAkAawMFAUQFDFZQX1BSRUNJU0lPTgUOdG9rZW5fcXVhbnRpdHkBCGxvZ19kYXRhAgFEA2FkZAQJdG90YWxfdm9sCQC3AgIJAKcDAQkBC3ZhbHVlT3JFbHNlAgkAoggBAgN2b2wCATAJALYCAQUDYWRkBBB0b3RhbF92b2xfc3RyaW5nCQCmAwEFCXRvdGFsX3ZvbAkAzggCCQDMCAIJAQtTdHJpbmdFbnRyeQICA3ZvbAUQdG90YWxfdm9sX3N0cmluZwUDbmlsBApsb2dfcGVyaW9kCQCkAwEJAGkCCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAUNVlBfTE9HX1BFUklPRAQHbG9nX2tleQkArAICAgRsb2dfBQpsb2dfcGVyaW9kAwkBCWlzRGVmaW5lZAEJAKIIAQUHbG9nX2tleQUDbmlsCQDMCAIJAQtTdHJpbmdFbnRyeQIFB2xvZ19rZXkJAKwCAgkArAICCQCsAgIJAKwCAgkApAMBCQENdmlydHVhbF9wcmljZQEFAUQCAV8FEHRvdGFsX3ZvbF9zdHJpbmcCAV8JAKQDAQgFCWxhc3RCbG9jawl0aW1lc3RhbXAFA25pbAEPZ2V0X25lYXJlc3RfbG9nAQZwZXJpb2QKAQRmb2xkAglsb2dfdmFsdWUEc3RlcAMJAQIhPQIFCWxvZ192YWx1ZQIABQlsb2dfdmFsdWUEB2xvZ19rZXkJAKwCAgIEbG9nXwkApAMBCQBlAgUGcGVyaW9kBQRzdGVwCQELdmFsdWVPckVsc2UCCQCiCAEFB2xvZ19rZXkCAAQHbGlzdDMwcAkAzAgCAAAJAMwIAgABCQDMCAIAAgkAzAgCAAMJAMwIAgAECQDMCAIABQkAzAgCAAYJAMwIAgAHCQDMCAIACAkAzAgCAAkJAMwIAgAKCQDMCAIACwkAzAgCAAwJAMwIAgANCQDMCAIADgkAzAgCAA8JAMwIAgAQCQDMCAIAEQkAzAgCABIJAMwIAgATCQDMCAIAFAkAzAgCABUJAMwIAgAWCQDMCAIAFwkAzAgCABgJAMwIAgAZCQDMCAIAGgkAzAgCABsJAMwIAgAcCQDMCAIAHQUDbmlsBAdsaXN0MzBtCQDMCAIA////////////AQkAzAgCAP7//////////wEJAMwIAgD9//////////8BCQDMCAIA/P//////////AQkAzAgCAPv//////////wEJAMwIAgD6//////////8BCQDMCAIA+f//////////AQkAzAgCAPj//////////wEJAMwIAgD3//////////8BCQDMCAIA9v//////////AQkAzAgCAPX//////////wEJAMwIAgD0//////////8BCQDMCAIA8///////////AQkAzAgCAPL//////////wEJAMwIAgDx//////////8BCQDMCAIA8P//////////AQkAzAgCAO///////////wEJAMwIAgDu//////////8BCQDMCAIA7f//////////AQkAzAgCAOz//////////wEJAMwIAgDr//////////8BCQDMCAIA6v//////////AQkAzAgCAOn//////////wEJAMwIAgDo//////////8BCQDMCAIA5///////////AQkAzAgCAOb//////////wEJAMwIAgDl//////////8BCQDMCAIA5P//////////AQkAzAgCAOP//////////wEJAMwIAgDi//////////8BBQNuaWwECHZhbHVlMzBwCgACJGwFB2xpc3QzMHAKAAIkcwkAkAMBBQIkbAoABSRhY2MwAgAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBBGZvbGQCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDMwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAwkBAiE9AgUIdmFsdWUzMHACAAUIdmFsdWUzMHAKAAIkbAUHbGlzdDMwbQoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMV8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEEZm9sZAIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMV8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMzAJAQUkZjFfMgIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgALAAwADQAOAA8AEAARABIAEwAUABUAFgAXABgAGQAaABsAHAAdAB4BFmdldF92aXJ0dWFsX3ByaWNlX2RpZmYBA190MAQDdnAxCQENdmlydHVhbF9wcmljZQEJAQVnZXRfRAIJAQNfeHAACQECX0EABA12cDFfdGltZXN0YW1wCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAQCdDADCQBnAgAABQNfdDAJAGQCBQ12cDFfdGltZXN0YW1wBQNfdDAFA190MAQJdDBfcGVyaW9kCQBpAgUCdDAFDVZQX0xPR19QRVJJT0QECWxvZ192YWx1ZQkBD2dldF9uZWFyZXN0X2xvZwEFCXQwX3BlcmlvZAMJAAACBQlsb2dfdmFsdWUCAAkAlQoDBQN2cDEFA3ZwMQAABAhsb2dfbGlzdAkAtQkCBQlsb2dfdmFsdWUCAV8EA3ZwMAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFCGxvZ19saXN0AAAEDXZwMF90aW1lc3RhbXAJAQ1wYXJzZUludFZhbHVlAQkAkQMCBQhsb2dfbGlzdAACCQCVCgMFA3ZwMQUDdnAwCQBlAgUNdnAxX3RpbWVzdGFtcAUNdnAwX3RpbWVzdGFtcAEPZ2V0X3ZvbHVtZV9kaWZmAQNfdDAEBHZvbDEJAKcDAQkBC3ZhbHVlT3JFbHNlAgkAoggBAgN2b2wCATAEDnZvbDFfdGltZXN0YW1wCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAQCdDADCQBnAgAABQNfdDAJAGQCBQ52b2wxX3RpbWVzdGFtcAUDX3QwBQNfdDAECXQwX3BlcmlvZAkAaQIFAnQwBQ1WUF9MT0dfUEVSSU9EBAlsb2dfdmFsdWUJAQ9nZXRfbmVhcmVzdF9sb2cBBQl0MF9wZXJpb2QDCQAAAgUJbG9nX3ZhbHVlAgAJAJUKAwUEdm9sMQUEdm9sMQAABAhsb2dfbGlzdAkAtQkCBQlsb2dfdmFsdWUCAV8EBHZvbDAJAKcDAQkAkQMCBQhsb2dfbGlzdAABBA52b2wwX3RpbWVzdGFtcAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFCGxvZ19saXN0AAIJAJUKAwUEdm9sMQUEdm9sMAkAZQIFDnZvbDFfdGltZXN0YW1wBQ52b2wwX3RpbWVzdGFtcBsDbXNnAQFEBAN4cDADeHAxA3hwMgNhbXAEAUQJAQ5nZXRfRF9pbnRlcm5hbAQJALYCAQUDeHAwCQC2AgEFA3hwMQkAtgIBBQN4cDIFA2FtcAkAlAoCCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgFEBQFEBQNuaWwFAUQDbXNnAQRpbml0BgZfb3duZXIGX2NvaW5zC19wb29sX3Rva2VuAl9BBF9mZWUKX2FkbWluX2ZlZQMJAQEhAQkAnggBBQR0aGlzCQACAQITQWxyZWFkeSBpbml0aWFsaXplZAMJAQIhPQIIBQNtc2cGY2FsbGVyBQR0aGlzCQACAQIYU2VsZiBpbml0aWFsaXphdGlvbiBvbmx5CgEJY2hlY2tDb2luAQZjb2luNTgEBGNvaW4JARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkA2QQBBQZjb2luNTgJAKwCAgISZnJvbUJhc2U1OFN0cmluZzogBQZjb2luNTgECGRlY2ltYWxzCAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQDsBwEFBGNvaW4JAKwCAgILYXNzZXRJbmZvOiAFBmNvaW41OAhkZWNpbWFscwMJAQIhPQIFCGRlY2ltYWxzBQhERUNJTUFMUwkAAgECDndyb25nIGRlY2ltYWxzBQRjb2luBAVjb2lucwkAtQkCBQZfY29pbnMCASwDCQECIT0CCQCQAwEFBWNvaW5zBQdOX0NPSU5TCQACAQkArAICAhFzaXplKCBjb2lucyApICE9IAkApAMBBQdOX0NPSU5TBAppc3N1ZVRva2VuCQDCCAUFC19wb29sX3Rva2VuCQCsAgICDUxQIHRva2VuIGZvciAFBl9jb2lucwAABQhERUNJTUFMUwYEB3Rva2VuSWQJANgEAQkAuAgBBQppc3N1ZVRva2VuCQDMCAIJAQtTdHJpbmdFbnRyeQICB2NvaW5zXzAJANgEAQkBCWNoZWNrQ29pbgEJAJEDAgUFY29pbnMAAAkAzAgCCQELU3RyaW5nRW50cnkCAgdjb2luc18xCQDYBAEJAQljaGVja0NvaW4BCQCRAwIFBWNvaW5zAAEJAMwIAgkBC1N0cmluZ0VudHJ5AgIHY29pbnNfMgkA2AQBCQEJY2hlY2tDb2luAQkAkQMCBQVjb2lucwACCQDMCAIJAQxJbnRlZ2VyRW50cnkCAglpbml0aWFsX0EFAl9BCQDMCAIJAQxJbnRlZ2VyRW50cnkCAghmdXR1cmVfQQUCX0EJAMwIAgkBDEludGVnZXJFbnRyeQICA2ZlZQUEX2ZlZQkAzAgCCQEMSW50ZWdlckVudHJ5AgIJYWRtaW5fZmVlBQpfYWRtaW5fZmVlCQDMCAIJAQtTdHJpbmdFbnRyeQICBW93bmVyCQEMY2hlY2tBZGRyZXNzAQUGX293bmVyCQDMCAIJAQxJbnRlZ2VyRW50cnkCAg1raWxsX2RlYWRsaW5lCQBkAgUGaGVpZ2h0BRBLSUxMX0RFQURMSU5FX0RUCQDMCAIJAQtTdHJpbmdFbnRyeQICBXRva2VuBQd0b2tlbklkCQDMCAIJAQxCb29sZWFuRW50cnkCAglpc19raWxsZWQHCQDMCAIJAQxCb29sZWFuRW50cnkCAgxpc19hdXRvX2ZlZXMGCQDMCAIFCmlzc3VlVG9rZW4FA25pbANtc2cBDWFkZF9saXF1aWRpdHkBD21pbl9taW50X2Ftb3VudAMJAQZhc3NlcnQBCQEBIQEFCWlzX2tpbGxlZAkAAgECCWlzIGtpbGxlZAQEX2ZlZQkAaQIJAGgCBQNmZWUFB05fQ09JTlMJAGgCAAQJAGUCBQdOX0NPSU5TAAEEA2FtcAkBAl9BAAQMdG9rZW5fc3VwcGx5BQ50b2tlbl9xdWFudGl0eQQCRDADCQAAAgUMdG9rZW5fc3VwcGx5AAAAAAkBCWdldF9EX21lbQQFCmJhbGFuY2VzXzAFCmJhbGFuY2VzXzEFCmJhbGFuY2VzXzIFA2FtcAQIcGF5bWVudHMIBQNtc2cIcGF5bWVudHMEAW4JAJADAQUIcGF5bWVudHMKAQphc3NldFZhbGlkAQRjb2luAwMJAAACBQRjb2luBQdjb2luc18wBgkAAAIFBGNvaW4FB2NvaW5zXzEGCQAAAgUEY29pbgUHY29pbnNfMgoBDXZhbGlkUGF5bWVudHMCCHBheW1lbnRzAW4DCQBmAgUBbgADCQACAQIRcGF5bWVudHMgc2l6ZSA+IDMDCQBmAgABBQFuCQACAQIRcGF5bWVudHMgc2l6ZSA8IDEDCQEBIQEJAQphc3NldFZhbGlkAQgJAJEDAgUIcGF5bWVudHMAAAdhc3NldElkCQACAQIXdW5rbm93biBwYXltZW50IDEgdG9rZW4DAwkAZgIFAW4AAQkBASEBCQEKYXNzZXRWYWxpZAEICQCRAwIFCHBheW1lbnRzAAEHYXNzZXRJZAcJAAIBAhd1bmtub3duIHBheW1lbnQgMiB0b2tlbgMDCQBmAgUBbgACCQEBIQEJAQphc3NldFZhbGlkAQgJAJEDAgUIcGF5bWVudHMAAgdhc3NldElkBwkAAgECF3Vua25vd24gcGF5bWVudCAzIHRva2VuBgoBDXBheW1lbnRBbW91bnQDBGNvaW4IcGF5bWVudHMBbgkAZAIJAGQCAwkAAAIICQCRAwIFCHBheW1lbnRzAAAHYXNzZXRJZAUEY29pbggJAJEDAgUIcGF5bWVudHMAAAZhbW91bnQAAAMDCQBmAgUBbgABCQAAAggJAJEDAgUIcGF5bWVudHMAAQdhc3NldElkBQRjb2luBwgJAJEDAgUIcGF5bWVudHMAAQZhbW91bnQAAAMDCQBmAgUBbgACCQAAAggJAJEDAgUIcGF5bWVudHMAAgdhc3NldElkBQRjb2luBwgJAJEDAgUIcGF5bWVudHMAAgZhbW91bnQAAAMJAQEhAQkBDXZhbGlkUGF5bWVudHMCBQhwYXltZW50cwUBbgkBBXRocm93AAQJYW1vdW50c18wCQENcGF5bWVudEFtb3VudAMFB2NvaW5zXzAFCHBheW1lbnRzBQFuBAlhbW91bnRzXzEJAQ1wYXltZW50QW1vdW50AwUHY29pbnNfMQUIcGF5bWVudHMFAW4ECWFtb3VudHNfMgkBDXBheW1lbnRBbW91bnQDBQdjb2luc18yBQhwYXltZW50cwUBbgMDCQAAAgUMdG9rZW5fc3VwcGx5AAADAwkAAAIFCWFtb3VudHNfMAAABgkAAAIFCWFtb3VudHNfMQAABgkAAAIFCWFtb3VudHNfMgAABwkAAgECImluaXRpYWwgZGVwb3NpdCByZXF1aXJlcyBhbGwgY29pbnMEDm5ld19iYWxhbmNlc18wCQBkAgUKYmFsYW5jZXNfMAUJYW1vdW50c18wBA5uZXdfYmFsYW5jZXNfMQkAZAIFCmJhbGFuY2VzXzEFCWFtb3VudHNfMQQObmV3X2JhbGFuY2VzXzIJAGQCBQpiYWxhbmNlc18yBQlhbW91bnRzXzIEAkQxCQEJZ2V0X0RfbWVtBAUObmV3X2JhbGFuY2VzXzAFDm5ld19iYWxhbmNlc18xBQ5uZXdfYmFsYW5jZXNfMgUDYW1wAwkBBmFzc2VydAEJAGYCBQJEMQUCRDAJAAIBAgdEMSA+IEQwBA0kdDAxNTM5MTE3NDMwAwkAZgIFDHRva2VuX3N1cHBseQAACgEIZmVlc1Byb2MCC29sZF9iYWxhbmNlC25ld19iYWxhbmNlBA1pZGVhbF9iYWxhbmNlCQBrAwUCRDEFC29sZF9iYWxhbmNlBQJEMAQKZGlmZmVyZW5jZQMJAGYCBQ1pZGVhbF9iYWxhbmNlBQtuZXdfYmFsYW5jZQkAZQIFDWlkZWFsX2JhbGFuY2UFC25ld19iYWxhbmNlCQBlAgULbmV3X2JhbGFuY2UFDWlkZWFsX2JhbGFuY2UJAGsDBQRfZmVlBQpkaWZmZXJlbmNlBQ9GRUVfREVOT01JTkFUT1IEBmZlZXNfMAkBCGZlZXNQcm9jAgUKYmFsYW5jZXNfMAUObmV3X2JhbGFuY2VzXzAEBmZlZXNfMQkBCGZlZXNQcm9jAgUKYmFsYW5jZXNfMQUObmV3X2JhbGFuY2VzXzEEBmZlZXNfMgkBCGZlZXNQcm9jAgUKYmFsYW5jZXNfMgUObmV3X2JhbGFuY2VzXzIEDGFkbWluX2ZlZXNfMAkAawMFBmZlZXNfMAUJYWRtaW5fZmVlBQ9GRUVfREVOT01JTkFUT1IEDGFkbWluX2ZlZXNfMQkAawMFBmZlZXNfMQUJYWRtaW5fZmVlBQ9GRUVfREVOT01JTkFUT1IEDGFkbWluX2ZlZXNfMgkAawMFBmZlZXNfMgUJYWRtaW5fZmVlBQ9GRUVfREVOT01JTkFUT1IJAJcKBQkAZQIFDm5ld19iYWxhbmNlc18wBQxhZG1pbl9mZWVzXzAJAGUCBQ5uZXdfYmFsYW5jZXNfMQUMYWRtaW5fZmVlc18xCQBlAgUObmV3X2JhbGFuY2VzXzIFDGFkbWluX2ZlZXNfMgkBCWdldF9EX21lbQQJAGUCBQ5uZXdfYmFsYW5jZXNfMAUGZmVlc18wCQBlAgUObmV3X2JhbGFuY2VzXzEFBmZlZXNfMQkAZQIFDm5ld19iYWxhbmNlc18yBQZmZWVzXzIFA2FtcAkAzggCCQDOCAIJAM4IAgkAzggCBQNuaWwDAwUMaXNfYXV0b19mZWVzCQBmAgUMYWRtaW5fZmVlc18wAAAHCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMFBW93bmVyBQxhZG1pbl9mZWVzXzAFB2NvaW5zXzAFA25pbAUDbmlsAwMFDGlzX2F1dG9fZmVlcwkAZgIFDGFkbWluX2ZlZXNfMQAABwkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQVvd25lcgUMYWRtaW5fZmVlc18xBQdjb2luc18xBQNuaWwFA25pbAMDBQxpc19hdXRvX2ZlZXMJAGYCBQxhZG1pbl9mZWVzXzIAAAcJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwUFb3duZXIFDGFkbWluX2ZlZXNfMgUHY29pbnNfMgUDbmlsBQNuaWwJAQhsb2dfZGF0YQIFAkQwCQBrAwkAZAIJAGQCBQZmZWVzXzAFBmZlZXNfMQUGZmVlc18yBQ9GRUVfREVOT01JTkFUT1IFA2ZlZQkAlwoFBQ5uZXdfYmFsYW5jZXNfMAUObmV3X2JhbGFuY2VzXzEFDm5ld19iYWxhbmNlc18yBQJEMQUDbmlsBBBmaW5hbF9iYWxhbmNlc18wCAUNJHQwMTUzOTExNzQzMAJfMQQQZmluYWxfYmFsYW5jZXNfMQgFDSR0MDE1MzkxMTc0MzACXzIEEGZpbmFsX2JhbGFuY2VzXzIIBQ0kdDAxNTM5MTE3NDMwAl8zBAJEMggFDSR0MDE1MzkxMTc0MzACXzQEDGZlZXNfYWN0aW9ucwgFDSR0MDE1MzkxMTc0MzACXzUEC21pbnRfYW1vdW50AwkAAAIFDHRva2VuX3N1cHBseQAABQJEMQkAawMFDHRva2VuX3N1cHBseQkAZQIFAkQyBQJEMAUCRDADCQEGYXNzZXJ0AQkAZwIFC21pbnRfYW1vdW50BQ9taW5fbWludF9hbW91bnQJAAIBAhRTbGlwcGFnZSBzY3Jld2VkIHlvdQkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgpiYWxhbmNlc18wBRBmaW5hbF9iYWxhbmNlc18wCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgpiYWxhbmNlc18xBRBmaW5hbF9iYWxhbmNlc18xCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgpiYWxhbmNlc18yBRBmaW5hbF9iYWxhbmNlc18yCQDMCAIJAQdSZWlzc3VlAwUFdG9rZW4FC21pbnRfYW1vdW50BgkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUDbXNnBmNhbGxlcgULbWludF9hbW91bnQFBXRva2VuBQNuaWwFDGZlZXNfYWN0aW9ucwNtc2cBBmdldF9keQMBaQFqAmR4BAJ4cAkBA194cAAEBHhwX2kJAQN4X2kCBQJ4cAUBaQQEeHBfagkBA3hfaQIFAnhwBQFqBAF4CQBkAgUEeHBfaQUCZHgEDSR0MDE4MTMwMTgxNjUJAQVnZXRfeQQFAWkFAWoFAXgFAnhwBAF5CAUNJHQwMTgxMzAxODE2NQJfMQQBRAgFDSR0MDE4MTMwMTgxNjUCXzIEAmR5CQBlAgkAZQIFBHhwX2oFAXkAAQQEX2ZlZQkAawMFA2ZlZQUCZHkFD0ZFRV9ERU5PTUlOQVRPUgkAlAoCBQNuaWwJAGUCBQJkeQUEX2ZlZQNtc2cBCGV4Y2hhbmdlAgFqBm1pbl9keQMJAQZhc3NlcnQBCQEBIQEFCWlzX2tpbGxlZAkAAgECCWlzIGtpbGxlZAMJAQIhPQIJAJADAQgFA21zZwhwYXltZW50cwABCQACAQIVc2l6ZSggcGF5bWVudHMgKSAhPSAxBAdwYXltZW50CQCRAwIIBQNtc2cIcGF5bWVudHMAAAQHdG9rZW5JbggFB3BheW1lbnQHYXNzZXRJZAQCZHgIBQdwYXltZW50BmFtb3VudAQBaQMJAAACBQd0b2tlbkluBQdjb2luc18wAAADCQAAAgUHdG9rZW5JbgUHY29pbnNfMQABAwkAAAIFB3Rva2VuSW4FB2NvaW5zXzIAAgkAAgECEHVua25vd24gdG9rZW4gaW4ECHRva2VuT3V0AwkAAAIFAWoAAAUHY29pbnNfMAMJAAACBQFqAAEFB2NvaW5zXzEDCQAAAgUBagACBQdjb2luc18yCQACAQIRdW5rbm93biB0b2tlbiBvdXQEAnhwCQEHX3hwX21lbQEFCGJhbGFuY2VzBAR4cF9pCQEDeF9pAgUCeHAFAWkEBHhwX2oJAQN4X2kCBQJ4cAUBagQBeAkAZAIFBHhwX2kFAmR4BA0kdDAxOTA0MDE5MDc1CQEFZ2V0X3kEBQFpBQFqBQF4BQJ4cAQBeQgFDSR0MDE5MDQwMTkwNzUCXzEEAUQIBQ0kdDAxOTA0MDE5MDc1Al8yBANfZHkJAGUCCQBlAgUEeHBfagUBeQABBAZkeV9mZWUJAGsDBQNfZHkFA2ZlZQUPRkVFX0RFTk9NSU5BVE9SBAJkeQkAZQIFA19keQUGZHlfZmVlAwkBBmFzc2VydAEJAGcCBQJkeQUGbWluX2R5CQACAQIuRXhjaGFuZ2UgcmVzdWx0ZWQgaW4gZmV3ZXIgY29pbnMgdGhhbiBleHBlY3RlZAQMZHlfYWRtaW5fZmVlCQBrAwUGZHlfZmVlBQlhZG1pbl9mZWUFD0ZFRV9ERU5PTUlOQVRPUgoBC2JhbGFuY2VQcm9jAgtvbGRfYmFsYW5jZQJfaQMJAAACBQJfaQUBaQkAZAIFC29sZF9iYWxhbmNlBQJkeAMJAAACBQJfaQUBagkAZQIJAGUCBQtvbGRfYmFsYW5jZQUCZHkFDGR5X2FkbWluX2ZlZQULb2xkX2JhbGFuY2UEEGZpbmFsX2JhbGFuY2VzXzAJAQtiYWxhbmNlUHJvYwIFCmJhbGFuY2VzXzAAAAQQZmluYWxfYmFsYW5jZXNfMQkBC2JhbGFuY2VQcm9jAgUKYmFsYW5jZXNfMQABBBBmaW5hbF9iYWxhbmNlc18yCQELYmFsYW5jZVByb2MCBQpiYWxhbmNlc18yAAIJAJQKAgkAzggCCQDOCAIJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzAFEGZpbmFsX2JhbGFuY2VzXzAJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzEFEGZpbmFsX2JhbGFuY2VzXzEJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzIFEGZpbmFsX2JhbGFuY2VzXzIJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFA21zZwZjYWxsZXIFAmR5BQh0b2tlbk91dAUDbmlsAwMFDGlzX2F1dG9fZmVlcwkAZgIFDGR5X2FkbWluX2ZlZQAABwkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQVvd25lcgUMZHlfYWRtaW5fZmVlBQh0b2tlbk91dAUDbmlsBQNuaWwJAQhsb2dfZGF0YQIFAUQFAmR4BQJkeQNtc2cBEHJlbW92ZV9saXF1aWRpdHkDDW1pbl9hbW91bnRzXzANbWluX2Ftb3VudHNfMQ1taW5fYW1vdW50c18yAwkBAiE9AgkAkAMBCAUDbXNnCHBheW1lbnRzAAEJAAIBAhVzaXplKCBwYXltZW50cyApICE9IDEEB3BheW1lbnQJAJEDAggFA21zZwhwYXltZW50cwAABAd0b2tlbkluCAUHcGF5bWVudAdhc3NldElkAwkBAiE9AgUHdG9rZW5JbgUFdG9rZW4JAAIBAg11bmtub3duIHRva2VuBAdfYW1vdW50CAUHcGF5bWVudAZhbW91bnQEDHRvdGFsX3N1cHBseQUOdG9rZW5fcXVhbnRpdHkEB3ZhbHVlXzAJAGsDBQpiYWxhbmNlc18wBQdfYW1vdW50BQx0b3RhbF9zdXBwbHkEB3ZhbHVlXzEJAGsDBQpiYWxhbmNlc18xBQdfYW1vdW50BQx0b3RhbF9zdXBwbHkEB3ZhbHVlXzIJAGsDBQpiYWxhbmNlc18yBQdfYW1vdW50BQx0b3RhbF9zdXBwbHkDCQEGYXNzZXJ0AQMDCQBnAgUHdmFsdWVfMAUNbWluX2Ftb3VudHNfMAkAZwIFB3ZhbHVlXzEFDW1pbl9hbW91bnRzXzEHCQBnAgUHdmFsdWVfMgUNbWluX2Ftb3VudHNfMgcJAAIBAjBXaXRoZHJhd2FsIHJlc3VsdGVkIGluIGZld2VyIGNvaW5zIHRoYW4gZXhwZWN0ZWQEEGZpbmFsX2JhbGFuY2VzXzAJAGUCBQpiYWxhbmNlc18wBQd2YWx1ZV8wBBBmaW5hbF9iYWxhbmNlc18xCQBlAgUKYmFsYW5jZXNfMQUHdmFsdWVfMQQQZmluYWxfYmFsYW5jZXNfMgkAZQIFCmJhbGFuY2VzXzIFB3ZhbHVlXzIJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzAFEGZpbmFsX2JhbGFuY2VzXzAJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzEFEGZpbmFsX2JhbGFuY2VzXzEJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzIFEGZpbmFsX2JhbGFuY2VzXzIJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFA21zZwZjYWxsZXIFB3ZhbHVlXzAFB2NvaW5zXzAJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFA21zZwZjYWxsZXIFB3ZhbHVlXzEFB2NvaW5zXzEJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFA21zZwZjYWxsZXIFB3ZhbHVlXzIFB2NvaW5zXzIJAMwIAgkBBEJ1cm4CBQV0b2tlbgUHX2Ftb3VudAUDbmlsA21zZwEWY2FsY193aXRoZHJhd19vbmVfY29pbgINX3Rva2VuX2Ftb3VudAFpCQCUCgIFA25pbAgJARdfY2FsY193aXRoZHJhd19vbmVfY29pbgIFDV90b2tlbl9hbW91bnQFAWkCXzEDbXNnARlyZW1vdmVfbGlxdWlkaXR5X29uZV9jb2luAgFpCm1pbl9hbW91bnQDCQEGYXNzZXJ0AQkBASEBBQlpc19raWxsZWQJAAIBAglpcyBraWxsZWQDCQECIT0CCQCQAwEIBQNtc2cIcGF5bWVudHMAAQkAAgECFXNpemUoIHBheW1lbnRzICkgIT0gMQQHcGF5bWVudAkAkQMCCAUDbXNnCHBheW1lbnRzAAAEB3Rva2VuSW4IBQdwYXltZW50B2Fzc2V0SWQDCQECIT0CBQd0b2tlbkluBQV0b2tlbgkAAgECDXVua25vd24gdG9rZW4EDV90b2tlbl9hbW91bnQIBQdwYXltZW50BmFtb3VudAQNJHQwMjIwMzMyMjEwMAkBF19jYWxjX3dpdGhkcmF3X29uZV9jb2luAgUNX3Rva2VuX2Ftb3VudAUBaQQCZHkIBQ0kdDAyMjAzMzIyMTAwAl8xBAZkeV9mZWUIBQ0kdDAyMjAzMzIyMTAwAl8yBAFECAUNJHQwMjIwMzMyMjEwMAJfMwMJAQZhc3NlcnQBCQBnAgUCZHkFCm1pbl9hbW91bnQJAAIBAhhOb3QgZW5vdWdoIGNvaW5zIHJlbW92ZWQEDGR5X2FkbWluX2ZlZQkAawMFBmR5X2ZlZQUJYWRtaW5fZmVlBQ9GRUVfREVOT01JTkFUT1IECmR5X2FuZF9mZWUJAGQCBQJkeQUMZHlfYWRtaW5fZmVlBA0kdDAyMjI5OTIyODA2AwkAAAIFAWkAAAkAlAoCCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgpiYWxhbmNlc18wCQBlAgUKYmFsYW5jZXNfMAUKZHlfYW5kX2ZlZQkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUDbXNnBmNhbGxlcgUCZHkFB2NvaW5zXzAFA25pbAUHY29pbnNfMAMJAAACBQFpAAEJAJQKAgkAzAgCCQEMSW50ZWdlckVudHJ5AgIKYmFsYW5jZXNfMQkAZQIFCmJhbGFuY2VzXzEFCmR5X2FuZF9mZWUJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwgFA21zZwZjYWxsZXIFAmR5BQdjb2luc18xBQNuaWwFB2NvaW5zXzEDCQAAAgUBaQACCQCUCgIJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzIJAGUCBQpiYWxhbmNlc18yBQpkeV9hbmRfZmVlCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMIBQNtc2cGY2FsbGVyBQJkeQUHY29pbnNfMgUDbmlsBQdjb2luc18yCQACAQIUaW5kZXggb3V0IG9mIE5fQ09JTlMEDGJhc2VfYWN0aW9ucwgFDSR0MDIyMjk5MjI4MDYCXzEECHRva2VuT3V0CAUNJHQwMjIyOTkyMjgwNgJfMgkAzggCCQDOCAIJAM4IAgUMYmFzZV9hY3Rpb25zCQDMCAIJAQRCdXJuAgUFdG9rZW4FDV90b2tlbl9hbW91bnQFA25pbAMDBQxpc19hdXRvX2ZlZXMJAGYCBQxkeV9hZG1pbl9mZWUAAAcJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwUFb3duZXIFDGR5X2FkbWluX2ZlZQUIdG9rZW5PdXQFA25pbAUDbmlsCQEIbG9nX2RhdGECBQFECQBrAwUGZHlfZmVlBQ9GRUVfREVOT01JTkFUT1IFA2ZlZQNtc2cBAUEACQCUCgIFA25pbAkBAl9BAANtc2cBEWdldF92aXJ0dWFsX3ByaWNlAAQBRAkBBWdldF9EAgkBA194cAAJAQJfQQAJAJQKAgUDbmlsCQENdmlydHVhbF9wcmljZQEFAUQDbXNnARFjYWxjX3Rva2VuX2Ftb3VudAQJYW1vdW50c18wCWFtb3VudHNfMQlhbW91bnRzXzIHZGVwb3NpdAQDYW1wCQECX0EABAJEMAkBCWdldF9EX21lbQQFCmJhbGFuY2VzXzAFCmJhbGFuY2VzXzEFCmJhbGFuY2VzXzIFA2FtcAQObmV3X2JhbGFuY2VzXzAJAGQCBQpiYWxhbmNlc18wAwUHZGVwb3NpdAUJYW1vdW50c18wCQEBLQEFCWFtb3VudHNfMAQObmV3X2JhbGFuY2VzXzEJAGQCBQpiYWxhbmNlc18xAwUHZGVwb3NpdAUJYW1vdW50c18xCQEBLQEFCWFtb3VudHNfMQQObmV3X2JhbGFuY2VzXzIJAGQCBQpiYWxhbmNlc18yAwUHZGVwb3NpdAUJYW1vdW50c18yCQEBLQEFCWFtb3VudHNfMgQCRDEJAQlnZXRfRF9tZW0EBQ5uZXdfYmFsYW5jZXNfMAUObmV3X2JhbGFuY2VzXzEFDm5ld19iYWxhbmNlc18yBQNhbXAEDHRva2VuX2Ftb3VudAUOdG9rZW5fcXVhbnRpdHkEBGRpZmYDBQdkZXBvc2l0CQBlAgUCRDEFAkQwCQBlAgUCRDAFAkQxCQCUCgIFA25pbAkAawMFBGRpZmYFDHRva2VuX2Ftb3VudAUCRDADbXNnAQZyYW1wX0ECCV9mdXR1cmVfQQxfZnV0dXJlX3RpbWUDCQEGYXNzZXJ0AQkAAAIIBQNtc2cGY2FsbGVyBQVvd25lcgkAAgECCm9ubHkgb3duZXIDCQEGYXNzZXJ0AQkAZwIFD2Jsb2NrX3RpbWVzdGFtcAkAZAIFDmluaXRpYWxfQV90aW1lBQ1NSU5fUkFNUF9USU1FCQACAQIJdG9vIG9mdGVuAwkBBmFzc2VydAEJAGcCBQxfZnV0dXJlX3RpbWUJAGQCBQ9ibG9ja190aW1lc3RhbXAFDU1JTl9SQU1QX1RJTUUJAAIBAhFpbnN1ZmZpY2llbnQgdGltZQQKX2luaXRpYWxfQQkBAl9BAAMJAQZhc3NlcnQBAwkAZgIFCV9mdXR1cmVfQQAACQBmAgUFTUFYX0EFCV9mdXR1cmVfQQcJAAIBAhFvdXQgb2YgYmFzZSByYW5nZQMJAQZhc3NlcnQBAwMJAGcCBQlfZnV0dXJlX0EFCl9pbml0aWFsX0EJAGcCCQBoAgUKX2luaXRpYWxfQQUMTUFYX0FfQ0hBTkdFBQlfZnV0dXJlX0EHBgMJAGYCBQpfaW5pdGlhbF9BBQlfZnV0dXJlX0EJAGcCCQBoAgUJX2Z1dHVyZV9BBQxNQVhfQV9DSEFOR0UFCl9pbml0aWFsX0EHCQACAQIMb3V0IG9mIHJhbmdlCQDMCAIJAQxJbnRlZ2VyRW50cnkCAglpbml0aWFsX0EFCl9pbml0aWFsX0EJAMwIAgkBDEludGVnZXJFbnRyeQICCGZ1dHVyZV9BBQlfZnV0dXJlX0EJAMwIAgkBDEludGVnZXJFbnRyeQICDmluaXRpYWxfQV90aW1lBQ9ibG9ja190aW1lc3RhbXAJAMwIAgkBDEludGVnZXJFbnRyeQICDWZ1dHVyZV9BX3RpbWUFDF9mdXR1cmVfdGltZQUDbmlsA21zZwELc3RvcF9yYW1wX0EAAwkBBmFzc2VydAEJAAACCAUDbXNnBmNhbGxlcgUFb3duZXIJAAIBAgpvbmx5IG93bmVyBAljdXJyZW50X0EJAQJfQQAJAMwIAgkBDEludGVnZXJFbnRyeQICCWluaXRpYWxfQQUJY3VycmVudF9BCQDMCAIJAQxJbnRlZ2VyRW50cnkCAghmdXR1cmVfQQUJY3VycmVudF9BCQDMCAIJAQxJbnRlZ2VyRW50cnkCAg5pbml0aWFsX0FfdGltZQUPYmxvY2tfdGltZXN0YW1wCQDMCAIJAQxJbnRlZ2VyRW50cnkCAg1mdXR1cmVfQV90aW1lBQ9ibG9ja190aW1lc3RhbXAFA25pbANtc2cBDmNvbW1pdF9uZXdfZmVlAgduZXdfZmVlDW5ld19hZG1pbl9mZWUDCQEGYXNzZXJ0AQkAAAIIBQNtc2cGY2FsbGVyBQVvd25lcgkAAgECCm9ubHkgb3duZXIDCQEGYXNzZXJ0AQkAAAIFFmFkbWluX2FjdGlvbnNfZGVhZGxpbmUAAAkAAgECDWFjdGl2ZSBhY3Rpb24DCQEGYXNzZXJ0AQkAZwIFB01BWF9GRUUFB25ld19mZWUJAAIBAhNmZWUgZXhjZWVkcyBtYXhpbXVtAwkBBmFzc2VydAEJAGcCBQ1NQVhfQURNSU5fRkVFBQ1uZXdfYWRtaW5fZmVlCQACAQIZYWRtaW4gZmVlIGV4Y2VlZHMgbWF4aW11bQQJX2RlYWRsaW5lCQBkAgUPYmxvY2tfdGltZXN0YW1wBRNBRE1JTl9BQ1RJT05TX0RFTEFZCQDMCAIJAQxJbnRlZ2VyRW50cnkCAhZhZG1pbl9hY3Rpb25zX2RlYWRsaW5lBQlfZGVhZGxpbmUJAMwIAgkBDEludGVnZXJFbnRyeQICCmZ1dHVyZV9mZWUFB25ld19mZWUJAMwIAgkBDEludGVnZXJFbnRyeQICEGZ1dHVyZV9hZG1pbl9mZWUFDW5ld19hZG1pbl9mZWUFA25pbANtc2cBDWFwcGx5X25ld19mZWUAAwkBBmFzc2VydAEJAAACCAUDbXNnBmNhbGxlcgUFb3duZXIJAAIBAgpvbmx5IG93bmVyAwkBBmFzc2VydAEJAGcCBQ9ibG9ja190aW1lc3RhbXAFFmFkbWluX2FjdGlvbnNfZGVhZGxpbmUJAAIBAhFpbnN1ZmZpY2llbnQgdGltZQMJAQZhc3NlcnQBCQECIT0CBRZhZG1pbl9hY3Rpb25zX2RlYWRsaW5lAAAJAAIBAhBubyBhY3RpdmUgYWN0aW9uCQDMCAIJAQxJbnRlZ2VyRW50cnkCAhZhZG1pbl9hY3Rpb25zX2RlYWRsaW5lAAAJAMwIAgkBDEludGVnZXJFbnRyeQICA2ZlZQUKZnV0dXJlX2ZlZQkAzAgCCQEMSW50ZWdlckVudHJ5AgIJYWRtaW5fZmVlBRBmdXR1cmVfYWRtaW5fZmVlBQNuaWwDbXNnARVyZXZlcnRfbmV3X3BhcmFtZXRlcnMAAwkBBmFzc2VydAEJAAACCAUDbXNnBmNhbGxlcgUFb3duZXIJAAIBAgpvbmx5IG93bmVyCQDMCAIJAQxJbnRlZ2VyRW50cnkCAhZhZG1pbl9hY3Rpb25zX2RlYWRsaW5lAAAFA25pbANtc2cBGWNvbW1pdF90cmFuc2Zlcl9vd25lcnNoaXABBl9vd25lcgMJAQZhc3NlcnQBCQAAAggFA21zZwZjYWxsZXIFBW93bmVyCQACAQIKb25seSBvd25lcgMJAQZhc3NlcnQBCQAAAgUbdHJhbnNmZXJfb3duZXJzaGlwX2RlYWRsaW5lAAAJAAIBAg9hY3RpdmUgdHJhbnNmZXIECV9kZWFkbGluZQkAZAIFD2Jsb2NrX3RpbWVzdGFtcAUTQURNSU5fQUNUSU9OU19ERUxBWQkAzAgCCQEMSW50ZWdlckVudHJ5AgIbdHJhbnNmZXJfb3duZXJzaGlwX2RlYWRsaW5lBQlfZGVhZGxpbmUJAMwIAgkBC1N0cmluZ0VudHJ5AgIMZnV0dXJlX293bmVyCQEMY2hlY2tBZGRyZXNzAQUGX293bmVyBQNuaWwDbXNnARhhcHBseV90cmFuc2Zlcl9vd25lcnNoaXAAAwkBBmFzc2VydAEJAAACCAUDbXNnBmNhbGxlcgUFb3duZXIJAAIBAgpvbmx5IG93bmVyAwkBBmFzc2VydAEJAGcCBQ9ibG9ja190aW1lc3RhbXAFG3RyYW5zZmVyX293bmVyc2hpcF9kZWFkbGluZQkAAgECEWluc3VmZmljaWVudCB0aW1lAwkBBmFzc2VydAEJAQIhPQIFG3RyYW5zZmVyX293bmVyc2hpcF9kZWFkbGluZQAACQACAQISbm8gYWN0aXZlIHRyYW5zZmVyCQDMCAIJAQxJbnRlZ2VyRW50cnkCAht0cmFuc2Zlcl9vd25lcnNoaXBfZGVhZGxpbmUAAAkAzAgCCQELU3RyaW5nRW50cnkCAgVvd25lcgUMZnV0dXJlX293bmVyBQNuaWwDbXNnARlyZXZlcnRfdHJhbnNmZXJfb3duZXJzaGlwAAMJAQZhc3NlcnQBCQAAAggFA21zZwZjYWxsZXIFBW93bmVyCQACAQIKb25seSBvd25lcgkAzAgCCQEMSW50ZWdlckVudHJ5AgIbdHJhbnNmZXJfb3duZXJzaGlwX2RlYWRsaW5lAAAFA25pbANtc2cBDmFkbWluX2JhbGFuY2VzAQFpCQCUCgIFA25pbAkBDWFkbWluX2JhbGFuY2UBBQFpA21zZwETd2l0aGRyYXdfYWRtaW5fZmVlcwADCQEGYXNzZXJ0AQkAAAIIBQNtc2cGY2FsbGVyBQVvd25lcgkAAgECCm9ubHkgb3duZXIJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwUFb3duZXIJAQ1hZG1pbl9iYWxhbmNlAQAABQdjb2luc18wCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMFBW93bmVyCQENYWRtaW5fYmFsYW5jZQEAAQUHY29pbnNfMQkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQVvd25lcgkBDWFkbWluX2JhbGFuY2UBAAIFB2NvaW5zXzIFA25pbANtc2cBEWRvbmF0ZV9hZG1pbl9mZWVzAAMJAQZhc3NlcnQBCQAAAggFA21zZwZjYWxsZXIFBW93bmVyCQACAQIKb25seSBvd25lcgkAzAgCCQEMSW50ZWdlckVudHJ5AgIKYmFsYW5jZXNfMAkAZAIJAQ1hZG1pbl9iYWxhbmNlAQAABQpiYWxhbmNlc18wCQDMCAIJAQxJbnRlZ2VyRW50cnkCAgpiYWxhbmNlc18xCQBkAgkBDWFkbWluX2JhbGFuY2UBAAEFCmJhbGFuY2VzXzEJAMwIAgkBDEludGVnZXJFbnRyeQICCmJhbGFuY2VzXzIJAGQCCQENYWRtaW5fYmFsYW5jZQEAAgUKYmFsYW5jZXNfMgUDbmlsA21zZwENc2V0X2F1dG9fZmVlcwENX2lzX2F1dG9fZmVlcwMJAQZhc3NlcnQBCQAAAggFA21zZwZjYWxsZXIFBW93bmVyCQACAQIKb25seSBvd25lcgkAzAgCCQEMQm9vbGVhbkVudHJ5AgIMaXNfYXV0b19mZWVzBQ1faXNfYXV0b19mZWVzBQNuaWwDbXNnAQdraWxsX21lAAMJAQZhc3NlcnQBCQAAAggFA21zZwZjYWxsZXIFBW93bmVyCQACAQIKb25seSBvd25lcgMJAQZhc3NlcnQBCQBmAgUNa2lsbF9kZWFkbGluZQUPYmxvY2tfdGltZXN0YW1wCQACAQITZGVhZGxpbmUgaGFzIHBhc3NlZAkAzAgCCQEMQm9vbGVhbkVudHJ5AgIJaXNfa2lsbGVkBgUDbmlsA21zZwEJdW5raWxsX21lAAMJAQZhc3NlcnQBCQAAAggFA21zZwZjYWxsZXIFBW93bmVyCQACAQIKb25seSBvd25lcgkAzAgCCQEMQm9vbGVhbkVudHJ5AgIJaXNfa2lsbGVkBwUDbmlsA21zZwESc2V0X2hlaWdodF9hZGRyZXNzAQ5faGVpZ2h0QWRkcmVzcwMJAQIhPQIIBQNtc2cGY2FsbGVyBQVvd25lcgkAAgECCk93bmVyIG9ubHkJAMwIAgkBC1N0cmluZ0VudHJ5AgINaGVpZ2h0QWRkcmVzcwkBDGNoZWNrQWRkcmVzcwEFDl9oZWlnaHRBZGRyZXNzBQNuaWwBaQEMc2V0X3ZlcmlmaWVyAQh2ZXJpZmllcgMJAQIhPQIIBQFpBmNhbGxlcgUEdGhpcwkAAgECDnNlbGYgY2FsbCBvbmx5BAlhZGRyZXNzT0sEByRtYXRjaDAJAKYIAQUIdmVyaWZpZXIDCQABAgUHJG1hdGNoMAIHQWRkcmVzcwQBYQUHJG1hdGNoMAYHAwkBASEBBQlhZGRyZXNzT0sJAAIBCQCsAgICF3ZlcmlmaWVyIHdyb25nIGFkZHJlc3MgBQh2ZXJpZmllcgMJAQlpc0RlZmluZWQBCQCdCAIFBHRoaXMCCHZlcmlmaWVyCQACAQIYdmVyaWZpZXIgYWxyZWFkeSBkZWZpbmVkCQDMCAIJAQtTdHJpbmdFbnRyeQICCHZlcmlmaWVyBQh2ZXJpZmllcgUDbmlsAQJ0eAEGdmVyaWZ5AAQHJG1hdGNoMAkAnQgCBQR0aGlzAgh2ZXJpZmllcgMJAAECBQckbWF0Y2gwAgZTdHJpbmcECHZlcmlmaWVyBQckbWF0Y2gwCQELdmFsdWVPckVsc2UCCQCbCAIJARFAZXh0ck5hdGl2ZSgxMDYyKQEFCHZlcmlmaWVyCQCsAgIJAKwCAgkArAICAgdzdGF0dXNfCQClCAEFBHRoaXMCAV8JANgEAQgFAnR4AmlkBwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAgFAnR4D3NlbmRlclB1YmxpY0tleZLqGyE=", "chainId": 84, "height": 2236265, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: none Next: none Full:
Old | New | Differences | |
---|---|---|---|
1 | - | # no script | |
1 | + | {-# STDLIB_VERSION 6 #-} | |
2 | + | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | + | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let N_COINS = 3 | |
5 | + | ||
6 | + | let FEE_DENOMINATOR = 10000000000 | |
7 | + | ||
8 | + | let PRECISION = 1000000 | |
9 | + | ||
10 | + | let DECIMALS = 6 | |
11 | + | ||
12 | + | let MAX_ADMIN_FEE = 10000000000 | |
13 | + | ||
14 | + | let MAX_FEE = 5000000000 | |
15 | + | ||
16 | + | let MAX_A = 1000000 | |
17 | + | ||
18 | + | let MAX_A_CHANGE = 10 | |
19 | + | ||
20 | + | let ADMIN_ACTIONS_DELAY = ((3 * 86400) / 60) | |
21 | + | ||
22 | + | let MIN_RAMP_TIME = (86400 / 60) | |
23 | + | ||
24 | + | let VP_LOG_PERIOD = 86400000 | |
25 | + | ||
26 | + | let VP_PRECISION = 1000000000000 | |
27 | + | ||
28 | + | let coins_0 = fromBase58String(getStringValue(this, "coins_0")) | |
29 | + | ||
30 | + | let coins_1 = fromBase58String(getStringValue(this, "coins_1")) | |
31 | + | ||
32 | + | let coins_2 = fromBase58String(getStringValue(this, "coins_2")) | |
33 | + | ||
34 | + | let balances_0 = valueOrElse(getInteger(this, "balances_0"), 0) | |
35 | + | ||
36 | + | let balances_1 = valueOrElse(getInteger(this, "balances_1"), 0) | |
37 | + | ||
38 | + | let balances_2 = valueOrElse(getInteger(this, "balances_2"), 0) | |
39 | + | ||
40 | + | let balances = $Tuple3(balances_0, balances_1, balances_2) | |
41 | + | ||
42 | + | let fee = getIntegerValue(this, "fee") | |
43 | + | ||
44 | + | let admin_fee = getIntegerValue(this, "admin_fee") | |
45 | + | ||
46 | + | let is_auto_fees = getBooleanValue(this, "is_auto_fees") | |
47 | + | ||
48 | + | let owner = addressFromStringValue(getStringValue(this, "owner")) | |
49 | + | ||
50 | + | let token = fromBase58String(getStringValue(this, "token")) | |
51 | + | ||
52 | + | let token_quantity = value(assetInfo(token)).quantity | |
53 | + | ||
54 | + | let initial_A = getIntegerValue(this, "initial_A") | |
55 | + | ||
56 | + | let future_A = getIntegerValue(this, "future_A") | |
57 | + | ||
58 | + | let initial_A_time = valueOrElse(getInteger(this, "initial_A_time"), 0) | |
59 | + | ||
60 | + | let future_A_time = valueOrElse(getInteger(this, "future_A_time"), 0) | |
61 | + | ||
62 | + | let admin_actions_deadline = valueOrElse(getInteger(this, "admin_actions_deadline"), 0) | |
63 | + | ||
64 | + | let transfer_ownership_deadline = valueOrElse(getInteger(this, "transfer_ownership_deadline"), 0) | |
65 | + | ||
66 | + | let future_fee = getIntegerValue(this, "future_fee") | |
67 | + | ||
68 | + | let future_admin_fee = getIntegerValue(this, "future_admin_fee") | |
69 | + | ||
70 | + | let future_owner = getStringValue(this, "future_owner") | |
71 | + | ||
72 | + | let is_killed = getBooleanValue(this, "is_killed") | |
73 | + | ||
74 | + | let kill_deadline = getIntegerValue(this, "kill_deadline") | |
75 | + | ||
76 | + | let KILL_DEADLINE_DT = (((2 * 30) * 86400) / 60) | |
77 | + | ||
78 | + | let big0 = toBigInt(0) | |
79 | + | ||
80 | + | let big1 = toBigInt(1) | |
81 | + | ||
82 | + | let big2 = toBigInt(2) | |
83 | + | ||
84 | + | let big3 = toBigInt(3) | |
85 | + | ||
86 | + | let big4 = toBigInt(4) | |
87 | + | ||
88 | + | let big27 = toBigInt(27) | |
89 | + | ||
90 | + | let heightAddress = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(this, "heightAddress"), "no settings defined")), "bad settings address") | |
91 | + | ||
92 | + | let HEIGHT = valueOrElse(getInteger(heightAddress, "__MOCK_HEIGHT"), 0) | |
93 | + | ||
94 | + | let block_timestamp = HEIGHT | |
95 | + | ||
96 | + | func assert (a) = if (a) | |
97 | + | then false | |
98 | + | else true | |
99 | + | ||
100 | + | ||
101 | + | func x_i (x,i) = if ((i == 0)) | |
102 | + | then x._1 | |
103 | + | else if ((i == 1)) | |
104 | + | then x._2 | |
105 | + | else if ((i == 2)) | |
106 | + | then x._3 | |
107 | + | else throw("index out of N_COINS") | |
108 | + | ||
109 | + | ||
110 | + | func admin_balance (i) = if ((i == 0)) | |
111 | + | then (assetBalance(this, coins_0) - balances_0) | |
112 | + | else if ((i == 1)) | |
113 | + | then (assetBalance(this, coins_1) - balances_1) | |
114 | + | else if ((i == 2)) | |
115 | + | then (assetBalance(this, coins_2) - balances_2) | |
116 | + | else throw("index out of N_COINS") | |
117 | + | ||
118 | + | ||
119 | + | func checkAddress (a58) = { | |
120 | + | let a = addressFromStringValue(a58) | |
121 | + | toString(a) | |
122 | + | } | |
123 | + | ||
124 | + | ||
125 | + | func _A () = { | |
126 | + | let t1 = future_A_time | |
127 | + | let A1 = future_A | |
128 | + | if ((t1 > block_timestamp)) | |
129 | + | then { | |
130 | + | let A0 = initial_A | |
131 | + | let t0 = initial_A_time | |
132 | + | if ((A1 > A0)) | |
133 | + | then (A0 + (((A1 - A0) * (block_timestamp - t0)) / (t1 - t0))) | |
134 | + | else (A0 - (((A0 - A1) * (block_timestamp - t0)) / (t1 - t0))) | |
135 | + | } | |
136 | + | else A1 | |
137 | + | } | |
138 | + | ||
139 | + | ||
140 | + | func _xp () = balances | |
141 | + | ||
142 | + | ||
143 | + | func _xp_mem (_balances) = _balances | |
144 | + | ||
145 | + | ||
146 | + | func get_D (xp,amp) = { | |
147 | + | let @ = invoke(this, "D", [xp._1, xp._2, xp._3, amp], nil) | |
148 | + | if ($isInstanceOf(@, "Int")) | |
149 | + | then @ | |
150 | + | else throw(($getType(invoke(this, "D", [xp._1, xp._2, xp._3, amp], nil)) + " couldn't be cast to Int")) | |
151 | + | } | |
152 | + | ||
153 | + | ||
154 | + | func get_D_internal (xp0,xp1,xp2,amp) = { | |
155 | + | let S = ((xp0 + xp1) + xp2) | |
156 | + | if ((toInt(S) == 0)) | |
157 | + | then 0 | |
158 | + | else { | |
159 | + | let Ann = (amp * N_COINS) | |
160 | + | let AnnS = (toBigInt(Ann) * S) | |
161 | + | let Ann1 = toBigInt((Ann - 1)) | |
162 | + | let xd = (((xp0 * xp1) * xp2) * big27) | |
163 | + | func Dproc (acc,i) = if ((acc._2 == true)) | |
164 | + | then acc | |
165 | + | else { | |
166 | + | let Dprev = acc._1 | |
167 | + | let DprevDprev = (Dprev * Dprev) | |
168 | + | let D_P = ((DprevDprev * DprevDprev) / xd) | |
169 | + | let D = fraction((AnnS + (big3 * D_P)), Dprev, ((Ann1 * Dprev) + (big4 * D_P))) | |
170 | + | if ((D > Dprev)) | |
171 | + | then if ((1 >= toInt((D - Dprev)))) | |
172 | + | then $Tuple2(D, true) | |
173 | + | else $Tuple2(D, false) | |
174 | + | else if ((1 >= toInt((Dprev - D)))) | |
175 | + | then $Tuple2(D, true) | |
176 | + | else $Tuple2(D, false) | |
177 | + | } | |
178 | + | ||
179 | + | let $t047124819 = { | |
180 | + | let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14] | |
181 | + | let $s = size($l) | |
182 | + | let $acc0 = $Tuple2(S, false) | |
183 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
184 | + | then $a | |
185 | + | else Dproc($a, $l[$i]) | |
186 | + | ||
187 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
188 | + | then $a | |
189 | + | else throw("List size exceeds 15") | |
190 | + | ||
191 | + | $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) | |
192 | + | } | |
193 | + | let D = $t047124819._1 | |
194 | + | let finished = $t047124819._2 | |
195 | + | if ((finished == false)) | |
196 | + | then throw(("get_D() not finished with " + toString(D))) | |
197 | + | else toInt(D) | |
198 | + | } | |
199 | + | } | |
200 | + | ||
201 | + | ||
202 | + | func get_D_mem (_balances_0,_balances_1,_balances_2,amp) = get_D(_xp_mem($Tuple3(_balances_0, _balances_1, _balances_2)), amp) | |
203 | + | ||
204 | + | ||
205 | + | func get_y (i,j,x,xp_) = if (assert((i != j))) | |
206 | + | then throw("same coin") | |
207 | + | else if (assert(if ((j >= 0)) | |
208 | + | then (i >= 0) | |
209 | + | else false)) | |
210 | + | then throw("below zero") | |
211 | + | else if (assert(if ((N_COINS > j)) | |
212 | + | then (N_COINS > i) | |
213 | + | else false)) | |
214 | + | then throw("above N_COINS") | |
215 | + | else { | |
216 | + | let amp = _A() | |
217 | + | let D = get_D(xp_, amp) | |
218 | + | let Ann = (amp * N_COINS) | |
219 | + | func xs (acc,_i) = if ((_i == i)) | |
220 | + | then (acc ++ [x]) | |
221 | + | else if ((_i != j)) | |
222 | + | then (acc ++ [x_i(xp_, _i)]) | |
223 | + | else acc | |
224 | + | ||
225 | + | let ab = { | |
226 | + | let $l = [0, 1, 2] | |
227 | + | let $s = size($l) | |
228 | + | let $acc0 = nil | |
229 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
230 | + | then $a | |
231 | + | else xs($a, $l[$i]) | |
232 | + | ||
233 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
234 | + | then $a | |
235 | + | else throw("List size exceeds 3") | |
236 | + | ||
237 | + | $f0_2($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3) | |
238 | + | } | |
239 | + | func S_c (a,b) = { | |
240 | + | let S_ = (a + b) | |
241 | + | let ca = fraction(D, D, (a * N_COINS)) | |
242 | + | let cb = fraction(ca, D, (b * N_COINS)) | |
243 | + | $Tuple2(S_, cb) | |
244 | + | } | |
245 | + | ||
246 | + | let $t058855921 = S_c(ab[0], ab[1]) | |
247 | + | let S_ = $t058855921._1 | |
248 | + | let c_ = $t058855921._2 | |
249 | + | let c = fraction(toBigInt(c_), toBigInt(D), toBigInt((Ann * N_COINS))) | |
250 | + | let bD = toBigInt(((S_ + (D / Ann)) - D)) | |
251 | + | func y_proc (acc,_i) = if ((acc._2 == true)) | |
252 | + | then acc | |
253 | + | else { | |
254 | + | let y_prev = acc._1 | |
255 | + | let y = (((y_prev * y_prev) + c) / ((big2 * y_prev) + bD)) | |
256 | + | if ((y > y_prev)) | |
257 | + | then if ((1 >= toInt((y - y_prev)))) | |
258 | + | then $Tuple2(y, true) | |
259 | + | else $Tuple2(y, false) | |
260 | + | else if ((1 >= toInt((y_prev - y)))) | |
261 | + | then $Tuple2(y, true) | |
262 | + | else $Tuple2(y, false) | |
263 | + | } | |
264 | + | ||
265 | + | let $t064426565 = { | |
266 | + | let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15] | |
267 | + | let $s = size($l) | |
268 | + | let $acc0 = $Tuple2(toBigInt(D), false) | |
269 | + | func $f1_1 ($a,$i) = if (($i >= $s)) | |
270 | + | then $a | |
271 | + | else y_proc($a, $l[$i]) | |
272 | + | ||
273 | + | func $f1_2 ($a,$i) = if (($i >= $s)) | |
274 | + | then $a | |
275 | + | else throw("List size exceeds 16") | |
276 | + | ||
277 | + | $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($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16) | |
278 | + | } | |
279 | + | let y = $t064426565._1 | |
280 | + | let finished = $t064426565._2 | |
281 | + | if ((finished == false)) | |
282 | + | then throw(("get_y() not finished with " + toString(y))) | |
283 | + | else $Tuple2(toInt(y), D) | |
284 | + | } | |
285 | + | ||
286 | + | ||
287 | + | func get_y_D (A_,i,xp,D) = if (assert((i >= 0))) | |
288 | + | then throw("i below zero") | |
289 | + | else if (assert((N_COINS > i))) | |
290 | + | then throw("i above N_COINS") | |
291 | + | else { | |
292 | + | let Ann = (A_ * N_COINS) | |
293 | + | func S_c (a,b) = { | |
294 | + | let S_ = (a + b) | |
295 | + | let ca = fraction(D, D, (a * N_COINS)) | |
296 | + | let cb = fraction(ca, D, (b * N_COINS)) | |
297 | + | $Tuple2(S_, cb) | |
298 | + | } | |
299 | + | ||
300 | + | let $t071067319 = if ((i == 0)) | |
301 | + | then S_c(xp._2, xp._3) | |
302 | + | else if ((i == 1)) | |
303 | + | then S_c(xp._1, xp._3) | |
304 | + | else if ((i == 2)) | |
305 | + | then S_c(xp._1, xp._2) | |
306 | + | else throw("index out of N_COINS") | |
307 | + | let S_ = $t071067319._1 | |
308 | + | let c_ = $t071067319._2 | |
309 | + | let c = fraction(toBigInt(c_), toBigInt(D), toBigInt((Ann * N_COINS))) | |
310 | + | let bD = toBigInt(((S_ + (D / Ann)) - D)) | |
311 | + | func y_D_proc (acc,_i) = if ((acc._2 == true)) | |
312 | + | then acc | |
313 | + | else { | |
314 | + | let y_prev = acc._1 | |
315 | + | let y = (((y_prev * y_prev) + c) / ((big2 * y_prev) + bD)) | |
316 | + | if ((y > y_prev)) | |
317 | + | then if ((1 >= toInt((y - y_prev)))) | |
318 | + | then $Tuple2(y, true) | |
319 | + | else $Tuple2(y, false) | |
320 | + | else if ((1 >= toInt((y_prev - y)))) | |
321 | + | then $Tuple2(y, true) | |
322 | + | else $Tuple2(y, false) | |
323 | + | } | |
324 | + | ||
325 | + | let $t078437968 = { | |
326 | + | let $l = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15] | |
327 | + | let $s = size($l) | |
328 | + | let $acc0 = $Tuple2(toBigInt(D), false) | |
329 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
330 | + | then $a | |
331 | + | else y_D_proc($a, $l[$i]) | |
332 | + | ||
333 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
334 | + | then $a | |
335 | + | else throw("List size exceeds 16") | |
336 | + | ||
337 | + | $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($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16) | |
338 | + | } | |
339 | + | let y = $t078437968._1 | |
340 | + | let finished = $t078437968._2 | |
341 | + | if ((finished == false)) | |
342 | + | then throw(("get_y_D() not finished with " + toString(y))) | |
343 | + | else toInt(y) | |
344 | + | } | |
345 | + | ||
346 | + | ||
347 | + | func _calc_withdraw_one_coin (_token_amount,i) = { | |
348 | + | let amp = _A() | |
349 | + | let _fee = ((fee * N_COINS) / (4 * (N_COINS - 1))) | |
350 | + | let total_supply = token_quantity | |
351 | + | let xp = _xp() | |
352 | + | let D0 = get_D(xp, amp) | |
353 | + | let D1 = (D0 - fraction(_token_amount, D0, total_supply)) | |
354 | + | let new_y = get_y_D(amp, i, xp, D1) | |
355 | + | let dy_0 = (x_i(xp, i) - new_y) | |
356 | + | func xp_reduced_proc (ji,xp_j) = { | |
357 | + | let dx_expected = if (ji) | |
358 | + | then (fraction(xp_j, D1, D0) - new_y) | |
359 | + | else (xp_j - fraction(xp_j, D1, D0)) | |
360 | + | (xp_j - fraction(_fee, dx_expected, FEE_DENOMINATOR)) | |
361 | + | } | |
362 | + | ||
363 | + | let xp_reduced = $Tuple3(xp_reduced_proc((i == 0), xp._1), xp_reduced_proc((i == 1), xp._2), xp_reduced_proc((i == 2), xp._3)) | |
364 | + | let xp_reduced_i = x_i(xp_reduced, i) | |
365 | + | let dy = ((xp_reduced_i - get_y_D(amp, i, xp_reduced, D1)) - 1) | |
366 | + | $Tuple3(dy, (dy_0 - dy), D0) | |
367 | + | } | |
368 | + | ||
369 | + | ||
370 | + | func virtual_price (D) = fraction(D, VP_PRECISION, token_quantity) | |
371 | + | ||
372 | + | ||
373 | + | func log_data (D,add) = { | |
374 | + | let total_vol = (parseBigIntValue(valueOrElse(getString("vol"), "0")) + toBigInt(add)) | |
375 | + | let total_vol_string = toString(total_vol) | |
376 | + | ([StringEntry("vol", total_vol_string)] ++ { | |
377 | + | let log_period = toString((lastBlock.timestamp / VP_LOG_PERIOD)) | |
378 | + | let log_key = ("log_" + log_period) | |
379 | + | if (isDefined(getString(log_key))) | |
380 | + | then nil | |
381 | + | else [StringEntry(log_key, ((((toString(virtual_price(D)) + "_") + total_vol_string) + "_") + toString(lastBlock.timestamp)))] | |
382 | + | }) | |
383 | + | } | |
384 | + | ||
385 | + | ||
386 | + | func get_nearest_log (period) = { | |
387 | + | func fold (log_value,step) = if ((log_value != "")) | |
388 | + | then log_value | |
389 | + | else { | |
390 | + | let log_key = ("log_" + toString((period - step))) | |
391 | + | valueOrElse(getString(log_key), "") | |
392 | + | } | |
393 | + | ||
394 | + | let list30p = [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] | |
395 | + | let list30m = [-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] | |
396 | + | let value30p = { | |
397 | + | let $l = list30p | |
398 | + | let $s = size($l) | |
399 | + | let $acc0 = "" | |
400 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
401 | + | then $a | |
402 | + | else fold($a, $l[$i]) | |
403 | + | ||
404 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
405 | + | then $a | |
406 | + | else throw("List size exceeds 30") | |
407 | + | ||
408 | + | $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($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) | |
409 | + | } | |
410 | + | if ((value30p != "")) | |
411 | + | then value30p | |
412 | + | else { | |
413 | + | let $l = list30m | |
414 | + | let $s = size($l) | |
415 | + | let $acc0 = "" | |
416 | + | func $f1_1 ($a,$i) = if (($i >= $s)) | |
417 | + | then $a | |
418 | + | else fold($a, $l[$i]) | |
419 | + | ||
420 | + | func $f1_2 ($a,$i) = if (($i >= $s)) | |
421 | + | then $a | |
422 | + | else throw("List size exceeds 30") | |
423 | + | ||
424 | + | $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($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) | |
425 | + | } | |
426 | + | } | |
427 | + | ||
428 | + | ||
429 | + | func get_virtual_price_diff (_t0) = { | |
430 | + | let vp1 = virtual_price(get_D(_xp(), _A())) | |
431 | + | let vp1_timestamp = lastBlock.timestamp | |
432 | + | let t0 = if ((0 >= _t0)) | |
433 | + | then (vp1_timestamp + _t0) | |
434 | + | else _t0 | |
435 | + | let t0_period = (t0 / VP_LOG_PERIOD) | |
436 | + | let log_value = get_nearest_log(t0_period) | |
437 | + | if ((log_value == "")) | |
438 | + | then $Tuple3(vp1, vp1, 0) | |
439 | + | else { | |
440 | + | let log_list = split(log_value, "_") | |
441 | + | let vp0 = parseIntValue(log_list[0]) | |
442 | + | let vp0_timestamp = parseIntValue(log_list[2]) | |
443 | + | $Tuple3(vp1, vp0, (vp1_timestamp - vp0_timestamp)) | |
444 | + | } | |
445 | + | } | |
446 | + | ||
447 | + | ||
448 | + | func get_volume_diff (_t0) = { | |
449 | + | let vol1 = parseBigIntValue(valueOrElse(getString("vol"), "0")) | |
450 | + | let vol1_timestamp = lastBlock.timestamp | |
451 | + | let t0 = if ((0 >= _t0)) | |
452 | + | then (vol1_timestamp + _t0) | |
453 | + | else _t0 | |
454 | + | let t0_period = (t0 / VP_LOG_PERIOD) | |
455 | + | let log_value = get_nearest_log(t0_period) | |
456 | + | if ((log_value == "")) | |
457 | + | then $Tuple3(vol1, vol1, 0) | |
458 | + | else { | |
459 | + | let log_list = split(log_value, "_") | |
460 | + | let vol0 = parseBigIntValue(log_list[1]) | |
461 | + | let vol0_timestamp = parseIntValue(log_list[2]) | |
462 | + | $Tuple3(vol1, vol0, (vol1_timestamp - vol0_timestamp)) | |
463 | + | } | |
464 | + | } | |
465 | + | ||
466 | + | ||
467 | + | @Callable(msg) | |
468 | + | func D (xp0,xp1,xp2,amp) = { | |
469 | + | let D = get_D_internal(toBigInt(xp0), toBigInt(xp1), toBigInt(xp2), amp) | |
470 | + | $Tuple2([IntegerEntry("D", D)], D) | |
471 | + | } | |
472 | + | ||
473 | + | ||
474 | + | ||
475 | + | @Callable(msg) | |
476 | + | func init (_owner,_coins,_pool_token,_A,_fee,_admin_fee) = if (!(isDataStorageUntouched(this))) | |
477 | + | then throw("Already initialized") | |
478 | + | else if ((msg.caller != this)) | |
479 | + | then throw("Self initialization only") | |
480 | + | else { | |
481 | + | func checkCoin (coin58) = { | |
482 | + | let coin = valueOrErrorMessage(fromBase58String(coin58), ("fromBase58String: " + coin58)) | |
483 | + | let decimals = valueOrErrorMessage(assetInfo(coin), ("assetInfo: " + coin58)).decimals | |
484 | + | if ((decimals != DECIMALS)) | |
485 | + | then throw("wrong decimals") | |
486 | + | else coin | |
487 | + | } | |
488 | + | ||
489 | + | let coins = split(_coins, ",") | |
490 | + | if ((size(coins) != N_COINS)) | |
491 | + | then throw(("size( coins ) != " + toString(N_COINS))) | |
492 | + | else { | |
493 | + | let issueToken = Issue(_pool_token, ("LP token for " + _coins), 0, DECIMALS, true) | |
494 | + | let tokenId = toBase58String(calculateAssetId(issueToken)) | |
495 | + | [StringEntry("coins_0", toBase58String(checkCoin(coins[0]))), StringEntry("coins_1", toBase58String(checkCoin(coins[1]))), StringEntry("coins_2", toBase58String(checkCoin(coins[2]))), IntegerEntry("initial_A", _A), IntegerEntry("future_A", _A), IntegerEntry("fee", _fee), IntegerEntry("admin_fee", _admin_fee), StringEntry("owner", checkAddress(_owner)), IntegerEntry("kill_deadline", (height + KILL_DEADLINE_DT)), StringEntry("token", tokenId), BooleanEntry("is_killed", false), BooleanEntry("is_auto_fees", true), issueToken] | |
496 | + | } | |
497 | + | } | |
498 | + | ||
499 | + | ||
500 | + | ||
501 | + | @Callable(msg) | |
502 | + | func add_liquidity (min_mint_amount) = if (assert(!(is_killed))) | |
503 | + | then throw("is killed") | |
504 | + | else { | |
505 | + | let _fee = ((fee * N_COINS) / (4 * (N_COINS - 1))) | |
506 | + | let amp = _A() | |
507 | + | let token_supply = token_quantity | |
508 | + | let D0 = if ((token_supply == 0)) | |
509 | + | then 0 | |
510 | + | else get_D_mem(balances_0, balances_1, balances_2, amp) | |
511 | + | let payments = msg.payments | |
512 | + | let n = size(payments) | |
513 | + | func assetValid (coin) = if (if ((coin == coins_0)) | |
514 | + | then true | |
515 | + | else (coin == coins_1)) | |
516 | + | then true | |
517 | + | else (coin == coins_2) | |
518 | + | ||
519 | + | func validPayments (payments,n) = if ((n > 3)) | |
520 | + | then throw("payments size > 3") | |
521 | + | else if ((1 > n)) | |
522 | + | then throw("payments size < 1") | |
523 | + | else if (!(assetValid(payments[0].assetId))) | |
524 | + | then throw("unknown payment 1 token") | |
525 | + | else if (if ((n > 1)) | |
526 | + | then !(assetValid(payments[1].assetId)) | |
527 | + | else false) | |
528 | + | then throw("unknown payment 2 token") | |
529 | + | else if (if ((n > 2)) | |
530 | + | then !(assetValid(payments[2].assetId)) | |
531 | + | else false) | |
532 | + | then throw("unknown payment 3 token") | |
533 | + | else true | |
534 | + | ||
535 | + | func paymentAmount (coin,payments,n) = (((if ((payments[0].assetId == coin)) | |
536 | + | then payments[0].amount | |
537 | + | else 0) + (if (if ((n > 1)) | |
538 | + | then (payments[1].assetId == coin) | |
539 | + | else false) | |
540 | + | then payments[1].amount | |
541 | + | else 0)) + (if (if ((n > 2)) | |
542 | + | then (payments[2].assetId == coin) | |
543 | + | else false) | |
544 | + | then payments[2].amount | |
545 | + | else 0)) | |
546 | + | ||
547 | + | if (!(validPayments(payments, n))) | |
548 | + | then throw() | |
549 | + | else { | |
550 | + | let amounts_0 = paymentAmount(coins_0, payments, n) | |
551 | + | let amounts_1 = paymentAmount(coins_1, payments, n) | |
552 | + | let amounts_2 = paymentAmount(coins_2, payments, n) | |
553 | + | if (if ((token_supply == 0)) | |
554 | + | then if (if ((amounts_0 == 0)) | |
555 | + | then true | |
556 | + | else (amounts_1 == 0)) | |
557 | + | then true | |
558 | + | else (amounts_2 == 0) | |
559 | + | else false) | |
560 | + | then throw("initial deposit requires all coins") | |
561 | + | else { | |
562 | + | let new_balances_0 = (balances_0 + amounts_0) | |
563 | + | let new_balances_1 = (balances_1 + amounts_1) | |
564 | + | let new_balances_2 = (balances_2 + amounts_2) | |
565 | + | let D1 = get_D_mem(new_balances_0, new_balances_1, new_balances_2, amp) | |
566 | + | if (assert((D1 > D0))) | |
567 | + | then throw("D1 > D0") | |
568 | + | else { | |
569 | + | let $t01539117430 = if ((token_supply > 0)) | |
570 | + | then { | |
571 | + | func feesProc (old_balance,new_balance) = { | |
572 | + | let ideal_balance = fraction(D1, old_balance, D0) | |
573 | + | let difference = if ((ideal_balance > new_balance)) | |
574 | + | then (ideal_balance - new_balance) | |
575 | + | else (new_balance - ideal_balance) | |
576 | + | fraction(_fee, difference, FEE_DENOMINATOR) | |
577 | + | } | |
578 | + | ||
579 | + | let fees_0 = feesProc(balances_0, new_balances_0) | |
580 | + | let fees_1 = feesProc(balances_1, new_balances_1) | |
581 | + | let fees_2 = feesProc(balances_2, new_balances_2) | |
582 | + | let admin_fees_0 = fraction(fees_0, admin_fee, FEE_DENOMINATOR) | |
583 | + | let admin_fees_1 = fraction(fees_1, admin_fee, FEE_DENOMINATOR) | |
584 | + | let admin_fees_2 = fraction(fees_2, admin_fee, FEE_DENOMINATOR) | |
585 | + | $Tuple5((new_balances_0 - admin_fees_0), (new_balances_1 - admin_fees_1), (new_balances_2 - admin_fees_2), get_D_mem((new_balances_0 - fees_0), (new_balances_1 - fees_1), (new_balances_2 - fees_2), amp), ((((nil ++ (if (if (is_auto_fees) | |
586 | + | then (admin_fees_0 > 0) | |
587 | + | else false) | |
588 | + | then [ScriptTransfer(owner, admin_fees_0, coins_0)] | |
589 | + | else nil)) ++ (if (if (is_auto_fees) | |
590 | + | then (admin_fees_1 > 0) | |
591 | + | else false) | |
592 | + | then [ScriptTransfer(owner, admin_fees_1, coins_1)] | |
593 | + | else nil)) ++ (if (if (is_auto_fees) | |
594 | + | then (admin_fees_2 > 0) | |
595 | + | else false) | |
596 | + | then [ScriptTransfer(owner, admin_fees_2, coins_2)] | |
597 | + | else nil)) ++ log_data(D0, fraction(((fees_0 + fees_1) + fees_2), FEE_DENOMINATOR, fee)))) | |
598 | + | } | |
599 | + | else $Tuple5(new_balances_0, new_balances_1, new_balances_2, D1, nil) | |
600 | + | let final_balances_0 = $t01539117430._1 | |
601 | + | let final_balances_1 = $t01539117430._2 | |
602 | + | let final_balances_2 = $t01539117430._3 | |
603 | + | let D2 = $t01539117430._4 | |
604 | + | let fees_actions = $t01539117430._5 | |
605 | + | let mint_amount = if ((token_supply == 0)) | |
606 | + | then D1 | |
607 | + | else fraction(token_supply, (D2 - D0), D0) | |
608 | + | if (assert((mint_amount >= min_mint_amount))) | |
609 | + | then throw("Slippage screwed you") | |
610 | + | else ([IntegerEntry("balances_0", final_balances_0), IntegerEntry("balances_1", final_balances_1), IntegerEntry("balances_2", final_balances_2), Reissue(token, mint_amount, true), ScriptTransfer(msg.caller, mint_amount, token)] ++ fees_actions) | |
611 | + | } | |
612 | + | } | |
613 | + | } | |
614 | + | } | |
615 | + | ||
616 | + | ||
617 | + | ||
618 | + | @Callable(msg) | |
619 | + | func get_dy (i,j,dx) = { | |
620 | + | let xp = _xp() | |
621 | + | let xp_i = x_i(xp, i) | |
622 | + | let xp_j = x_i(xp, j) | |
623 | + | let x = (xp_i + dx) | |
624 | + | let $t01813018165 = get_y(i, j, x, xp) | |
625 | + | let y = $t01813018165._1 | |
626 | + | let D = $t01813018165._2 | |
627 | + | let dy = ((xp_j - y) - 1) | |
628 | + | let _fee = fraction(fee, dy, FEE_DENOMINATOR) | |
629 | + | $Tuple2(nil, (dy - _fee)) | |
630 | + | } | |
631 | + | ||
632 | + | ||
633 | + | ||
634 | + | @Callable(msg) | |
635 | + | func exchange (j,min_dy) = if (assert(!(is_killed))) | |
636 | + | then throw("is killed") | |
637 | + | else if ((size(msg.payments) != 1)) | |
638 | + | then throw("size( payments ) != 1") | |
639 | + | else { | |
640 | + | let payment = msg.payments[0] | |
641 | + | let tokenIn = payment.assetId | |
642 | + | let dx = payment.amount | |
643 | + | let i = if ((tokenIn == coins_0)) | |
644 | + | then 0 | |
645 | + | else if ((tokenIn == coins_1)) | |
646 | + | then 1 | |
647 | + | else if ((tokenIn == coins_2)) | |
648 | + | then 2 | |
649 | + | else throw("unknown token in") | |
650 | + | let tokenOut = if ((j == 0)) | |
651 | + | then coins_0 | |
652 | + | else if ((j == 1)) | |
653 | + | then coins_1 | |
654 | + | else if ((j == 2)) | |
655 | + | then coins_2 | |
656 | + | else throw("unknown token out") | |
657 | + | let xp = _xp_mem(balances) | |
658 | + | let xp_i = x_i(xp, i) | |
659 | + | let xp_j = x_i(xp, j) | |
660 | + | let x = (xp_i + dx) | |
661 | + | let $t01904019075 = get_y(i, j, x, xp) | |
662 | + | let y = $t01904019075._1 | |
663 | + | let D = $t01904019075._2 | |
664 | + | let _dy = ((xp_j - y) - 1) | |
665 | + | let dy_fee = fraction(_dy, fee, FEE_DENOMINATOR) | |
666 | + | let dy = (_dy - dy_fee) | |
667 | + | if (assert((dy >= min_dy))) | |
668 | + | then throw("Exchange resulted in fewer coins than expected") | |
669 | + | else { | |
670 | + | let dy_admin_fee = fraction(dy_fee, admin_fee, FEE_DENOMINATOR) | |
671 | + | func balanceProc (old_balance,_i) = if ((_i == i)) | |
672 | + | then (old_balance + dx) | |
673 | + | else if ((_i == j)) | |
674 | + | then ((old_balance - dy) - dy_admin_fee) | |
675 | + | else old_balance | |
676 | + | ||
677 | + | let final_balances_0 = balanceProc(balances_0, 0) | |
678 | + | let final_balances_1 = balanceProc(balances_1, 1) | |
679 | + | let final_balances_2 = balanceProc(balances_2, 2) | |
680 | + | $Tuple2((([IntegerEntry("balances_0", final_balances_0), IntegerEntry("balances_1", final_balances_1), IntegerEntry("balances_2", final_balances_2), ScriptTransfer(msg.caller, dy, tokenOut)] ++ (if (if (is_auto_fees) | |
681 | + | then (dy_admin_fee > 0) | |
682 | + | else false) | |
683 | + | then [ScriptTransfer(owner, dy_admin_fee, tokenOut)] | |
684 | + | else nil)) ++ log_data(D, dx)), dy) | |
685 | + | } | |
686 | + | } | |
687 | + | ||
688 | + | ||
689 | + | ||
690 | + | @Callable(msg) | |
691 | + | func remove_liquidity (min_amounts_0,min_amounts_1,min_amounts_2) = if ((size(msg.payments) != 1)) | |
692 | + | then throw("size( payments ) != 1") | |
693 | + | else { | |
694 | + | let payment = msg.payments[0] | |
695 | + | let tokenIn = payment.assetId | |
696 | + | if ((tokenIn != token)) | |
697 | + | then throw("unknown token") | |
698 | + | else { | |
699 | + | let _amount = payment.amount | |
700 | + | let total_supply = token_quantity | |
701 | + | let value_0 = fraction(balances_0, _amount, total_supply) | |
702 | + | let value_1 = fraction(balances_1, _amount, total_supply) | |
703 | + | let value_2 = fraction(balances_2, _amount, total_supply) | |
704 | + | if (assert(if (if ((value_0 >= min_amounts_0)) | |
705 | + | then (value_1 >= min_amounts_1) | |
706 | + | else false) | |
707 | + | then (value_2 >= min_amounts_2) | |
708 | + | else false)) | |
709 | + | then throw("Withdrawal resulted in fewer coins than expected") | |
710 | + | else { | |
711 | + | let final_balances_0 = (balances_0 - value_0) | |
712 | + | let final_balances_1 = (balances_1 - value_1) | |
713 | + | let final_balances_2 = (balances_2 - value_2) | |
714 | + | [IntegerEntry("balances_0", final_balances_0), IntegerEntry("balances_1", final_balances_1), IntegerEntry("balances_2", final_balances_2), ScriptTransfer(msg.caller, value_0, coins_0), ScriptTransfer(msg.caller, value_1, coins_1), ScriptTransfer(msg.caller, value_2, coins_2), Burn(token, _amount)] | |
715 | + | } | |
716 | + | } | |
717 | + | } | |
718 | + | ||
719 | + | ||
720 | + | ||
721 | + | @Callable(msg) | |
722 | + | func calc_withdraw_one_coin (_token_amount,i) = $Tuple2(nil, _calc_withdraw_one_coin(_token_amount, i)._1) | |
723 | + | ||
724 | + | ||
725 | + | ||
726 | + | @Callable(msg) | |
727 | + | func remove_liquidity_one_coin (i,min_amount) = if (assert(!(is_killed))) | |
728 | + | then throw("is killed") | |
729 | + | else if ((size(msg.payments) != 1)) | |
730 | + | then throw("size( payments ) != 1") | |
731 | + | else { | |
732 | + | let payment = msg.payments[0] | |
733 | + | let tokenIn = payment.assetId | |
734 | + | if ((tokenIn != token)) | |
735 | + | then throw("unknown token") | |
736 | + | else { | |
737 | + | let _token_amount = payment.amount | |
738 | + | let $t02203322100 = _calc_withdraw_one_coin(_token_amount, i) | |
739 | + | let dy = $t02203322100._1 | |
740 | + | let dy_fee = $t02203322100._2 | |
741 | + | let D = $t02203322100._3 | |
742 | + | if (assert((dy >= min_amount))) | |
743 | + | then throw("Not enough coins removed") | |
744 | + | else { | |
745 | + | let dy_admin_fee = fraction(dy_fee, admin_fee, FEE_DENOMINATOR) | |
746 | + | let dy_and_fee = (dy + dy_admin_fee) | |
747 | + | let $t02229922806 = if ((i == 0)) | |
748 | + | then $Tuple2([IntegerEntry("balances_0", (balances_0 - dy_and_fee)), ScriptTransfer(msg.caller, dy, coins_0)], coins_0) | |
749 | + | else if ((i == 1)) | |
750 | + | then $Tuple2([IntegerEntry("balances_1", (balances_1 - dy_and_fee)), ScriptTransfer(msg.caller, dy, coins_1)], coins_1) | |
751 | + | else if ((i == 2)) | |
752 | + | then $Tuple2([IntegerEntry("balances_2", (balances_2 - dy_and_fee)), ScriptTransfer(msg.caller, dy, coins_2)], coins_2) | |
753 | + | else throw("index out of N_COINS") | |
754 | + | let base_actions = $t02229922806._1 | |
755 | + | let tokenOut = $t02229922806._2 | |
756 | + | (((base_actions ++ [Burn(token, _token_amount)]) ++ (if (if (is_auto_fees) | |
757 | + | then (dy_admin_fee > 0) | |
758 | + | else false) | |
759 | + | then [ScriptTransfer(owner, dy_admin_fee, tokenOut)] | |
760 | + | else nil)) ++ log_data(D, fraction(dy_fee, FEE_DENOMINATOR, fee))) | |
761 | + | } | |
762 | + | } | |
763 | + | } | |
764 | + | ||
765 | + | ||
766 | + | ||
767 | + | @Callable(msg) | |
768 | + | func A () = $Tuple2(nil, _A()) | |
769 | + | ||
770 | + | ||
771 | + | ||
772 | + | @Callable(msg) | |
773 | + | func get_virtual_price () = { | |
774 | + | let D = get_D(_xp(), _A()) | |
775 | + | $Tuple2(nil, virtual_price(D)) | |
776 | + | } | |
777 | + | ||
778 | + | ||
779 | + | ||
780 | + | @Callable(msg) | |
781 | + | func calc_token_amount (amounts_0,amounts_1,amounts_2,deposit) = { | |
782 | + | let amp = _A() | |
783 | + | let D0 = get_D_mem(balances_0, balances_1, balances_2, amp) | |
784 | + | let new_balances_0 = (balances_0 + (if (deposit) | |
785 | + | then amounts_0 | |
786 | + | else -(amounts_0))) | |
787 | + | let new_balances_1 = (balances_1 + (if (deposit) | |
788 | + | then amounts_1 | |
789 | + | else -(amounts_1))) | |
790 | + | let new_balances_2 = (balances_2 + (if (deposit) | |
791 | + | then amounts_2 | |
792 | + | else -(amounts_2))) | |
793 | + | let D1 = get_D_mem(new_balances_0, new_balances_1, new_balances_2, amp) | |
794 | + | let token_amount = token_quantity | |
795 | + | let diff = if (deposit) | |
796 | + | then (D1 - D0) | |
797 | + | else (D0 - D1) | |
798 | + | $Tuple2(nil, fraction(diff, token_amount, D0)) | |
799 | + | } | |
800 | + | ||
801 | + | ||
802 | + | ||
803 | + | @Callable(msg) | |
804 | + | func ramp_A (_future_A,_future_time) = if (assert((msg.caller == owner))) | |
805 | + | then throw("only owner") | |
806 | + | else if (assert((block_timestamp >= (initial_A_time + MIN_RAMP_TIME)))) | |
807 | + | then throw("too often") | |
808 | + | else if (assert((_future_time >= (block_timestamp + MIN_RAMP_TIME)))) | |
809 | + | then throw("insufficient time") | |
810 | + | else { | |
811 | + | let _initial_A = _A() | |
812 | + | if (assert(if ((_future_A > 0)) | |
813 | + | then (MAX_A > _future_A) | |
814 | + | else false)) | |
815 | + | then throw("out of base range") | |
816 | + | else if (assert(if (if ((_future_A >= _initial_A)) | |
817 | + | then ((_initial_A * MAX_A_CHANGE) >= _future_A) | |
818 | + | else false) | |
819 | + | then true | |
820 | + | else if ((_initial_A > _future_A)) | |
821 | + | then ((_future_A * MAX_A_CHANGE) >= _initial_A) | |
822 | + | else false)) | |
823 | + | then throw("out of range") | |
824 | + | else [IntegerEntry("initial_A", _initial_A), IntegerEntry("future_A", _future_A), IntegerEntry("initial_A_time", block_timestamp), IntegerEntry("future_A_time", _future_time)] | |
825 | + | } | |
826 | + | ||
827 | + | ||
828 | + | ||
829 | + | @Callable(msg) | |
830 | + | func stop_ramp_A () = if (assert((msg.caller == owner))) | |
831 | + | then throw("only owner") | |
832 | + | else { | |
833 | + | let current_A = _A() | |
834 | + | [IntegerEntry("initial_A", current_A), IntegerEntry("future_A", current_A), IntegerEntry("initial_A_time", block_timestamp), IntegerEntry("future_A_time", block_timestamp)] | |
835 | + | } | |
836 | + | ||
837 | + | ||
838 | + | ||
839 | + | @Callable(msg) | |
840 | + | func commit_new_fee (new_fee,new_admin_fee) = if (assert((msg.caller == owner))) | |
841 | + | then throw("only owner") | |
842 | + | else if (assert((admin_actions_deadline == 0))) | |
843 | + | then throw("active action") | |
844 | + | else if (assert((MAX_FEE >= new_fee))) | |
845 | + | then throw("fee exceeds maximum") | |
846 | + | else if (assert((MAX_ADMIN_FEE >= new_admin_fee))) | |
847 | + | then throw("admin fee exceeds maximum") | |
848 | + | else { | |
849 | + | let _deadline = (block_timestamp + ADMIN_ACTIONS_DELAY) | |
850 | + | [IntegerEntry("admin_actions_deadline", _deadline), IntegerEntry("future_fee", new_fee), IntegerEntry("future_admin_fee", new_admin_fee)] | |
851 | + | } | |
852 | + | ||
853 | + | ||
854 | + | ||
855 | + | @Callable(msg) | |
856 | + | func apply_new_fee () = if (assert((msg.caller == owner))) | |
857 | + | then throw("only owner") | |
858 | + | else if (assert((block_timestamp >= admin_actions_deadline))) | |
859 | + | then throw("insufficient time") | |
860 | + | else if (assert((admin_actions_deadline != 0))) | |
861 | + | then throw("no active action") | |
862 | + | else [IntegerEntry("admin_actions_deadline", 0), IntegerEntry("fee", future_fee), IntegerEntry("admin_fee", future_admin_fee)] | |
863 | + | ||
864 | + | ||
865 | + | ||
866 | + | @Callable(msg) | |
867 | + | func revert_new_parameters () = if (assert((msg.caller == owner))) | |
868 | + | then throw("only owner") | |
869 | + | else [IntegerEntry("admin_actions_deadline", 0)] | |
870 | + | ||
871 | + | ||
872 | + | ||
873 | + | @Callable(msg) | |
874 | + | func commit_transfer_ownership (_owner) = if (assert((msg.caller == owner))) | |
875 | + | then throw("only owner") | |
876 | + | else if (assert((transfer_ownership_deadline == 0))) | |
877 | + | then throw("active transfer") | |
878 | + | else { | |
879 | + | let _deadline = (block_timestamp + ADMIN_ACTIONS_DELAY) | |
880 | + | [IntegerEntry("transfer_ownership_deadline", _deadline), StringEntry("future_owner", checkAddress(_owner))] | |
881 | + | } | |
882 | + | ||
883 | + | ||
884 | + | ||
885 | + | @Callable(msg) | |
886 | + | func apply_transfer_ownership () = if (assert((msg.caller == owner))) | |
887 | + | then throw("only owner") | |
888 | + | else if (assert((block_timestamp >= transfer_ownership_deadline))) | |
889 | + | then throw("insufficient time") | |
890 | + | else if (assert((transfer_ownership_deadline != 0))) | |
891 | + | then throw("no active transfer") | |
892 | + | else [IntegerEntry("transfer_ownership_deadline", 0), StringEntry("owner", future_owner)] | |
893 | + | ||
894 | + | ||
895 | + | ||
896 | + | @Callable(msg) | |
897 | + | func revert_transfer_ownership () = if (assert((msg.caller == owner))) | |
898 | + | then throw("only owner") | |
899 | + | else [IntegerEntry("transfer_ownership_deadline", 0)] | |
900 | + | ||
901 | + | ||
902 | + | ||
903 | + | @Callable(msg) | |
904 | + | func admin_balances (i) = $Tuple2(nil, admin_balance(i)) | |
905 | + | ||
906 | + | ||
907 | + | ||
908 | + | @Callable(msg) | |
909 | + | func withdraw_admin_fees () = if (assert((msg.caller == owner))) | |
910 | + | then throw("only owner") | |
911 | + | else [ScriptTransfer(owner, admin_balance(0), coins_0), ScriptTransfer(owner, admin_balance(1), coins_1), ScriptTransfer(owner, admin_balance(2), coins_2)] | |
912 | + | ||
913 | + | ||
914 | + | ||
915 | + | @Callable(msg) | |
916 | + | func donate_admin_fees () = if (assert((msg.caller == owner))) | |
917 | + | then throw("only owner") | |
918 | + | else [IntegerEntry("balances_0", (admin_balance(0) + balances_0)), IntegerEntry("balances_1", (admin_balance(1) + balances_1)), IntegerEntry("balances_2", (admin_balance(2) + balances_2))] | |
919 | + | ||
920 | + | ||
921 | + | ||
922 | + | @Callable(msg) | |
923 | + | func set_auto_fees (_is_auto_fees) = if (assert((msg.caller == owner))) | |
924 | + | then throw("only owner") | |
925 | + | else [BooleanEntry("is_auto_fees", _is_auto_fees)] | |
926 | + | ||
927 | + | ||
928 | + | ||
929 | + | @Callable(msg) | |
930 | + | func kill_me () = if (assert((msg.caller == owner))) | |
931 | + | then throw("only owner") | |
932 | + | else if (assert((kill_deadline > block_timestamp))) | |
933 | + | then throw("deadline has passed") | |
934 | + | else [BooleanEntry("is_killed", true)] | |
935 | + | ||
936 | + | ||
937 | + | ||
938 | + | @Callable(msg) | |
939 | + | func unkill_me () = if (assert((msg.caller == owner))) | |
940 | + | then throw("only owner") | |
941 | + | else [BooleanEntry("is_killed", false)] | |
942 | + | ||
943 | + | ||
944 | + | ||
945 | + | @Callable(msg) | |
946 | + | func set_height_address (_heightAddress) = if ((msg.caller != owner)) | |
947 | + | then throw("Owner only") | |
948 | + | else [StringEntry("heightAddress", checkAddress(_heightAddress))] | |
949 | + | ||
950 | + | ||
951 | + | ||
952 | + | @Callable(i) | |
953 | + | func set_verifier (verifier) = if ((i.caller != this)) | |
954 | + | then throw("self call only") | |
955 | + | else { | |
956 | + | let addressOK = match addressFromString(verifier) { | |
957 | + | case a: Address => | |
958 | + | true | |
959 | + | case _ => | |
960 | + | false | |
961 | + | } | |
962 | + | if (!(addressOK)) | |
963 | + | then throw(("verifier wrong address " + verifier)) | |
964 | + | else if (isDefined(getString(this, "verifier"))) | |
965 | + | then throw("verifier already defined") | |
966 | + | else [StringEntry("verifier", verifier)] | |
967 | + | } | |
968 | + | ||
969 | + | ||
970 | + | @Verifier(tx) | |
971 | + | func verify () = match getString(this, "verifier") { | |
972 | + | case verifier: String => | |
973 | + | valueOrElse(getBoolean(addressFromStringValue(verifier), ((("status_" + toString(this)) + "_") + toBase58String(tx.id))), false) | |
974 | + | case _ => | |
975 | + | sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
976 | + | } | |
977 | + |
github/deemru/w8io/c3f4982 57.39 ms ◑