2023.09.02 15:31 [3803431] smart account 3P5Bfd58PPfNvBM2Hy8QfbcDqMeNtzg7KfP > SELF 0.00000000 Waves

{ "type": 13, "id": "7LR9Pr52xmmaZu6o3EnnbbGb2z7jQ4krESZFp9Dpyx8J", "fee": 2200000, "feeAssetId": null, "timestamp": 1693662190390, "version": 2, "chainId": 87, "sender": "3P5Bfd58PPfNvBM2Hy8QfbcDqMeNtzg7KfP", "senderPublicKey": "GqXuX2WHNr3WUqTaeH2YCySFY45NAJoE9RmY9bEWkzh", "proofs": [ "aMNF2h4u8tMSXHRQrWJvnJpnkmoQ7SVLpC7iuRu3JgF92QANfcvAUaLGL5mMG7j4zfkKVQv68ftiMNLKha7ifST", "2KtZR82RQcvm5MkNpsM2oZnFUQuS1Rvc2Brcv8BxX7oa71yHc2ZzrvCBNB6vrDRHqvatQt7Msf7im8KK4Zffukhs", "", "5N7QoQYeeVUouXN8jvBoiV5ckFiX3VkTs5rXXEhCVq6Jd8L2eFVDuiFZY7PvjebFHosQKVRU7DPMrLTYGTZfHL9p" ], "script": "base64:BgIvCAISEQoPCAgICAgICAgICAgICAgIEgMKAQgSDAoKAQIBAgECAQIBAhIFCgMBAQg/AAtyZXZpc2lvbk51bQIAAANTRVACAl9fAQhhc1N0cmluZwEBdgQHJG1hdGNoMAUBdgMJAAECBQckbWF0Y2gwAgZTdHJpbmcEAXMFByRtYXRjaDAFAXMJAAIBAhhmYWlsIHRvIGNhc3QgaW50byBTdHJpbmcBBWFzSW50AQF2BAckbWF0Y2gwBQF2AwkAAQIFByRtYXRjaDACA0ludAQBaQUHJG1hdGNoMAUBaQkAAgECFWZhaWwgdG8gY2FzdCBpbnRvIEludAAZSWR4Q29udHJvbENmZ05ldXRyaW5vRGFwcAABABhJZHhDb250cm9sQ2ZnQXVjdGlvbkRhcHAAAgAUSWR4Q29udHJvbENmZ1JwZERhcHAAAwAVSWR4Q29udHJvbENmZ01hdGhEYXBwAAQAHElkeENvbnRyb2xDZmdMaXF1aWRhdGlvbkRhcHAABQAVSWR4Q29udHJvbENmZ1Jlc3REYXBwAAYAHUlkeENvbnRyb2xDZmdOb2RlUmVnaXN0cnlEYXBwAAcAHElkeENvbnRyb2xDZmdOc2J0U3Rha2luZ0RhcHAACAAZSWR4Q29udHJvbENmZ01lZGlhdG9yRGFwcAAJABxJZHhDb250cm9sQ2ZnU3VyZlN0YWtpbmdEYXBwAAoAIElkeENvbnRyb2xDZmdHbnNidENvbnRyb2xsZXJEYXBwAAsAF0lkeENvbnRyb2xDZmdSZXN0VjJEYXBwAAwAG0lkeENvbnRyb2xDZmdHb3Zlcm5hbmNlRGFwcAANABxJZHhDb250cm9sQ2ZnUGVnUHJvdmlkZXJEYXBwAA4BEGtleUNvbnRyb2xDb25maWcAAhElc19fY29udHJvbENvbmZpZwEPa2V5UHJpY2VCeUFzc2V0AQphc3NldElkU3RyCQC5CQIJAMwIAgIcJXMlcyVzX19jb21tb25fX3ByaWNlQnlBc3NldAkAzAgCBQphc3NldElkU3RyBQNuaWwFA1NFUAEVa2V5QmxhY2tTd2FuVGhyZXNob2xkAAInJXMlc19fY29udHJvbENvbmZpZ19fYmxhY2tTd2FuVGhyZXNob2xkAQ5kYXRhQ29udHJvbENmZw8QbmV1dHJpbm9Db250cmFjdA9hdWN0aW9uQ29udHJhY3QLcnBkQ29udHJhY3QMbWF0aENvbnRyYWN0E2xpcXVpZGF0aW9uQ29udHJhY3QMcmVzdENvbnRyYWN0FG5vZGVSZWdpc3RyeUNvbnRyYWN0E25zYnRTdGFraW5nQ29udHJhY3QQbWVkaWF0b3JDb250cmFjdBNzdXJmU3Rha2luZ0NvbnRyYWN0F2duc2J0Q29udHJvbGxlckNvbnRyYWN0DnJlc3RWMkNvbnRyYWN0EmdvdmVybmFuY2VDb250cmFjdAxkb3JhQ29udHJhY3QTcG9vbHNGYWNhZGVDb250cmFjdAkAugkCCQDMCAICHiVzJXMlcyVzJXMlcyVzJXMlcyVzJXMlcyVzJXMlcwkAzAgCBRBuZXV0cmlub0NvbnRyYWN0CQDMCAIFD2F1Y3Rpb25Db250cmFjdAkAzAgCBQtycGRDb250cmFjdAkAzAgCBQxtYXRoQ29udHJhY3QJAMwIAgUTbGlxdWlkYXRpb25Db250cmFjdAkAzAgCBQxyZXN0Q29udHJhY3QJAMwIAgUUbm9kZVJlZ2lzdHJ5Q29udHJhY3QJAMwIAgUTbnNidFN0YWtpbmdDb250cmFjdAkAzAgCBRBtZWRpYXRvckNvbnRyYWN0CQDMCAIFE3N1cmZTdGFraW5nQ29udHJhY3QJAMwIAgUXZ25zYnRDb250cm9sbGVyQ29udHJhY3QJAMwIAgUOcmVzdFYyQ29udHJhY3QJAMwIAgUSZ292ZXJuYW5jZUNvbnRyYWN0CQDMCAIFDGRvcmFDb250cmFjdAkAzAgCBRNwb29sc0ZhY2FkZUNvbnRyYWN0BQNuaWwFA1NFUAEWY29udmVydFdhdmVzVG9OZXV0cmlubwIGYW1vdW50BXByaWNlCQBrAwkAawMFBmFtb3VudAUFcHJpY2UAwIQ9AMCEPQCAwtcvARhnZXROdW1iZXJCeUFkZHJlc3NBbmRLZXkCB2FkZHJlc3MDa2V5CQELdmFsdWVPckVsc2UCCQCaCAIJARFAZXh0ck5hdGl2ZSgxMDYyKQEFB2FkZHJlc3MFA2tleQAAARhnZXRTdHJpbmdCeUFkZHJlc3NBbmRLZXkCB2FkZHJlc3MDa2V5CQELdmFsdWVPckVsc2UCCQCdCAIFB2FkZHJlc3MFA2tleQIAARZjb252ZXJ0SnNvbkFycmF5VG9MaXN0AQlqc29uQXJyYXkJALUJAgUJanNvbkFycmF5AgEsAAVwcmljZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzAgVwcmljZQAAAApwcmljZUluZGV4CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMCC3ByaWNlX2luZGV4AAAACWlzQmxvY2tlZAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzAgppc19ibG9ja2VkBwAScGVyY2VudFByaWNlT2Zmc2V0CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJARVrZXlCbGFja1N3YW5UaHJlc2hvbGQAAAcADXB1YktleU9yYWNsZXMJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwIHb3JhY2xlcwIAABFwdWJLZXlPcmFjbGVzTGlzdAkBFmNvbnZlcnRKc29uQXJyYXlUb0xpc3QBBQ1wdWJLZXlPcmFjbGVzAA9uZXV0cmlub0FkZHJlc3MJAQdBZGRyZXNzAQEaAVdwBGKmR5vprVZolMvvhYwwgiAomggUlrIAEmxpcXVpZGF0aW9uQWRkcmVzcwkBB0FkZHJlc3MBARoBVxrqScv6Cn7KCH8o1Z+Vjgx7abWpOJEfSAANbmV1dHJpbm9Bc3NldAEgtiYpwwT1zlORpA5LdSQvZIxRsfrfr1QpvUjSHSqyqtEAD2NvbnRyb2xDb250cmFjdAUEdGhpcwEPZ2V0U3RyaW5nT3JGYWlsAgdhZGRyZXNzA2tleQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCdCAIFB2FkZHJlc3MFA2tleQkAuQkCCQDMCAICCm1hbmRhdG9yeSAJAMwIAgkApQgBBQdhZGRyZXNzCQDMCAICAS4JAMwIAgUDa2V5CQDMCAICDyBpcyBub3QgZGVmaW5lZAUDbmlsAgABGGdldENvbnRyYWN0QWRkcmVzc09yRmFpbAIKY29udHJvbENmZwNpZHgJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQCRAwIFCmNvbnRyb2xDZmcFA2lkeAkArAICAi1Db250cm9sIGNmZyBkb2Vzbid0IGNvbnRhaW4gYWRkcmVzcyBhdCBpbmRleCAJAKQDAQUDaWR4AApjb250cm9sQ2ZnCQC8CQIJAQ9nZXRTdHJpbmdPckZhaWwCBQR0aGlzCQEQa2V5Q29udHJvbENvbmZpZwAFA1NFUAASZ292ZXJuYW5jZUNvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUbSWR4Q29udHJvbENmZ0dvdmVybmFuY2VEYXBwABVuZXV0cmlub0xvY2tlZEJhbGFuY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUPbmV1dHJpbm9BZGRyZXNzAhViYWxhbmNlX2xvY2tfbmV1dHJpbm8AAAASd2F2ZXNMb2NrZWRCYWxhbmNlCQELdmFsdWVPckVsc2UCCQCaCAIFD25ldXRyaW5vQWRkcmVzcwISYmFsYW5jZV9sb2NrX3dhdmVzAAAAB3Jlc2VydmUJAGUCCAkA7wcBBQ9uZXV0cmlub0FkZHJlc3MHcmVndWxhcgUSd2F2ZXNMb2NrZWRCYWxhbmNlAA5uZXV0cmlub1N1cHBseQkAZQIJAGUCCQBkAgUVbmV1dHJpbm9Mb2NrZWRCYWxhbmNlCAkBBXZhbHVlAQkA7AcBBQ1uZXV0cmlub0Fzc2V0CHF1YW50aXR5CQDwBwIFD25ldXRyaW5vQWRkcmVzcwUNbmV1dHJpbm9Bc3NldAkA8AcCBRJsaXF1aWRhdGlvbkFkZHJlc3MFDW5ldXRyaW5vQXNzZXQAB2RlZmljaXQJAGUCBQ5uZXV0cmlub1N1cHBseQkBFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8CBQdyZXNlcnZlBQVwcmljZQAOZG9yYUFkZHJlc3NTdHICIzNQS2tvaktkZDZCQnpUZjFSWGJRVmZVRHJhTkZYWEhLelFGAAtkb3JhQWRkcmVzcwkBEUBleHRyTmF0aXZlKDEwNjIpAQUOZG9yYUFkZHJlc3NTdHIADXh0blVzZFByaWNlWDYJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQtkb3JhQWRkcmVzcwIWJXMlc19fcHJpY2VfX1VTRE4tVVNEVAIkRE9SQSBkb2Vzbid0IGNvbnRhaW4gVVNETi9VU0RUIHByaWNlABF3eF9yZXN0QWRkcmVzc1N0cgIjM1A4TW9QbnNhdXJvZmsxVnloc2RBRmtlUTZpanBKWVhDcFcAC3d4X3Jlc3REYXBwCQERQGV4dHJOYXRpdmUoMTA2MikBBRF3eF9yZXN0QWRkcmVzc1N0cgAMc3dvcFJlc3REYXBwCQERQGV4dHJOYXRpdmUoMTA2MikBAiMzUDU2ak5RekVDWG5yV3BuYmJTSkt3N0Vvb282ZmtVYU1QcAAFeHRuSWQFDW5ldXRyaW5vQXNzZXQACHh0bklkU3RyCQDYBAEFBXh0bklkAAd3eElkU3RyAixBdHF2NTlFWXpqRkd1aXRLVm5NUms2SDhGdWtqb1Yza3RQb3JiRXlzMjVvbgAEd3hJZAkA2QQBBQd3eElkU3RyAAp2aXJlc0lkU3RyAixEU2JiaExzU1RlRGc1THNpdWZrMkFuZWgzRGpWcUp1UHIyTTl1VTFnd3k1cAAHdmlyZXNJZAkA2QQBBQp2aXJlc0lkU3RyARFmaW5kUHJpY2VzSW5SYW5nZQEGcHJpY2VzBA9taW5QZXJjZW50Qm91bmQAWgQPbWF4UGVyY2VudEJvdW5kAG4EAnAwCQCRAwIFBnByaWNlcwAABAZjaGVjazADCQBnAgAACQCRAwIFBnByaWNlcwAACQDMCAIAAAUDbmlsBANwMDEJAGkCCQBoAgkAkQMCBQZwcmljZXMAAQBkBQJwMAQDcDAyCQBpAgkAaAIJAJEDAgUGcHJpY2VzAAIAZAUCcDAEA3AwMwkAaQIJAGgCCQCRAwIFBnByaWNlcwADAGQFAnAwBANwMDQJAGkCCQBoAgkAkQMCBQZwcmljZXMABABkBQJwMAQGYXJyYXkxAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AwMQkAZgIFA3AwMQUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAEJAMwIAgAABQNuaWwJAMwIAgAABQNuaWwEBmFycmF5MgMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwMDIJAGYCBQNwMDIFD21pblBlcmNlbnRCb3VuZAcJAMwIAgACBQZhcnJheTEFBmFycmF5MQQGYXJyYXkzAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AwMwkAZgIFA3AwMwUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAMFBmFycmF5MgUGYXJyYXkyAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AwNAkAZgIFA3AwNAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAQFBmFycmF5MwUGYXJyYXkzBAZjaGVjazEDCQBnAgkAkAMBBQZjaGVjazAAAwUGY2hlY2swBAJwMQkAkQMCBQZwcmljZXMAAQMJAGcCAAAFAnAxCQDMCAIAAQUDbmlsBANwMTAJAGkCCQBoAgkAkQMCBQZwcmljZXMAAABkBQJwMQQDcDEyCQBpAgkAaAIJAJEDAgUGcHJpY2VzAAIAZAUCcDEEA3AxMwkAaQIJAGgCCQCRAwIFBnByaWNlcwADAGQFAnAxBANwMTQJAGkCCQBoAgkAkQMCBQZwcmljZXMABABkBQJwMQQGYXJyYXkxAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AxMAkAZgIFA3AxMAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAAJAMwIAgABBQNuaWwJAMwIAgABBQNuaWwEBmFycmF5MgMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwMTIJAGYCBQNwMTIFD21pblBlcmNlbnRCb3VuZAcJAMwIAgACBQZhcnJheTEFBmFycmF5MQQGYXJyYXkzAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AxMwkAZgIFA3AxMwUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAMFBmFycmF5MgUGYXJyYXkyAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AxNAkAZgIFA3AxNAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAQFBmFycmF5MwUGYXJyYXkzBAZjaGVjazIDCQBnAgkAkAMBBQZjaGVjazEAAwUGY2hlY2sxBAJwMgkAkQMCBQZwcmljZXMAAgMJAGcCAAAFAnAyCQDMCAIAAgUDbmlsBANwMjAJAGkCCQBoAgkAkQMCBQZwcmljZXMAAABkBQJwMgQDcDIxCQBpAgkAaAIJAJEDAgUGcHJpY2VzAAEAZAUCcDIEA3AyMwkAaQIJAGgCCQCRAwIFBnByaWNlcwADAGQFAnAyBANwMjQJAGkCCQBoAgkAkQMCBQZwcmljZXMABABkBQJwMgQGYXJyYXkxAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AyMAkAZgIFA3AyMAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAAJAMwIAgACBQNuaWwJAMwIAgACBQNuaWwEBmFycmF5MgMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwMjEJAGYCBQNwMjEFD21pblBlcmNlbnRCb3VuZAcJAMwIAgABBQZhcnJheTEFBmFycmF5MQQGYXJyYXkzAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AyMwkAZgIFA3AyMwUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAMFBmFycmF5MgUGYXJyYXkyAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AyNAkAZgIFA3AyNAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAQFBmFycmF5MwUGYXJyYXkzBAZjaGVjazMDCQBnAgkAkAMBBQZjaGVjazIAAwUGY2hlY2syBAJwMwkAkQMCBQZwcmljZXMAAwMJAGcCAAAFAnAzCQDMCAIAAwUDbmlsBANwMzAJAGkCCQBoAgkAkQMCBQZwcmljZXMAAABkBQJwMwQDcDMxCQBpAgkAaAIJAJEDAgUGcHJpY2VzAAEAZAUCcDMEA3AzMgkAaQIJAGgCCQCRAwIFBnByaWNlcwACAGQFAnAzBANwMzQJAGkCCQBoAgkAkQMCBQZwcmljZXMABABkBQJwMwQGYXJyYXkxAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AzMAkAZgIFA3AzMAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAAJAMwIAgADBQNuaWwJAMwIAgADBQNuaWwEBmFycmF5MgMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwMzEJAGYCBQNwMzEFD21pblBlcmNlbnRCb3VuZAcJAMwIAgABBQZhcnJheTEFBmFycmF5MQQGYXJyYXkzAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AzMgkAZgIFA3AzMgUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAIFBmFycmF5MgUGYXJyYXkyAwMJAGYCBQ9tYXhQZXJjZW50Qm91bmQFA3AzNAkAZgIFA3AzNAUPbWluUGVyY2VudEJvdW5kBwkAzAgCAAQFBmFycmF5MwUGYXJyYXkzAwkAZwIJAJADAQUGY2hlY2szAAMFBmNoZWNrMwQCcDQJAJEDAgUGcHJpY2VzAAQDCQBnAgAABQJwNAkAzAgCAAQFA25pbAQDcDQwCQBpAgkAaAIJAJEDAgUGcHJpY2VzAAAAZAUCcDQEA3A0MQkAaQIJAGgCCQCRAwIFBnByaWNlcwABAGQFAnA0BANwNDIJAGkCCQBoAgkAkQMCBQZwcmljZXMAAgBkBQJwNAQDcDQzCQBpAgkAaAIJAJEDAgUGcHJpY2VzAAMAZAUCcDQEBmFycmF5MQMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwNDAJAGYCBQNwNDAFD21pblBlcmNlbnRCb3VuZAcJAMwIAgAACQDMCAIABAUDbmlsCQDMCAIABAUDbmlsBAZhcnJheTIDAwkAZgIFD21heFBlcmNlbnRCb3VuZAUDcDQxCQBmAgUDcDQxBQ9taW5QZXJjZW50Qm91bmQHCQDMCAIAAQUGYXJyYXkxBQZhcnJheTEEBmFycmF5MwMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwNDIJAGYCBQNwNDIFD21pblBlcmNlbnRCb3VuZAcJAMwIAgACBQZhcnJheTIFBmFycmF5MgMDCQBmAgUPbWF4UGVyY2VudEJvdW5kBQNwNDMJAGYCBQNwNDMFD21pblBlcmNlbnRCb3VuZAcJAMwIAgADBQZhcnJheTMFBmFycmF5MwESZm9ybWF0dGluZ1ByaWNlTXNnAQVwcmljZQkAmwMBCQCsAgIJAKwCAgkArAICCQCsAgICE1dBVkVTTkVVVFJJTk9QUkVGSVgCAV8JAKQDAQUGaGVpZ2h0AgFfCQCkAwEFBXByaWNlAQpQcmljZUVudHJ5AwhuZXdQcmljZQphc3NldElkU3RyDW5ld1ByaWNlSW5kZXgECWhlaWdodFN0cgkApAMBBQZoZWlnaHQEEG5ld1ByaWNlSW5kZXhTdHIJAKQDAQUNbmV3UHJpY2VJbmRleAQQcHJpY2VCeUhlaWdodEtFWQkAuQkCCQDMCAICHyVzJXMlcyVkX19jb21tb25fX3ByaWNlQnlIZWlnaHQJAMwIAgUKYXNzZXRJZFN0cgkAzAgCBQloZWlnaHRTdHIFA25pbAUDU0VQBA9wcmljZUJ5SW5kZXhLRVkJALkJAgkAzAgCAh4lcyVzJXMlZF9fY29tbW9uX19wcmljZUJ5SW5kZXgJAMwIAgUKYXNzZXRJZFN0cgkAzAgCBRBuZXdQcmljZUluZGV4U3RyBQNuaWwFA1NFUAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBD2tleVByaWNlQnlBc3NldAEFCmFzc2V0SWRTdHIFCG5ld1ByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ9wcmljZUJ5SW5kZXhLRVkFCG5ld1ByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRBwcmljZUJ5SGVpZ2h0S0VZBQhuZXdQcmljZQUDbmlsARdmaW5hbGl6ZVZpcmVzVXNkT25jaGFpbgEPd2F2ZXNVc2RQcmljZVg2BBB3eF9WaXJlc1dhdmVzX2xwAixFM1loVVJOc0NtalVhWU5pZ0g2QWF5THhiaDl1ODF1VkdVM1p2SEt6blk1dgQMd3hSZXN0UmVzdWx0CQC1CQIJAQhhc1N0cmluZwEJAPwHBAULd3hfcmVzdERhcHACEXBvb2xTdGF0c1JFQURPTkxZCQDMCAIFEHd4X1ZpcmVzV2F2ZXNfbHAFA25pbAUDbmlsAgJfXwQVd3hfVmlyZXNXYXZlc19XQVZFU3g4CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUMd3hSZXN0UmVzdWx0AAIEFXd4X1ZpcmVzV2F2ZXNfVklSRVN4OAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFDHd4UmVzdFJlc3VsdAABBB13eF9WaXJlc1dhdmVzX1ZpcmVzVXNkUHJpY2VYNgkAawMFFXd4X1ZpcmVzV2F2ZXNfV0FWRVN4OAUPd2F2ZXNVc2RQcmljZVg2BRV3eF9WaXJlc1dhdmVzX1ZJUkVTeDgEFHd4X1ZpcmVzV2F2ZXNfV2VpZ2h0BRV3eF9WaXJlc1dhdmVzX1ZJUkVTeDgEFnd4X1ZpcmVzWHRuX0FkZHJlc3NTdHICIzNQNUhqUG93Z2lwaVZCM1V2WHBoRGlQdktzNDI0V0M5eFF3BBN3eF9WaXJlc1h0bl9BZGRyZXNzCQERQGV4dHJOYXRpdmUoMTA2MikBBRZ3eF9WaXJlc1h0bl9BZGRyZXNzU3RyBBF3eF9WaXJlc1h0bl9YVE54NgkA8AcCBRN3eF9WaXJlc1h0bl9BZGRyZXNzBQV4dG5JZAQTd3hfVmlyZXNYdG5fVklSRVN4OAkA8AcCBRN3eF9WaXJlc1h0bl9BZGRyZXNzBQd2aXJlc0lkBBt3eF9WaXJlc1h0bl9WaXJlc1VzZFByaWNlWDYJAGsDCQBoAgURd3hfVmlyZXNYdG5fWFROeDYAZAUNeHRuVXNkUHJpY2VYNgUTd3hfVmlyZXNYdG5fVklSRVN4OAQSd3hfVmlyZXNYdG5fV2VpZ2h0BRN3eF9WaXJlc1h0bl9WSVJFU3g4BBhzd29wX1ZpcmVzWHRuX0FkZHJlc3NTdHICIzNQSjQ4UDNwMnd2V1VqZ1FhUWlaMmNGYnI4cW14TW9rQkdkBBVzd29wX1ZpcmVzWHRuX0FkZHJlc3MJARFAZXh0ck5hdGl2ZSgxMDYyKQEFGHN3b3BfVmlyZXNYdG5fQWRkcmVzc1N0cgQdc3dvcF9WaXJlc1h0bl9WaXJlc1h0blByaWNlWDYJAQVhc0ludAEJAPwHBAUMc3dvcFJlc3REYXBwAhFjYWxjR2V0QW1vdW50Q1BNTQkAzAgCBRhzd29wX1ZpcmVzWHRuX0FkZHJlc3NTdHIJAMwIAgIjM1A4OHFrMUt6RjFCS2pEN2ZDN0xqTlZBS000ZXpmZjVXRTYJAMwIAgUKdmlyZXNJZFN0cgkAzAgCAIDC1y8FA25pbAUDbmlsBB1zd29wX1ZpcmVzWHRuX1ZpcmVzVXNkUHJpY2VYNgkAawMFHXN3b3BfVmlyZXNYdG5fVmlyZXNYdG5QcmljZVg2BQ14dG5Vc2RQcmljZVg2AMCEPQQUc3dvcF9WaXJlc1h0bl9XZWlnaHQJARFAZXh0ck5hdGl2ZSgxMDUwKQIFFXN3b3BfVmlyZXNYdG5fQWRkcmVzcwIPQV9hc3NldF9iYWxhbmNlBAFXCQBkAgkAZAIFFHd4X1ZpcmVzV2F2ZXNfV2VpZ2h0BRJ3eF9WaXJlc1h0bl9XZWlnaHQFFHN3b3BfVmlyZXNYdG5fV2VpZ2h0BA92aXJlc1VzZFByaWNlWDYJAGQCCQBkAgkAawMFHXd4X1ZpcmVzV2F2ZXNfVmlyZXNVc2RQcmljZVg2BRR3eF9WaXJlc1dhdmVzX1dlaWdodAUBVwkAawMFG3d4X1ZpcmVzWHRuX1ZpcmVzVXNkUHJpY2VYNgUSd3hfVmlyZXNYdG5fV2VpZ2h0BQFXCQBrAwUdc3dvcF9WaXJlc1h0bl9WaXJlc1VzZFByaWNlWDYFFHN3b3BfVmlyZXNYdG5fV2VpZ2h0BQFXBAVkZWJ1ZwkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAg54dG5Vc2RQcmljZVg2PQkApAMBBQ14dG5Vc2RQcmljZVg2Ah8gd3hfVmlyZXNXYXZlc19WaXJlc1VzZFByaWNlWDY9CQCkAwEFHXd4X1ZpcmVzV2F2ZXNfVmlyZXNVc2RQcmljZVg2Ah0gd3hfVmlyZXNYdG5fVmlyZXNVc2RQcmljZVg2PQkApAMBBRt3eF9WaXJlc1h0bl9WaXJlc1VzZFByaWNlWDYCHyBzd29wX1ZpcmVzWHRuX1ZpcmVzVXNkUHJpY2VYNj0JAKQDAQUdc3dvcF9WaXJlc1h0bl9WaXJlc1VzZFByaWNlWDYJAJQKAgUPdmlyZXNVc2RQcmljZVg2BQVkZWJ1ZwEUZmluYWxpemVXeFVzZE9uY2hhaW4BD3dhdmVzVXNkUHJpY2VYNgQNd3hfV3hXYXZlc19scAIsQmlTekZlOG5TTDc4b1phZWJmb2luNXZCWjVQemU2ZDdrYWVpakxxcjV4WmUEDHd4UmVzdFJlc3VsdAkAtQkCCQEIYXNTdHJpbmcBCQD8BwQFC3d4X3Jlc3REYXBwAhFwb29sU3RhdHNSRUFET05MWQkAzAgCBQ13eF9XeFdhdmVzX2xwBQNuaWwFA25pbAICX18EEXd4X1d4V2F2ZXNfV2VpZ2h0ANCGAwQSd3hfV3hXYXZlc19XQVZFU3g4CQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUMd3hSZXN0UmVzdWx0AAIED3d4X1d4V2F2ZXNfV1h4OAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFDHd4UmVzdFJlc3VsdAABBBd3eF9XeFdhdmVzX1d4VXNkUHJpY2VYNgkAawMFEnd4X1d4V2F2ZXNfV0FWRVN4OAUPd2F2ZXNVc2RQcmljZVg2BQ93eF9XeFdhdmVzX1dYeDgEE3d4X1d4WHRuX0FkZHJlc3NTdHICIzNQQ0VOcEVLZThhdHdFTFo3b0NTbWNkRWZjUnVLVHJVeDk5BBB3eF9XeFh0bl9BZGRyZXNzCQERQGV4dHJOYXRpdmUoMTA2MikBBRN3eF9XeFh0bl9BZGRyZXNzU3RyBA93eF9XeFh0bl9XZWlnaHQA0IYDBA53eF9XeFh0bl9YVE54NgkA8AcCBRB3eF9XeFh0bl9BZGRyZXNzBQV4dG5JZAQNd3hfV3hYdG5fV1h4OAkA8AcCBRB3eF9XeFh0bl9BZGRyZXNzBQR3eElkBBV3eF9XeFh0bl9XeFVzZFByaWNlWDYJAGsDCQBoAgUOd3hfV3hYdG5fWFROeDYAZAUNeHRuVXNkUHJpY2VYNgUNd3hfV3hYdG5fV1h4OAQVc3dvcF9XeFh0bl9BZGRyZXNzU3RyAiMzUEtpNEczVlgyazQyWlNtTk5ybXZnZERIN0p6UmFVaFk3UgQRc3dvcF9XeFh0bl9XZWlnaHQAiCcEF3N3b3BfV3hYdG5fV3hYdG5QcmljZVg2CQEFYXNJbnQBCQD8BwQFDHN3b3BSZXN0RGFwcAIRY2FsY0dldEFtb3VudENQTU0JAMwIAgUVc3dvcF9XeFh0bl9BZGRyZXNzU3RyCQDMCAICIzNQODhxazFLekYxQktqRDdmQzdMak5WQUtNNGV6ZmY1V0U2CQDMCAIFB3d4SWRTdHIJAMwIAgCAwtcvBQNuaWwFA25pbAQXc3dvcF9XeFh0bl9XeFVzZFByaWNlWDYJAGsDBRdzd29wX1d4WHRuX1d4WHRuUHJpY2VYNgUNeHRuVXNkUHJpY2VYNgDAhD0EAVcJAGQCCQBkAgURd3hfV3hXYXZlc19XZWlnaHQFD3d4X1d4WHRuX1dlaWdodAURc3dvcF9XeFh0bl9XZWlnaHQEDHd4VXNkUHJpY2VYNgkAZAIJAGQCCQBrAwUXd3hfV3hXYXZlc19XeFVzZFByaWNlWDYFEXd4X1d4V2F2ZXNfV2VpZ2h0BQFXCQBrAwUVd3hfV3hYdG5fV3hVc2RQcmljZVg2BQ93eF9XeFh0bl9XZWlnaHQFAVcJAGsDBRdzd29wX1d4WHRuX1d4VXNkUHJpY2VYNgURc3dvcF9XeFh0bl9XZWlnaHQFAVcEBWRlYnVnCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICDnh0blVzZFByaWNlWDY9CQCkAwEFDXh0blVzZFByaWNlWDYCGSB3eF9XeFdhdmVzX1d4VXNkUHJpY2VYNj0JAKQDAQUXd3hfV3hXYXZlc19XeFVzZFByaWNlWDYCFyB3eF9XeFh0bl9XeFVzZFByaWNlWDY9CQCkAwEFFXd4X1d4WHRuX1d4VXNkUHJpY2VYNgIZIHN3b3BfV3hYdG5fV3hVc2RQcmljZVg2PQkApAMBBRdzd29wX1d4WHRuX1d4VXNkUHJpY2VYNgkAlAoCBQx3eFVzZFByaWNlWDYFBWRlYnVnAQtjaGVja1ByaWNlcwMNbmV3V2F2ZXNQcmljZQpuZXdXeFByaWNlDW5ld1ZpcmVzUHJpY2UKAQtpc0JsYWNrU3dhbgIFY3VyUHIFbmV3UHIDCQBnAgUFbmV3UHIJAGQCBQVjdXJQcgkAaQIJAGgCBQVjdXJQcgUScGVyY2VudFByaWNlT2Zmc2V0AGQGCQBnAgkAZQIFBWN1clByCQBpAgkAaAIFBWN1clByBRJwZXJjZW50UHJpY2VPZmZzZXQAZAUFbmV3UHIEB3d4UHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJ8IAQkBD2tleVByaWNlQnlBc3NldAEFB3d4SWRTdHIAAAQKdmlyZXNQcmljZQkBC3ZhbHVlT3JFbHNlAgkAnwgBCQEPa2V5UHJpY2VCeUFzc2V0AQUKdmlyZXNJZFN0cgAAAwkBC2lzQmxhY2tTd2FuAgUFcHJpY2UFDW5ld1dhdmVzUHJpY2UJAJUKAwYCBVdBVkVTBQ1uZXdXYXZlc1ByaWNlAwkBC2lzQmxhY2tTd2FuAgUHd3hQcmljZQUKbmV3V3hQcmljZQkAlQoDBgUHd3hJZFN0cgUKbmV3V3hQcmljZQMJAQtpc0JsYWNrU3dhbgIFCnZpcmVzUHJpY2UFDW5ld1ZpcmVzUHJpY2UJAJUKAwYFCnZpcmVzSWRTdHIFDW5ld1ZpcmVzUHJpY2UJAJUKAwcCAAAABAFpAQ1jb25zdHJ1Y3RvclYxDxBuZXV0cmlub0NvbnRyYWN0D2F1Y3Rpb25Db250cmFjdAtycGRDb250cmFjdAxtYXRoQ29udHJhY3QTbGlxdWlkYXRpb25Db250cmFjdAxyZXN0Q29udHJhY3QUbm9kZVJlZ2lzdHJ5Q29udHJhY3QTbnNidFN0YWtpbmdDb250cmFjdBBtZWRpYXRvckNvbnRyYWN0E3N1cmZTdGFraW5nQ29udHJhY3QXZ25zYnRDb250cm9sbGVyQ29udHJhY3QOcmVzdFYyQ29udHJhY3QSZ292ZXJuYW5jZUNvbnRyYWN0DGRvcmFDb250cmFjdBNwb29sc0ZhY2FkZUNvbnRyYWN0AwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIScGVybWlzc2lvbnMgZGVuaWVkCQDMCAIJAQtTdHJpbmdFbnRyeQIJARBrZXlDb250cm9sQ29uZmlnAAkBDmRhdGFDb250cm9sQ2ZnDwUQbmV1dHJpbm9Db250cmFjdAUPYXVjdGlvbkNvbnRyYWN0BQtycGRDb250cmFjdAUMbWF0aENvbnRyYWN0BRNsaXF1aWRhdGlvbkNvbnRyYWN0BQxyZXN0Q29udHJhY3QFFG5vZGVSZWdpc3RyeUNvbnRyYWN0BRNuc2J0U3Rha2luZ0NvbnRyYWN0BRBtZWRpYXRvckNvbnRyYWN0BRNzdXJmU3Rha2luZ0NvbnRyYWN0BRdnbnNidENvbnRyb2xsZXJDb250cmFjdAUOcmVzdFYyQ29udHJhY3QFEmdvdmVybmFuY2VDb250cmFjdAUMZG9yYUNvbnRyYWN0BRNwb29sc0ZhY2FkZUNvbnRyYWN0BQNuaWwBaQEVY2FsbEVtZXJnZW5jeVNodXRkb3duAQZyZWFzb24EGkF1dG9FbWVyZ2VuY3lPcmFjbGVBZGRyZXNzAiMzUDdpaEZWeEJOYkhLMjM3VE5kUHhUMXhIRXU4cEhleFhUcgQNY2FsbGVyQWRkcmVzcwkApQgBCAUBaQZjYWxsZXIDAwkBAiE9AgUaQXV0b0VtZXJnZW5jeU9yYWNsZUFkZHJlc3MFDWNhbGxlckFkZHJlc3MJAQIhPQIJAKUIAQUSZ292ZXJuYW5jZUNvbnRyYWN0BQ1jYWxsZXJBZGRyZXNzBwkAAgECPWNhbGxlciBtdXN0IGJlIG9uZSBhbiBlbWVyZ2VuY3kgb3JhY2xlIG9yIEdvdmVybmFuY2UgY29udHJhY3QJAMwIAgkBDEJvb2xlYW5FbnRyeQICCmlzX2Jsb2NrZWQGCQDMCAIJAQtTdHJpbmdFbnRyeQICEWlzX2Jsb2NrZWRfY2FsbGVyBQ1jYWxsZXJBZGRyZXNzCQDMCAIJAQtTdHJpbmdFbnRyeQICEWlzX2Jsb2NrZWRfcmVhc29uBQZyZWFzb24FA25pbAFpARRmaW5hbGl6ZUN1cnJlbnRQcmljZQoGcHJpY2UxBXNpZ24xBnByaWNlMgVzaWduMgZwcmljZTMFc2lnbjMGcHJpY2U0BXNpZ240BnByaWNlNQVzaWduNQMFCWlzQmxvY2tlZAkAAgECWmNvbnRyYWN0IGlzIGJsb2NrZWQgYnkgRU1FUkdFTkNZIFNIVVRET1dOIGFjdGlvbnMgdW50aWxsIHJlYWN0aXZhdGlvbiBieSBlbWVyZ2VuY3kgb3JhY2xlcwMJAQIhPQIJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkArAICAgZwcmljZV8JAKQDAQUGaGVpZ2h0AAAAAAkAAgECD3dhaXQgbmV4dCBibG9jawMJAQIhPQIJAJEDAgURcHViS2V5T3JhY2xlc0xpc3QJAGoCBQZoZWlnaHQABQkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkJAAIBCQCsAgIJAKwCAgkArAICAhpPdXQgb2YgdHVybiBmaW5hbGl6YXRpb246IAkApAMBBQZoZWlnaHQCHSBibG9jayBzaG91bGQgYmUgZmluYWxpemUgYnkgCQCRAwIFEXB1YktleU9yYWNsZXNMaXN0CQBqAgUGaGVpZ2h0AAUEBnByaWNlcwkAzAgCAwkAxBMDCQESZm9ybWF0dGluZ1ByaWNlTXNnAQUGcHJpY2UxBQVzaWduMQkA2QQBCQCRAwIFEXB1YktleU9yYWNsZXNMaXN0AAAFBnByaWNlMQAACQDMCAIDCQDEEwMJARJmb3JtYXR0aW5nUHJpY2VNc2cBBQZwcmljZTIFBXNpZ24yCQDZBAEJAJEDAgURcHViS2V5T3JhY2xlc0xpc3QAAQUGcHJpY2UyAAAJAMwIAgMJAMQTAwkBEmZvcm1hdHRpbmdQcmljZU1zZwEFBnByaWNlMwUFc2lnbjMJANkEAQkAkQMCBRFwdWJLZXlPcmFjbGVzTGlzdAACBQZwcmljZTMAAAkAzAgCAwkAxBMDCQESZm9ybWF0dGluZ1ByaWNlTXNnAQUGcHJpY2U0BQVzaWduNAkA2QQBCQCRAwIFEXB1YktleU9yYWNsZXNMaXN0AAMFBnByaWNlNAAACQDMCAIDCQDEEwMJARJmb3JtYXR0aW5nUHJpY2VNc2cBBQZwcmljZTUFBXNpZ241CQDZBAEJAJEDAgURcHViS2V5T3JhY2xlc0xpc3QABAUGcHJpY2U1AAAFA25pbAQOemVyb1ByaWNlQ291bnQDCQAAAgkAkQMCBQZwcmljZXMAAAAAAAEJAGQCAAADCQAAAgkAkQMCBQZwcmljZXMAAQAAAAEJAGQCAAADCQAAAgkAkQMCBQZwcmljZXMAAgAAAAEJAGQCAAADCQAAAgkAkQMCBQZwcmljZXMAAwAAAAEJAGQCAAADCQAAAgkAkQMCBQZwcmljZXMABAAAAAEAAAMJAGcCBQ56ZXJvUHJpY2VDb3VudAADCQACAQIgMyBwcmljZXMgb3IgbW9yZSBhcmUgZXF1YWxzIHRvIDAEDXByaWNlc0luUmFuZ2UJARFmaW5kUHJpY2VzSW5SYW5nZQEFBnByaWNlcwQTcHJpY2VQcm92aWRpbmdDb3VudAkAkAMBBQ1wcmljZXNJblJhbmdlAwkAZgIAAwUTcHJpY2VQcm92aWRpbmdDb3VudAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICOkNvdWxkIG5vdCBmaW5hbGl6ZSBwcmljZSBiZWNhdXNlIG9mIGJpZyB2YXJpYXRpb246IGhlaWdodD0JAKQDAQUGaGVpZ2h0AgEKCQCRAwIFEXB1YktleU9yYWNsZXNMaXN0AAACAT0JAKQDAQkAkQMCBQZwcmljZXMAAAIBCgkAkQMCBRFwdWJLZXlPcmFjbGVzTGlzdAABAgE9CQCkAwEJAJEDAgUGcHJpY2VzAAECAQoJAJEDAgURcHViS2V5T3JhY2xlc0xpc3QAAgIBPQkApAMBCQCRAwIFBnByaWNlcwACAgEKCQCRAwIFEXB1YktleU9yYWNsZXNMaXN0AAMCAT0JAKQDAQkAkQMCBQZwcmljZXMAAwIBCgkAkQMCBRFwdWJLZXlPcmFjbGVzTGlzdAAEAgE9CQCkAwEJAJEDAgUGcHJpY2VzAAQEBHN1bTEJAGQCCQBkAgkAkQMCBQZwcmljZXMJAJEDAgUNcHJpY2VzSW5SYW5nZQAACQCRAwIFBnByaWNlcwkAkQMCBQ1wcmljZXNJblJhbmdlAAEJAJEDAgUGcHJpY2VzCQCRAwIFDXByaWNlc0luUmFuZ2UAAgQEc3VtMgMJAGcCBRNwcmljZVByb3ZpZGluZ0NvdW50AAQJAGQCBQRzdW0xCQCRAwIFBnByaWNlcwkAkQMCBQ1wcmljZXNJblJhbmdlAAMFBHN1bTEECHByaWNlU3VtAwkAZwIFE3ByaWNlUHJvdmlkaW5nQ291bnQABQkAZAIFBHN1bTIJAJEDAgUGcHJpY2VzCQCRAwIFDXByaWNlc0luUmFuZ2UABAUEc3VtMgMJAGcCBRNwcmljZVByb3ZpZGluZ0NvdW50AAYJAAIBAh5JbnZhbGlkIHByaWNlc0luUmFuZ2UgY3JlYXRpb24ECG5ld1ByaWNlCQBpAgUIcHJpY2VTdW0FE3ByaWNlUHJvdmlkaW5nQ291bnQECnd4VXNkVHVwbGUJARRmaW5hbGl6ZVd4VXNkT25jaGFpbgEFCG5ld1ByaWNlBA1uZXdXeFVzZFByaWNlCAUKd3hVc2RUdXBsZQJfMQQHd3hEZWJ1ZwgFCnd4VXNkVHVwbGUCXzIEDXZpcmVzVXNkVHVwbGUJARdmaW5hbGl6ZVZpcmVzVXNkT25jaGFpbgEFCG5ld1ByaWNlBBBuZXdWaXJlc1VzZFByaWNlCAUNdmlyZXNVc2RUdXBsZQJfMQQKdmlyZXNEZWJ1ZwgFDXZpcmVzVXNkVHVwbGUCXzIEDSR0MDE5NDUwMTk1NTQJAQtjaGVja1ByaWNlcwMFCG5ld1ByaWNlBQ1uZXdXeFVzZFByaWNlBRBuZXdWaXJlc1VzZFByaWNlBAtzaG91bGRCbG9jawgFDSR0MDE5NDUwMTk1NTQCXzEEDWJsb2NraW5nQXNzZXQIBQ0kdDAxOTQ1MDE5NTU0Al8yBA1ibG9ja2luZ1ByaWNlCAUNJHQwMTk0NTAxOTU1NAJfMwMFC3Nob3VsZEJsb2NrBAZyZWFzb24CP2F1dG9tYXRpYyBlbWVyZ2VuY3kgc2h1dGRvd24gYmVjYXVzZSBvZiBsYXJnZSBwcmljZSB2YXJpYWJpbGl0eQkAzAgCCQEMQm9vbGVhbkVudHJ5AgIKaXNfYmxvY2tlZAYJAMwIAgkBC1N0cmluZ0VudHJ5AgIRaXNfYmxvY2tlZF9jYWxsZXIJAKUIAQUEdGhpcwkAzAgCCQELU3RyaW5nRW50cnkCAhFpc19ibG9ja2VkX3JlYXNvbgUGcmVhc29uCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICEmJsYWNrX3N3YXJtX3ByaWNlXwkApAMBBQZoZWlnaHQFDWJsb2NraW5nUHJpY2UJAMwIAgkBC1N0cmluZ0VudHJ5AgIQYmxhY2tfc3dhbl90b2tlbgUNYmxvY2tpbmdBc3NldAUDbmlsBA1uZXdQcmljZUluZGV4CQBkAgUKcHJpY2VJbmRleAABCQDNCAIJAM0IAgkAzggCCQDOCAIJAM4IAgkAzAgCCQEMSW50ZWdlckVudHJ5AgIFcHJpY2UFCG5ld1ByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICBnByaWNlXwkApAMBBQZoZWlnaHQFCG5ld1ByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICDHByaWNlX2luZGV4XwkApAMBBQ1uZXdQcmljZUluZGV4BQZoZWlnaHQJAMwIAgkBDEludGVnZXJFbnRyeQICC3ByaWNlX2luZGV4BQ1uZXdQcmljZUluZGV4CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICDmluZGV4QnlIZWlnaHRfCQCkAwEFBmhlaWdodAUNbmV3UHJpY2VJbmRleAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg1wcmljZUJ5SW5kZXhfCQCkAwEFDW5ld1ByaWNlSW5kZXgFCG5ld1ByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgICCGRlZmljaXRfCQCkAwEFBmhlaWdodAUHZGVmaWNpdAkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAg9uZXV0cmlub1N1cHBseV8JAKQDAQUGaGVpZ2h0BQ5uZXV0cmlub1N1cHBseQkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICAhBkZWZpY2l0X3BlcmNlbnRfCQCkAwEFBmhlaWdodAMJAQIhPQIFDm5ldXRyaW5vU3VwcGx5AAAJAGkCCQBoAgUHZGVmaWNpdABkBQ5uZXV0cmlub1N1cHBseQAABQNuaWwJAQpQcmljZUVudHJ5AwUIbmV3UHJpY2UCBVdBVkVTBQ1uZXdQcmljZUluZGV4CQEKUHJpY2VFbnRyeQMFDW5ld1d4VXNkUHJpY2UFB3d4SWRTdHIFDW5ld1ByaWNlSW5kZXgJAQpQcmljZUVudHJ5AwUQbmV3VmlyZXNVc2RQcmljZQUKdmlyZXNJZFN0cgUNbmV3UHJpY2VJbmRleAkBC1N0cmluZ0VudHJ5AgIPZGVidWdfd3hVc2RDYWxjBQd3eERlYnVnCQELU3RyaW5nRW50cnkCAhJkZWJ1Z192aXJlc1VzZENhbGMFCnZpcmVzRGVidWcBaQENdmFsaWRhdGVQcmljZQMRYXZlcmFnaW5nUGVyaW9kTXMLdG9sZXJhbmNlWDYKYXNzZXRJZFN0cgkAlAoCBQNuaWwGAQJ0eAEGdmVyaWZ5AAQTcHViS2V5QWRtaW5zTGlzdFN0cgkAuQkCCQDMCAICLEdKZExTYUxpdjVLN3h1ZWphYzhtY1JjSG95bzNkUHJFU3J2a3RHM2E2TUFSCQDMCAICLEVZd1ptVVJkNUtLYVFSQmpzVmE2ZzhEUGlzRm9TNlNvdlJKdEZpTDVnTUhVCQDMCAICLER0bUFmdURkQ3JISzhzcGRBZUFZenE2TXNaZWdlRDlnbnNycHVUUmtDYlZBCQDMCAICLDVXUlhGU2p3Y1RiTmZLY0pzOFpxWG1TU1dZc1NWSlV0TXZNcVpqNWhINE5jBQNuaWwFA1NFUAQQcHViS2V5QWRtaW5zTGlzdAkAtQkCCQELdmFsdWVPckVsc2UCCQCdCAIFD2NvbnRyb2xDb250cmFjdAIMJXNfX211bHRpc2lnBRNwdWJLZXlBZG1pbnNMaXN0U3RyBQNTRVAEBWNvdW50CQBkAgkAZAIJAGQCAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAkA2QQBCQCRAwIFEHB1YktleUFkbWluc0xpc3QAAAABAAADCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwABCQDZBAEJAJEDAgUQcHViS2V5QWRtaW5zTGlzdAABAAEAAAMJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAIJANkEAQkAkQMCBRBwdWJLZXlBZG1pbnNMaXN0AAIAAQAAAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAwkA2QQBCQCRAwIFEHB1YktleUFkbWluc0xpc3QAAwACAAAJAGcCBQVjb3VudAADzOzxSA==", "height": 3803431, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 6DsWgcVbmrjw7GQUAUbenBdxPEEXhsTR1at5x7B4Hw7o Next: 9oz6cQypNLsm7kn9pfwe21UVdRN4bpPUmsTeitxEAdnJ Diff:
OldNewDifferences
5252 func keyControlConfig () = "%s__controlConfig"
5353
5454
55+func keyPriceByAsset (assetIdStr) = makeString(["%s%s%s__common__priceByAsset", assetIdStr], SEP)
56+
57+
58+func keyBlackSwanThreshold () = "%s%s__controlConfig__blackSwanThreshold"
59+
60+
5561 func dataControlCfg (neutrinoContract,auctionContract,rpdContract,mathContract,liquidationContract,restContract,nodeRegistryContract,nsbtStakingContract,mediatorContract,surfStakingContract,gnsbtControllerContract,restV2Contract,governanceContract,doraContract,poolsFacadeContract) = makeString_2C(["%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s", neutrinoContract, auctionContract, rpdContract, mathContract, liquidationContract, restContract, nodeRegistryContract, nsbtStakingContract, mediatorContract, surfStakingContract, gnsbtControllerContract, restV2Contract, governanceContract, doraContract, poolsFacadeContract], SEP)
5662
5763
7379
7480 let isBlocked = valueOrElse(getBoolean(this, "is_blocked"), false)
7581
76-let percentPriceOffset = 7
82+let percentPriceOffset = valueOrElse(getInteger(this, keyBlackSwanThreshold()), 7)
7783
7884 let pubKeyOracles = valueOrElse(getString(this, "oracles"), "")
7985
304310 func PriceEntry (newPrice,assetIdStr,newPriceIndex) = {
305311 let heightStr = toString(height)
306312 let newPriceIndexStr = toString(newPriceIndex)
307- let priceByAssetKEY = makeString(["%s%s%s__common__priceByAsset", assetIdStr], SEP)
308313 let priceByHeightKEY = makeString(["%s%s%s%d__common__priceByHeight", assetIdStr, heightStr], SEP)
309314 let priceByIndexKEY = makeString(["%s%s%s%d__common__priceByIndex", assetIdStr, newPriceIndexStr], SEP)
310-[IntegerEntry(priceByAssetKEY, newPrice), IntegerEntry(priceByIndexKEY, newPrice), IntegerEntry(priceByHeightKEY, newPrice)]
315+[IntegerEntry(keyPriceByAsset(assetIdStr), newPrice), IntegerEntry(priceByIndexKEY, newPrice), IntegerEntry(priceByHeightKEY, newPrice)]
311316 }
312317
313318
357362 let wxUsdPriceX6 = ((fraction(wx_WxWaves_WxUsdPriceX6, wx_WxWaves_Weight, W) + fraction(wx_WxXtn_WxUsdPriceX6, wx_WxXtn_Weight, W)) + fraction(swop_WxXtn_WxUsdPriceX6, swop_WxXtn_Weight, W))
358363 let debug = ((((((("xtnUsdPriceX6=" + toString(xtnUsdPriceX6)) + " wx_WxWaves_WxUsdPriceX6=") + toString(wx_WxWaves_WxUsdPriceX6)) + " wx_WxXtn_WxUsdPriceX6=") + toString(wx_WxXtn_WxUsdPriceX6)) + " swop_WxXtn_WxUsdPriceX6=") + toString(swop_WxXtn_WxUsdPriceX6))
359364 $Tuple2(wxUsdPriceX6, debug)
365+ }
366+
367+
368+func checkPrices (newWavesPrice,newWxPrice,newViresPrice) = {
369+ func isBlackSwan (curPr,newPr) = if ((newPr >= (curPr + ((curPr * percentPriceOffset) / 100))))
370+ then true
371+ else ((curPr - ((curPr * percentPriceOffset) / 100)) >= newPr)
372+
373+ let wxPrice = valueOrElse(getInteger(keyPriceByAsset(wxIdStr)), 0)
374+ let viresPrice = valueOrElse(getInteger(keyPriceByAsset(viresIdStr)), 0)
375+ if (isBlackSwan(price, newWavesPrice))
376+ then $Tuple3(true, "WAVES", newWavesPrice)
377+ else if (isBlackSwan(wxPrice, newWxPrice))
378+ then $Tuple3(true, wxIdStr, newWxPrice)
379+ else if (isBlackSwan(viresPrice, newViresPrice))
380+ then $Tuple3(true, viresIdStr, newViresPrice)
381+ else $Tuple3(false, "", 0)
360382 }
361383
362384
440462 let viresUsdTuple = finalizeViresUsdOnchain(newPrice)
441463 let newViresUsdPrice = viresUsdTuple._1
442464 let viresDebug = viresUsdTuple._2
443- if (if ((newPrice >= (price + ((price * percentPriceOffset) / 100))))
444- then true
445- else ((price - ((price * percentPriceOffset) / 100)) >= newPrice))
465+ let $t01945019554 = checkPrices(newPrice, newWxUsdPrice, newViresUsdPrice)
466+ let shouldBlock = $t01945019554._1
467+ let blockingAsset = $t01945019554._2
468+ let blockingPrice = $t01945019554._3
469+ if (shouldBlock)
446470 then {
447471 let reason = "automatic emergency shutdown because of large price variability"
448-[BooleanEntry("is_blocked", true), StringEntry("is_blocked_caller", toString(this)), StringEntry("is_blocked_reason", reason), IntegerEntry((("black_swarm_price" + "_") + toString(height)), newPrice)]
472+[BooleanEntry("is_blocked", true), StringEntry("is_blocked_caller", toString(this)), StringEntry("is_blocked_reason", reason), IntegerEntry(("black_swarm_price_" + toString(height)), blockingPrice), StringEntry("black_swan_token", blockingAsset)]
449473 }
450474 else {
451475 let newPriceIndex = (priceIndex + 1)
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let revisionNum = ""
55
66 let SEP = "__"
77
88 func asString (v) = match v {
99 case s: String =>
1010 s
1111 case _ =>
1212 throw("fail to cast into String")
1313 }
1414
1515
1616 func asInt (v) = match v {
1717 case i: Int =>
1818 i
1919 case _ =>
2020 throw("fail to cast into Int")
2121 }
2222
2323
2424 let IdxControlCfgNeutrinoDapp = 1
2525
2626 let IdxControlCfgAuctionDapp = 2
2727
2828 let IdxControlCfgRpdDapp = 3
2929
3030 let IdxControlCfgMathDapp = 4
3131
3232 let IdxControlCfgLiquidationDapp = 5
3333
3434 let IdxControlCfgRestDapp = 6
3535
3636 let IdxControlCfgNodeRegistryDapp = 7
3737
3838 let IdxControlCfgNsbtStakingDapp = 8
3939
4040 let IdxControlCfgMediatorDapp = 9
4141
4242 let IdxControlCfgSurfStakingDapp = 10
4343
4444 let IdxControlCfgGnsbtControllerDapp = 11
4545
4646 let IdxControlCfgRestV2Dapp = 12
4747
4848 let IdxControlCfgGovernanceDapp = 13
4949
5050 let IdxControlCfgPegProviderDapp = 14
5151
5252 func keyControlConfig () = "%s__controlConfig"
5353
5454
55+func keyPriceByAsset (assetIdStr) = makeString(["%s%s%s__common__priceByAsset", assetIdStr], SEP)
56+
57+
58+func keyBlackSwanThreshold () = "%s%s__controlConfig__blackSwanThreshold"
59+
60+
5561 func dataControlCfg (neutrinoContract,auctionContract,rpdContract,mathContract,liquidationContract,restContract,nodeRegistryContract,nsbtStakingContract,mediatorContract,surfStakingContract,gnsbtControllerContract,restV2Contract,governanceContract,doraContract,poolsFacadeContract) = makeString_2C(["%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s", neutrinoContract, auctionContract, rpdContract, mathContract, liquidationContract, restContract, nodeRegistryContract, nsbtStakingContract, mediatorContract, surfStakingContract, gnsbtControllerContract, restV2Contract, governanceContract, doraContract, poolsFacadeContract], SEP)
5662
5763
5864 func convertWavesToNeutrino (amount,price) = fraction(fraction(amount, price, 1000000), 1000000, 100000000)
5965
6066
6167 func getNumberByAddressAndKey (address,key) = valueOrElse(getInteger(addressFromStringValue(address), key), 0)
6268
6369
6470 func getStringByAddressAndKey (address,key) = valueOrElse(getString(address, key), "")
6571
6672
6773 func convertJsonArrayToList (jsonArray) = split(jsonArray, ",")
6874
6975
7076 let price = valueOrElse(getInteger(this, "price"), 0)
7177
7278 let priceIndex = valueOrElse(getInteger(this, "price_index"), 0)
7379
7480 let isBlocked = valueOrElse(getBoolean(this, "is_blocked"), false)
7581
76-let percentPriceOffset = 7
82+let percentPriceOffset = valueOrElse(getInteger(this, keyBlackSwanThreshold()), 7)
7783
7884 let pubKeyOracles = valueOrElse(getString(this, "oracles"), "")
7985
8086 let pubKeyOraclesList = convertJsonArrayToList(pubKeyOracles)
8187
8288 let neutrinoAddress = Address(base58'3PC9BfRwJWWiw9AREE2B3eWzCks3CYtg4yo')
8389
8490 let liquidationAddress = Address(base58'3P4PCxsJqMzQBALo8zANHtBDZRRquobHQp7')
8591
8692 let neutrinoAsset = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
8793
8894 let controlContract = this
8995
9096 func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
9197
9298
9399 func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
94100
95101
96102 let controlCfg = split_4C(getStringOrFail(this, keyControlConfig()), SEP)
97103
98104 let governanceContract = getContractAddressOrFail(controlCfg, IdxControlCfgGovernanceDapp)
99105
100106 let neutrinoLockedBalance = valueOrElse(getInteger(neutrinoAddress, "balance_lock_neutrino"), 0)
101107
102108 let wavesLockedBalance = valueOrElse(getInteger(neutrinoAddress, "balance_lock_waves"), 0)
103109
104110 let reserve = (wavesBalance(neutrinoAddress).regular - wavesLockedBalance)
105111
106112 let neutrinoSupply = (((neutrinoLockedBalance + value(assetInfo(neutrinoAsset)).quantity) - assetBalance(neutrinoAddress, neutrinoAsset)) - assetBalance(liquidationAddress, neutrinoAsset))
107113
108114 let deficit = (neutrinoSupply - convertWavesToNeutrino(reserve, price))
109115
110116 let doraAddressStr = "3PKkojKdd6BBzTf1RXbQVfUDraNFXXHKzQF"
111117
112118 let doraAddress = addressFromStringValue(doraAddressStr)
113119
114120 let xtnUsdPriceX6 = valueOrErrorMessage(getInteger(doraAddress, "%s%s__price__USDN-USDT"), "DORA doesn't contain USDN/USDT price")
115121
116122 let wx_restAddressStr = "3P8MoPnsaurofk1VyhsdAFkeQ6ijpJYXCpW"
117123
118124 let wx_restDapp = addressFromStringValue(wx_restAddressStr)
119125
120126 let swopRestDapp = addressFromStringValue("3P56jNQzECXnrWpnbbSJKw7Eooo6fkUaMPp")
121127
122128 let xtnId = neutrinoAsset
123129
124130 let xtnIdStr = toBase58String(xtnId)
125131
126132 let wxIdStr = "Atqv59EYzjFGuitKVnMRk6H8FukjoV3ktPorbEys25on"
127133
128134 let wxId = fromBase58String(wxIdStr)
129135
130136 let viresIdStr = "DSbbhLsSTeDg5Lsiufk2Aneh3DjVqJuPr2M9uU1gwy5p"
131137
132138 let viresId = fromBase58String(viresIdStr)
133139
134140 func findPricesInRange (prices) = {
135141 let minPercentBound = 90
136142 let maxPercentBound = 110
137143 let p0 = prices[0]
138144 let check0 = if ((0 >= prices[0]))
139145 then [0]
140146 else {
141147 let p01 = ((prices[1] * 100) / p0)
142148 let p02 = ((prices[2] * 100) / p0)
143149 let p03 = ((prices[3] * 100) / p0)
144150 let p04 = ((prices[4] * 100) / p0)
145151 let array1 = if (if ((maxPercentBound > p01))
146152 then (p01 > minPercentBound)
147153 else false)
148154 then [1, 0]
149155 else [0]
150156 let array2 = if (if ((maxPercentBound > p02))
151157 then (p02 > minPercentBound)
152158 else false)
153159 then 2 :: array1
154160 else array1
155161 let array3 = if (if ((maxPercentBound > p03))
156162 then (p03 > minPercentBound)
157163 else false)
158164 then 3 :: array2
159165 else array2
160166 if (if ((maxPercentBound > p04))
161167 then (p04 > minPercentBound)
162168 else false)
163169 then 4 :: array3
164170 else array3
165171 }
166172 let check1 = if ((size(check0) >= 3))
167173 then check0
168174 else {
169175 let p1 = prices[1]
170176 if ((0 >= p1))
171177 then [1]
172178 else {
173179 let p10 = ((prices[0] * 100) / p1)
174180 let p12 = ((prices[2] * 100) / p1)
175181 let p13 = ((prices[3] * 100) / p1)
176182 let p14 = ((prices[4] * 100) / p1)
177183 let array1 = if (if ((maxPercentBound > p10))
178184 then (p10 > minPercentBound)
179185 else false)
180186 then [0, 1]
181187 else [1]
182188 let array2 = if (if ((maxPercentBound > p12))
183189 then (p12 > minPercentBound)
184190 else false)
185191 then 2 :: array1
186192 else array1
187193 let array3 = if (if ((maxPercentBound > p13))
188194 then (p13 > minPercentBound)
189195 else false)
190196 then 3 :: array2
191197 else array2
192198 if (if ((maxPercentBound > p14))
193199 then (p14 > minPercentBound)
194200 else false)
195201 then 4 :: array3
196202 else array3
197203 }
198204 }
199205 let check2 = if ((size(check1) >= 3))
200206 then check1
201207 else {
202208 let p2 = prices[2]
203209 if ((0 >= p2))
204210 then [2]
205211 else {
206212 let p20 = ((prices[0] * 100) / p2)
207213 let p21 = ((prices[1] * 100) / p2)
208214 let p23 = ((prices[3] * 100) / p2)
209215 let p24 = ((prices[4] * 100) / p2)
210216 let array1 = if (if ((maxPercentBound > p20))
211217 then (p20 > minPercentBound)
212218 else false)
213219 then [0, 2]
214220 else [2]
215221 let array2 = if (if ((maxPercentBound > p21))
216222 then (p21 > minPercentBound)
217223 else false)
218224 then 1 :: array1
219225 else array1
220226 let array3 = if (if ((maxPercentBound > p23))
221227 then (p23 > minPercentBound)
222228 else false)
223229 then 3 :: array2
224230 else array2
225231 if (if ((maxPercentBound > p24))
226232 then (p24 > minPercentBound)
227233 else false)
228234 then 4 :: array3
229235 else array3
230236 }
231237 }
232238 let check3 = if ((size(check2) >= 3))
233239 then check2
234240 else {
235241 let p3 = prices[3]
236242 if ((0 >= p3))
237243 then [3]
238244 else {
239245 let p30 = ((prices[0] * 100) / p3)
240246 let p31 = ((prices[1] * 100) / p3)
241247 let p32 = ((prices[2] * 100) / p3)
242248 let p34 = ((prices[4] * 100) / p3)
243249 let array1 = if (if ((maxPercentBound > p30))
244250 then (p30 > minPercentBound)
245251 else false)
246252 then [0, 3]
247253 else [3]
248254 let array2 = if (if ((maxPercentBound > p31))
249255 then (p31 > minPercentBound)
250256 else false)
251257 then 1 :: array1
252258 else array1
253259 let array3 = if (if ((maxPercentBound > p32))
254260 then (p32 > minPercentBound)
255261 else false)
256262 then 2 :: array2
257263 else array2
258264 if (if ((maxPercentBound > p34))
259265 then (p34 > minPercentBound)
260266 else false)
261267 then 4 :: array3
262268 else array3
263269 }
264270 }
265271 if ((size(check3) >= 3))
266272 then check3
267273 else {
268274 let p4 = prices[4]
269275 if ((0 >= p4))
270276 then [4]
271277 else {
272278 let p40 = ((prices[0] * 100) / p4)
273279 let p41 = ((prices[1] * 100) / p4)
274280 let p42 = ((prices[2] * 100) / p4)
275281 let p43 = ((prices[3] * 100) / p4)
276282 let array1 = if (if ((maxPercentBound > p40))
277283 then (p40 > minPercentBound)
278284 else false)
279285 then [0, 4]
280286 else [4]
281287 let array2 = if (if ((maxPercentBound > p41))
282288 then (p41 > minPercentBound)
283289 else false)
284290 then 1 :: array1
285291 else array1
286292 let array3 = if (if ((maxPercentBound > p42))
287293 then (p42 > minPercentBound)
288294 else false)
289295 then 2 :: array2
290296 else array2
291297 if (if ((maxPercentBound > p43))
292298 then (p43 > minPercentBound)
293299 else false)
294300 then 3 :: array3
295301 else array3
296302 }
297303 }
298304 }
299305
300306
301307 func formattingPriceMsg (price) = toBytes((((("WAVESNEUTRINOPREFIX" + "_") + toString(height)) + "_") + toString(price)))
302308
303309
304310 func PriceEntry (newPrice,assetIdStr,newPriceIndex) = {
305311 let heightStr = toString(height)
306312 let newPriceIndexStr = toString(newPriceIndex)
307- let priceByAssetKEY = makeString(["%s%s%s__common__priceByAsset", assetIdStr], SEP)
308313 let priceByHeightKEY = makeString(["%s%s%s%d__common__priceByHeight", assetIdStr, heightStr], SEP)
309314 let priceByIndexKEY = makeString(["%s%s%s%d__common__priceByIndex", assetIdStr, newPriceIndexStr], SEP)
310-[IntegerEntry(priceByAssetKEY, newPrice), IntegerEntry(priceByIndexKEY, newPrice), IntegerEntry(priceByHeightKEY, newPrice)]
315+[IntegerEntry(keyPriceByAsset(assetIdStr), newPrice), IntegerEntry(priceByIndexKEY, newPrice), IntegerEntry(priceByHeightKEY, newPrice)]
311316 }
312317
313318
314319 func finalizeViresUsdOnchain (wavesUsdPriceX6) = {
315320 let wx_ViresWaves_lp = "E3YhURNsCmjUaYNigH6AayLxbh9u81uVGU3ZvHKznY5v"
316321 let wxRestResult = split(asString(invoke(wx_restDapp, "poolStatsREADONLY", [wx_ViresWaves_lp], nil)), "__")
317322 let wx_ViresWaves_WAVESx8 = parseIntValue(wxRestResult[2])
318323 let wx_ViresWaves_VIRESx8 = parseIntValue(wxRestResult[1])
319324 let wx_ViresWaves_ViresUsdPriceX6 = fraction(wx_ViresWaves_WAVESx8, wavesUsdPriceX6, wx_ViresWaves_VIRESx8)
320325 let wx_ViresWaves_Weight = wx_ViresWaves_VIRESx8
321326 let wx_ViresXtn_AddressStr = "3P5HjPowgipiVB3UvXphDiPvKs424WC9xQw"
322327 let wx_ViresXtn_Address = addressFromStringValue(wx_ViresXtn_AddressStr)
323328 let wx_ViresXtn_XTNx6 = assetBalance(wx_ViresXtn_Address, xtnId)
324329 let wx_ViresXtn_VIRESx8 = assetBalance(wx_ViresXtn_Address, viresId)
325330 let wx_ViresXtn_ViresUsdPriceX6 = fraction((wx_ViresXtn_XTNx6 * 100), xtnUsdPriceX6, wx_ViresXtn_VIRESx8)
326331 let wx_ViresXtn_Weight = wx_ViresXtn_VIRESx8
327332 let swop_ViresXtn_AddressStr = "3PJ48P3p2wvWUjgQaQiZ2cFbr8qmxMokBGd"
328333 let swop_ViresXtn_Address = addressFromStringValue(swop_ViresXtn_AddressStr)
329334 let swop_ViresXtn_ViresXtnPriceX6 = asInt(invoke(swopRestDapp, "calcGetAmountCPMM", [swop_ViresXtn_AddressStr, "3P88qk1KzF1BKjD7fC7LjNVAKM4ezff5WE6", viresIdStr, 100000000], nil))
330335 let swop_ViresXtn_ViresUsdPriceX6 = fraction(swop_ViresXtn_ViresXtnPriceX6, xtnUsdPriceX6, 1000000)
331336 let swop_ViresXtn_Weight = getIntegerValue(swop_ViresXtn_Address, "A_asset_balance")
332337 let W = ((wx_ViresWaves_Weight + wx_ViresXtn_Weight) + swop_ViresXtn_Weight)
333338 let viresUsdPriceX6 = ((fraction(wx_ViresWaves_ViresUsdPriceX6, wx_ViresWaves_Weight, W) + fraction(wx_ViresXtn_ViresUsdPriceX6, wx_ViresXtn_Weight, W)) + fraction(swop_ViresXtn_ViresUsdPriceX6, swop_ViresXtn_Weight, W))
334339 let debug = ((((((("xtnUsdPriceX6=" + toString(xtnUsdPriceX6)) + " wx_ViresWaves_ViresUsdPriceX6=") + toString(wx_ViresWaves_ViresUsdPriceX6)) + " wx_ViresXtn_ViresUsdPriceX6=") + toString(wx_ViresXtn_ViresUsdPriceX6)) + " swop_ViresXtn_ViresUsdPriceX6=") + toString(swop_ViresXtn_ViresUsdPriceX6))
335340 $Tuple2(viresUsdPriceX6, debug)
336341 }
337342
338343
339344 func finalizeWxUsdOnchain (wavesUsdPriceX6) = {
340345 let wx_WxWaves_lp = "BiSzFe8nSL78oZaebfoin5vBZ5Pze6d7kaeijLqr5xZe"
341346 let wxRestResult = split(asString(invoke(wx_restDapp, "poolStatsREADONLY", [wx_WxWaves_lp], nil)), "__")
342347 let wx_WxWaves_Weight = 50000
343348 let wx_WxWaves_WAVESx8 = parseIntValue(wxRestResult[2])
344349 let wx_WxWaves_WXx8 = parseIntValue(wxRestResult[1])
345350 let wx_WxWaves_WxUsdPriceX6 = fraction(wx_WxWaves_WAVESx8, wavesUsdPriceX6, wx_WxWaves_WXx8)
346351 let wx_WxXtn_AddressStr = "3PCENpEKe8atwELZ7oCSmcdEfcRuKTrUx99"
347352 let wx_WxXtn_Address = addressFromStringValue(wx_WxXtn_AddressStr)
348353 let wx_WxXtn_Weight = 50000
349354 let wx_WxXtn_XTNx6 = assetBalance(wx_WxXtn_Address, xtnId)
350355 let wx_WxXtn_WXx8 = assetBalance(wx_WxXtn_Address, wxId)
351356 let wx_WxXtn_WxUsdPriceX6 = fraction((wx_WxXtn_XTNx6 * 100), xtnUsdPriceX6, wx_WxXtn_WXx8)
352357 let swop_WxXtn_AddressStr = "3PKi4G3VX2k42ZSmNNrmvgdDH7JzRaUhY7R"
353358 let swop_WxXtn_Weight = 5000
354359 let swop_WxXtn_WxXtnPriceX6 = asInt(invoke(swopRestDapp, "calcGetAmountCPMM", [swop_WxXtn_AddressStr, "3P88qk1KzF1BKjD7fC7LjNVAKM4ezff5WE6", wxIdStr, 100000000], nil))
355360 let swop_WxXtn_WxUsdPriceX6 = fraction(swop_WxXtn_WxXtnPriceX6, xtnUsdPriceX6, 1000000)
356361 let W = ((wx_WxWaves_Weight + wx_WxXtn_Weight) + swop_WxXtn_Weight)
357362 let wxUsdPriceX6 = ((fraction(wx_WxWaves_WxUsdPriceX6, wx_WxWaves_Weight, W) + fraction(wx_WxXtn_WxUsdPriceX6, wx_WxXtn_Weight, W)) + fraction(swop_WxXtn_WxUsdPriceX6, swop_WxXtn_Weight, W))
358363 let debug = ((((((("xtnUsdPriceX6=" + toString(xtnUsdPriceX6)) + " wx_WxWaves_WxUsdPriceX6=") + toString(wx_WxWaves_WxUsdPriceX6)) + " wx_WxXtn_WxUsdPriceX6=") + toString(wx_WxXtn_WxUsdPriceX6)) + " swop_WxXtn_WxUsdPriceX6=") + toString(swop_WxXtn_WxUsdPriceX6))
359364 $Tuple2(wxUsdPriceX6, debug)
365+ }
366+
367+
368+func checkPrices (newWavesPrice,newWxPrice,newViresPrice) = {
369+ func isBlackSwan (curPr,newPr) = if ((newPr >= (curPr + ((curPr * percentPriceOffset) / 100))))
370+ then true
371+ else ((curPr - ((curPr * percentPriceOffset) / 100)) >= newPr)
372+
373+ let wxPrice = valueOrElse(getInteger(keyPriceByAsset(wxIdStr)), 0)
374+ let viresPrice = valueOrElse(getInteger(keyPriceByAsset(viresIdStr)), 0)
375+ if (isBlackSwan(price, newWavesPrice))
376+ then $Tuple3(true, "WAVES", newWavesPrice)
377+ else if (isBlackSwan(wxPrice, newWxPrice))
378+ then $Tuple3(true, wxIdStr, newWxPrice)
379+ else if (isBlackSwan(viresPrice, newViresPrice))
380+ then $Tuple3(true, viresIdStr, newViresPrice)
381+ else $Tuple3(false, "", 0)
360382 }
361383
362384
363385 @Callable(i)
364386 func constructorV1 (neutrinoContract,auctionContract,rpdContract,mathContract,liquidationContract,restContract,nodeRegistryContract,nsbtStakingContract,mediatorContract,surfStakingContract,gnsbtControllerContract,restV2Contract,governanceContract,doraContract,poolsFacadeContract) = if ((i.caller != this))
365387 then throw("permissions denied")
366388 else [StringEntry(keyControlConfig(), dataControlCfg(neutrinoContract, auctionContract, rpdContract, mathContract, liquidationContract, restContract, nodeRegistryContract, nsbtStakingContract, mediatorContract, surfStakingContract, gnsbtControllerContract, restV2Contract, governanceContract, doraContract, poolsFacadeContract))]
367389
368390
369391
370392 @Callable(i)
371393 func callEmergencyShutdown (reason) = {
372394 let AutoEmergencyOracleAddress = "3P7ihFVxBNbHK237TNdPxT1xHEu8pHexXTr"
373395 let callerAddress = toString(i.caller)
374396 if (if ((AutoEmergencyOracleAddress != callerAddress))
375397 then (toString(governanceContract) != callerAddress)
376398 else false)
377399 then throw("caller must be one an emergency oracle or Governance contract")
378400 else [BooleanEntry("is_blocked", true), StringEntry("is_blocked_caller", callerAddress), StringEntry("is_blocked_reason", reason)]
379401 }
380402
381403
382404
383405 @Callable(i)
384406 func finalizeCurrentPrice (price1,sign1,price2,sign2,price3,sign3,price4,sign4,price5,sign5) = if (isBlocked)
385407 then throw("contract is blocked by EMERGENCY SHUTDOWN actions untill reactivation by emergency oracles")
386408 else if ((valueOrElse(getInteger(this, ("price_" + toString(height))), 0) != 0))
387409 then throw("wait next block")
388410 else if ((pubKeyOraclesList[(height % 5)] != toBase58String(i.callerPublicKey)))
389411 then throw(((("Out of turn finalization: " + toString(height)) + " block should be finalize by ") + pubKeyOraclesList[(height % 5)]))
390412 else {
391413 let prices = [if (sigVerify_8Kb(formattingPriceMsg(price1), sign1, fromBase58String(pubKeyOraclesList[0])))
392414 then price1
393415 else 0, if (sigVerify_8Kb(formattingPriceMsg(price2), sign2, fromBase58String(pubKeyOraclesList[1])))
394416 then price2
395417 else 0, if (sigVerify_8Kb(formattingPriceMsg(price3), sign3, fromBase58String(pubKeyOraclesList[2])))
396418 then price3
397419 else 0, if (sigVerify_8Kb(formattingPriceMsg(price4), sign4, fromBase58String(pubKeyOraclesList[3])))
398420 then price4
399421 else 0, if (sigVerify_8Kb(formattingPriceMsg(price5), sign5, fromBase58String(pubKeyOraclesList[4])))
400422 then price5
401423 else 0]
402424 let zeroPriceCount = if ((prices[0] == 0))
403425 then 1
404426 else (0 + (if ((prices[1] == 0))
405427 then 1
406428 else (0 + (if ((prices[2] == 0))
407429 then 1
408430 else (0 + (if ((prices[3] == 0))
409431 then 1
410432 else (0 + (if ((prices[4] == 0))
411433 then 1
412434 else 0))))))))
413435 if ((zeroPriceCount >= 3))
414436 then throw("3 prices or more are equals to 0")
415437 else {
416438 let pricesInRange = findPricesInRange(prices)
417439 let priceProvidingCount = size(pricesInRange)
418440 if ((3 > priceProvidingCount))
419441 then throw(((((((((((((((((((((("Could not finalize price because of big variation: height=" + toString(height)) + "
420442 ") + pubKeyOraclesList[0]) + "=") + toString(prices[0])) + "
421443 ") + pubKeyOraclesList[1]) + "=") + toString(prices[1])) + "
422444 ") + pubKeyOraclesList[2]) + "=") + toString(prices[2])) + "
423445 ") + pubKeyOraclesList[3]) + "=") + toString(prices[3])) + "
424446 ") + pubKeyOraclesList[4]) + "=") + toString(prices[4])))
425447 else {
426448 let sum1 = ((prices[pricesInRange[0]] + prices[pricesInRange[1]]) + prices[pricesInRange[2]])
427449 let sum2 = if ((priceProvidingCount >= 4))
428450 then (sum1 + prices[pricesInRange[3]])
429451 else sum1
430452 let priceSum = if ((priceProvidingCount >= 5))
431453 then (sum2 + prices[pricesInRange[4]])
432454 else sum2
433455 if ((priceProvidingCount >= 6))
434456 then throw("Invalid pricesInRange creation")
435457 else {
436458 let newPrice = (priceSum / priceProvidingCount)
437459 let wxUsdTuple = finalizeWxUsdOnchain(newPrice)
438460 let newWxUsdPrice = wxUsdTuple._1
439461 let wxDebug = wxUsdTuple._2
440462 let viresUsdTuple = finalizeViresUsdOnchain(newPrice)
441463 let newViresUsdPrice = viresUsdTuple._1
442464 let viresDebug = viresUsdTuple._2
443- if (if ((newPrice >= (price + ((price * percentPriceOffset) / 100))))
444- then true
445- else ((price - ((price * percentPriceOffset) / 100)) >= newPrice))
465+ let $t01945019554 = checkPrices(newPrice, newWxUsdPrice, newViresUsdPrice)
466+ let shouldBlock = $t01945019554._1
467+ let blockingAsset = $t01945019554._2
468+ let blockingPrice = $t01945019554._3
469+ if (shouldBlock)
446470 then {
447471 let reason = "automatic emergency shutdown because of large price variability"
448-[BooleanEntry("is_blocked", true), StringEntry("is_blocked_caller", toString(this)), StringEntry("is_blocked_reason", reason), IntegerEntry((("black_swarm_price" + "_") + toString(height)), newPrice)]
472+[BooleanEntry("is_blocked", true), StringEntry("is_blocked_caller", toString(this)), StringEntry("is_blocked_reason", reason), IntegerEntry(("black_swarm_price_" + toString(height)), blockingPrice), StringEntry("black_swan_token", blockingAsset)]
449473 }
450474 else {
451475 let newPriceIndex = (priceIndex + 1)
452476 ((((([IntegerEntry("price", newPrice), IntegerEntry(("price_" + toString(height)), newPrice), IntegerEntry(("price_index_" + toString(newPriceIndex)), height), IntegerEntry("price_index", newPriceIndex), IntegerEntry(("indexByHeight_" + toString(height)), newPriceIndex), IntegerEntry(("priceByIndex_" + toString(newPriceIndex)), newPrice), IntegerEntry(("deficit_" + toString(height)), deficit), IntegerEntry(("neutrinoSupply_" + toString(height)), neutrinoSupply), IntegerEntry(("deficit_percent_" + toString(height)), if ((neutrinoSupply != 0))
453477 then ((deficit * 100) / neutrinoSupply)
454478 else 0)] ++ PriceEntry(newPrice, "WAVES", newPriceIndex)) ++ PriceEntry(newWxUsdPrice, wxIdStr, newPriceIndex)) ++ PriceEntry(newViresUsdPrice, viresIdStr, newPriceIndex)) :+ StringEntry("debug_wxUsdCalc", wxDebug)) :+ StringEntry("debug_viresUsdCalc", viresDebug))
455479 }
456480 }
457481 }
458482 }
459483 }
460484
461485
462486
463487 @Callable(i)
464488 func validatePrice (averagingPeriodMs,toleranceX6,assetIdStr) = $Tuple2(nil, true)
465489
466490
467491 @Verifier(tx)
468492 func verify () = {
469493 let pubKeyAdminsListStr = makeString(["GJdLSaLiv5K7xuejac8mcRcHoyo3dPrESrvktG3a6MAR", "EYwZmURd5KKaQRBjsVa6g8DPisFoS6SovRJtFiL5gMHU", "DtmAfuDdCrHK8spdAeAYzq6MsZegeD9gnsrpuTRkCbVA", "5WRXFSjwcTbNfKcJs8ZqXmSSWYsSVJUtMvMqZj5hH4Nc"], SEP)
470494 let pubKeyAdminsList = split(valueOrElse(getString(controlContract, "%s__multisig"), pubKeyAdminsListStr), SEP)
471495 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
472496 then 1
473497 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
474498 then 1
475499 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[2], fromBase58String(pubKeyAdminsList[2])))
476500 then 1
477501 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
478502 then 2
479503 else 0))
480504 (count >= 3)
481505 }
482506

github/deemru/w8io/786bc32 
53.30 ms