tx · 7KQw4kh2beZiNBzh6oaNLRYC1knFuaJ59iyMs67ue238

3Mp66Sufwhc33Wg9L4fWvD2GR729YxSQPCh:  -0.07500000 Waves

2023.05.19 18:36 [2585036] smart account 3Mp66Sufwhc33Wg9L4fWvD2GR729YxSQPCh > SELF 0.00000000 Waves

{ "type": 13, "id": "7KQw4kh2beZiNBzh6oaNLRYC1knFuaJ59iyMs67ue238", "fee": 7500000, "feeAssetId": null, "timestamp": 1684510562324, "version": 2, "chainId": 84, "sender": "3Mp66Sufwhc33Wg9L4fWvD2GR729YxSQPCh", "senderPublicKey": "42U12q4TTBTVkhjqmAhNWDrxkFhHrg9yFuNjqqg5mF8G", "proofs": [ "6fWrYKT5Lz9cbegvgvqDWWiqVf1dqSRgWJfQY66kkkByxoF1PKuQqPGTG2SWu7w1YL9VAC1NFBZuJhifnLfr4gd" ], "script": "base64:BgKOAQgCEgASABIAEgASAwoBARIRCg8BAQEBAQEBAQEBAQEBAQESFgoUAQEBAQEBAQgICAEBAQEBAQEBAQESBwoFAQEBCAgSBAoCAQgSBQoDAQEIEgcKBQEBAQQIEgUKAwgBCBIDCgEIEgMKAQgSABIAEgMKAQgSBQoDCAEIEgMKAQESABIDCgEIEgASBAoCCAizAQAMa19iYXNlT3JhY2xlAgxrX2Jhc2VPcmFjbGUADWtfcXVvdGVPcmFjbGUCDWtfcXVvdGVPcmFjbGUACWtfYmFsYW5jZQIJa19iYWxhbmNlAAprX3NlcXVlbmNlAgprX3NlcXVlbmNlAA5rX3Bvc2l0aW9uU2l6ZQIOa19wb3NpdGlvblNpemUAEGtfcG9zaXRpb25NYXJnaW4CEGtfcG9zaXRpb25NYXJnaW4AFmtfcG9zaXRpb25PcGVuTm90aW9uYWwCFmtfcG9zaXRpb25PcGVuTm90aW9uYWwALmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24CEmtfcG9zaXRpb25GcmFjdGlvbgASa19wb3NpdGlvblNlcXVlbmNlAhJrX3Bvc2l0aW9uU2VxdWVuY2UADWtfcG9zaXRpb25GZWUCDWtfcG9zaXRpb25GZWUAHmtfcG9zaXRpb25MYXN0VXBkYXRlZFRpbWVzdGFtcAITa19wb3NpdGlvblRpbWVzdGFtcAANa19pbml0aWFsaXplZAINa19pbml0aWFsaXplZAAIa19wYXVzZWQCCGtfcGF1c2VkAAtrX2Nsb3NlT25seQILa19jbG9zZU9ubHkABWtfZmVlAgVrX2ZlZQANa19yb2xsb3ZlckZlZQIOa19yb2xsb3Zlcl9mZWUAD2tfZnVuZGluZ1BlcmlvZAIPa19mdW5kaW5nUGVyaW9kABFrX2luaXRNYXJnaW5SYXRpbwIRa19pbml0TWFyZ2luUmF0aW8AGGtfbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwIFa19tbXIAFWtfbGlxdWlkYXRpb25GZWVSYXRpbwIVa19saXF1aWRhdGlvbkZlZVJhdGlvABlrX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvAhZrX3BhcnRMaXF1aWRhdGlvblJhdGlvAA1rX3NwcmVhZExpbWl0Ag1rX3NwcmVhZExpbWl0ABBrX21heFByaWNlSW1wYWN0AhBrX21heFByaWNlSW1wYWN0ABBrX21heFByaWNlU3ByZWFkAhBrX21heFByaWNlU3ByZWFkABFrX21heE9wZW5Ob3Rpb25hbAIRa19tYXhPcGVuTm90aW9uYWwAFWtfZmVlVG9TdGFrZXJzUGVyY2VudAIVa19mZWVUb1N0YWtlcnNQZXJjZW50ABBrX21heE9yYWNsZURlbGF5AhBrX21heE9yYWNsZURlbGF5AA1rX2Z1bmRpbmdNb2RlAg1rX2Z1bmRpbmdNb2RlAAxrX29yYWNsZU1vZGUCDGtfb3JhY2xlTW9kZQAla19sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgIba19sYXRlc3RMb25nUHJlbWl1bUZyYWN0aW9uACZrX2xhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgIca19sYXRlc3RTaG9ydFByZW1pdW1GcmFjdGlvbgASa19uZXh0RnVuZGluZ0Jsb2NrAh5rX25leHRGdW5kaW5nQmxvY2tNaW5UaW1lc3RhbXAAEWtfbG9uZ0Z1bmRpbmdSYXRlAhFrX2xvbmdGdW5kaW5nUmF0ZQASa19zaG9ydEZ1bmRpbmdSYXRlAhJrX3Nob3J0RnVuZGluZ1JhdGUAE2tfcXVvdGVBc3NldFJlc2VydmUCCGtfcXRBc3RSABJrX2Jhc2VBc3NldFJlc2VydmUCCGtfYnNBc3RSABJrX3F1b3RlQXNzZXRXZWlnaHQCCGtfcXRBc3RXABFrX2Jhc2VBc3NldFdlaWdodAIIa19ic0FzdFcAE2tfdG90YWxQb3NpdGlvblNpemUCE2tfdG90YWxQb3NpdGlvblNpemUAF2tfdG90YWxMb25nUG9zaXRpb25TaXplAhdrX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQAYa190b3RhbFNob3J0UG9zaXRpb25TaXplAhhrX3RvdGFsU2hvcnRQb3NpdGlvblNpemUAFmtfb3BlbkludGVyZXN0Tm90aW9uYWwCFmtfb3BlbkludGVyZXN0Tm90aW9uYWwAE2tfb3BlbkludGVyZXN0U2hvcnQCE2tfb3BlbkludGVyZXN0U2hvcnQAEmtfb3BlbkludGVyZXN0TG9uZwISa19vcGVuSW50ZXJlc3RMb25nAAhrX2xhc3RUeAIIa19sYXN0VHgAFGtfY29vcmRpbmF0b3JBZGRyZXNzAhRrX2Nvb3JkaW5hdG9yQWRkcmVzcwAPa192YXVsdF9hZGRyZXNzAg9rX3ZhdWx0X2FkZHJlc3MAD2tfYWRtaW5fYWRkcmVzcwIPa19hZG1pbl9hZGRyZXNzAA1rX3F1b3RlX2Fzc2V0Ag1rX3F1b3RlX2Fzc2V0ABFrX3N0YWtpbmdfYWRkcmVzcwIRa19zdGFraW5nX2FkZHJlc3MAD2tfbWluZXJfYWRkcmVzcwIPa19taW5lcl9hZGRyZXNzABBrX29yZGVyc19hZGRyZXNzAhBrX29yZGVyc19hZGRyZXNzABJrX3JlZmVycmFsX2FkZHJlc3MCEmtfcmVmZXJyYWxfYWRkcmVzcwAVa19uZnRfbWFuYWdlcl9hZGRyZXNzAhVrX25mdF9tYW5hZ2VyX2FkZHJlc3MBDnRvQ29tcG9zaXRlS2V5AgRfa2V5CF9hZGRyZXNzCQCsAgIJAKwCAgUEX2tleQIBXwUIX2FkZHJlc3MBC2Nvb3JkaW5hdG9yAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMFFGtfY29vcmRpbmF0b3JBZGRyZXNzAhNDb29yZGluYXRvciBub3Qgc2V0AQxhZG1pbkFkZHJlc3MACQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFD2tfYWRtaW5fYWRkcmVzcwEKcXVvdGVBc3NldAAJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUNa19xdW90ZV9hc3NldAEOc3Rha2luZ0FkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAURa19zdGFraW5nX2FkZHJlc3MCD1N0YWtpbmcgbm90IHNldAEMdmF1bHRBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFD2tfdmF1bHRfYWRkcmVzcwINVmF1bHQgbm90IHNldAEMbWluZXJBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFD2tfbWluZXJfYWRkcmVzcwINTWluZXIgbm90IHNldAENb3JkZXJzQWRkcmVzcwAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABRBrX29yZGVyc19hZGRyZXNzAg5PcmRlcnMgbm90IHNldAEPcmVmZXJyYWxBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFEmtfcmVmZXJyYWxfYWRkcmVzcwIQUmVmZXJyYWwgbm90IHNldAERbmZ0TWFuYWdlckFkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUVa19uZnRfbWFuYWdlcl9hZGRyZXNzAhNORlQgTWFuYWdlciBub3Qgc2V0AA1rX3Rva2VuX3BhcmFtAg1rX3Rva2VuX3BhcmFtAAxrX3Rva2VuX3R5cGUCDGtfdG9rZW5fdHlwZQAYRkVFX1JFRFVDVElPTl9UT0tFTl9UWVBFAg1mZWVfcmVkdWN0aW9uAAhESVJfTE9ORwABAAlESVJfU0hPUlQAAgAHU0VDT05EUwDoBwAPREVDSU1BTF9OVU1CRVJTAAYADERFQ0lNQUxfVU5JVAkAaAIAAQkAaAIJAGgCCQBoAgkAaAIJAGgCAAoACgAKAAoACgAKAA9NSU5VVEVTX0lOX1lFQVIJAGgCAKCKIAUMREVDSU1BTF9VTklUAAdPTkVfREFZCQBoAgCAowUFDERFQ0lNQUxfVU5JVAAPUE5MX09QVElPTl9TUE9UAAEAEVBOTF9PUFRJT05fT1JBQ0xFAAIAEkZVTkRJTkdfQVNZTU1FVFJJQwABABFGVU5ESU5HX1NZTU1FVFJJQwACAAxPUkFDTEVfUExBSU4AAQAKT1JBQ0xFX0pJVAACAQFzAQJfeAkArAICCQCkAwEFAl94AgEsAQRkaXZkAgJfeAJfeQkAbgQFAl94BQxERUNJTUFMX1VOSVQFAl95BQhIQUxGRVZFTgEEbXVsZAICX3gCX3kJAG4EBQJfeAUCX3kFDERFQ0lNQUxfVU5JVAUISEFMRkVWRU4BBWJkaXZkAgJfeAJfeQkAvQIEBQJfeAkAtgIBBQxERUNJTUFMX1VOSVQFAl95BQhIQUxGRVZFTgEFYm11bGQCAl94Al95CQC9AgQFAl94BQJfeQkAtgIBBQxERUNJTUFMX1VOSVQFCEhBTEZFVkVOAQNhYnMBAl94AwkAZgIFAl94AAAFAl94CQEBLQEFAl94AQR2bWF4AgJfeAJfeQMJAGcCBQJfeAUCX3kFAl94BQJfeQEDaW50AQFrCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUEdGhpcwUBawkArAICAg1ubyB2YWx1ZSBmb3IgBQFrAQVpbnRPcgIBawNkZWYJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUBawUDZGVmAQRzdHJBAghfYWRkcmVzcwRfa2V5BAN2YWwJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQhfYWRkcmVzcwUEX2tleQkArAICAhFObyB2YWx1ZSBmb3Iga2V5IAUEX2tleQUDdmFsAQRpbnRBAghfYWRkcmVzcwRfa2V5BAN2YWwJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQhfYWRkcmVzcwUEX2tleQkArAICAhFObyB2YWx1ZSBmb3Iga2V5IAUEX2tleQUDdmFsAQhjYmFsYW5jZQAJAQNpbnQBBQlrX2JhbGFuY2UBA2ZlZQAJAQNpbnQBBQVrX2ZlZQEPcm9sbG92ZXJGZWVSYXRlAAkBA2ludAEFDWtfcm9sbG92ZXJGZWUBD2luaXRNYXJnaW5SYXRpbwAJAQNpbnQBBRFrX2luaXRNYXJnaW5SYXRpbwEGcXRBc3RSAAkBA2ludAEFE2tfcXVvdGVBc3NldFJlc2VydmUBBmJzQXN0UgAJAQNpbnQBBRJrX2Jhc2VBc3NldFJlc2VydmUBBnF0QXN0VwAJAQVpbnRPcgIFEmtfcXVvdGVBc3NldFdlaWdodAUMREVDSU1BTF9VTklUAQZic0FzdFcACQEFaW50T3ICBRFrX2Jhc2VBc3NldFdlaWdodAUMREVDSU1BTF9VTklUARF0b3RhbFBvc2l0aW9uU2l6ZQAJAQNpbnQBBRNrX3RvdGFsUG9zaXRpb25TaXplARRvcGVuSW50ZXJlc3ROb3Rpb25hbAAJAQNpbnQBBRZrX29wZW5JbnRlcmVzdE5vdGlvbmFsARFvcGVuSW50ZXJlc3RTaG9ydAAJAQNpbnQBBRNrX29wZW5JbnRlcmVzdFNob3J0ARBvcGVuSW50ZXJlc3RMb25nAAkBA2ludAEFEmtfb3BlbkludGVyZXN0TG9uZwEZbmV4dEZ1bmRpbmdCbG9ja1RpbWVzdGFtcAAJAQNpbnQBBRJrX25leHRGdW5kaW5nQmxvY2sBEGZ1bmRpbmdQZXJpb2RSYXcACQEDaW50AQUPa19mdW5kaW5nUGVyaW9kARRmdW5kaW5nUGVyaW9kRGVjaW1hbAAJAGgCCQEQZnVuZGluZ1BlcmlvZFJhdwAFDERFQ0lNQUxfVU5JVAEUZnVuZGluZ1BlcmlvZFNlY29uZHMACQBoAgkBEGZ1bmRpbmdQZXJpb2RSYXcABQdTRUNPTkRTARZtYWludGVuYW5jZU1hcmdpblJhdGlvAAkBA2ludAEFGGtfbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwETbGlxdWlkYXRpb25GZWVSYXRpbwAJAQNpbnQBBRVrX2xpcXVpZGF0aW9uRmVlUmF0aW8BF3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvAAkBA2ludAEFGWtfcGFydGlhbExpcXVpZGF0aW9uUmF0aW8BC3NwcmVhZExpbWl0AAkBA2ludAEFDWtfc3ByZWFkTGltaXQBDm1heFByaWNlSW1wYWN0AAkBA2ludAEFEGtfbWF4UHJpY2VJbXBhY3QBDm1heFByaWNlU3ByZWFkAAkBA2ludAEFEGtfbWF4UHJpY2VTcHJlYWQBD21heE9wZW5Ob3Rpb25hbAAJAQNpbnQBBRFrX21heE9wZW5Ob3Rpb25hbAEjbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24ACQEDaW50AQUla19sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgEkbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAAkBA2ludAEFJmtfbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAkBA2ludAEFGGtfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQEVdG90YWxMb25nUG9zaXRpb25TaXplAAkBA2ludAEFF2tfdG90YWxMb25nUG9zaXRpb25TaXplAQxsYXN0U2VxdWVuY2UACQEFaW50T3ICBQprX3NlcXVlbmNlAAABE2ZlZVRvU3Rha2Vyc1BlcmNlbnQACQEDaW50AQUVa19mZWVUb1N0YWtlcnNQZXJjZW50AQ5tYXhPcmFjbGVEZWxheQAJAQNpbnQBBRBrX21heE9yYWNsZURlbGF5AQtmdW5kaW5nTW9kZQAJAQVpbnRPcgIFDWtfZnVuZGluZ01vZGUFEkZVTkRJTkdfQVNZTU1FVFJJQwEKb3JhY2xlTW9kZQAJAQVpbnRPcgIFDGtfb3JhY2xlTW9kZQUMT1JBQ0xFX1BMQUlOAQ1sYXN0VGltZXN0YW1wAAgFCWxhc3RCbG9jawl0aW1lc3RhbXABD2dldEFjdHVhbENhbGxlcgEBaQkBC3ZhbHVlT3JFbHNlAgkAnQgCCQENb3JkZXJzQWRkcmVzcwACCGtfc2VuZGVyCQClCAEIBQFpBmNhbGxlcgEWcmVxdWlyZU1vcmVNYXJnaW5SYXRpbwMMX21hcmdpblJhdGlvEF9iYXNlTWFyZ2luUmF0aW8UX2xhcmdlclRoYW5PckVxdWFsVG8EFHJlbWFpbmluZ01hcmdpblJhdGlvCQBlAgUMX21hcmdpblJhdGlvBRBfYmFzZU1hcmdpblJhdGlvAwMFFF9sYXJnZXJUaGFuT3JFcXVhbFRvCQBmAgAABRRyZW1haW5pbmdNYXJnaW5SYXRpbwcJAAIBCQCsAgIJAKwCAgkArAICAhBJbnZhbGlkIG1hcmdpbjogCQCkAwEFDF9tYXJnaW5SYXRpbwIDIDwgCQCkAwEFEF9iYXNlTWFyZ2luUmF0aW8DAwkBASEBBRRfbGFyZ2VyVGhhbk9yRXF1YWxUbwkAZwIFFHJlbWFpbmluZ01hcmdpblJhdGlvAAAHCQACAQkArAICCQCsAgIJAKwCAgIQSW52YWxpZCBtYXJnaW46IAkApAMBBQxfbWFyZ2luUmF0aW8CAyA+IAkApAMBBRBfYmFzZU1hcmdpblJhdGlvBgEfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgENX3Bvc2l0aW9uU2l6ZQMJAAACBQ1fcG9zaXRpb25TaXplAAAJAAIBAixTaG91bGQgbm90IGJlIGNhbGxlZCB3aXRoIF9wb3NpdGlvblNpemUgPT0gMAMJAGYCBQ1fcG9zaXRpb25TaXplAAAJASNsYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgAJASRsYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24AAQtnZXRQb3NpdGlvbgIHX3RyYWRlcgpfZGlyZWN0aW9uBAtwb3NpdGlvbktleQkArAICCQCsAgIFB190cmFkZXICAV8JAKQDAQUKX2RpcmVjdGlvbgQPcG9zaXRpb25TaXplT3B0CQCaCAIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFDmtfcG9zaXRpb25TaXplBQtwb3NpdGlvbktleQQHJG1hdGNoMAUPcG9zaXRpb25TaXplT3B0AwkAAQIFByRtYXRjaDACA0ludAQMcG9zaXRpb25TaXplBQckbWF0Y2gwCQCXCgUFDHBvc2l0aW9uU2l6ZQkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUQa19wb3NpdGlvbk1hcmdpbgULcG9zaXRpb25LZXkJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFFmtfcG9zaXRpb25PcGVuTm90aW9uYWwFC3Bvc2l0aW9uS2V5CQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBS5rX3Bvc2l0aW9uTGFzdFVwZGF0ZWRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBQtwb3NpdGlvbktleQkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUea19wb3NpdGlvbkxhc3RVcGRhdGVkVGltZXN0YW1wBQtwb3NpdGlvbktleQkAlwoFAAAAAAAAAAAAAAEMZ2V0RGlyZWN0aW9uAQ1fcG9zaXRpb25TaXplAwkAZgIAAAUNX3Bvc2l0aW9uU2l6ZQUJRElSX1NIT1JUBQhESVJfTE9ORwEOZ2V0UG9zaXRpb25GZWUCB190cmFkZXIKX2RpcmVjdGlvbgQLcG9zaXRpb25LZXkJAKwCAgkArAICBQdfdHJhZGVyAgFfCQCkAwEFCl9kaXJlY3Rpb24EDnBvc2l0aW9uRmVlT3B0CQCaCAIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFDWtfcG9zaXRpb25GZWUFC3Bvc2l0aW9uS2V5BAckbWF0Y2gwBQ5wb3NpdGlvbkZlZU9wdAMJAAECBQckbWF0Y2gwAgNJbnQEC3Bvc2l0aW9uRmVlBQckbWF0Y2gwBQtwb3NpdGlvbkZlZQkBA2ZlZQABE3JlcXVpcmVPcGVuUG9zaXRpb24CB190cmFkZXIKX2RpcmVjdGlvbgMJAAACCAkBC2dldFBvc2l0aW9uAgUHX3RyYWRlcgUKX2RpcmVjdGlvbgJfMQAACQACAQIQTm8gb3BlbiBwb3NpdGlvbgYBDWdldE9yYWNsZURhdGEBA2tleQQNb3JhY2xlRGF0YVN0cgkAnQgCBQR0aGlzBQNrZXkDAwkBCWlzRGVmaW5lZAEFDW9yYWNsZURhdGFTdHIJAQIhPQIJAQV2YWx1ZQEFDW9yYWNsZURhdGFTdHICAAcECm9yYWNsZURhdGEJALUJAgkBBXZhbHVlAQUNb3JhY2xlRGF0YVN0cgIBLAQNb3JhY2xlQWRkcmVzcwkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJAJEDAgUKb3JhY2xlRGF0YQAACQCsAgICG0ludmFsaWQgb3JhY2xlIGFkZHJlc3MgaW46IAkBBXZhbHVlAQUNb3JhY2xlRGF0YVN0cgQIcHJpY2VLZXkJAJEDAgUKb3JhY2xlRGF0YQABBAhibG9ja0tleQkAkQMCBQpvcmFjbGVEYXRhAAIEB29wZW5LZXkJAJEDAgUKb3JhY2xlRGF0YQADCQCWCgQFDW9yYWNsZUFkZHJlc3MFCHByaWNlS2V5BQhibG9ja0tleQUHb3BlbktleQUEdW5pdAELaW5pdGlhbGl6ZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFDWtfaW5pdGlhbGl6ZWQHAQZwYXVzZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFCGtfcGF1c2VkBwEJY2xvc2VPbmx5AAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzBQtrX2Nsb3NlT25seQcBDXVwZGF0ZVJlc2VydmUDBl9pc0FkZBFfcXVvdGVBc3NldEFtb3VudBBfYmFzZUFzc2V0QW1vdW50AwUGX2lzQWRkBAduZXdCYXNlCQBlAgkBBmJzQXN0UgAFEF9iYXNlQXNzZXRBbW91bnQDCQBnAgAABQduZXdCYXNlCQACAQIqVHggbGVhZCB0byBiYXNlIGFzc2V0IHJlc2VydmUgPD0gMCwgcmV2ZXJ0CQCVCgMJAGQCCQEGcXRBc3RSAAURX3F1b3RlQXNzZXRBbW91bnQFB25ld0Jhc2UJAGQCCQERdG90YWxQb3NpdGlvblNpemUABRBfYmFzZUFzc2V0QW1vdW50BAhuZXdRdW90ZQkAZQIJAQZxdEFzdFIABRFfcXVvdGVBc3NldEFtb3VudAMJAGcCAAAFCG5ld1F1b3RlCQACAQIqVHggbGVhZCB0byBiYXNlIHF1b3RlIHJlc2VydmUgPD0gMCwgcmV2ZXJ0CQCVCgMFCG5ld1F1b3RlCQBkAgkBBmJzQXN0UgAFEF9iYXNlQXNzZXRBbW91bnQJAGUCCQERdG90YWxQb3NpdGlvblNpemUABRBfYmFzZUFzc2V0QW1vdW50AQ1jYWxjSW52YXJpYW50AgdfcXRBc3RSB19ic0FzdFIEB2JxdEFzdFIJALYCAQUHX3F0QXN0UgQHYmJzQXN0UgkAtgIBBQdfYnNBc3RSCQEFYm11bGQCBQdicXRBc3RSBQdiYnNBc3RSAQlzd2FwSW5wdXQCBl9pc0FkZBFfcXVvdGVBc3NldEFtb3VudAQHX3F0QXN0UgkBBnF0QXN0UgAEB19ic0FzdFIJAQZic0FzdFIABAdfcXRBc3RXCQEGcXRBc3RXAAQHX2JzQXN0VwkBBmJzQXN0VwAEGHF1b3RlQXNzZXRBbW91bnRBZGp1c3RlZAkBBGRpdmQCBRFfcXVvdGVBc3NldEFtb3VudAUHX3F0QXN0VwQBawkBDWNhbGNJbnZhcmlhbnQCBQdfcXRBc3RSBQdfYnNBc3RSBBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyAwUGX2lzQWRkCQBkAgUHX3F0QXN0UgUYcXVvdGVBc3NldEFtb3VudEFkanVzdGVkCQBlAgUHX3F0QXN0UgUYcXVvdGVBc3NldEFtb3VudEFkanVzdGVkBBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIJAKADAQkBBWJkaXZkAgUBawkAtgIBBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBBhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMJAQNhYnMBCQBlAgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBQdfYnNBc3RSBBVhbW91bnRCYXNlQXNzZXRCb3VnaHQDBQZfaXNBZGQFGGFtb3VudEJhc2VBc3NldEJvdWdodEFicwkBAS0BBRhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMEDSR0MDE1Nzg2MTU5NTYJAQ11cGRhdGVSZXNlcnZlAwUGX2lzQWRkBRhxdW90ZUFzc2V0QW1vdW50QWRqdXN0ZWQFGGFtb3VudEJhc2VBc3NldEJvdWdodEFicwQXcXVvdGVBc3NldFJlc2VydmVBZnRlcjEIBQ0kdDAxNTc4NjE1OTU2Al8xBBZiYXNlQXNzZXRSZXNlcnZlQWZ0ZXIxCAUNJHQwMTU3ODYxNTk1NgJfMgQXdG90YWxQb3NpdGlvblNpemVBZnRlcjEIBQ0kdDAxNTc4NjE1OTU2Al8zBAtwcmljZUJlZm9yZQkBBGRpdmQCCQEEbXVsZAIFB19xdEFzdFIFB19xdEFzdFcJAQRtdWxkAgUHX2JzQXN0UgUHX2JzQXN0VwQLbWFya2V0UHJpY2UJAQRkaXZkAgURX3F1b3RlQXNzZXRBbW91bnQFGGFtb3VudEJhc2VBc3NldEJvdWdodEFicwQJcHJpY2VEaWZmCQEDYWJzAQkAZQIFC3ByaWNlQmVmb3JlBQttYXJrZXRQcmljZQQLcHJpY2VJbXBhY3QJAGUCBQxERUNJTUFMX1VOSVQJAQRkaXZkAgULcHJpY2VCZWZvcmUJAGQCBQtwcmljZUJlZm9yZQUJcHJpY2VEaWZmBBNtYXhQcmljZUltcGFjdFZhbHVlCQEObWF4UHJpY2VJbXBhY3QAAwkAZgIFC3ByaWNlSW1wYWN0BRNtYXhQcmljZUltcGFjdFZhbHVlCQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAg1QcmljZSBpbXBhY3QgCQCkAwEFC3ByaWNlSW1wYWN0AhQgPiBtYXggcHJpY2UgaW1wYWN0IAkApAMBBRNtYXhQcmljZUltcGFjdFZhbHVlAhUgYmVmb3JlIHF1b3RlIGFzc2V0OiAJAKQDAQUHX3F0QXN0UgIUIGJlZm9yZSBiYXNlIGFzc2V0OiAJAKQDAQUHX2JzQXN0UgIhIHF1b3RlIGFzc2V0IGFtb3VudCB0byBleGNoYW5nZTogCQCkAwEFEV9xdW90ZUFzc2V0QW1vdW50Ag8gcHJpY2UgYmVmb3JlOiAJAKQDAQULcHJpY2VCZWZvcmUCDiBtYXJrZXRQcmljZTogCQCkAwEFC21hcmtldFByaWNlCQCWCgQFFWFtb3VudEJhc2VBc3NldEJvdWdodAUXcXVvdGVBc3NldFJlc2VydmVBZnRlcjEFFmJhc2VBc3NldFJlc2VydmVBZnRlcjEFF3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIxAQ9jYWxjUm9sbG92ZXJGZWUCEl9vbGRQb3NpdGlvbk1hcmdpbiBfb2xkUG9zaXRpb25MYXN0VXBkYXRlZFRpbWVzdGFtcAQPcG9zaXRpb25NaW51dGVzCQBoAgkAaQIJAGkCCQBlAgkBDWxhc3RUaW1lc3RhbXAABSBfb2xkUG9zaXRpb25MYXN0VXBkYXRlZFRpbWVzdGFtcADoBwA8BQxERUNJTUFMX1VOSVQEC3JvbGxvdmVyRmVlCQEEZGl2ZAIJAQRtdWxkAgkBBG11bGQCBRJfb2xkUG9zaXRpb25NYXJnaW4FD3Bvc2l0aW9uTWludXRlcwkBD3JvbGxvdmVyRmVlUmF0ZQAFD01JTlVURVNfSU5fWUVBUgULcm9sbG92ZXJGZWUBMGNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnRBbmRSb2xsb3ZlckZlZQUQX29sZFBvc2l0aW9uU2l6ZRJfb2xkUG9zaXRpb25NYXJnaW4lX29sZFBvc2l0aW9uQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbiBfb2xkUG9zaXRpb25MYXN0VXBkYXRlZFRpbWVzdGFtcAxfbWFyZ2luRGVsdGEEDmZ1bmRpbmdQYXltZW50AwkBAiE9AgUQX29sZFBvc2l0aW9uU2l6ZQAABCBfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBRBfb2xkUG9zaXRpb25TaXplCQEEbXVsZAIJAGUCBSBfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgUlX29sZFBvc2l0aW9uQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgUQX29sZFBvc2l0aW9uU2l6ZQAABAtyb2xsb3ZlckZlZQkBD2NhbGNSb2xsb3ZlckZlZQIFEl9vbGRQb3NpdGlvbk1hcmdpbgUgX29sZFBvc2l0aW9uTGFzdFVwZGF0ZWRUaW1lc3RhbXAEDHNpZ25lZE1hcmdpbgkAZAIJAGUCCQBlAgUMX21hcmdpbkRlbHRhBQtyb2xsb3ZlckZlZQUOZnVuZGluZ1BheW1lbnQFEl9vbGRQb3NpdGlvbk1hcmdpbgQNJHQwMTg2MjMxODc1MAMJAGYCAAAFDHNpZ25lZE1hcmdpbgkAlAoCAAAJAQNhYnMBBQxzaWduZWRNYXJnaW4JAJQKAgkBA2FicwEFDHNpZ25lZE1hcmdpbgAABAxyZW1haW5NYXJnaW4IBQ0kdDAxODYyMzE4NzUwAl8xBAdiYWREZWJ0CAUNJHQwMTg2MjMxODc1MAJfMgkAlgoEBQxyZW1haW5NYXJnaW4FB2JhZERlYnQFDmZ1bmRpbmdQYXltZW50BQtyb2xsb3ZlckZlZQEWc3dhcE91dHB1dFdpdGhSZXNlcnZlcwcGX2lzQWRkEF9iYXNlQXNzZXRBbW91bnQUX2NoZWNrTWF4UHJpY2VJbXBhY3QSX3F1b3RlQXNzZXRSZXNlcnZlEV9xdW90ZUFzc2V0V2VpZ2h0EV9iYXNlQXNzZXRSZXNlcnZlEF9iYXNlQXNzZXRXZWlnaHQEC3ByaWNlQmVmb3JlCQEEZGl2ZAIJAQRtdWxkAgUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfcXVvdGVBc3NldFdlaWdodAkBBG11bGQCBRFfYmFzZUFzc2V0UmVzZXJ2ZQUQX2Jhc2VBc3NldFdlaWdodAMJAAACBRBfYmFzZUFzc2V0QW1vdW50AAAJAAIBAhlJbnZhbGlkIGJhc2UgYXNzZXQgYW1vdW50BAFrCQENY2FsY0ludmFyaWFudAIFEl9xdW90ZUFzc2V0UmVzZXJ2ZQURX2Jhc2VBc3NldFJlc2VydmUEGGJhc2VBc3NldFBvb2xBbW91bnRBZnRlcgMFBl9pc0FkZAkAZAIFEV9iYXNlQXNzZXRSZXNlcnZlBRBfYmFzZUFzc2V0QW1vdW50CQBlAgURX2Jhc2VBc3NldFJlc2VydmUFEF9iYXNlQXNzZXRBbW91bnQED3F1b3RlQXNzZXRBZnRlcgkAoAMBCQEFYmRpdmQCBQFrCQC2AgEFGGJhc2VBc3NldFBvb2xBbW91bnRBZnRlcgQPcXVvdGVBc3NldERlbHRhCQEDYWJzAQkAZQIFD3F1b3RlQXNzZXRBZnRlcgUSX3F1b3RlQXNzZXRSZXNlcnZlBA5xdW90ZUFzc2V0U29sZAkBBG11bGQCBQ9xdW90ZUFzc2V0RGVsdGEFEV9xdW90ZUFzc2V0V2VpZ2h0BBNtYXhQcmljZUltcGFjdFZhbHVlCQEObWF4UHJpY2VJbXBhY3QABA0kdDAyMDAxMjIwMTc0CQENdXBkYXRlUmVzZXJ2ZQMJAQEhAQUGX2lzQWRkBQ9xdW90ZUFzc2V0RGVsdGEFEF9iYXNlQXNzZXRBbW91bnQEF3F1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIxCAUNJHQwMjAwMTIyMDE3NAJfMQQWYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyMQgFDSR0MDIwMDEyMjAxNzQCXzIEF3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIxCAUNJHQwMjAwMTIyMDE3NAJfMwQLbWFya2V0UHJpY2UJAQRkaXZkAgUOcXVvdGVBc3NldFNvbGQFEF9iYXNlQXNzZXRBbW91bnQECXByaWNlRGlmZgkBA2FicwEJAGUCBQtwcmljZUJlZm9yZQULbWFya2V0UHJpY2UEC3ByaWNlSW1wYWN0CQBlAgUMREVDSU1BTF9VTklUCQEEZGl2ZAIFC3ByaWNlQmVmb3JlCQBkAgULcHJpY2VCZWZvcmUFCXByaWNlRGlmZgMDCQBmAgULcHJpY2VJbXBhY3QFE21heFByaWNlSW1wYWN0VmFsdWUFFF9jaGVja01heFByaWNlSW1wYWN0BwkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgINUHJpY2UgaW1wYWN0IAkApAMBBQtwcmljZUltcGFjdAIUID4gbWF4IHByaWNlIGltcGFjdCAJAKQDAQUTbWF4UHJpY2VJbXBhY3RWYWx1ZQIVIGJlZm9yZSBxdW90ZSBhc3NldDogCQCkAwEFEl9xdW90ZUFzc2V0UmVzZXJ2ZQIUIGJlZm9yZSBiYXNlIGFzc2V0OiAJAKQDAQURX2Jhc2VBc3NldFJlc2VydmUCICBiYXNlIGFzc2V0IGFtb3VudCB0byBleGNoYW5nZTogCQCkAwEFEF9iYXNlQXNzZXRBbW91bnQCDyBwcmljZSBiZWZvcmU6IAkApAMBBQtwcmljZUJlZm9yZQIPIG1hcmtldCBwcmljZTogCQCkAwEFC21hcmtldFByaWNlCQCZCgcFDnF1b3RlQXNzZXRTb2xkBRdxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyMQUWYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyMQUXdG90YWxQb3NpdGlvblNpemVBZnRlcjEJAGUCCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAMFBl9pc0FkZAkBA2FicwEFEF9iYXNlQXNzZXRBbW91bnQAAAkAZQIJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAMJAQEhAQUGX2lzQWRkCQEDYWJzAQUQX2Jhc2VBc3NldEFtb3VudAAABQtwcmljZUltcGFjdAEKc3dhcE91dHB1dAMGX2lzQWRkEF9iYXNlQXNzZXRBbW91bnQUX2NoZWNrTWF4UHJpY2VJbXBhY3QJARZzd2FwT3V0cHV0V2l0aFJlc2VydmVzBwUGX2lzQWRkBRBfYmFzZUFzc2V0QW1vdW50BRRfY2hlY2tNYXhQcmljZUltcGFjdAkBBnF0QXN0UgAJAQZxdEFzdFcACQEGYnNBc3RSAAkBBmJzQXN0VwABE2dldE9yYWNsZVByaWNlVmFsdWUDBm9yYWNsZQhwcmljZUtleQhibG9ja0tleQQJbGFzdFZhbHVlCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUGb3JhY2xlBQhwcmljZUtleQkArAICCQCsAgIJAKwCAgIiQ2FuIG5vdCBnZXQgb3JhY2xlIHByaWNlLiBPcmFjbGU6IAkApQgBBQZvcmFjbGUCBiBrZXk6IAUIcHJpY2VLZXkDCQECIT0CBQhibG9ja0tleQIABAxjdXJyZW50QmxvY2sFBmhlaWdodAQPbGFzdE9yYWNsZUJsb2NrCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUGb3JhY2xlBQhibG9ja0tleQkArAICCQCsAgIJAKwCAgIiQ2FuIG5vdCBnZXQgb3JhY2xlIGJsb2NrLiBPcmFjbGU6IAkApQgBBQZvcmFjbGUCBiBrZXk6IAUIYmxvY2tLZXkDCQBmAgkAZQIFDGN1cnJlbnRCbG9jawUPbGFzdE9yYWNsZUJsb2NrCQEObWF4T3JhY2xlRGVsYXkACQACAQkArAICCQCsAgIJAKwCAgImT3JhY2xlIHN0YWxlIGRhdGEuIExhc3Qgb3JhY2xlIGJsb2NrOiAJAKQDAQUPbGFzdE9yYWNsZUJsb2NrAhAgY3VycmVudCBibG9jazogCQCkAwEFDGN1cnJlbnRCbG9jawUJbGFzdFZhbHVlBQlsYXN0VmFsdWUBDmdldE9yYWNsZVByaWNlAAQKYmFzZU9yYWNsZQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQENZ2V0T3JhY2xlRGF0YQEFDGtfYmFzZU9yYWNsZQIZTm8gYmFzZSBhc3NldCBvcmFjbGUgZGF0YQQPYmFzZU9yYWNsZVByaWNlCQETZ2V0T3JhY2xlUHJpY2VWYWx1ZQMIBQpiYXNlT3JhY2xlAl8xCAUKYmFzZU9yYWNsZQJfMggFCmJhc2VPcmFjbGUCXzMEC3F1b3RlT3JhY2xlCQENZ2V0T3JhY2xlRGF0YQEFDWtfcXVvdGVPcmFjbGUEEHF1b3RlT3JhY2xlUHJpY2UDCQEJaXNEZWZpbmVkAQULcXVvdGVPcmFjbGUEDHF1b3RlT3JhY2xlVgkBBXZhbHVlAQULcXVvdGVPcmFjbGUJARNnZXRPcmFjbGVQcmljZVZhbHVlAwgFDHF1b3RlT3JhY2xlVgJfMQgFDHF1b3RlT3JhY2xlVgJfMggFDHF1b3RlT3JhY2xlVgJfMwUMREVDSU1BTF9VTklUCQEEZGl2ZAIFD2Jhc2VPcmFjbGVQcmljZQUQcXVvdGVPcmFjbGVQcmljZQEOaXNNYXJrZXRDbG9zZWQABApiYXNlT3JhY2xlCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAQ1nZXRPcmFjbGVEYXRhAQUMa19iYXNlT3JhY2xlAhlObyBiYXNlIGFzc2V0IG9yYWNsZSBkYXRhBAZvcmFjbGUIBQpiYXNlT3JhY2xlAl8xBAdvcGVuS2V5CAUKYmFzZU9yYWNsZQJfNAMJAQIhPQIFB29wZW5LZXkCAAQGaXNPcGVuCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJsIAgUGb3JhY2xlBQdvcGVuS2V5CQCsAgIJAKwCAgkArAICAitDYW4gbm90IGdldCBvcmFjbGUgaXMgb3Blbi9jbG9zZWQuIE9yYWNsZTogCQClCAEFBm9yYWNsZQIGIGtleTogBQdvcGVuS2V5CQEBIQEFBmlzT3BlbgcBDGFic1ByaWNlRGlmZgUMX29yYWNsZVByaWNlEl9xdW90ZUFzc2V0UmVzZXJ2ZRFfYmFzZUFzc2V0UmVzZXJ2ZQdfcXRBc3RXB19ic0FzdFcECnByaWNlQWZ0ZXIJAQRkaXZkAgkBBG11bGQCBRJfcXVvdGVBc3NldFJlc2VydmUFB19xdEFzdFcJAQRtdWxkAgURX2Jhc2VBc3NldFJlc2VydmUFB19ic0FzdFcEDGF2ZXJhZ2VQcmljZQkBBGRpdmQCCQBkAgUMX29yYWNsZVByaWNlBQpwcmljZUFmdGVyCQBoAgACBQxERUNJTUFMX1VOSVQEDGFic1ByaWNlRGlmZgkBBGRpdmQCCQEDYWJzAQkAZQIFDF9vcmFjbGVQcmljZQUKcHJpY2VBZnRlcgUMYXZlcmFnZVByaWNlBQxhYnNQcmljZURpZmYBGXJlcXVpcmVOb3RPdmVyU3ByZWFkTGltaXQCEl9xdW90ZUFzc2V0UmVzZXJ2ZRFfYmFzZUFzc2V0UmVzZXJ2ZQQLb3JhY2xlUHJpY2UJAQ5nZXRPcmFjbGVQcmljZQAEB19xdEFzdFcJAQZxdEFzdFcABAdfYnNBc3RXCQEGYnNBc3RXAAQSYWJzUHJpY2VEaWZmQmVmb3JlCQEMYWJzUHJpY2VEaWZmBQULb3JhY2xlUHJpY2UJAQZxdEFzdFIACQEGYnNBc3RSAAUHX3F0QXN0VwUHX2JzQXN0VwQRYWJzUHJpY2VEaWZmQWZ0ZXIJAQxhYnNQcmljZURpZmYFBQtvcmFjbGVQcmljZQUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfYmFzZUFzc2V0UmVzZXJ2ZQUHX3F0QXN0VwUHX2JzQXN0VwMDCQBmAgURYWJzUHJpY2VEaWZmQWZ0ZXIJAQ5tYXhQcmljZVNwcmVhZAAJAGYCBRFhYnNQcmljZURpZmZBZnRlcgUSYWJzUHJpY2VEaWZmQmVmb3JlBwkAAgEJAKwCAgkArAICCQCsAgICDVByaWNlIHNwcmVhZCAJAKQDAQURYWJzUHJpY2VEaWZmQWZ0ZXICFCA+IG1heCBwcmljZSBzcHJlYWQgCQCkAwEJAQ5tYXhQcmljZVNwcmVhZAAGAR1yZXF1aXJlTm90T3Zlck1heE9wZW5Ob3Rpb25hbAIRX2xvbmdPcGVuTm90aW9uYWwSX3Nob3J0T3Blbk5vdGlvbmFsBBBfbWF4T3Blbk5vdGlvbmFsCQEPbWF4T3Blbk5vdGlvbmFsAAMJAGYCBRFfbG9uZ09wZW5Ob3Rpb25hbAUQX21heE9wZW5Ob3Rpb25hbAkAAgEJAKwCAgkArAICCQCsAgICE0xvbmcgb3BlbiBub3Rpb25hbCAJAKQDAQURX2xvbmdPcGVuTm90aW9uYWwCFSA+IG1heCBvcGVuIG5vdGlvbmFsIAkApAMBBRBfbWF4T3Blbk5vdGlvbmFsAwkAZgIFEl9zaG9ydE9wZW5Ob3Rpb25hbAUQX21heE9wZW5Ob3Rpb25hbAkAAgEJAKwCAgkArAICCQCsAgICFFNob3J0IG9wZW4gbm90aW9uYWwgCQCkAwEFEl9zaG9ydE9wZW5Ob3Rpb25hbAIVID4gbWF4IG9wZW4gbm90aW9uYWwgCQCkAwEFEF9tYXhPcGVuTm90aW9uYWwGAQxnZXRTcG90UHJpY2UABBJfcXVvdGVBc3NldFJlc2VydmUJAQZxdEFzdFIABBFfYmFzZUFzc2V0UmVzZXJ2ZQkBBmJzQXN0UgAEB19xdEFzdFcJAQZxdEFzdFcABAdfYnNBc3RXCQEGYnNBc3RXAAkBBGRpdmQCCQEEbXVsZAIFEl9xdW90ZUFzc2V0UmVzZXJ2ZQUHX3F0QXN0VwkBBG11bGQCBRFfYmFzZUFzc2V0UmVzZXJ2ZQUHX2JzQXN0VwEWaXNPdmVyRmx1Y3R1YXRpb25MaW1pdAAEC29yYWNsZVByaWNlCQEOZ2V0T3JhY2xlUHJpY2UABAxjdXJyZW50UHJpY2UJAQxnZXRTcG90UHJpY2UACQBmAgkBBGRpdmQCCQEDYWJzAQkAZQIFC29yYWNsZVByaWNlBQxjdXJyZW50UHJpY2UFC29yYWNsZVByaWNlCQELc3ByZWFkTGltaXQAAR9nZXRQb3NpdGlvbkFkanVzdGVkT3Blbk5vdGlvbmFsBg1fcG9zaXRpb25TaXplB19vcHRpb24SX3F1b3RlQXNzZXRSZXNlcnZlEV9xdW90ZUFzc2V0V2VpZ2h0EV9iYXNlQXNzZXRSZXNlcnZlEF9iYXNlQXNzZXRXZWlnaHQED3Bvc2l0aW9uU2l6ZUFicwkBA2FicwEFDV9wb3NpdGlvblNpemUEB2lzU2hvcnQJAGYCAAAFDV9wb3NpdGlvblNpemUEEHBvc2l0aW9uTm90aW9uYWwDCQAAAgUHX29wdGlvbgUPUE5MX09QVElPTl9TUE9UBBNvdXRQb3NpdGlvbk5vdGlvbmFsCAkBFnN3YXBPdXRwdXRXaXRoUmVzZXJ2ZXMHCQEBIQEFB2lzU2hvcnQFD3Bvc2l0aW9uU2l6ZUFicwcFEl9xdW90ZUFzc2V0UmVzZXJ2ZQURX3F1b3RlQXNzZXRXZWlnaHQFEV9iYXNlQXNzZXRSZXNlcnZlBRBfYmFzZUFzc2V0V2VpZ2h0Al8xBRNvdXRQb3NpdGlvbk5vdGlvbmFsCQEEbXVsZAIFD3Bvc2l0aW9uU2l6ZUFicwkBDmdldE9yYWNsZVByaWNlAAUQcG9zaXRpb25Ob3Rpb25hbAErZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmxCeVZhbHVlcwcNX3Bvc2l0aW9uU2l6ZRVfcG9zaXRpb25PcGVuTm90aW9uYWwSX3F1b3RlQXNzZXRSZXNlcnZlEV9xdW90ZUFzc2V0V2VpZ2h0EV9iYXNlQXNzZXRSZXNlcnZlEF9iYXNlQXNzZXRXZWlnaHQHX29wdGlvbgMJAAACBQ1fcG9zaXRpb25TaXplAAAJAAIBAhVJbnZhbGlkIHBvc2l0aW9uIHNpemUEB2lzU2hvcnQJAGYCAAAFDV9wb3NpdGlvblNpemUEEHBvc2l0aW9uTm90aW9uYWwJAR9nZXRQb3NpdGlvbkFkanVzdGVkT3Blbk5vdGlvbmFsBgUNX3Bvc2l0aW9uU2l6ZQUHX29wdGlvbgUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfcXVvdGVBc3NldFdlaWdodAURX2Jhc2VBc3NldFJlc2VydmUFEF9iYXNlQXNzZXRXZWlnaHQEDXVucmVhbGl6ZWRQbmwDBQdpc1Nob3J0CQBlAgUVX3Bvc2l0aW9uT3Blbk5vdGlvbmFsBRBwb3NpdGlvbk5vdGlvbmFsCQBlAgUQcG9zaXRpb25Ob3Rpb25hbAUVX3Bvc2l0aW9uT3Blbk5vdGlvbmFsCQCUCgIFEHBvc2l0aW9uTm90aW9uYWwFDXVucmVhbGl6ZWRQbmwBI2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sAwdfdHJhZGVyCl9kaXJlY3Rpb24HX29wdGlvbgQNJHQwMjgwNTkyODE5OQkBC2dldFBvc2l0aW9uAgUHX3RyYWRlcgUKX2RpcmVjdGlvbgQMcG9zaXRpb25TaXplCAUNJHQwMjgwNTkyODE5OQJfMQQOcG9zaXRpb25NYXJnaW4IBQ0kdDAyODA1OTI4MTk5Al8yBBRwb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDI4MDU5MjgxOTkCXzMEEXBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwMjgwNTkyODE5OQJfNAkBK2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sQnlWYWx1ZXMHBQxwb3NpdGlvblNpemUFFHBvc2l0aW9uT3Blbk5vdGlvbmFsCQEGcXRBc3RSAAkBBnF0QXN0VwAJAQZic0FzdFIACQEGYnNBc3RXAAUHX29wdGlvbgEPY2FsY01hcmdpblJhdGlvAw1fcmVtYWluTWFyZ2luCF9iYWREZWJ0EV9wb3NpdGlvbk5vdGlvbmFsCQEEZGl2ZAIJAGUCBQ1fcmVtYWluTWFyZ2luBQhfYmFkRGVidAURX3Bvc2l0aW9uTm90aW9uYWwBFmdldE1hcmdpblJhdGlvQnlPcHRpb24DB190cmFkZXIKX2RpcmVjdGlvbgdfb3B0aW9uBA0kdDAyODczMTI4ODg0CQELZ2V0UG9zaXRpb24CBQdfdHJhZGVyBQpfZGlyZWN0aW9uBAxwb3NpdGlvblNpemUIBQ0kdDAyODczMTI4ODg0Al8xBA5wb3NpdGlvbk1hcmdpbggFDSR0MDI4NzMxMjg4ODQCXzIEA3BvbggFDSR0MDI4NzMxMjg4ODQCXzMEFnBvc2l0aW9uTGFzdFVwZGF0ZWRDUEYIBQ0kdDAyODczMTI4ODg0Al80BBFwb3NpdGlvblRpbWVzdGFtcAgFDSR0MDI4NzMxMjg4ODQCXzUEDSR0MDI4ODkwMjg5OTUJASNnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubAMFB190cmFkZXIFCl9kaXJlY3Rpb24FB19vcHRpb24EEHBvc2l0aW9uTm90aW9uYWwIBQ0kdDAyODg5MDI4OTk1Al8xBA11bnJlYWxpemVkUG5sCAUNJHQwMjg4OTAyODk5NQJfMgQNJHQwMjkwMDAyOTIxMgkBMGNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnRBbmRSb2xsb3ZlckZlZQUFDHBvc2l0aW9uU2l6ZQUOcG9zaXRpb25NYXJnaW4FFnBvc2l0aW9uTGFzdFVwZGF0ZWRDUEYFEXBvc2l0aW9uVGltZXN0YW1wBQ11bnJlYWxpemVkUG5sBAxyZW1haW5NYXJnaW4IBQ0kdDAyOTAwMDI5MjEyAl8xBAdiYWREZWJ0CAUNJHQwMjkwMDAyOTIxMgJfMgkBD2NhbGNNYXJnaW5SYXRpbwMFDHJlbWFpbk1hcmdpbgUHYmFkRGVidAUQcG9zaXRpb25Ob3Rpb25hbAEOZ2V0TWFyZ2luUmF0aW8CB190cmFkZXIKX2RpcmVjdGlvbgkBFmdldE1hcmdpblJhdGlvQnlPcHRpb24DBQdfdHJhZGVyBQpfZGlyZWN0aW9uBQ9QTkxfT1BUSU9OX1NQT1QBG2dldFBhcnRpYWxMaXF1aWRhdGlvbkFtb3VudAIHX3RyYWRlcg1fcG9zaXRpb25TaXplBAxtYXhpbXVtUmF0aW8JAQR2bWF4AgkBF3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvAAkAZQIFDERFQ0lNQUxfVU5JVAkBBGRpdmQCCQEOZ2V0TWFyZ2luUmF0aW8CBQdfdHJhZGVyCQEMZ2V0RGlyZWN0aW9uAQUNX3Bvc2l0aW9uU2l6ZQkBFm1haW50ZW5hbmNlTWFyZ2luUmF0aW8ABBhtYXhFeGNoYW5nZWRQb3NpdGlvblNpemUJAQRtdWxkAgkBA2FicwEFDV9wb3NpdGlvblNpemUFDG1heGltdW1SYXRpbwQKc3dhcFJlc3VsdAkBCnN3YXBPdXRwdXQDCQBmAgUNX3Bvc2l0aW9uU2l6ZQAABRhtYXhFeGNoYW5nZWRQb3NpdGlvblNpemUHBBxtYXhFeGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50CAUKc3dhcFJlc3VsdAJfMQQLcHJpY2VJbXBhY3QIBQpzd2FwUmVzdWx0Al83AwkAZgIJAQ5tYXhQcmljZUltcGFjdAAFC3ByaWNlSW1wYWN0BRhtYXhFeGNoYW5nZWRQb3NpdGlvblNpemUJAQRtdWxkAgkBA2FicwEFDV9wb3NpdGlvblNpemUJARdwYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwABFWludGVybmFsQ2xvc2VQb3NpdGlvbggHX3RyYWRlcgpfZGlyZWN0aW9uBV9zaXplBF9mZWUUX21pblF1b3RlQXNzZXRBbW91bnQMX2FkZFRvTWFyZ2luFF9jaGVja01heFByaWNlSW1wYWN0Cl9saXF1aWRhdGUEDSR0MDMwMzc1MzA1NDMJAQtnZXRQb3NpdGlvbgIFB190cmFkZXIFCl9kaXJlY3Rpb24ED29sZFBvc2l0aW9uU2l6ZQgFDSR0MDMwMzc1MzA1NDMCXzEEEW9sZFBvc2l0aW9uTWFyZ2luCAUNJHQwMzAzNzUzMDU0MwJfMgQXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDAzMDM3NTMwNTQzAl8zBBRvbGRQb3NpdGlvbkxzdFVwZENQRggFDSR0MDMwMzc1MzA1NDMCXzQEFG9sZFBvc2l0aW9uVGltZXN0YW1wCAUNJHQwMzAzNzUzMDU0MwJfNQQOaXNMb25nUG9zaXRpb24JAGYCBQ9vbGRQb3NpdGlvblNpemUAAAQSYWJzT2xkUG9zaXRpb25TaXplCQEDYWJzAQUPb2xkUG9zaXRpb25TaXplAwMJAGcCBRJhYnNPbGRQb3NpdGlvblNpemUFBV9zaXplCQBmAgUFX3NpemUAAAcEDmlzUGFydGlhbENsb3NlCQBmAgUSYWJzT2xkUG9zaXRpb25TaXplBQVfc2l6ZQQNJHQwMzA4MzUzMTI4NgkBCnN3YXBPdXRwdXQDCQBmAgUPb2xkUG9zaXRpb25TaXplAAAFBV9zaXplBRRfY2hlY2tNYXhQcmljZUltcGFjdAQZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAgFDSR0MDMwODM1MzEyODYCXzEEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDAzMDgzNTMxMjg2Al8yBBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDAzMDgzNTMxMjg2Al8zBBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwMzA4MzUzMTI4NgJfNAQVZXhjaGFuZ2VkUG9zaXRpb25TaXplAwkAZgIFD29sZFBvc2l0aW9uU2l6ZQAACQEBLQEFBV9zaXplBQVfc2l6ZQQNJHQwMzE1MDEzMTcyNQkBI2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sAwUHX3RyYWRlcgUKX2RpcmVjdGlvbgUPUE5MX09QVElPTl9TUE9UBBNvbGRQb3NpdGlvbk5vdGlvbmFsCAUNJHQwMzE1MDEzMTcyNQJfMQQNdW5yZWFsaXplZFBubAgFDSR0MDMxNTAxMzE3MjUCXzIEDXJlYWxpemVkUmF0aW8JAQRkaXZkAgkBA2FicwEFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQUSYWJzT2xkUG9zaXRpb25TaXplBAtyZWFsaXplZFBubAkBBG11bGQCBQ11bnJlYWxpemVkUG5sBQ1yZWFsaXplZFJhdGlvBA0kdDAzMjA2NjMyMzEyCQEwY2FsY1JlbWFpbk1hcmdpbldpdGhGdW5kaW5nUGF5bWVudEFuZFJvbGxvdmVyRmVlBQUPb2xkUG9zaXRpb25TaXplBRFvbGRQb3NpdGlvbk1hcmdpbgUUb2xkUG9zaXRpb25Mc3RVcGRDUEYFFG9sZFBvc2l0aW9uVGltZXN0YW1wBQ11bnJlYWxpemVkUG5sBBJyZW1haW5NYXJnaW5CZWZvcmUIBQ0kdDAzMjA2NjMyMzEyAl8xBAJ4MQgFDSR0MDMyMDY2MzIzMTICXzIEAngyCAUNJHQwMzIwNjYzMjMxMgJfMwQLcm9sbG92ZXJGZWUIBQ0kdDAzMjA2NjMyMzEyAl80BA9wb3NpdGlvbkJhZERlYnQICQEwY2FsY1JlbWFpbk1hcmdpbldpdGhGdW5kaW5nUGF5bWVudEFuZFJvbGxvdmVyRmVlBQUPb2xkUG9zaXRpb25TaXplBRFvbGRQb3NpdGlvbk1hcmdpbgUUb2xkUG9zaXRpb25Mc3RVcGRDUEYFFG9sZFBvc2l0aW9uVGltZXN0YW1wBQtyZWFsaXplZFBubAJfMgQQcmVhbGl6ZWRDbG9zZUZlZQkBBG11bGQCCQEEbXVsZAIFE29sZFBvc2l0aW9uTm90aW9uYWwFDXJlYWxpemVkUmF0aW8FBF9mZWUEEnVucmVhbGl6ZWRQbmxBZnRlcgkAZQIFDXVucmVhbGl6ZWRQbmwFC3JlYWxpemVkUG5sBBJyZW1haW5PcGVuTm90aW9uYWwDCQBmAgUPb2xkUG9zaXRpb25TaXplAAAJAGUCCQBlAgUTb2xkUG9zaXRpb25Ob3Rpb25hbAUZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAUSdW5yZWFsaXplZFBubEFmdGVyCQBlAgkAZAIFEnVucmVhbGl6ZWRQbmxBZnRlcgUTb2xkUG9zaXRpb25Ob3Rpb25hbAUZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAQPbmV3UG9zaXRpb25TaXplCQBkAgUPb2xkUG9zaXRpb25TaXplBRVleGNoYW5nZWRQb3NpdGlvblNpemUEDSR0MDMzNzE4MzQxMDQDCQAAAgUPbmV3UG9zaXRpb25TaXplAAAJAJQKAgAAAAAJAJQKAgkBA2FicwEFEnJlbWFpbk9wZW5Ob3Rpb25hbAkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBQ9uZXdQb3NpdGlvblNpemUEF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwMzM3MTgzNDEwNAJfMQQUbmV3UG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDAzMzcxODM0MTA0Al8yBBFvcGVuTm90aW9uYWxEZWx0YQkAZQIFF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAQLbWFyZ2luUmF0aW8JARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAwUHX3RyYWRlcgUKX2RpcmVjdGlvbgUPUE5MX09QVElPTl9TUE9UBB5uZXdQb3NpdGlvbk1hcmdpbldpdGhTYW1lUmF0aW8DCQBmAgUPb2xkUG9zaXRpb25TaXplAAAJAGUCCQEEbXVsZAIJAGQCBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUSdW5yZWFsaXplZFBubEFmdGVyBQttYXJnaW5SYXRpbwUSdW5yZWFsaXplZFBubEFmdGVyCQBlAgkBBG11bGQCCQBlAgUXbmV3UG9zaXRpb25PcGVuTm90aW9uYWwFEnVucmVhbGl6ZWRQbmxBZnRlcgULbWFyZ2luUmF0aW8FEnVucmVhbGl6ZWRQbmxBZnRlcgQRbWFyZ2luVG9UcmFkZXJSYXcJAGUCCQBlAgUScmVtYWluTWFyZ2luQmVmb3JlCQBkAgUebmV3UG9zaXRpb25NYXJnaW5XaXRoU2FtZVJhdGlvBRJ1bnJlYWxpemVkUG5sQWZ0ZXIFEHJlYWxpemVkQ2xvc2VGZWUEDm1hcmdpblRvVHJhZGVyAwkAZgIAAAURbWFyZ2luVG9UcmFkZXJSYXcDBQpfbGlxdWlkYXRlAAAJAAIBAjdJbnZhbGlkIGludGVybmFsQ2xvc2VQb3NpdGlvbiBwYXJhbXM6IHVuYWJsZSB0byBwYXkgZmVlBRFtYXJnaW5Ub1RyYWRlclJhdwQRbmV3UG9zaXRpb25NYXJnaW4DBQxfYWRkVG9NYXJnaW4JAGQCBR5uZXdQb3NpdGlvbk1hcmdpbldpdGhTYW1lUmF0aW8FDm1hcmdpblRvVHJhZGVyBR5uZXdQb3NpdGlvbk1hcmdpbldpdGhTYW1lUmF0aW8DAwkBAiE9AgUUX21pblF1b3RlQXNzZXRBbW91bnQAAAkAZgIFFF9taW5RdW90ZUFzc2V0QW1vdW50BRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BwkAAgEJAKwCAgkArAICCQCsAgICDUxpbWl0IGVycm9yOiAJAKQDAQUZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAIDIDwgCQCkAwEFFF9taW5RdW90ZUFzc2V0QW1vdW50CQCjChEFD25ld1Bvc2l0aW9uU2l6ZQURbmV3UG9zaXRpb25NYXJnaW4FF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBRRuZXdQb3NpdGlvbkxzdFVwZENQRgUPcG9zaXRpb25CYWREZWJ0BQtyZWFsaXplZFBubAMDBQxfYWRkVG9NYXJnaW4FDmlzUGFydGlhbENsb3NlBwAABQ5tYXJnaW5Ub1RyYWRlcgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBRZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCQBlAgkBFG9wZW5JbnRlcmVzdE5vdGlvbmFsAAURb3Blbk5vdGlvbmFsRGVsdGEJAGUCCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAMFDmlzTG9uZ1Bvc2l0aW9uCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplAAAJAGUCCQEWdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQADCQEBIQEFDmlzTG9uZ1Bvc2l0aW9uCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplAAAJAGUCCQEQb3BlbkludGVyZXN0TG9uZwADBQ5pc0xvbmdQb3NpdGlvbgURb3Blbk5vdGlvbmFsRGVsdGEAAAkAZQIJARFvcGVuSW50ZXJlc3RTaG9ydAADCQEBIQEFDmlzTG9uZ1Bvc2l0aW9uBRFvcGVuTm90aW9uYWxEZWx0YQAACQBkAgUQcmVhbGl6ZWRDbG9zZUZlZQULcm9sbG92ZXJGZWUFGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQJAAIBCQCsAgIJAKwCAgkArAICAj1JbnZhbGlkIGludGVybmFsQ2xvc2VQb3NpdGlvbiBwYXJhbXM6IGludmFsaWQgcG9zaXRpb24gc2l6ZTogCQCkAwEFBV9zaXplAgYgbWF4OiAJAKQDAQUSYWJzT2xkUG9zaXRpb25TaXplARNnZXRUZXJtaW5hbEFtbVN0YXRlAAQNX3Bvc2l0aW9uU2l6ZQkBEXRvdGFsUG9zaXRpb25TaXplAAMJAAACBQ1fcG9zaXRpb25TaXplAAAJAJQKAgkBBnF0QXN0UgAJAQZic0FzdFIABAlkaXJlY3Rpb24JAGYCBQ1fcG9zaXRpb25TaXplAAAEDSR0MDM3MzUxMzc1MzAJAQpzd2FwT3V0cHV0AwUJZGlyZWN0aW9uCQEDYWJzAQUNX3Bvc2l0aW9uU2l6ZQcEFWN1cnJlbnROZXRNYXJrZXRWYWx1ZQgFDSR0MDM3MzUxMzc1MzACXzEEGXRlcm1pbmFsUXVvdGVBc3NldFJlc2VydmUIBQ0kdDAzNzM1MTM3NTMwAl8yBBh0ZXJtaW5hbEJhc2VBc3NldFJlc2VydmUIBQ0kdDAzNzM1MTM3NTMwAl8zCQCUCgIFGXRlcm1pbmFsUXVvdGVBc3NldFJlc2VydmUFGHRlcm1pbmFsQmFzZUFzc2V0UmVzZXJ2ZQETZ2V0UXVvdGVBc3NldFdlaWdodAQQYmFzZUFzc2V0UmVzZXJ2ZRF0b3RhbFBvc2l0aW9uU2l6ZRFxdW90ZUFzc2V0UmVzZXJ2ZQt0YXJnZXRQcmljZQQBYgkAtgIBBRBiYXNlQXNzZXRSZXNlcnZlBAJzegkAtgIBBRF0b3RhbFBvc2l0aW9uU2l6ZQQBcQkAtgIBBRFxdW90ZUFzc2V0UmVzZXJ2ZQQBcAkAtgIBBQt0YXJnZXRQcmljZQQBawkBBWJtdWxkAgUBcQUBYgQEbmV3QgkAtwICBQFiBQJzegQEbmV3UQkBBWJkaXZkAgUBawUEbmV3QgQBegkBBWJkaXZkAgUEbmV3UQUEbmV3QgQGcmVzdWx0CQEFYmRpdmQCBQFwBQF6CQCgAwEFBnJlc3VsdAEUZ2V0U3luY1Rlcm1pbmFsUHJpY2UDDl90ZXJtaW5hbFByaWNlB19xdEFzdFIHX2JzQXN0UgQNX3Bvc2l0aW9uU2l6ZQkBEXRvdGFsUG9zaXRpb25TaXplAAMJAAACBQ1fcG9zaXRpb25TaXplAAAECW5ld1F0QXN0VwkBBGRpdmQCCQEEbXVsZAIFDl90ZXJtaW5hbFByaWNlBQdfYnNBc3RSBQdfcXRBc3RSCQCVCgMFCW5ld1F0QXN0VwUMREVDSU1BTF9VTklUAAAECWRpcmVjdGlvbgkAZgIFDV9wb3NpdGlvblNpemUAAAQVY3VycmVudE5ldE1hcmtldFZhbHVlCAkBCnN3YXBPdXRwdXQDBQlkaXJlY3Rpb24JAQNhYnMBBQ1fcG9zaXRpb25TaXplBwJfMQQJbmV3UXRBc3RXCQETZ2V0UXVvdGVBc3NldFdlaWdodAQFB19ic0FzdFIFDV9wb3NpdGlvblNpemUFB19xdEFzdFIFDl90ZXJtaW5hbFByaWNlBAluZXdCc0FzdFcFDERFQ0lNQUxfVU5JVAQNbWFyZ2luVG9WYXVsdAgJAStnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubEJ5VmFsdWVzBwUNX3Bvc2l0aW9uU2l6ZQUVY3VycmVudE5ldE1hcmtldFZhbHVlBQdfcXRBc3RSBQluZXdRdEFzdFcFB19ic0FzdFIFCW5ld0JzQXN0VwUPUE5MX09QVElPTl9TUE9UAl8yCQCVCgMFCW5ld1F0QXN0VwUJbmV3QnNBc3RXBQ1tYXJnaW5Ub1ZhdWx0AQpnZXRGdW5kaW5nAAQPdW5kZXJseWluZ1ByaWNlCQEOZ2V0T3JhY2xlUHJpY2UABAlzcG90UHJpY2UJAQxnZXRTcG90UHJpY2UABAdwcmVtaXVtCQBlAgUJc3BvdFByaWNlBQ91bmRlcmx5aW5nUHJpY2UDAwkBDmlzTWFya2V0Q2xvc2VkAAYDCQAAAgkBC2Z1bmRpbmdNb2RlAAUSRlVORElOR19BU1lNTUVUUklDAwkAAAIJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAAABgkAAAIJARV0b3RhbExvbmdQb3NpdGlvblNpemUAAAAHCQCVCgMAAAAAAAADCQBmAgAABQdwcmVtaXVtBBRzaG9ydFByZW1pdW1GcmFjdGlvbgkBBGRpdmQCCQEEbXVsZAIFB3ByZW1pdW0JARRmdW5kaW5nUGVyaW9kRGVjaW1hbAAFB09ORV9EQVkDCQAAAgkBC2Z1bmRpbmdNb2RlAAUSRlVORElOR19BU1lNTUVUUklDBBNsb25nUHJlbWl1bUZyYWN0aW9uCQEEZGl2ZAIJAQRtdWxkAgUUc2hvcnRQcmVtaXVtRnJhY3Rpb24JARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQAJAJUKAwUUc2hvcnRQcmVtaXVtRnJhY3Rpb24FE2xvbmdQcmVtaXVtRnJhY3Rpb24AAAQZc2hvcnRUb3RhbFByZW1pdW1GcmFjdGlvbgkBA2FicwEJAQRtdWxkAgUUc2hvcnRQcmVtaXVtRnJhY3Rpb24JARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAQYbG9uZ1RvdGFsUHJlbWl1bUZyYWN0aW9uCQEDYWJzAQkBBG11bGQCBRRzaG9ydFByZW1pdW1GcmFjdGlvbgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQAEDnByZW1pdW1Ub1ZhdWx0CQBlAgUZc2hvcnRUb3RhbFByZW1pdW1GcmFjdGlvbgUYbG9uZ1RvdGFsUHJlbWl1bUZyYWN0aW9uCQCVCgMFFHNob3J0UHJlbWl1bUZyYWN0aW9uBRRzaG9ydFByZW1pdW1GcmFjdGlvbgUOcHJlbWl1bVRvVmF1bHQEE2xvbmdQcmVtaXVtRnJhY3Rpb24JAQRkaXZkAgkBBG11bGQCBQdwcmVtaXVtCQEUZnVuZGluZ1BlcmlvZERlY2ltYWwABQdPTkVfREFZAwkAAAIJAQtmdW5kaW5nTW9kZQAFEkZVTkRJTkdfQVNZTU1FVFJJQwQUc2hvcnRQcmVtaXVtRnJhY3Rpb24JAQRkaXZkAgkBBG11bGQCBRNsb25nUHJlbWl1bUZyYWN0aW9uCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUACQCVCgMFFHNob3J0UHJlbWl1bUZyYWN0aW9uBRNsb25nUHJlbWl1bUZyYWN0aW9uAAAEGGxvbmdUb3RhbFByZW1pdW1GcmFjdGlvbgkBA2FicwEJAQRtdWxkAgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQAEGXNob3J0VG90YWxQcmVtaXVtRnJhY3Rpb24JAQNhYnMBCQEEbXVsZAIFE2xvbmdQcmVtaXVtRnJhY3Rpb24JARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAQOcHJlbWl1bVRvVmF1bHQJAGUCBRhsb25nVG90YWxQcmVtaXVtRnJhY3Rpb24FGXNob3J0VG90YWxQcmVtaXVtRnJhY3Rpb24JAJUKAwUTbG9uZ1ByZW1pdW1GcmFjdGlvbgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgUOcHJlbWl1bVRvVmF1bHQBDmdldEFkanVzdGVkRmVlAgtfYXJ0aWZhY3RJZBBfYmFzZUZlZURpc2NvdW50BApiYXNlRmVlUmF3CQEDZmVlAAQHYmFzZUZlZQkBBG11bGQCBQpiYXNlRmVlUmF3BRBfYmFzZUZlZURpc2NvdW50BA0kdDA0MjE5NDQyNjg5AwkBAiE9AgULX2FydGlmYWN0SWQCAAQMYXJ0aWZhY3RLaW5kCQEEc3RyQQIJARFuZnRNYW5hZ2VyQWRkcmVzcwAJAQ50b0NvbXBvc2l0ZUtleQIFDGtfdG9rZW5fdHlwZQULX2FydGlmYWN0SWQDCQAAAgUMYXJ0aWZhY3RLaW5kBRhGRUVfUkVEVUNUSU9OX1RPS0VOX1RZUEUECXJlZHVjdGlvbgkBBGludEECCQERbmZ0TWFuYWdlckFkZHJlc3MACQEOdG9Db21wb3NpdGVLZXkCBQ1rX3Rva2VuX3BhcmFtBQtfYXJ0aWZhY3RJZAQLYWRqdXN0ZWRGZWUJAQRtdWxkAgUHYmFzZUZlZQUJcmVkdWN0aW9uCQCUCgIFC2FkanVzdGVkRmVlBgkAAgECGUludmFsaWQgYXR0YWNoZWQgYXJ0aWZhY3QJAJQKAgUHYmFzZUZlZQcEC2FkanVzdGVkRmVlCAUNJHQwNDIxOTQ0MjY4OQJfMQQMYnVybkFydGlmYWN0CAUNJHQwNDIxOTQ0MjY4OQJfMgkAlAoCBQthZGp1c3RlZEZlZQUMYnVybkFydGlmYWN0ARhnZXRGb3JUcmFkZXJXaXRoQXJ0aWZhY3QCB190cmFkZXILX2FydGlmYWN0SWQEEGRvR2V0RmVlRGlzY291bnQJAPwHBAkBDG1pbmVyQWRkcmVzcwACEmNvbXB1dGVGZWVEaXNjb3VudAkAzAgCBQdfdHJhZGVyBQNuaWwFA25pbAMJAAACBRBkb0dldEZlZURpc2NvdW50BRBkb0dldEZlZURpc2NvdW50BAtmZWVEaXNjb3VudAQHJG1hdGNoMAUQZG9HZXRGZWVEaXNjb3VudAMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAiFJbnZhbGlkIGNvbXB1dGVGZWVEaXNjb3VudCByZXN1bHQEDSR0MDQzMDM1NDMxMDkJAQ5nZXRBZGp1c3RlZEZlZQIFC19hcnRpZmFjdElkBQtmZWVEaXNjb3VudAQLYWRqdXN0ZWRGZWUIBQ0kdDA0MzAzNTQzMTA5Al8xBAxidXJuQXJ0aWZhY3QIBQ0kdDA0MzAzNTQzMTA5Al8yCQCUCgIFC2FkanVzdGVkRmVlBQxidXJuQXJ0aWZhY3QJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BDWdldEFydGlmYWN0SWQBAWkECmFydGlmYWN0SWQDCQBmAgkAkAMBCAUBaQhwYXltZW50cwABCQDYBAEJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAggJAJEDAggFAWkIcGF5bWVudHMAAQdhc3NldElkAhJJbnZhbGlkIGFydGlmYWN0SWQCAAUKYXJ0aWZhY3RJZAENZGlzdHJpYnV0ZUZlZQEKX2ZlZUFtb3VudAQMZmVlVG9TdGFrZXJzCQEEbXVsZAIFCl9mZWVBbW91bnQJARNmZWVUb1N0YWtlcnNQZXJjZW50AAQKZmVlVG9WYXVsdAkAZQIFCl9mZWVBbW91bnQFDGZlZVRvU3Rha2VycwkAlAoCBQxmZWVUb1N0YWtlcnMFCmZlZVRvVmF1bHQBDnVwZGF0ZVNldHRpbmdzDxBfaW5pdE1hcmdpblJhdGlvBF9tbXIUX2xpcXVpZGF0aW9uRmVlUmF0aW8OX2Z1bmRpbmdQZXJpb2QEX2ZlZQxfc3ByZWFkTGltaXQPX21heFByaWNlSW1wYWN0GF9wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbw9fbWF4UHJpY2VTcHJlYWQQX21heE9wZW5Ob3Rpb25hbBRfZmVlVG9TdGFrZXJzUGVyY2VudA9fbWF4T3JhY2xlRGVsYXkMX3JvbGxvdmVyRmVlDF9mdW5kaW5nTW9kZQtfb3JhY2xlTW9kZQkAzAgCCQEMSW50ZWdlckVudHJ5AgURa19pbml0TWFyZ2luUmF0aW8FEF9pbml0TWFyZ2luUmF0aW8JAMwIAgkBDEludGVnZXJFbnRyeQIFGGtfbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwUEX21tcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUVa19saXF1aWRhdGlvbkZlZVJhdGlvBRRfbGlxdWlkYXRpb25GZWVSYXRpbwkAzAgCCQEMSW50ZWdlckVudHJ5AgUPa19mdW5kaW5nUGVyaW9kBQ5fZnVuZGluZ1BlcmlvZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUFa19mZWUFBF9mZWUJAMwIAgkBDEludGVnZXJFbnRyeQIFDWtfc3ByZWFkTGltaXQFDF9zcHJlYWRMaW1pdAkAzAgCCQEMSW50ZWdlckVudHJ5AgUQa19tYXhQcmljZUltcGFjdAUPX21heFByaWNlSW1wYWN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCBRlrX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvBRhfcGFydGlhbExpcXVpZGF0aW9uUmF0aW8JAMwIAgkBDEludGVnZXJFbnRyeQIFEGtfbWF4UHJpY2VTcHJlYWQFD19tYXhQcmljZVNwcmVhZAkAzAgCCQEMSW50ZWdlckVudHJ5AgURa19tYXhPcGVuTm90aW9uYWwFEF9tYXhPcGVuTm90aW9uYWwJAMwIAgkBDEludGVnZXJFbnRyeQIFFWtfZmVlVG9TdGFrZXJzUGVyY2VudAUUX2ZlZVRvU3Rha2Vyc1BlcmNlbnQJAMwIAgkBDEludGVnZXJFbnRyeQIFEGtfbWF4T3JhY2xlRGVsYXkFD19tYXhPcmFjbGVEZWxheQkAzAgCCQEMSW50ZWdlckVudHJ5AgUNa19yb2xsb3ZlckZlZQUMX3JvbGxvdmVyRmVlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ1rX2Z1bmRpbmdNb2RlBQxfZnVuZGluZ01vZGUJAMwIAgkBDEludGVnZXJFbnRyeQIFDGtfb3JhY2xlTW9kZQULX29yYWNsZU1vZGUFA25pbAENdXBkYXRlRnVuZGluZwURX25leHRGdW5kaW5nQmxvY2skX2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uJV9sYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24QX2xvbmdGdW5kaW5nUmF0ZRFfc2hvcnRGdW5kaW5nUmF0ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUSa19uZXh0RnVuZGluZ0Jsb2NrBRFfbmV4dEZ1bmRpbmdCbG9jawkAzAgCCQEMSW50ZWdlckVudHJ5AgUla19sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgUkX2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uCQDMCAIJAQxJbnRlZ2VyRW50cnkCBSZrX2xhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgUlX2xhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgkAzAgCCQEMSW50ZWdlckVudHJ5AgURa19sb25nRnVuZGluZ1JhdGUFEF9sb25nRnVuZGluZ1JhdGUJAMwIAgkBDEludGVnZXJFbnRyeQIFEmtfc2hvcnRGdW5kaW5nUmF0ZQURX3Nob3J0RnVuZGluZ1JhdGUFA25pbAEfaW5jcmVtZW50UG9zaXRpb25TZXF1ZW5jZU51bWJlcgMOX2lzTmV3UG9zaXRpb24HX3RyYWRlcgpfZGlyZWN0aW9uBAtwb3NpdGlvbktleQkArAICCQCsAgIFB190cmFkZXICAV8JAKQDAQUKX2RpcmVjdGlvbgMFDl9pc05ld1Bvc2l0aW9uBA9jdXJyZW50U2VxdWVuY2UJAQxsYXN0U2VxdWVuY2UACQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRJrX3Bvc2l0aW9uU2VxdWVuY2UFC3Bvc2l0aW9uS2V5CQBkAgUPY3VycmVudFNlcXVlbmNlAAEJAMwIAgkBDEludGVnZXJFbnRyeQIFCmtfc2VxdWVuY2UJAGQCBQ9jdXJyZW50U2VxdWVuY2UAAQUDbmlsBQNuaWwBEXVwZGF0ZVBvc2l0aW9uRmVlBA5faXNOZXdQb3NpdGlvbgdfdHJhZGVyCl9kaXJlY3Rpb24EX2ZlZQQLcG9zaXRpb25LZXkJAKwCAgkArAICBQdfdHJhZGVyAgFfCQCkAwEFCl9kaXJlY3Rpb24DBQ5faXNOZXdQb3NpdGlvbgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUNa19wb3NpdGlvbkZlZQULcG9zaXRpb25LZXkFBF9mZWUFA25pbAUDbmlsAQ51cGRhdGVQb3NpdGlvbgYHX3RyYWRlcgVfc2l6ZQdfbWFyZ2luDV9vcGVuTm90aW9uYWwgX2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24QX2xhdGVzdFRpbWVzdGFtcAQJZGlyZWN0aW9uCQEMZ2V0RGlyZWN0aW9uAQUFX3NpemUEC3Bvc2l0aW9uS2V5CQCsAgIJAKwCAgUHX3RyYWRlcgIBXwkApAMBBQlkaXJlY3Rpb24JAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFDmtfcG9zaXRpb25TaXplBQtwb3NpdGlvbktleQUFX3NpemUJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFEGtfcG9zaXRpb25NYXJnaW4FC3Bvc2l0aW9uS2V5BQdfbWFyZ2luCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsBQtwb3NpdGlvbktleQUNX29wZW5Ob3Rpb25hbAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUua19wb3NpdGlvbkxhc3RVcGRhdGVkQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgULcG9zaXRpb25LZXkFIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBR5rX3Bvc2l0aW9uTGFzdFVwZGF0ZWRUaW1lc3RhbXAFC3Bvc2l0aW9uS2V5BRBfbGF0ZXN0VGltZXN0YW1wBQNuaWwBEXVwZGF0ZUFtbVJlc2VydmVzAgdfcXRBc3RSB19ic0FzdFIDAwkAZgIAAAUHX3F0QXN0UgYJAGYCAAAFB19ic0FzdFIJAAIBAiFJbnZhbGlkIGFtb3VudCB0byB1cGRhdGUgcmVzZXJ2ZXMJAMwIAgkBDEludGVnZXJFbnRyeQIFE2tfcXVvdGVBc3NldFJlc2VydmUFB19xdEFzdFIJAMwIAgkBDEludGVnZXJFbnRyeQIFEmtfYmFzZUFzc2V0UmVzZXJ2ZQUHX2JzQXN0UgUDbmlsARB1cGRhdGVBbW1XZWlnaHRzAgdfcXRBc3RXB19ic0FzdFcJAMwIAgkBDEludGVnZXJFbnRyeQIFEmtfcXVvdGVBc3NldFdlaWdodAUHX3F0QXN0VwkAzAgCCQEMSW50ZWdlckVudHJ5AgURa19iYXNlQXNzZXRXZWlnaHQFB19ic0FzdFcFA25pbAEJdXBkYXRlQW1tCAdfcXRBc3RSB19ic0FzdFIXX3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIVX29wZW5JbnRlcmVzdE5vdGlvbmFsFl90b3RhbExvbmdQb3NpdGlvblNpemUXX3RvdGFsU2hvcnRQb3NpdGlvblNpemUWX3RvdGFsTG9uZ09wZW5Ob3Rpb25hbBdfdG90YWxTaG9ydE9wZW5Ob3Rpb25hbAQHX3F0QXN0VwkBBnF0QXN0VwAEB19ic0FzdFcJAQZic0FzdFcAAwkBAiE9AgkAZQIFFl90b3RhbExvbmdQb3NpdGlvblNpemUFF190b3RhbFNob3J0UG9zaXRpb25TaXplBRdfdG90YWxQb3NpdGlvblNpemVBZnRlcgkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAhhJbnZhbGlkIEFNTSBzdGF0ZSBkYXRhOiAJAKQDAQUWX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQIEICsgIAkApAMBBRdfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQIEICE9IAkApAMBBRdfdG90YWxQb3NpdGlvblNpemVBZnRlcgkAzggCCQERdXBkYXRlQW1tUmVzZXJ2ZXMCBQdfcXRBc3RSBQdfYnNBc3RSCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRNrX3RvdGFsUG9zaXRpb25TaXplBRdfdG90YWxQb3NpdGlvblNpemVBZnRlcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUWa19vcGVuSW50ZXJlc3ROb3Rpb25hbAUVX29wZW5JbnRlcmVzdE5vdGlvbmFsCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRdrX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQUWX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUYa190b3RhbFNob3J0UG9zaXRpb25TaXplBRdfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUSa19vcGVuSW50ZXJlc3RMb25nBRZfdG90YWxMb25nT3Blbk5vdGlvbmFsCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRNrX29wZW5JbnRlcmVzdFNob3J0BRdfdG90YWxTaG9ydE9wZW5Ob3Rpb25hbAUDbmlsAQ5kZWxldGVQb3NpdGlvbgIHX3RyYWRlcgpfZGlyZWN0aW9uBAtwb3NpdGlvbktleQkArAICCQCsAgIFB190cmFkZXICAV8JAKQDAQUKX2RpcmVjdGlvbgkAzAgCCQELRGVsZXRlRW50cnkBCQEOdG9Db21wb3NpdGVLZXkCBQ5rX3Bvc2l0aW9uU2l6ZQULcG9zaXRpb25LZXkJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUQa19wb3NpdGlvbk1hcmdpbgULcG9zaXRpb25LZXkJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUWa19wb3NpdGlvbk9wZW5Ob3Rpb25hbAULcG9zaXRpb25LZXkJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUua19wb3NpdGlvbkxhc3RVcGRhdGVkQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgULcG9zaXRpb25LZXkJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUNa19wb3NpdGlvbkZlZQULcG9zaXRpb25LZXkJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUea19wb3NpdGlvbkxhc3RVcGRhdGVkVGltZXN0YW1wBQtwb3NpdGlvbktleQUDbmlsAQh3aXRoZHJhdwIIX2FkZHJlc3MHX2Ftb3VudAQHYmFsYW5jZQkA8AcCBQR0aGlzCQEKcXVvdGVBc3NldAADCQBmAgUHX2Ftb3VudAUHYmFsYW5jZQkAAgEJAKwCAgkArAICCQCsAgICE1VuYWJsZSB0byB3aXRoZHJhdyAJAKQDAQUHX2Ftb3VudAIXIGZyb20gY29udHJhY3QgYmFsYW5jZSAJAKQDAQUHYmFsYW5jZQkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQhfYWRkcmVzcwUHX2Ftb3VudAkBCnF1b3RlQXNzZXQABQNuaWwBDXVwZGF0ZUJhbGFuY2UBAWkDCQBmAgAABQFpCQACAQIHQmFsYW5jZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUJa19iYWxhbmNlBQFpBQNuaWwBC3RyYW5zZmVyRmVlAQFpCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMJAQ5zdGFraW5nQWRkcmVzcwAFAWkJAQpxdW90ZUFzc2V0AAUDbmlsAQ5kb0J1cm5BcnRpZmFjdAINX2J1cm5BcnRpZmFjdAFpAwUNX2J1cm5BcnRpZmFjdAkAzAgCCQEEQnVybgIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAggJAJEDAggFAWkIcGF5bWVudHMAAQdhc3NldElkAhBJbnZhbGlkIGFydGlmYWN0AAEFA25pbAUDbmlsFwFpAQVwYXVzZQADCQECIT0CCAUBaQZjYWxsZXIJAQxhZG1pbkFkZHJlc3MACQACAQIUSW52YWxpZCBwYXVzZSBwYXJhbXMJAMwIAgkBDEJvb2xlYW5FbnRyeQIFCGtfcGF1c2VkBgUDbmlsAWkBB3VucGF1c2UAAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAkAAgECFkludmFsaWQgdW5wYXVzZSBwYXJhbXMJAMwIAgkBDEJvb2xlYW5FbnRyeQIFCGtfcGF1c2VkBwUDbmlsAWkBDHNldENsb3NlT25seQADCQECIT0CCAUBaQZjYWxsZXIJAQxhZG1pbkFkZHJlc3MACQACAQIbSW52YWxpZCBzZXRDbG9zZU9ubHkgcGFyYW1zCQDMCAIJAQxCb29sZWFuRW50cnkCBQtrX2Nsb3NlT25seQYFA25pbAFpAQ51bnNldENsb3NlT25seQADCQECIT0CCAUBaQZjYWxsZXIJAQxhZG1pbkFkZHJlc3MACQACAQIdSW52YWxpZCB1bnNldENsb3NlT25seSBwYXJhbXMJAMwIAgkBDEJvb2xlYW5FbnRyeQIFC2tfY2xvc2VPbmx5BwUDbmlsAWkBD2NoYW5nZUxpcXVpZGl0eQERX3F1b3RlQXNzZXRBbW91bnQDAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAYJAAACBRFfcXVvdGVBc3NldEFtb3VudAAACQACAQIeSW52YWxpZCBjaGFuZ2VMaXF1aWRpdHkgcGFyYW1zBAdfcXRBc3RSCQEGcXRBc3RSAAQHX2JzQXN0UgkBBmJzQXN0UgAEB19xdEFzdFcJAQZxdEFzdFcABAdfYnNBc3RXCQEGYnNBc3RXAAQFcHJpY2UJAQRkaXZkAgkBBG11bGQCBQdfcXRBc3RSBQdfcXRBc3RXCQEEbXVsZAIFB19ic0FzdFIFB19ic0FzdFcEC3F0QXN0UkFmdGVyCQBkAgUHX3F0QXN0UgURX3F1b3RlQXNzZXRBbW91bnQEFGJhc2VBc3NldEFtb3VudFRvQWRkCQBlAgkBBGRpdmQCCQEEbXVsZAIFC3F0QXN0UkFmdGVyBQdfcXRBc3RXBQVwcmljZQUHX2JzQXN0UgQLYnNBc3RSQWZ0ZXIJAGQCBQdfYnNBc3RSBRRiYXNlQXNzZXRBbW91bnRUb0FkZAQNJHQwNTE1NDM1MTY5NAkBFGdldFN5bmNUZXJtaW5hbFByaWNlAwkBDmdldE9yYWNsZVByaWNlAAULcXRBc3RSQWZ0ZXIFC2JzQXN0UkFmdGVyBBNuZXdRdW90ZUFzc2V0V2VpZ2h0CAUNJHQwNTE1NDM1MTY5NAJfMQQSbmV3QmFzZUFzc2V0V2VpZ2h0CAUNJHQwNTE1NDM1MTY5NAJfMgQNbWFyZ2luVG9WYXVsdAgFDSR0MDUxNTQzNTE2OTQCXzMEDWRvRXhjaGFuZ2VQbkwDCQECIT0CBQ1tYXJnaW5Ub1ZhdWx0AAAEDWRvRXhjaGFuZ2VQbkwJAPwHBAkBDHZhdWx0QWRkcmVzcwACFWV4Y2hhbmdlRnJlZUFuZExvY2tlZAkAzAgCBQ1tYXJnaW5Ub1ZhdWx0BQNuaWwFA25pbAMJAAACBQ1kb0V4Y2hhbmdlUG5MBQ1kb0V4Y2hhbmdlUG5MBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBQ1kb0V4Y2hhbmdlUG5MBQ1kb0V4Y2hhbmdlUG5MCQDOCAIJARF1cGRhdGVBbW1SZXNlcnZlcwIFC3F0QXN0UkFmdGVyBQtic0FzdFJBZnRlcgkBEHVwZGF0ZUFtbVdlaWdodHMCBRNuZXdRdW90ZUFzc2V0V2VpZ2h0BRJuZXdCYXNlQXNzZXRXZWlnaHQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEOY2hhbmdlU2V0dGluZ3MPEF9pbml0TWFyZ2luUmF0aW8EX21tchRfbGlxdWlkYXRpb25GZWVSYXRpbw5fZnVuZGluZ1BlcmlvZARfZmVlDF9zcHJlYWRMaW1pdA9fbWF4UHJpY2VJbXBhY3QYX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvD19tYXhQcmljZVNwcmVhZBBfbWF4T3Blbk5vdGlvbmFsFF9mZWVUb1N0YWtlcnNQZXJjZW50D19tYXhPcmFjbGVEZWxheQxfcm9sbG92ZXJGZWUMX2Z1bmRpbmdNb2RlC19vcmFjbGVNb2RlAwMDAwMDAwMDAwMDAwMDAwMDCQBnAgAABQ5fZnVuZGluZ1BlcmlvZAYJAGcCAAAFEF9pbml0TWFyZ2luUmF0aW8GCQBnAgAABQRfbW1yBgkAZwIAAAUUX2xpcXVpZGF0aW9uRmVlUmF0aW8GCQBnAgAABQRfZmVlBgkAZwIAAAUMX3NwcmVhZExpbWl0BgkAZwIAAAUPX21heFByaWNlSW1wYWN0BgkAZwIAAAUYX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvBgkAZwIAAAUPX21heFByaWNlU3ByZWFkBgkAZwIAAAUQX21heE9wZW5Ob3Rpb25hbAYJAGcCAAAFFF9mZWVUb1N0YWtlcnNQZXJjZW50BgkAZgIFFF9mZWVUb1N0YWtlcnNQZXJjZW50BQxERUNJTUFMX1VOSVQGCQBnAgAABQ9fbWF4T3JhY2xlRGVsYXkGCQBnAgAABQxfcm9sbG92ZXJGZWUGAwkBAiE9AgUMX2Z1bmRpbmdNb2RlBRFGVU5ESU5HX1NZTU1FVFJJQwkBAiE9AgUMX2Z1bmRpbmdNb2RlBRJGVU5ESU5HX0FTWU1NRVRSSUMHBgMJAQIhPQIFC19vcmFjbGVNb2RlBQxPUkFDTEVfUExBSU4JAQIhPQIFC19vcmFjbGVNb2RlBQpPUkFDTEVfSklUBwYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAJAAIBAh1JbnZhbGlkIGNoYW5nZVNldHRpbmdzIHBhcmFtcwkBDnVwZGF0ZVNldHRpbmdzDwUQX2luaXRNYXJnaW5SYXRpbwUEX21tcgUUX2xpcXVpZGF0aW9uRmVlUmF0aW8FDl9mdW5kaW5nUGVyaW9kBQRfZmVlBQxfc3ByZWFkTGltaXQFD19tYXhQcmljZUltcGFjdAUYX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvBQ9fbWF4UHJpY2VTcHJlYWQFEF9tYXhPcGVuTm90aW9uYWwFFF9mZWVUb1N0YWtlcnNQZXJjZW50BQ9fbWF4T3JhY2xlRGVsYXkFDF9yb2xsb3ZlckZlZQUMX2Z1bmRpbmdNb2RlBQtfb3JhY2xlTW9kZQFpAQppbml0aWFsaXplFAdfcXRBc3RSB19ic0FzdFIOX2Z1bmRpbmdQZXJpb2QQX2luaXRNYXJnaW5SYXRpbwRfbW1yFF9saXF1aWRhdGlvbkZlZVJhdGlvBF9mZWUPX2Jhc2VPcmFjbGVEYXRhEF9xdW90ZU9yYWNsZURhdGEMX2Nvb3JkaW5hdG9yDF9zcHJlYWRMaW1pdA9fbWF4UHJpY2VJbXBhY3QYX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvD19tYXhQcmljZVNwcmVhZBBfbWF4T3Blbk5vdGlvbmFsFF9mZWVUb1N0YWtlcnNQZXJjZW50D19tYXhPcmFjbGVEZWxheQxfcm9sbG92ZXJGZWUMX2Z1bmRpbmdNb2RlC19vcmFjbGVNb2RlAwMDAwMDAwMDAwMDAwMDAwMDAwMJAGcCAAAFB19xdEFzdFIGCQBnAgAABQdfYnNBc3RSBgkAZwIAAAUOX2Z1bmRpbmdQZXJpb2QGCQBnAgAABRBfaW5pdE1hcmdpblJhdGlvBgkAZwIAAAUEX21tcgYJAGcCAAAFFF9saXF1aWRhdGlvbkZlZVJhdGlvBgkAZwIAAAUEX2ZlZQYJAGcCAAAFDF9zcHJlYWRMaW1pdAYJAGcCAAAFD19tYXhQcmljZUltcGFjdAYJAGcCAAAFGF9wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwYJAGcCAAAFD19tYXhQcmljZVNwcmVhZAYJAGcCAAAFEF9tYXhPcGVuTm90aW9uYWwGCQBnAgAABRRfZmVlVG9TdGFrZXJzUGVyY2VudAYJAGYCBRRfZmVlVG9TdGFrZXJzUGVyY2VudAUMREVDSU1BTF9VTklUBgkAZgIAAAUPX21heE9yYWNsZURlbGF5BgkAZwIAAAUMX3JvbGxvdmVyRmVlBgMJAQIhPQIFDF9mdW5kaW5nTW9kZQURRlVORElOR19TWU1NRVRSSUMJAQIhPQIFDF9mdW5kaW5nTW9kZQUSRlVORElOR19BU1lNTUVUUklDBwYDCQECIT0CBQtfb3JhY2xlTW9kZQUMT1JBQ0xFX1BMQUlOCQECIT0CBQtfb3JhY2xlTW9kZQUKT1JBQ0xFX0pJVAcGCQELaW5pdGlhbGl6ZWQABgkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIdSW52YWxpZCBpbml0aWFsaXplIHBhcmFtZXRlcnMJAM4IAgkAzggCCQDOCAIJAM4IAgkBCXVwZGF0ZUFtbQgFB19xdEFzdFIFB19ic0FzdFIAAAAAAAAAAAAAAAAJAQ51cGRhdGVTZXR0aW5ncw8FEF9pbml0TWFyZ2luUmF0aW8FBF9tbXIFFF9saXF1aWRhdGlvbkZlZVJhdGlvBQ5fZnVuZGluZ1BlcmlvZAUEX2ZlZQUMX3NwcmVhZExpbWl0BQ9fbWF4UHJpY2VJbXBhY3QFGF9wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwUPX21heFByaWNlU3ByZWFkBRBfbWF4T3Blbk5vdGlvbmFsBRRfZmVlVG9TdGFrZXJzUGVyY2VudAUPX21heE9yYWNsZURlbGF5BQxfcm9sbG92ZXJGZWUFDF9mdW5kaW5nTW9kZQULX29yYWNsZU1vZGUJAQ11cGRhdGVGdW5kaW5nBQkAZAIJAQ1sYXN0VGltZXN0YW1wAAUOX2Z1bmRpbmdQZXJpb2QAAAAAAAAAAAkBDXVwZGF0ZUJhbGFuY2UBAAAJAMwIAgkBDEJvb2xlYW5FbnRyeQIFDWtfaW5pdGlhbGl6ZWQGCQDMCAIJAQtTdHJpbmdFbnRyeQIFDGtfYmFzZU9yYWNsZQUPX2Jhc2VPcmFjbGVEYXRhCQDMCAIJAQtTdHJpbmdFbnRyeQIFDWtfcXVvdGVPcmFjbGUFEF9xdW90ZU9yYWNsZURhdGEJAMwIAgkBC1N0cmluZ0VudHJ5AgUUa19jb29yZGluYXRvckFkZHJlc3MJAKUIAQkBEUBleHRyTmF0aXZlKDEwNjIpAQUMX2Nvb3JkaW5hdG9yBQNuaWwBaQEQaW5jcmVhc2VQb3NpdGlvbgUKX2RpcmVjdGlvbglfbGV2ZXJhZ2UTX21pbkJhc2VBc3NldEFtb3VudAhfcmVmTGluawxfcHJpY2VVcGRhdGUEDHVwZGF0ZU9yYWNsZQkA/AcEBQR0aGlzAgx1cGRhdGVPcmFjbGUJAMwIAgUMX3ByaWNlVXBkYXRlBQNuaWwFA25pbAMJAAACBQx1cGRhdGVPcmFjbGUFDHVwZGF0ZU9yYWNsZQQEc3luYwkA/AcEBQR0aGlzAhlzeW5jVGVybWluYWxQcmljZVRvT3JhY2xlBQNuaWwFA25pbAMJAAACBQRzeW5jBQRzeW5jBBBlbnN1cmVDYWxsZWRPbmNlCQD8BwQFBHRoaXMCEGVuc3VyZUNhbGxlZE9uY2UFA25pbAUDbmlsAwkAAAIFEGVuc3VyZUNhbGxlZE9uY2UFEGVuc3VyZUNhbGxlZE9uY2UEB190cmFkZXIJAQ9nZXRBY3R1YWxDYWxsZXIBBQFpBApfcmF3QW1vdW50CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAQIX2Fzc2V0SWQICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQLX2Fzc2V0SWRTdHIJANgEAQkBBXZhbHVlAQUIX2Fzc2V0SWQEDGlzUXVvdGVBc3NldAkAAAIFCF9hc3NldElkCQEKcXVvdGVBc3NldAADAwMDAwMDAwMJAQIhPQIFCl9kaXJlY3Rpb24FCERJUl9MT05HCQECIT0CBQpfZGlyZWN0aW9uBQlESVJfU0hPUlQHBgkAZwIAAAUKX3Jhd0Ftb3VudAYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQEhAQUMaXNRdW90ZUFzc2V0BgkBASEBCQEWcmVxdWlyZU1vcmVNYXJnaW5SYXRpbwMJAQRkaXZkAgUMREVDSU1BTF9VTklUBQlfbGV2ZXJhZ2UJAQ9pbml0TWFyZ2luUmF0aW8ABgYJAQZwYXVzZWQABgkBCWNsb3NlT25seQAGCQEOaXNNYXJrZXRDbG9zZWQACQACAQIjSW52YWxpZCBpbmNyZWFzZVBvc2l0aW9uIHBhcmFtZXRlcnMEDSR0MDU4MDcwNTgyMTkJARhnZXRGb3JUcmFkZXJXaXRoQXJ0aWZhY3QCBQdfdHJhZGVyCQENZ2V0QXJ0aWZhY3RJZAEFAWkEC2FkanVzdGVkRmVlCAUNJHQwNTgwNzA1ODIxOQJfMQQMYnVybkFydGlmYWN0CAUNJHQwNTgwNzA1ODIxOQJfMgQHX2Ftb3VudAkBBGRpdmQCBQpfcmF3QW1vdW50CQBkAgkBBG11bGQCBQthZGp1c3RlZEZlZQUJX2xldmVyYWdlBQxERUNJTUFMX1VOSVQEE2Rpc3RyaWJ1dGVGZWVBbW91bnQJAGUCBQpfcmF3QW1vdW50BQdfYW1vdW50BA5yZWZlcnJlckZlZUFueQkA/AcECQEPcmVmZXJyYWxBZGRyZXNzAAIVYWNjZXB0UGF5bWVudFdpdGhMaW5rCQDMCAIFB190cmFkZXIJAMwIAgUIX3JlZkxpbmsFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABRNkaXN0cmlidXRlRmVlQW1vdW50BQNuaWwDCQAAAgUOcmVmZXJyZXJGZWVBbnkFDnJlZmVycmVyRmVlQW55BAtyZWZlcnJlckZlZQQHJG1hdGNoMAUOcmVmZXJyZXJGZWVBbnkDCQABAgUHJG1hdGNoMAIDSW50BAF4BQckbWF0Y2gwBQF4CQACAQITSW52YWxpZCByZWZlcnJlckZlZQQJZmVlQW1vdW50CQBlAgUTZGlzdHJpYnV0ZUZlZUFtb3VudAULcmVmZXJyZXJGZWUEDSR0MDU4NzE1NTg4OTUJAQtnZXRQb3NpdGlvbgIFB190cmFkZXIFCl9kaXJlY3Rpb24ED29sZFBvc2l0aW9uU2l6ZQgFDSR0MDU4NzE1NTg4OTUCXzEEEW9sZFBvc2l0aW9uTWFyZ2luCAUNJHQwNTg3MTU1ODg5NQJfMgQXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDA1ODcxNTU4ODk1Al8zBBRvbGRQb3NpdGlvbkxzdFVwZENQRggFDSR0MDU4NzE1NTg4OTUCXzQEFG9sZFBvc2l0aW9uVGltZXN0YW1wCAUNJHQwNTg3MTU1ODg5NQJfNQQNaXNOZXdQb3NpdGlvbgkAAAIFD29sZFBvc2l0aW9uU2l6ZQAABA9pc1NhbWVEaXJlY3Rpb24DCQBmAgUPb2xkUG9zaXRpb25TaXplAAAJAAACBQpfZGlyZWN0aW9uBQhESVJfTE9ORwkAAAIFCl9kaXJlY3Rpb24FCURJUl9TSE9SVAQOZXhwYW5kRXhpc3RpbmcDCQEBIQEFDWlzTmV3UG9zaXRpb24FD2lzU2FtZURpcmVjdGlvbgcEBWlzQWRkCQAAAgUKX2RpcmVjdGlvbgUIRElSX0xPTkcEDSR0MDU5MTg0NjIzMTcDAwUNaXNOZXdQb3NpdGlvbgYFDmV4cGFuZEV4aXN0aW5nBAxvcGVuTm90aW9uYWwJAQRtdWxkAgUHX2Ftb3VudAUJX2xldmVyYWdlBA0kdDA1OTY5MzU5ODY2CQEJc3dhcElucHV0AgUFaXNBZGQFDG9wZW5Ob3Rpb25hbAQVYW1vdW50QmFzZUFzc2V0Qm91Z2h0CAUNJHQwNTk2OTM1OTg2NgJfMQQWcXVvdGVBc3NldFJlc2VydmVBZnRlcggFDSR0MDU5NjkzNTk4NjYCXzIEFWJhc2VBc3NldFJlc2VydmVBZnRlcggFDSR0MDU5NjkzNTk4NjYCXzMEFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIIBQ0kdDA1OTY5MzU5ODY2Al80AwMJAQIhPQIFE19taW5CYXNlQXNzZXRBbW91bnQAAAkAZgIFE19taW5CYXNlQXNzZXRBbW91bnQJAQNhYnMBBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQHCQACAQkArAICCQCsAgIJAKwCAgINTGltaXQgZXJyb3I6IAkApAMBCQEDYWJzAQUVYW1vdW50QmFzZUFzc2V0Qm91Z2h0AgMgPCAJAKQDAQUTX21pbkJhc2VBc3NldEFtb3VudAQPbmV3UG9zaXRpb25TaXplCQBkAgUPb2xkUG9zaXRpb25TaXplBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQEGnRvdGFsTG9uZ09wZW5JbnRlcmVzdEFmdGVyCQBkAgkBEG9wZW5JbnRlcmVzdExvbmcAAwkAZgIFD25ld1Bvc2l0aW9uU2l6ZQAABQxvcGVuTm90aW9uYWwAAAQbdG90YWxTaG9ydE9wZW5JbnRlcmVzdEFmdGVyCQBkAgkBEW9wZW5JbnRlcmVzdFNob3J0AAMJAGYCAAAFD25ld1Bvc2l0aW9uU2l6ZQUMb3Blbk5vdGlvbmFsAAAEDSR0MDYwNDEyNjA2ODcJATBjYWxjUmVtYWluTWFyZ2luV2l0aEZ1bmRpbmdQYXltZW50QW5kUm9sbG92ZXJGZWUFBQ9vbGRQb3NpdGlvblNpemUFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvbkxzdFVwZENQRgUUb2xkUG9zaXRpb25UaW1lc3RhbXAFB19hbW91bnQEDHJlbWFpbk1hcmdpbggFDSR0MDYwNDEyNjA2ODcCXzEEAngxCAUNJHQwNjA0MTI2MDY4NwJfMgQCeDIIBQ0kdDA2MDQxMjYwNjg3Al8zBAtyb2xsb3ZlckZlZQgFDSR0MDYwNDEyNjA2ODcCXzQDCQEBIQEJARlyZXF1aXJlTm90T3ZlclNwcmVhZExpbWl0AgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCQACAQIVT3ZlciBtYXggc3ByZWFkIGxpbWl0AwkBASEBCQEdcmVxdWlyZU5vdE92ZXJNYXhPcGVuTm90aW9uYWwCBRp0b3RhbExvbmdPcGVuSW50ZXJlc3RBZnRlcgUbdG90YWxTaG9ydE9wZW5JbnRlcmVzdEFmdGVyCQACAQIWT3ZlciBtYXggb3BlbiBub3Rpb25hbAkAoAoOBQ9uZXdQb3NpdGlvblNpemUFDHJlbWFpbk1hcmdpbgkAZAIFF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsBQxvcGVuTm90aW9uYWwJAR9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAQUPbmV3UG9zaXRpb25TaXplCQENbGFzdFRpbWVzdGFtcAAFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgkAZAIJARRvcGVuSW50ZXJlc3ROb3Rpb25hbAAFDG9wZW5Ob3Rpb25hbAkAZAIJARV0b3RhbExvbmdQb3NpdGlvblNpemUAAwkAZgIFD25ld1Bvc2l0aW9uU2l6ZQAACQEDYWJzAQUVYW1vdW50QmFzZUFzc2V0Qm91Z2h0AAAJAGQCCQEWdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQADCQBmAgAABQ9uZXdQb3NpdGlvblNpemUJAQNhYnMBBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQAAAUadG90YWxMb25nT3BlbkludGVyZXN0QWZ0ZXIFG3RvdGFsU2hvcnRPcGVuSW50ZXJlc3RBZnRlcgULcm9sbG92ZXJGZWUEDG9wZW5Ob3Rpb25hbAkBBG11bGQCBQdfYW1vdW50BQlfbGV2ZXJhZ2UEDSR0MDYyMDA1NjIxMzMJASNnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubAMJAKUIAQgFAWkGY2FsbGVyBQpfZGlyZWN0aW9uBQ9QTkxfT1BUSU9OX1NQT1QEE29sZFBvc2l0aW9uTm90aW9uYWwIBQ0kdDA2MjAwNTYyMTMzAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwNjIwMDU2MjEzMwJfMgMJAGYCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBQxvcGVuTm90aW9uYWwJAAIBAi5Vc2UgZGVjcmVhc2VQb3NpdGlvbiB0byBkZWNyZWFzZSBwb3NpdGlvbiBzaXplCQACAQIUQ2xvc2UgcG9zaXRpb24gZmlyc3QED25ld1Bvc2l0aW9uU2l6ZQgFDSR0MDU5MTg0NjIzMTcCXzEEF25ld1Bvc2l0aW9uUmVtYWluTWFyZ2luCAUNJHQwNTkxODQ2MjMxNwJfMgQXbmV3UG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDA1OTE4NDYyMzE3Al8zBBRuZXdQb3NpdGlvbkxhdGVzdENQRggFDSR0MDU5MTg0NjIzMTcCXzQEFG5ld1Bvc2l0aW9uVGltZXN0YW1wCAUNJHQwNTkxODQ2MjMxNwJfNQQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNTkxODQ2MjMxNwJfNgQWcXVvdGVBc3NldFJlc2VydmVBZnRlcggFDSR0MDU5MTg0NjIzMTcCXzcEFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIIBQ0kdDA1OTE4NDYyMzE3Al84BBlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCAUNJHQwNTkxODQ2MjMxNwJfOQQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA1OTE4NDYyMzE3A18xMAQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNTkxODQ2MjMxNwNfMTEEGnRvdGFsTG9uZ09wZW5JbnRlcmVzdEFmdGVyCAUNJHQwNTkxODQ2MjMxNwNfMTIEG3RvdGFsU2hvcnRPcGVuSW50ZXJlc3RBZnRlcggFDSR0MDU5MTg0NjIzMTcDXzEzBAtyb2xsb3ZlckZlZQgFDSR0MDU5MTg0NjIzMTcDXzE0BA0kdDA2MjMyMzYyMzk0CQENZGlzdHJpYnV0ZUZlZQEJAGQCBQlmZWVBbW91bnQFC3JvbGxvdmVyRmVlBAxmZWVUb1N0YWtlcnMIBQ0kdDA2MjMyMzYyMzk0Al8xBApmZWVUb1ZhdWx0CAUNJHQwNjIzMjM2MjM5NAJfMgQFc3Rha2UDCQBnAgUHX2Ftb3VudAULcm9sbG92ZXJGZWUJAPwHBAkBDHZhdWx0QWRkcmVzcwACCWFkZExvY2tlZAUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQEKcXVvdGVBc3NldAAJAGUCBQdfYW1vdW50BQtyb2xsb3ZlckZlZQUDbmlsCQD8BwQJAQx2YXVsdEFkZHJlc3MAAg53aXRoZHJhd0xvY2tlZAkAzAgCCQBlAgULcm9sbG92ZXJGZWUFB19hbW91bnQFA25pbAUDbmlsAwkAAAIFBXN0YWtlBQVzdGFrZQQMZGVwb3NpdFZhdWx0CQD8BwQJAQx2YXVsdEFkZHJlc3MAAgdhZGRGcmVlBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUKZmVlVG9WYXVsdAUDbmlsAwkAAAIFDGRlcG9zaXRWYXVsdAUMZGVwb3NpdFZhdWx0BAlub3RpZnlGZWUJAPwHBAkBDG1pbmVyQWRkcmVzcwACCm5vdGlmeUZlZXMJAMwIAgUHX3RyYWRlcgkAzAgCBQlmZWVBbW91bnQFA25pbAUDbmlsAwkAAAIFCW5vdGlmeUZlZQUJbm90aWZ5RmVlBA5ub3RpZnlOb3Rpb25hbAkA/AcECQEMbWluZXJBZGRyZXNzAAIObm90aWZ5Tm90aW9uYWwJAMwIAgUHX3RyYWRlcgkAzAgCBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUDbmlsBQNuaWwDCQAAAgUObm90aWZ5Tm90aW9uYWwFDm5vdGlmeU5vdGlvbmFsCQDOCAIJAM4IAgkAzggCCQDOCAIJAM4IAgkAzggCCQEOdXBkYXRlUG9zaXRpb24GBQdfdHJhZGVyBQ9uZXdQb3NpdGlvblNpemUFF25ld1Bvc2l0aW9uUmVtYWluTWFyZ2luBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUUbmV3UG9zaXRpb25MYXRlc3RDUEYFFG5ld1Bvc2l0aW9uVGltZXN0YW1wCQEfaW5jcmVtZW50UG9zaXRpb25TZXF1ZW5jZU51bWJlcgMFDWlzTmV3UG9zaXRpb24FB190cmFkZXIFCl9kaXJlY3Rpb24JARF1cGRhdGVQb3NpdGlvbkZlZQQFDWlzTmV3UG9zaXRpb24FB190cmFkZXIFCl9kaXJlY3Rpb24FC2FkanVzdGVkRmVlCQEJdXBkYXRlQW1tCAUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBRZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyBRlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyBQ50b3RhbExvbmdBZnRlcgUPdG90YWxTaG9ydEFmdGVyBRp0b3RhbExvbmdPcGVuSW50ZXJlc3RBZnRlcgUbdG90YWxTaG9ydE9wZW5JbnRlcmVzdEFmdGVyCQELdHJhbnNmZXJGZWUBBQxmZWVUb1N0YWtlcnMJAQ11cGRhdGVCYWxhbmNlAQkAZQIJAGQCCQEIY2JhbGFuY2UABQdfYW1vdW50BQtyb2xsb3ZlckZlZQkBDmRvQnVybkFydGlmYWN0AgUMYnVybkFydGlmYWN0BQFpCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBCWFkZE1hcmdpbgIKX2RpcmVjdGlvbgxfcHJpY2VVcGRhdGUEDHVwZGF0ZU9yYWNsZQkA/AcEBQR0aGlzAgx1cGRhdGVPcmFjbGUJAMwIAgUMX3ByaWNlVXBkYXRlBQNuaWwFA25pbAMJAAACBQx1cGRhdGVPcmFjbGUFDHVwZGF0ZU9yYWNsZQQEc3luYwkA/AcEBQR0aGlzAhlzeW5jVGVybWluYWxQcmljZVRvT3JhY2xlBQNuaWwFA25pbAMJAAACBQRzeW5jBQRzeW5jBBBlbnN1cmVDYWxsZWRPbmNlCQD8BwQFBHRoaXMCEGVuc3VyZUNhbGxlZE9uY2UFA25pbAUDbmlsAwkAAAIFEGVuc3VyZUNhbGxlZE9uY2UFEGVuc3VyZUNhbGxlZE9uY2UEB190cmFkZXIJAKUIAQgFAWkGY2FsbGVyBAdfYW1vdW50CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAQIX2Fzc2V0SWQICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAQLX2Fzc2V0SWRTdHIJANgEAQkBBXZhbHVlAQUIX2Fzc2V0SWQEDGlzUXVvdGVBc3NldAkAAAIFCF9hc3NldElkCQEKcXVvdGVBc3NldAADAwMDAwkBASEBBQxpc1F1b3RlQXNzZXQGCQEBIQEJARNyZXF1aXJlT3BlblBvc2l0aW9uAgkApQgBCAUBaQZjYWxsZXIFCl9kaXJlY3Rpb24GCQEBIQEJAQtpbml0aWFsaXplZAAGCQEGcGF1c2VkAAYJAQ5pc01hcmtldENsb3NlZAAJAAIBAhxJbnZhbGlkIGFkZE1hcmdpbiBwYXJhbWV0ZXJzBA0kdDA2NDczNjY0OTE2CQELZ2V0UG9zaXRpb24CBQdfdHJhZGVyBQpfZGlyZWN0aW9uBA9vbGRQb3NpdGlvblNpemUIBQ0kdDA2NDczNjY0OTE2Al8xBBFvbGRQb3NpdGlvbk1hcmdpbggFDSR0MDY0NzM2NjQ5MTYCXzIEF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwNjQ3MzY2NDkxNgJfMwQUb2xkUG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDA2NDczNjY0OTE2Al80BBRvbGRQb3NpdGlvblRpbWVzdGFtcAgFDSR0MDY0NzM2NjQ5MTYCXzUEBXN0YWtlCQD8BwQJAQx2YXVsdEFkZHJlc3MAAglhZGRMb2NrZWQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQdfYW1vdW50BQNuaWwDCQAAAgUFc3Rha2UFBXN0YWtlBAtyb2xsb3ZlckZlZQkBD2NhbGNSb2xsb3ZlckZlZQIFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvblRpbWVzdGFtcAQWZG9UcmFuc2ZlckZlZVRvU3Rha2VycwMJAGYCBQtyb2xsb3ZlckZlZQAABA0kdDA2NTIwMTY1MjYwCQENZGlzdHJpYnV0ZUZlZQEFC3JvbGxvdmVyRmVlBAxmZWVUb1N0YWtlcnMIBQ0kdDA2NTIwMTY1MjYwAl8xBApmZWVUb1ZhdWx0CAUNJHQwNjUyMDE2NTI2MAJfMgQHdW5zdGFrZQkA/AcECQEMdmF1bHRBZGRyZXNzAAIOd2l0aGRyYXdMb2NrZWQJAMwIAgUMZmVlVG9TdGFrZXJzBQNuaWwFA25pbAMJAAACBQd1bnN0YWtlBQd1bnN0YWtlBAtsb2NrQmFkRGVidAkA/AcECQEMdmF1bHRBZGRyZXNzAAIVZXhjaGFuZ2VGcmVlQW5kTG9ja2VkCQDMCAIJAQEtAQUKZmVlVG9WYXVsdAUDbmlsBQNuaWwDCQAAAgULbG9ja0JhZERlYnQFC2xvY2tCYWREZWJ0CQELdHJhbnNmZXJGZWUBBQxmZWVUb1N0YWtlcnMJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBRZkb1RyYW5zZmVyRmVlVG9TdGFrZXJzBRZkb1RyYW5zZmVyRmVlVG9TdGFrZXJzCQDOCAIJAM4IAgkBDnVwZGF0ZVBvc2l0aW9uBgUHX3RyYWRlcgUPb2xkUG9zaXRpb25TaXplCQBkAgkAZQIFEW9sZFBvc2l0aW9uTWFyZ2luBQtyb2xsb3ZlckZlZQUHX2Ftb3VudAUXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwFFG9sZFBvc2l0aW9uTHN0VXBkQ1BGCQENbGFzdFRpbWVzdGFtcAAJAQ11cGRhdGVCYWxhbmNlAQkAZQIJAGQCCQEIY2JhbGFuY2UABQdfYW1vdW50BQtyb2xsb3ZlckZlZQUWZG9UcmFuc2ZlckZlZVRvU3Rha2VycwkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQxyZW1vdmVNYXJnaW4DB19hbW91bnQKX2RpcmVjdGlvbgxfcHJpY2VVcGRhdGUEDHVwZGF0ZU9yYWNsZQkA/AcEBQR0aGlzAgx1cGRhdGVPcmFjbGUJAMwIAgUMX3ByaWNlVXBkYXRlBQNuaWwFA25pbAMJAAACBQx1cGRhdGVPcmFjbGUFDHVwZGF0ZU9yYWNsZQQEc3luYwkA/AcEBQR0aGlzAhlzeW5jVGVybWluYWxQcmljZVRvT3JhY2xlBQNuaWwFA25pbAMJAAACBQRzeW5jBQRzeW5jBBBlbnN1cmVDYWxsZWRPbmNlCQD8BwQFBHRoaXMCEGVuc3VyZUNhbGxlZE9uY2UFA25pbAUDbmlsAwkAAAIFEGVuc3VyZUNhbGxlZE9uY2UFEGVuc3VyZUNhbGxlZE9uY2UEB190cmFkZXIJAKUIAQgFAWkGY2FsbGVyAwMDAwMJAGcCAAAFB19hbW91bnQGCQEBIQEJARNyZXF1aXJlT3BlblBvc2l0aW9uAgUHX3RyYWRlcgUKX2RpcmVjdGlvbgYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQZwYXVzZWQABgkBDmlzTWFya2V0Q2xvc2VkAAkAAgECH0ludmFsaWQgcmVtb3ZlTWFyZ2luIHBhcmFtZXRlcnMEDSR0MDY2NzYxNjY5NDEJAQtnZXRQb3NpdGlvbgIFB190cmFkZXIFCl9kaXJlY3Rpb24ED29sZFBvc2l0aW9uU2l6ZQgFDSR0MDY2NzYxNjY5NDECXzEEEW9sZFBvc2l0aW9uTWFyZ2luCAUNJHQwNjY3NjE2Njk0MQJfMgQXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDA2Njc2MTY2OTQxAl8zBBRvbGRQb3NpdGlvbkxzdFVwZENQRggFDSR0MDY2NzYxNjY5NDECXzQEFG9sZFBvc2l0aW9uVGltZXN0YW1wCAUNJHQwNjY3NjE2Njk0MQJfNQQNJHQwNjY5NDc2NzE5NgkBMGNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnRBbmRSb2xsb3ZlckZlZQUFD29sZFBvc2l0aW9uU2l6ZQURb2xkUG9zaXRpb25NYXJnaW4FFG9sZFBvc2l0aW9uTHN0VXBkQ1BGBRRvbGRQb3NpdGlvblRpbWVzdGFtcAkBAS0BBQdfYW1vdW50BAxyZW1haW5NYXJnaW4IBQ0kdDA2Njk0NzY3MTk2Al8xBAdiYWREZWJ0CAUNJHQwNjY5NDc2NzE5NgJfMgQOZnVuZGluZ1BheW1lbnQIBQ0kdDA2Njk0NzY3MTk2Al8zBAtyb2xsb3ZlckZlZQgFDSR0MDY2OTQ3NjcxOTYCXzQDCQECIT0CBQdiYWREZWJ0AAAJAAIBAh1JbnZhbGlkIHJlbW92ZWQgbWFyZ2luIGFtb3VudAQLbWFyZ2luUmF0aW8JAQ9jYWxjTWFyZ2luUmF0aW8DBQxyZW1haW5NYXJnaW4FB2JhZERlYnQFF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsAwkBASEBCQEWcmVxdWlyZU1vcmVNYXJnaW5SYXRpbwMFC21hcmdpblJhdGlvCQEPaW5pdE1hcmdpblJhdGlvAAYJAAIBCQCsAgIJAKwCAgkArAICAhlUb28gbXVjaCBtYXJnaW4gcmVtb3ZlZDogCQCkAwEFC21hcmdpblJhdGlvAgMgPCAJAKQDAQkBD2luaXRNYXJnaW5SYXRpbwAEDSR0MDY3NTgyNjc2NDEJAQ1kaXN0cmlidXRlRmVlAQULcm9sbG92ZXJGZWUEDGZlZVRvU3Rha2VycwgFDSR0MDY3NTgyNjc2NDECXzEECmZlZVRvVmF1bHQIBQ0kdDA2NzU4MjY3NjQxAl8yBBZkb1RyYW5zZmVyRmVlVG9TdGFrZXJzAwkAZgIFC3JvbGxvdmVyRmVlAAAEC2xvY2tCYWREZWJ0CQD8BwQJAQx2YXVsdEFkZHJlc3MAAhVleGNoYW5nZUZyZWVBbmRMb2NrZWQJAMwIAgkBAS0BBQpmZWVUb1ZhdWx0BQNuaWwFA25pbAMJAAACBQtsb2NrQmFkRGVidAULbG9ja0JhZERlYnQJAQt0cmFuc2ZlckZlZQEFDGZlZVRvU3Rha2VycwkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgUDbmlsAwkAAAIFFmRvVHJhbnNmZXJGZWVUb1N0YWtlcnMFFmRvVHJhbnNmZXJGZWVUb1N0YWtlcnMEB3Vuc3Rha2UJAPwHBAkBDHZhdWx0QWRkcmVzcwACDndpdGhkcmF3TG9ja2VkCQDMCAIJAGQCBQdfYW1vdW50BQxmZWVUb1N0YWtlcnMFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UJAM4IAgkAzggCCQDOCAIJAQ51cGRhdGVQb3NpdGlvbgYFB190cmFkZXIFD29sZFBvc2l0aW9uU2l6ZQUMcmVtYWluTWFyZ2luBRdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBQ9vbGRQb3NpdGlvblNpemUJAQ1sYXN0VGltZXN0YW1wAAkBCHdpdGhkcmF3AggFAWkGY2FsbGVyBQdfYW1vdW50CQENdXBkYXRlQmFsYW5jZQEJAGUCCQBlAgkBCGNiYWxhbmNlAAUHX2Ftb3VudAULcm9sbG92ZXJGZWUFFmRvVHJhbnNmZXJGZWVUb1N0YWtlcnMJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQENY2xvc2VQb3NpdGlvbgUFX3NpemUKX2RpcmVjdGlvbhRfbWluUXVvdGVBc3NldEFtb3VudAxfYWRkVG9NYXJnaW4MX3ByaWNlVXBkYXRlBAx1cGRhdGVPcmFjbGUJAPwHBAUEdGhpcwIMdXBkYXRlT3JhY2xlCQDMCAIFDF9wcmljZVVwZGF0ZQUDbmlsBQNuaWwDCQAAAgUMdXBkYXRlT3JhY2xlBQx1cGRhdGVPcmFjbGUEBHN5bmMJAPwHBAUEdGhpcwIZc3luY1Rlcm1pbmFsUHJpY2VUb09yYWNsZQUDbmlsBQNuaWwDCQAAAgUEc3luYwUEc3luYwQQZW5zdXJlQ2FsbGVkT25jZQkA/AcEBQR0aGlzAhBlbnN1cmVDYWxsZWRPbmNlBQNuaWwFA25pbAMJAAACBRBlbnN1cmVDYWxsZWRPbmNlBRBlbnN1cmVDYWxsZWRPbmNlBAdfdHJhZGVyCQEPZ2V0QWN0dWFsQ2FsbGVyAQUBaQQOX3RyYWRlckFkZHJlc3MJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBBQdfdHJhZGVyAg5JbnZhbGlkIGNhbGxlcgQLcG9zaXRpb25GZWUJAQ5nZXRQb3NpdGlvbkZlZQIFB190cmFkZXIFCl9kaXJlY3Rpb24DAwMDAwMJAQEhAQkBE3JlcXVpcmVPcGVuUG9zaXRpb24CBQdfdHJhZGVyBQpfZGlyZWN0aW9uBgkBASEBCQELaW5pdGlhbGl6ZWQABgkBBnBhdXNlZAAGCQBnAgAABQVfc2l6ZQYJAGYCAAAFFF9taW5RdW90ZUFzc2V0QW1vdW50BgkBDmlzTWFya2V0Q2xvc2VkAAkAAgECIEludmFsaWQgY2xvc2VQb3NpdGlvbiBwYXJhbWV0ZXJzBBRvbGRQb3NpdGlvblRpbWVzdGFtcAgJAQtnZXRQb3NpdGlvbgIFB190cmFkZXIFCl9kaXJlY3Rpb24CXzUEDSR0MDY5OTc2NzA1NzkJARVpbnRlcm5hbENsb3NlUG9zaXRpb24IBQdfdHJhZGVyBQpfZGlyZWN0aW9uBQVfc2l6ZQULcG9zaXRpb25GZWUFFF9taW5RdW90ZUFzc2V0QW1vdW50BQxfYWRkVG9NYXJnaW4GBgQPbmV3UG9zaXRpb25TaXplCAUNJHQwNjk5NzY3MDU3OQJfMQQRbmV3UG9zaXRpb25NYXJnaW4IBQ0kdDA2OTk3NjcwNTc5Al8yBBduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDY5OTc2NzA1NzkCXzMEFG5ld1Bvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwNjk5NzY3MDU3OQJfNAQPcG9zaXRpb25CYWREZWJ0CAUNJHQwNjk5NzY3MDU3OQJfNQQLcmVhbGl6ZWRQbmwIBQ0kdDA2OTk3NjcwNTc5Al82BA5tYXJnaW5Ub1RyYWRlcggFDSR0MDY5OTc2NzA1NzkCXzcEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA2OTk3NjcwNTc5Al84BBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA2OTk3NjcwNTc5Al85BBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwNjk5NzY3MDU3OQNfMTAEGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIIBQ0kdDA2OTk3NjcwNTc5A18xMQQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA2OTk3NjcwNTc5A18xMgQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNjk5NzY3MDU3OQNfMTMEGnRvdGFsTG9uZ09wZW5JbnRlcmVzdEFmdGVyCAUNJHQwNjk5NzY3MDU3OQNfMTQEG3RvdGFsU2hvcnRPcGVuSW50ZXJlc3RBZnRlcggFDSR0MDY5OTc2NzA1NzkDXzE1BAtyZWFsaXplZEZlZQgFDSR0MDY5OTc2NzA1NzkDXzE2AwkAZgIFD3Bvc2l0aW9uQmFkRGVidAAACQACAQIqSW52YWxpZCBjbG9zZVBvc2l0aW9uIHBhcmFtZXRlcnM6IGJhZCBkZWJ0AwkAZwIFFG9sZFBvc2l0aW9uVGltZXN0YW1wCQENbGFzdFRpbWVzdGFtcAAJAAIBAlNJbnZhbGlkIGNsb3NlUG9zaXRpb24gcGFyYW1ldGVyczogd2FpdCBhdCBsZWFzdCAxIGJsb2NrIGJlZm9yZSBjbG9zaW5nIHRoZSBwb3NpdGlvbgQOaXNQYXJ0aWFsQ2xvc2UJAQIhPQIFD25ld1Bvc2l0aW9uU2l6ZQAABA53aXRoZHJhd0Ftb3VudAkAZAIFDm1hcmdpblRvVHJhZGVyBQtyZWFsaXplZEZlZQQKYW1tQmFsYW5jZQkAZQIJAQhjYmFsYW5jZQAFDndpdGhkcmF3QW1vdW50BA1hbW1OZXdCYWxhbmNlAwkAZgIAAAUKYW1tQmFsYW5jZQAABQphbW1CYWxhbmNlBAd1bnN0YWtlCQD8BwQJAQx2YXVsdEFkZHJlc3MAAg53aXRoZHJhd0xvY2tlZAkAzAgCBQ53aXRoZHJhd0Ftb3VudAUDbmlsBQNuaWwDCQAAAgUHdW5zdGFrZQUHdW5zdGFrZQQOcmVmZXJyZXJGZWVBbnkJAPwHBAkBD3JlZmVycmFsQWRkcmVzcwACDWFjY2VwdFBheW1lbnQJAMwIAgUHX3RyYWRlcgUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQEKcXVvdGVBc3NldAAFC3JlYWxpemVkRmVlBQNuaWwDCQAAAgUOcmVmZXJyZXJGZWVBbnkFDnJlZmVycmVyRmVlQW55BAtyZWZlcnJlckZlZQQHJG1hdGNoMAUOcmVmZXJyZXJGZWVBbnkDCQABAgUHJG1hdGNoMAIDSW50BAF4BQckbWF0Y2gwBQF4CQACAQITSW52YWxpZCByZWZlcnJlckZlZQQNJHQwNzE1NTE3MTYyNAkBDWRpc3RyaWJ1dGVGZWUBCQBlAgULcmVhbGl6ZWRGZWUFC3JlZmVycmVyRmVlBAxmZWVUb1N0YWtlcnMIBQ0kdDA3MTU1MTcxNjI0Al8xBApmZWVUb1ZhdWx0CAUNJHQwNzE1NTE3MTYyNAJfMgQMZGVwb3NpdFZhdWx0CQD8BwQJAQx2YXVsdEFkZHJlc3MAAgdhZGRGcmVlBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUKZmVlVG9WYXVsdAUDbmlsAwkAAAIFDGRlcG9zaXRWYXVsdAUMZGVwb3NpdFZhdWx0BAlub3RpZnlGZWUJAPwHBAkBDG1pbmVyQWRkcmVzcwACCm5vdGlmeUZlZXMJAMwIAgUHX3RyYWRlcgkAzAgCBQtyZWFsaXplZEZlZQUDbmlsBQNuaWwDCQAAAgUJbm90aWZ5RmVlBQlub3RpZnlGZWUEDm5vdGlmeU5vdGlvbmFsCQD8BwQJAQxtaW5lckFkZHJlc3MAAg5ub3RpZnlOb3Rpb25hbAkAzAgCBQdfdHJhZGVyCQDMCAIFF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBQNuaWwFA25pbAMJAAACBQ5ub3RpZnlOb3Rpb25hbAUObm90aWZ5Tm90aW9uYWwJAM4IAgkAzggCCQDOCAIJAM4IAgMFDmlzUGFydGlhbENsb3NlCQEOdXBkYXRlUG9zaXRpb24GBQdfdHJhZGVyBQ9uZXdQb3NpdGlvblNpemUFEW5ld1Bvc2l0aW9uTWFyZ2luBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUUbmV3UG9zaXRpb25Mc3RVcGRDUEYJAQ1sYXN0VGltZXN0YW1wAAkBDmRlbGV0ZVBvc2l0aW9uAgUHX3RyYWRlcgUKX2RpcmVjdGlvbgkBCXVwZGF0ZUFtbQgFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUZb3BlbkludGVyZXN0Tm90aW9uYWxBZnRlcgUOdG90YWxMb25nQWZ0ZXIFD3RvdGFsU2hvcnRBZnRlcgUadG90YWxMb25nT3BlbkludGVyZXN0QWZ0ZXIFG3RvdGFsU2hvcnRPcGVuSW50ZXJlc3RBZnRlcgMJAGYCBQ5tYXJnaW5Ub1RyYWRlcgAACQEId2l0aGRyYXcCBQ5fdHJhZGVyQWRkcmVzcwUObWFyZ2luVG9UcmFkZXIFA25pbAkBDXVwZGF0ZUJhbGFuY2UBBQ1hbW1OZXdCYWxhbmNlCQELdHJhbnNmZXJGZWUBBQxmZWVUb1N0YWtlcnMJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEJbGlxdWlkYXRlAwdfdHJhZGVyCl9kaXJlY3Rpb24MX3ByaWNlVXBkYXRlBAx1cGRhdGVPcmFjbGUJAPwHBAUEdGhpcwIMdXBkYXRlT3JhY2xlCQDMCAIFDF9wcmljZVVwZGF0ZQUDbmlsBQNuaWwDCQAAAgUMdXBkYXRlT3JhY2xlBQx1cGRhdGVPcmFjbGUEBHN5bmMJAPwHBAUEdGhpcwIZc3luY1Rlcm1pbmFsUHJpY2VUb09yYWNsZQUDbmlsBQNuaWwDCQAAAgUEc3luYwUEc3luYwQPc3BvdE1hcmdpblJhdGlvCQEWZ2V0TWFyZ2luUmF0aW9CeU9wdGlvbgMFB190cmFkZXIFCl9kaXJlY3Rpb24FD1BOTF9PUFRJT05fU1BPVAQWbGlxdWlkYXRpb25NYXJnaW5SYXRpbwMJARZpc092ZXJGbHVjdHVhdGlvbkxpbWl0AAQRb3JhY2xlTWFyZ2luUmF0aW8JARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAwUHX3RyYWRlcgUKX2RpcmVjdGlvbgURUE5MX09QVElPTl9PUkFDTEUJAQR2bWF4AgUPc3BvdE1hcmdpblJhdGlvBRFvcmFjbGVNYXJnaW5SYXRpbwUPc3BvdE1hcmdpblJhdGlvAwMDAwMJAQEhAQkBFnJlcXVpcmVNb3JlTWFyZ2luUmF0aW8DBRZsaXF1aWRhdGlvbk1hcmdpblJhdGlvCQEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAHBgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgIFB190cmFkZXIFCl9kaXJlY3Rpb24GCQEBIQEJAQtpbml0aWFsaXplZAAGCQEGcGF1c2VkAAYJAQ5pc01hcmtldENsb3NlZAAJAAIBAhNVbmFibGUgdG8gbGlxdWlkYXRlBBRpc1BhcnRpYWxMaXF1aWRhdGlvbgMDCQBmAgUPc3BvdE1hcmdpblJhdGlvCQETbGlxdWlkYXRpb25GZWVSYXRpbwAJAGYCCQEXcGFydGlhbExpcXVpZGF0aW9uUmF0aW8AAAAHCQBmAgUMREVDSU1BTF9VTklUCQEXcGFydGlhbExpcXVpZGF0aW9uUmF0aW8ABwQPb2xkUG9zaXRpb25TaXplCAkBC2dldFBvc2l0aW9uAgUHX3RyYWRlcgUKX2RpcmVjdGlvbgJfMQQPcG9zaXRpb25TaXplQWJzCQEDYWJzAQUPb2xkUG9zaXRpb25TaXplBA0kdDA3NDIxNzc0NTQwAwUUaXNQYXJ0aWFsTGlxdWlkYXRpb24ED2xpcXVpZGF0aW9uU2l6ZQkBG2dldFBhcnRpYWxMaXF1aWRhdGlvbkFtb3VudAIFB190cmFkZXIFD29sZFBvc2l0aW9uU2l6ZQQQbGlxdWlkYXRpb25SYXRpbwkBBGRpdmQCCQEDYWJzAQUPbGlxdWlkYXRpb25TaXplBQ9wb3NpdGlvblNpemVBYnMJAJQKAgUQbGlxdWlkYXRpb25SYXRpbwkBA2FicwEFD2xpcXVpZGF0aW9uU2l6ZQkAlAoCAAAFD3Bvc2l0aW9uU2l6ZUFicwQQbGlxdWlkYXRpb25SYXRpbwgFDSR0MDc0MjE3NzQ1NDACXzEED2xpcXVpZGF0aW9uU2l6ZQgFDSR0MDc0MjE3NzQ1NDACXzIEDSR0MDc0NTQ2NzUyMDIJARVpbnRlcm5hbENsb3NlUG9zaXRpb24IBQdfdHJhZGVyBQpfZGlyZWN0aW9uAwUUaXNQYXJ0aWFsTGlxdWlkYXRpb24FD2xpcXVpZGF0aW9uU2l6ZQUPcG9zaXRpb25TaXplQWJzCQETbGlxdWlkYXRpb25GZWVSYXRpbwAAAAYHBgQPbmV3UG9zaXRpb25TaXplCAUNJHQwNzQ1NDY3NTIwMgJfMQQRbmV3UG9zaXRpb25NYXJnaW4IBQ0kdDA3NDU0Njc1MjAyAl8yBBduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDc0NTQ2NzUyMDICXzMEFG5ld1Bvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwNzQ1NDY3NTIwMgJfNAQPcG9zaXRpb25CYWREZWJ0CAUNJHQwNzQ1NDY3NTIwMgJfNQQLcmVhbGl6ZWRQbmwIBQ0kdDA3NDU0Njc1MjAyAl82BA5tYXJnaW5Ub1RyYWRlcggFDSR0MDc0NTQ2NzUyMDICXzcEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA3NDU0Njc1MjAyAl84BBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA3NDU0Njc1MjAyAl85BBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwNzQ1NDY3NTIwMgNfMTAEGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIIBQ0kdDA3NDU0Njc1MjAyA18xMQQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA3NDU0Njc1MjAyA18xMgQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNzQ1NDY3NTIwMgNfMTMEGnRvdGFsTG9uZ09wZW5JbnRlcmVzdEFmdGVyCAUNJHQwNzQ1NDY3NTIwMgNfMTQEG3RvdGFsU2hvcnRPcGVuSW50ZXJlc3RBZnRlcggFDSR0MDc0NTQ2NzUyMDIDXzE1BBJsaXF1aWRhdGlvblBlbmFsdHkIBQ0kdDA3NDU0Njc1MjAyA18xNgQPZmVlVG9MaXF1aWRhdG9yCQBpAgUSbGlxdWlkYXRpb25QZW5hbHR5AAIECmZlZVRvVmF1bHQJAGUCBRJsaXF1aWRhdGlvblBlbmFsdHkFD2ZlZVRvTGlxdWlkYXRvcgQKYW1tQmFsYW5jZQkAZQIJAQhjYmFsYW5jZQAFEmxpcXVpZGF0aW9uUGVuYWx0eQQNbmV3QW1tQmFsYW5jZQMJAGYCAAAFCmFtbUJhbGFuY2UAAAUKYW1tQmFsYW5jZQQLbG9ja0JhZERlYnQDCQBmAgUPcG9zaXRpb25CYWREZWJ0AAAEC2xvY2tCYWREZWJ0CQD8BwQJAQx2YXVsdEFkZHJlc3MAAhVleGNoYW5nZUZyZWVBbmRMb2NrZWQJAMwIAgkAZAIFD3Bvc2l0aW9uQmFkRGVidAUSbGlxdWlkYXRpb25QZW5hbHR5BQNuaWwFA25pbAMJAAACBQtsb2NrQmFkRGVidAULbG9ja0JhZERlYnQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgUDbmlsAwkAAAIFC2xvY2tCYWREZWJ0BQtsb2NrQmFkRGVidAQHdW5zdGFrZQkA/AcECQEMdmF1bHRBZGRyZXNzAAIOd2l0aGRyYXdMb2NrZWQJAMwIAgUSbGlxdWlkYXRpb25QZW5hbHR5BQNuaWwFA25pbAMJAAACBQd1bnN0YWtlBQd1bnN0YWtlBBBkZXBvc2l0SW5zdXJhbmNlCQD8BwQJAQx2YXVsdEFkZHJlc3MAAgdhZGRGcmVlBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUKZmVlVG9WYXVsdAUDbmlsAwkAAAIFEGRlcG9zaXRJbnN1cmFuY2UFEGRlcG9zaXRJbnN1cmFuY2UEDm5vdGlmeU5vdGlvbmFsCQD8BwQJAQxtaW5lckFkZHJlc3MAAg5ub3RpZnlOb3Rpb25hbAkAzAgCBQdfdHJhZGVyCQDMCAIFF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBQNuaWwFA25pbAMJAAACBQ5ub3RpZnlOb3Rpb25hbAUObm90aWZ5Tm90aW9uYWwJAM4IAgkAzggCCQDOCAIDBRRpc1BhcnRpYWxMaXF1aWRhdGlvbgkBDnVwZGF0ZVBvc2l0aW9uBgUHX3RyYWRlcgUPbmV3UG9zaXRpb25TaXplBRFuZXdQb3NpdGlvbk1hcmdpbgUXbmV3UG9zaXRpb25PcGVuTm90aW9uYWwFFG5ld1Bvc2l0aW9uTHN0VXBkQ1BGCQENbGFzdFRpbWVzdGFtcAAJAQ5kZWxldGVQb3NpdGlvbgIFB190cmFkZXIFCl9kaXJlY3Rpb24JAQl1cGRhdGVBbW0IBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBRViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIFFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIFGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIFDnRvdGFsTG9uZ0FmdGVyBQ90b3RhbFNob3J0QWZ0ZXIFGnRvdGFsTG9uZ09wZW5JbnRlcmVzdEFmdGVyBRt0b3RhbFNob3J0T3BlbkludGVyZXN0QWZ0ZXIJAQh3aXRoZHJhdwIIBQFpBmNhbGxlcgUPZmVlVG9MaXF1aWRhdG9yCQENdXBkYXRlQmFsYW5jZQEFDW5ld0FtbUJhbGFuY2UJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEKcGF5RnVuZGluZwEMX3ByaWNlVXBkYXRlBAx1cGRhdGVPcmFjbGUJAPwHBAUEdGhpcwIMdXBkYXRlT3JhY2xlCQDMCAIFDF9wcmljZVVwZGF0ZQUDbmlsBQNuaWwDCQAAAgUMdXBkYXRlT3JhY2xlBQx1cGRhdGVPcmFjbGUEBHN5bmMJAPwHBAUEdGhpcwIZc3luY1Rlcm1pbmFsUHJpY2VUb09yYWNsZQUDbmlsBQNuaWwDCQAAAgUEc3luYwUEc3luYwQVZnVuZGluZ0Jsb2NrVGltZXN0YW1wCQEZbmV4dEZ1bmRpbmdCbG9ja1RpbWVzdGFtcAADAwMJAGYCBRVmdW5kaW5nQmxvY2tUaW1lc3RhbXAJAQ1sYXN0VGltZXN0YW1wAAYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQZwYXVzZWQACQACAQkArAICCQCsAgIJAKwCAgIhSW52YWxpZCBmdW5kaW5nIGJsb2NrIHRpbWVzdGFtcDogCQCkAwEJAQ1sYXN0VGltZXN0YW1wAAIDIDwgCQCkAwEFFWZ1bmRpbmdCbG9ja1RpbWVzdGFtcAQPdW5kZXJseWluZ1ByaWNlCQEOZ2V0T3JhY2xlUHJpY2UABA0kdDA3NzQyMDc3NDk4CQEKZ2V0RnVuZGluZwAEFHNob3J0UHJlbWl1bUZyYWN0aW9uCAUNJHQwNzc0MjA3NzQ5OAJfMQQTbG9uZ1ByZW1pdW1GcmFjdGlvbggFDSR0MDc3NDIwNzc0OTgCXzIEDnByZW1pdW1Ub1ZhdWx0CAUNJHQwNzc0MjA3NzQ5OAJfMwQTZG9QYXlGdW5kaW5nVG9WYXVsdAMJAGYCBQ5wcmVtaXVtVG9WYXVsdAAABBNkb1BheUZ1bmRpbmdUb1ZhdWx0CQD8BwQJAQx2YXVsdEFkZHJlc3MAAhVleGNoYW5nZUZyZWVBbmRMb2NrZWQJAMwIAgkBAS0BBQ5wcmVtaXVtVG9WYXVsdAUDbmlsBQNuaWwDCQAAAgUTZG9QYXlGdW5kaW5nVG9WYXVsdAUTZG9QYXlGdW5kaW5nVG9WYXVsdAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUTZG9QYXlGdW5kaW5nVG9WYXVsdAUTZG9QYXlGdW5kaW5nVG9WYXVsdAkBDXVwZGF0ZUZ1bmRpbmcFCQBkAgUVZnVuZGluZ0Jsb2NrVGltZXN0YW1wCQEUZnVuZGluZ1BlcmlvZFNlY29uZHMACQBkAgkBI2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAAUTbG9uZ1ByZW1pdW1GcmFjdGlvbgkAZAIJASRsYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24ABRRzaG9ydFByZW1pdW1GcmFjdGlvbgkBBGRpdmQCBRNsb25nUHJlbWl1bUZyYWN0aW9uBQ91bmRlcmx5aW5nUHJpY2UJAQRkaXZkAgUUc2hvcnRQcmVtaXVtRnJhY3Rpb24FD3VuZGVybHlpbmdQcmljZQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQx1cGRhdGVPcmFjbGUBDF9wcmljZVVwZGF0ZQMJAAACCQEKb3JhY2xlTW9kZQAFDE9SQUNMRV9QTEFJTgUDbmlsBAxwcmljZVVwZGF0ZXMJALwJAgUMX3ByaWNlVXBkYXRlAgI6OgQKYmFzZU9yYWNsZQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQENZ2V0T3JhY2xlRGF0YQEFDGtfYmFzZU9yYWNsZQIZTm8gYmFzZSBhc3NldCBvcmFjbGUgZGF0YQQRYmFzZU9yYWNsZUFkZHJlc3MIBQpiYXNlT3JhY2xlAl8xBBJkb1VwZGF0ZUJhc2VPcmFjbGUJAPwHBAURYmFzZU9yYWNsZUFkZHJlc3MCCnVwZGF0ZURhdGEJAMwIAgkAkQMCBQxwcmljZVVwZGF0ZXMAAAUDbmlsBQNuaWwDCQAAAgUSZG9VcGRhdGVCYXNlT3JhY2xlBRJkb1VwZGF0ZUJhc2VPcmFjbGUEC3F1b3RlT3JhY2xlCQENZ2V0T3JhY2xlRGF0YQEFDWtfcXVvdGVPcmFjbGUEE2RvVXBkYXRlUXVvdGVPcmFjbGUDCQEJaXNEZWZpbmVkAQULcXVvdGVPcmFjbGUEDHF1b3RlT3JhY2xlVgkBBXZhbHVlAQULcXVvdGVPcmFjbGUEEnF1b3RlT3JhY2xlQWRkcmVzcwgFDHF1b3RlT3JhY2xlVgJfMQQTZG9VcGRhdGVRdW90ZU9yYWNsZQkA/AcEBRJxdW90ZU9yYWNsZUFkZHJlc3MCCnVwZGF0ZURhdGEJAMwIAgkAkQMCBQxwcmljZVVwZGF0ZXMAAQUDbmlsBQNuaWwDCQAAAgUTZG9VcGRhdGVRdW90ZU9yYWNsZQUTZG9VcGRhdGVRdW90ZU9yYWNsZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUTZG9VcGRhdGVRdW90ZU9yYWNsZQUTZG9VcGRhdGVRdW90ZU9yYWNsZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBGXN5bmNUZXJtaW5hbFByaWNlVG9PcmFjbGUABAdfcXRBc3RSCQEGcXRBc3RSAAQHX2JzQXN0UgkBBmJzQXN0UgAEDSR0MDc5MzIxNzk2ODcJARRnZXRTeW5jVGVybWluYWxQcmljZQMJAQ5nZXRPcmFjbGVQcmljZQAFB19xdEFzdFIFB19ic0FzdFIEE25ld1F1b3RlQXNzZXRXZWlnaHQIBQ0kdDA3OTMyMTc5Njg3Al8xBBJuZXdCYXNlQXNzZXRXZWlnaHQIBQ0kdDA3OTMyMTc5Njg3Al8yBA1tYXJnaW5Ub1ZhdWx0CAUNJHQwNzkzMjE3OTY4NwJfMwQQbWFyZ2luVG9WYXVsdEFkagMDCQBmAgAABQ1tYXJnaW5Ub1ZhdWx0CQBmAgkBA2FicwEFDW1hcmdpblRvVmF1bHQJAQhjYmFsYW5jZQAHCQEBLQEJAQhjYmFsYW5jZQAFDW1hcmdpblRvVmF1bHQEDWRvRXhjaGFuZ2VQbkwDCQECIT0CBRBtYXJnaW5Ub1ZhdWx0QWRqAAAEDWRvRXhjaGFuZ2VQbkwJAPwHBAkBDHZhdWx0QWRkcmVzcwACFWV4Y2hhbmdlRnJlZUFuZExvY2tlZAkAzAgCBRBtYXJnaW5Ub1ZhdWx0QWRqBQNuaWwFA25pbAMJAAACBQ1kb0V4Y2hhbmdlUG5MBQ1kb0V4Y2hhbmdlUG5MBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBQ1kb0V4Y2hhbmdlUG5MBQ1kb0V4Y2hhbmdlUG5MCQDOCAIJAQ11cGRhdGVCYWxhbmNlAQkAZAIJAQhjYmFsYW5jZQAFEG1hcmdpblRvVmF1bHRBZGoJARB1cGRhdGVBbW1XZWlnaHRzAgUTbmV3UXVvdGVBc3NldFdlaWdodAUSbmV3QmFzZUFzc2V0V2VpZ2h0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBEGVuc3VyZUNhbGxlZE9uY2UAAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIiSW52YWxpZCBzYXZlQ3VycmVudFR4SWQgcGFyYW1ldGVycwQEdHhJZAkA2AQBCAUBaQ10cmFuc2FjdGlvbklkBAZsYXN0VHgJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwUIa19sYXN0VHgCAAMJAQIhPQIFBmxhc3RUeAUEdHhJZAkAzAgCCQELU3RyaW5nRW50cnkCBQhrX2xhc3RUeAUEdHhJZAUDbmlsCQACAQIpQ2FuIG5vdCBjYWxsIHZBTU0gbWV0aG9kcyB0d2ljZSBpbiBvbmUgdHgBaQEPbWlncmF0ZVBvc2l0aW9uAQdfdHJhZGVyBA9wb3NpdGlvblNpemVPcHQJAJoIAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUOa19wb3NpdGlvblNpemUFB190cmFkZXIDAwkBCWlzRGVmaW5lZAEFD3Bvc2l0aW9uU2l6ZU9wdAkBCWlzRGVmaW5lZAEJAKYIAQUHX3RyYWRlcgcEBXBTaXplCQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBQ5rX3Bvc2l0aW9uU2l6ZQUHX3RyYWRlcgQHcE1hcmdpbgkBEUBleHRyTmF0aXZlKDEwNTApAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUQa19wb3NpdGlvbk1hcmdpbgUHX3RyYWRlcgQJcE5vdGlvbmFsCQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBRZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsBQdfdHJhZGVyBAlwRnJhY3Rpb24JARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFLmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FB190cmFkZXIECnBUaW1lc3RhbXAJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUea19wb3NpdGlvbkxhc3RVcGRhdGVkVGltZXN0YW1wBQdfdHJhZGVyCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAQEcEZlZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBQ1rX3Bvc2l0aW9uRmVlBQdfdHJhZGVyCQEDZmVlAAQJcFNlcXVlbmNlCQERQGV4dHJOYXRpdmUoMTA1MCkCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBRJrX3Bvc2l0aW9uU2VxdWVuY2UFB190cmFkZXIECnBEaXJlY3Rpb24JAQxnZXREaXJlY3Rpb24BBQVwU2l6ZQQLcG9zaXRpb25LZXkJAKwCAgkArAICBQdfdHJhZGVyAgFfCQCkAwEFCnBEaXJlY3Rpb24JAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUOa19wb3NpdGlvblNpemUFB190cmFkZXIJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUQa19wb3NpdGlvbk1hcmdpbgUHX3RyYWRlcgkAzAgCCQELRGVsZXRlRW50cnkBCQEOdG9Db21wb3NpdGVLZXkCBRZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsBQdfdHJhZGVyCQDMCAIJAQtEZWxldGVFbnRyeQEJAQ50b0NvbXBvc2l0ZUtleQIFLmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FB190cmFkZXIJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUea19wb3NpdGlvbkxhc3RVcGRhdGVkVGltZXN0YW1wBQdfdHJhZGVyCQDMCAIJAQtEZWxldGVFbnRyeQEJAQ50b0NvbXBvc2l0ZUtleQIFDWtfcG9zaXRpb25GZWUFB190cmFkZXIJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUSa19wb3NpdGlvblNlcXVlbmNlBQdfdHJhZGVyCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBQ5rX3Bvc2l0aW9uU2l6ZQULcG9zaXRpb25LZXkFBXBTaXplCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRBrX3Bvc2l0aW9uTWFyZ2luBQtwb3NpdGlvbktleQUHcE1hcmdpbgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUWa19wb3NpdGlvbk9wZW5Ob3Rpb25hbAULcG9zaXRpb25LZXkFCXBOb3Rpb25hbAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUua19wb3NpdGlvbkxhc3RVcGRhdGVkQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgULcG9zaXRpb25LZXkFCXBGcmFjdGlvbgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUea19wb3NpdGlvbkxhc3RVcGRhdGVkVGltZXN0YW1wBQtwb3NpdGlvbktleQUKcFRpbWVzdGFtcAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUNa19wb3NpdGlvbkZlZQULcG9zaXRpb25LZXkFBHBGZWUJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFEmtfcG9zaXRpb25TZXF1ZW5jZQULcG9zaXRpb25LZXkFCXBTZXF1ZW5jZQUDbmlsCQACAQkArAICAhdOb3RoaW5nIHRvIG1pZ3JhdGUgZm9yIAUHX3RyYWRlcgFpASd2aWV3X2NhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQDB190cmFkZXIKX2RpcmVjdGlvbgxfcHJpY2VVcGRhdGUEDHVwZGF0ZU9yYWNsZQkA/AcEBQR0aGlzAgx1cGRhdGVPcmFjbGUJAMwIAgUMX3ByaWNlVXBkYXRlBQNuaWwFA25pbAMJAAACBQx1cGRhdGVPcmFjbGUFDHVwZGF0ZU9yYWNsZQQEc3luYwkA/AcEBQR0aGlzAhlzeW5jVGVybWluYWxQcmljZVRvT3JhY2xlBQNuaWwFA25pbAMJAAACBQRzeW5jBQRzeW5jBA0kdDA4MzQ4MTgzNjE3CQELZ2V0UG9zaXRpb24CBQdfdHJhZGVyBQpfZGlyZWN0aW9uBAxwb3NpdGlvblNpemUIBQ0kdDA4MzQ4MTgzNjE3Al8xBA5wb3NpdGlvbk1hcmdpbggFDSR0MDgzNDgxODM2MTcCXzIEA3BvbggFDSR0MDgzNDgxODM2MTcCXzMEEXBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwODM0ODE4MzYxNwJfNAQRcG9zaXRpb25UaW1lc3RhbXAIBQ0kdDA4MzQ4MTgzNjE3Al81BA0kdDA4MzYyMDgzNzMzCQEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwDBQdfdHJhZGVyBQpfZGlyZWN0aW9uBQ9QTkxfT1BUSU9OX1NQT1QEEHBvc2l0aW9uTm90aW9uYWwIBQ0kdDA4MzYyMDgzNzMzAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwODM2MjA4MzczMwJfMgQNJHQwODM3MzY4Mzk2MAkBMGNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnRBbmRSb2xsb3ZlckZlZQUFDHBvc2l0aW9uU2l6ZQUOcG9zaXRpb25NYXJnaW4FEXBvc2l0aW9uTHN0VXBkQ1BGBRFwb3NpdGlvblRpbWVzdGFtcAUNdW5yZWFsaXplZFBubAQMcmVtYWluTWFyZ2luCAUNJHQwODM3MzY4Mzk2MAJfMQQHYmFkRGVidAgFDSR0MDgzNzM2ODM5NjACXzIEDmZ1bmRpbmdQYXltZW50CAUNJHQwODM3MzY4Mzk2MAJfMwQLcm9sbG92ZXJGZWUIBQ0kdDA4MzczNjgzOTYwAl80CQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkBAXMBBQxyZW1haW5NYXJnaW4JAQFzAQUOZnVuZGluZ1BheW1lbnQJAQFzAQkBDmdldE1hcmdpblJhdGlvAgUHX3RyYWRlcgUKX2RpcmVjdGlvbgkBAXMBBQ11bnJlYWxpemVkUG5sCQEBcwEFB2JhZERlYnQJAQFzAQUQcG9zaXRpb25Ob3Rpb25hbAkBAXMBBQtyb2xsb3ZlckZlZQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARV2aWV3X2dldFBlZ0FkanVzdENvc3QBBl9wcmljZQQHX3F0QXN0UgkBBnF0QXN0UgAEB19ic0FzdFIJAQZic0FzdFIABAZyZXN1bHQJARRnZXRTeW5jVGVybWluYWxQcmljZQMFBl9wcmljZQUHX3F0QXN0UgUHX2JzQXN0UgkAAgEJAKQDAQgFBnJlc3VsdAJfMwFpARh2aWV3X2dldFRlcm1pbmFsQW1tUHJpY2UABA0kdDA4NDY5NTg0Nzc2CQETZ2V0VGVybWluYWxBbW1TdGF0ZQAEGXRlcm1pbmFsUXVvdGVBc3NldFJlc2VydmUIBQ0kdDA4NDY5NTg0Nzc2Al8xBBh0ZXJtaW5hbEJhc2VBc3NldFJlc2VydmUIBQ0kdDA4NDY5NTg0Nzc2Al8yBAVwcmljZQkBBGRpdmQCCQEEbXVsZAIFGXRlcm1pbmFsUXVvdGVBc3NldFJlc2VydmUJAQZxdEFzdFcACQEEbXVsZAIFGHRlcm1pbmFsQmFzZUFzc2V0UmVzZXJ2ZQkBBmJzQXN0VwAJAAIBCQCkAwEFBXByaWNlAWkBD3ZpZXdfZ2V0RnVuZGluZwEMX3ByaWNlVXBkYXRlBAx1cGRhdGVPcmFjbGUJAPwHBAUEdGhpcwIMdXBkYXRlT3JhY2xlCQDMCAIFDF9wcmljZVVwZGF0ZQUDbmlsBQNuaWwDCQAAAgUMdXBkYXRlT3JhY2xlBQx1cGRhdGVPcmFjbGUEBHN5bmMJAPwHBAUEdGhpcwIZc3luY1Rlcm1pbmFsUHJpY2VUb09yYWNsZQUDbmlsBQNuaWwDCQAAAgUEc3luYwUEc3luYwQPdW5kZXJseWluZ1ByaWNlCQEOZ2V0T3JhY2xlUHJpY2UABA0kdDA4NTQyODg1NTA2CQEKZ2V0RnVuZGluZwAEFHNob3J0UHJlbWl1bUZyYWN0aW9uCAUNJHQwODU0Mjg4NTUwNgJfMQQTbG9uZ1ByZW1pdW1GcmFjdGlvbggFDSR0MDg1NDI4ODU1MDYCXzIEDnByZW1pdW1Ub1ZhdWx0CAUNJHQwODU0Mjg4NTUwNgJfMwQLbG9uZ0Z1bmRpbmcJAQRkaXZkAgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgUPdW5kZXJseWluZ1ByaWNlBAxzaG9ydEZ1bmRpbmcJAQRkaXZkAgUUc2hvcnRQcmVtaXVtRnJhY3Rpb24FD3VuZGVybHlpbmdQcmljZQkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkBAXMBBQtsb25nRnVuZGluZwkBAXMBBQxzaG9ydEZ1bmRpbmcJAQFzAQkBDGdldFNwb3RQcmljZQAJAQFzAQkBDmdldE9yYWNsZVByaWNlAAkBAXMBBQ5wcmVtaXVtVG9WYXVsdAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARBjb21wdXRlU3BvdFByaWNlAAQEc3luYwkA/AcEBQR0aGlzAhlzeW5jVGVybWluYWxQcmljZVRvT3JhY2xlBQNuaWwFA25pbAMJAAACBQRzeW5jBQRzeW5jBAZyZXN1bHQJAQxnZXRTcG90UHJpY2UACQCUCgIFA25pbAUGcmVzdWx0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBH2NvbXB1dGVGZWVGb3JUcmFkZXJXaXRoQXJ0aWZhY3QCB190cmFkZXILX2FydGlmYWN0SWQEBnJlc3VsdAkBGGdldEZvclRyYWRlcldpdGhBcnRpZmFjdAIFB190cmFkZXIFC19hcnRpZmFjdElkCQCUCgIFA25pbAUGcmVzdWx0AQJ0eAEGdmVyaWZ5AAQOY29vcmRpbmF0b3JTdHIJAJ0IAgUEdGhpcwUUa19jb29yZGluYXRvckFkZHJlc3MDCQEJaXNEZWZpbmVkAQUOY29vcmRpbmF0b3JTdHIEBWFkbWluCQCdCAIJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQV2YWx1ZQEFDmNvb3JkaW5hdG9yU3RyBQ9rX2FkbWluX2FkZHJlc3MDCQEJaXNEZWZpbmVkAQUFYWRtaW4JAQt2YWx1ZU9yRWxzZQIJAJsIAgkBEUBleHRyTmF0aXZlKDEwNjIpAQkBBXZhbHVlAQUFYWRtaW4JAKwCAgkArAICCQCsAgICB3N0YXR1c18JAKUIAQUEdGhpcwIBXwkA2AQBCAUCdHgCaWQHCQACAQIudW5hYmxlIHRvIHZlcmlmeTogYWRtaW4gbm90IHNldCBpbiBjb29yZGluYXRvcgkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAgFAnR4D3NlbmRlclB1YmxpY0tleQnYHyY=", "height": 2585036, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: BHVa92PCgsUSEH2mCm5DkqebcWNfngCnCHSdqLebYf2L Next: FcLwNePbozB24xgNfDQk9aT57i1ao82nqXT1yUSSfQ9r Diff:
OldNewDifferences
800800 let underlyingPrice = getOraclePrice()
801801 let spotPrice = getSpotPrice()
802802 let premium = (spotPrice - underlyingPrice)
803- if (if (if ((totalShortPositionSize() == 0))
803+ if (if (isMarketClosed())
804804 then true
805- else (totalLongPositionSize() == 0))
806- then true
807- else isMarketClosed())
805+ else if ((fundingMode() == FUNDING_ASYMMETRIC))
806+ then if ((totalShortPositionSize() == 0))
807+ then true
808+ else (totalLongPositionSize() == 0)
809+ else false)
808810 then $Tuple3(0, 0, 0)
809811 else if ((0 > premium))
810812 then {
841843 func getAdjustedFee (_artifactId,_baseFeeDiscount) = {
842844 let baseFeeRaw = fee()
843845 let baseFee = muld(baseFeeRaw, _baseFeeDiscount)
844- let $t04210542600 = if ((_artifactId != ""))
846+ let $t04219442689 = if ((_artifactId != ""))
845847 then {
846848 let artifactKind = strA(nftManagerAddress(), toCompositeKey(k_token_type, _artifactId))
847849 if ((artifactKind == FEE_REDUCTION_TOKEN_TYPE))
853855 else throw("Invalid attached artifact")
854856 }
855857 else $Tuple2(baseFee, false)
856- let adjustedFee = $t04210542600._1
857- let burnArtifact = $t04210542600._2
858+ let adjustedFee = $t04219442689._1
859+ let burnArtifact = $t04219442689._2
858860 $Tuple2(adjustedFee, burnArtifact)
859861 }
860862
869871 case _ =>
870872 throw("Invalid computeFeeDiscount result")
871873 }
872- let $t04294643020 = getAdjustedFee(_artifactId, feeDiscount)
873- let adjustedFee = $t04294643020._1
874- let burnArtifact = $t04294643020._2
874+ let $t04303543109 = getAdjustedFee(_artifactId, feeDiscount)
875+ let adjustedFee = $t04303543109._1
876+ let burnArtifact = $t04303543109._2
875877 $Tuple2(adjustedFee, burnArtifact)
876878 }
877879 else throw("Strict value is not equal to itself.")
10131015 let qtAstRAfter = (_qtAstR + _quoteAssetAmount)
10141016 let baseAssetAmountToAdd = (divd(muld(qtAstRAfter, _qtAstW), price) - _bsAstR)
10151017 let bsAstRAfter = (_bsAstR + baseAssetAmountToAdd)
1016- let $t05145451605 = getSyncTerminalPrice(getOraclePrice(), qtAstRAfter, bsAstRAfter)
1017- let newQuoteAssetWeight = $t05145451605._1
1018- let newBaseAssetWeight = $t05145451605._2
1019- let marginToVault = $t05145451605._3
1018+ let $t05154351694 = getSyncTerminalPrice(getOraclePrice(), qtAstRAfter, bsAstRAfter)
1019+ let newQuoteAssetWeight = $t05154351694._1
1020+ let newBaseAssetWeight = $t05154351694._2
1021+ let marginToVault = $t05154351694._3
10201022 let doExchangePnL = if ((marginToVault != 0))
10211023 then {
10221024 let doExchangePnL = invoke(vaultAddress(), "exchangeFreeAndLocked", [marginToVault], nil)
11611163 else isMarketClosed())
11621164 then throw("Invalid increasePosition parameters")
11631165 else {
1164- let $t05798158130 = getForTraderWithArtifact(_trader, getArtifactId(i))
1165- let adjustedFee = $t05798158130._1
1166- let burnArtifact = $t05798158130._2
1166+ let $t05807058219 = getForTraderWithArtifact(_trader, getArtifactId(i))
1167+ let adjustedFee = $t05807058219._1
1168+ let burnArtifact = $t05807058219._2
11671169 let _amount = divd(_rawAmount, (muld(adjustedFee, _leverage) + DECIMAL_UNIT))
11681170 let distributeFeeAmount = (_rawAmount - _amount)
11691171 let referrerFeeAny = invoke(referralAddress(), "acceptPaymentWithLink", [_trader, _refLink], [AttachedPayment(quoteAsset(), distributeFeeAmount)])
11761178 throw("Invalid referrerFee")
11771179 }
11781180 let feeAmount = (distributeFeeAmount - referrerFee)
1179- let $t05862658806 = getPosition(_trader, _direction)
1180- let oldPositionSize = $t05862658806._1
1181- let oldPositionMargin = $t05862658806._2
1182- let oldPositionOpenNotional = $t05862658806._3
1183- let oldPositionLstUpdCPF = $t05862658806._4
1184- let oldPositionTimestamp = $t05862658806._5
1181+ let $t05871558895 = getPosition(_trader, _direction)
1182+ let oldPositionSize = $t05871558895._1
1183+ let oldPositionMargin = $t05871558895._2
1184+ let oldPositionOpenNotional = $t05871558895._3
1185+ let oldPositionLstUpdCPF = $t05871558895._4
1186+ let oldPositionTimestamp = $t05871558895._5
11851187 let isNewPosition = (oldPositionSize == 0)
11861188 let isSameDirection = if ((oldPositionSize > 0))
11871189 then (_direction == DIR_LONG)
11901192 then isSameDirection
11911193 else false
11921194 let isAdd = (_direction == DIR_LONG)
1193- let $t05909562228 = if (if (isNewPosition)
1195+ let $t05918462317 = if (if (isNewPosition)
11941196 then true
11951197 else expandExisting)
11961198 then {
11971199 let openNotional = muld(_amount, _leverage)
1198- let $t05960459777 = swapInput(isAdd, openNotional)
1199- let amountBaseAssetBought = $t05960459777._1
1200- let quoteAssetReserveAfter = $t05960459777._2
1201- let baseAssetReserveAfter = $t05960459777._3
1202- let totalPositionSizeAfter = $t05960459777._4
1200+ let $t05969359866 = swapInput(isAdd, openNotional)
1201+ let amountBaseAssetBought = $t05969359866._1
1202+ let quoteAssetReserveAfter = $t05969359866._2
1203+ let baseAssetReserveAfter = $t05969359866._3
1204+ let totalPositionSizeAfter = $t05969359866._4
12031205 if (if ((_minBaseAssetAmount != 0))
12041206 then (_minBaseAssetAmount > abs(amountBaseAssetBought))
12051207 else false)
12121214 let totalShortOpenInterestAfter = (openInterestShort() + (if ((0 > newPositionSize))
12131215 then openNotional
12141216 else 0))
1215- let $t06032360598 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, _amount)
1216- let remainMargin = $t06032360598._1
1217- let x1 = $t06032360598._2
1218- let x2 = $t06032360598._3
1219- let rolloverFee = $t06032360598._4
1217+ let $t06041260687 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, _amount)
1218+ let remainMargin = $t06041260687._1
1219+ let x1 = $t06041260687._2
1220+ let x2 = $t06041260687._3
1221+ let rolloverFee = $t06041260687._4
12201222 if (!(requireNotOverSpreadLimit(quoteAssetReserveAfter, baseAssetReserveAfter)))
12211223 then throw("Over max spread limit")
12221224 else if (!(requireNotOverMaxOpenNotional(totalLongOpenInterestAfter, totalShortOpenInterestAfter)))
12301232 }
12311233 else {
12321234 let openNotional = muld(_amount, _leverage)
1233- let $t06191662044 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), _direction, PNL_OPTION_SPOT)
1234- let oldPositionNotional = $t06191662044._1
1235- let unrealizedPnl = $t06191662044._2
1235+ let $t06200562133 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), _direction, PNL_OPTION_SPOT)
1236+ let oldPositionNotional = $t06200562133._1
1237+ let unrealizedPnl = $t06200562133._2
12361238 if ((oldPositionNotional > openNotional))
12371239 then throw("Use decreasePosition to decrease position size")
12381240 else throw("Close position first")
12391241 }
1240- let newPositionSize = $t05909562228._1
1241- let newPositionRemainMargin = $t05909562228._2
1242- let newPositionOpenNotional = $t05909562228._3
1243- let newPositionLatestCPF = $t05909562228._4
1244- let newPositionTimestamp = $t05909562228._5
1245- let baseAssetReserveAfter = $t05909562228._6
1246- let quoteAssetReserveAfter = $t05909562228._7
1247- let totalPositionSizeAfter = $t05909562228._8
1248- let openInterestNotionalAfter = $t05909562228._9
1249- let totalLongAfter = $t05909562228._10
1250- let totalShortAfter = $t05909562228._11
1251- let totalLongOpenInterestAfter = $t05909562228._12
1252- let totalShortOpenInterestAfter = $t05909562228._13
1253- let rolloverFee = $t05909562228._14
1254- let $t06223462305 = distributeFee((feeAmount + rolloverFee))
1255- let feeToStakers = $t06223462305._1
1256- let feeToVault = $t06223462305._2
1242+ let newPositionSize = $t05918462317._1
1243+ let newPositionRemainMargin = $t05918462317._2
1244+ let newPositionOpenNotional = $t05918462317._3
1245+ let newPositionLatestCPF = $t05918462317._4
1246+ let newPositionTimestamp = $t05918462317._5
1247+ let baseAssetReserveAfter = $t05918462317._6
1248+ let quoteAssetReserveAfter = $t05918462317._7
1249+ let totalPositionSizeAfter = $t05918462317._8
1250+ let openInterestNotionalAfter = $t05918462317._9
1251+ let totalLongAfter = $t05918462317._10
1252+ let totalShortAfter = $t05918462317._11
1253+ let totalLongOpenInterestAfter = $t05918462317._12
1254+ let totalShortOpenInterestAfter = $t05918462317._13
1255+ let rolloverFee = $t05918462317._14
1256+ let $t06232362394 = distributeFee((feeAmount + rolloverFee))
1257+ let feeToStakers = $t06232362394._1
1258+ let feeToVault = $t06232362394._2
12571259 let stake = if ((_amount >= rolloverFee))
12581260 then invoke(vaultAddress(), "addLocked", nil, [AttachedPayment(quoteAsset(), (_amount - rolloverFee))])
12591261 else invoke(vaultAddress(), "withdrawLocked", [(rolloverFee - _amount)], nil)
13151317 else isMarketClosed())
13161318 then throw("Invalid addMargin parameters")
13171319 else {
1318- let $t06464764827 = getPosition(_trader, _direction)
1319- let oldPositionSize = $t06464764827._1
1320- let oldPositionMargin = $t06464764827._2
1321- let oldPositionOpenNotional = $t06464764827._3
1322- let oldPositionLstUpdCPF = $t06464764827._4
1323- let oldPositionTimestamp = $t06464764827._5
1320+ let $t06473664916 = getPosition(_trader, _direction)
1321+ let oldPositionSize = $t06473664916._1
1322+ let oldPositionMargin = $t06473664916._2
1323+ let oldPositionOpenNotional = $t06473664916._3
1324+ let oldPositionLstUpdCPF = $t06473664916._4
1325+ let oldPositionTimestamp = $t06473664916._5
13241326 let stake = invoke(vaultAddress(), "addLocked", nil, [AttachedPayment(quoteAsset(), _amount)])
13251327 if ((stake == stake))
13261328 then {
13271329 let rolloverFee = calcRolloverFee(oldPositionMargin, oldPositionTimestamp)
13281330 let doTransferFeeToStakers = if ((rolloverFee > 0))
13291331 then {
1330- let $t06511265171 = distributeFee(rolloverFee)
1331- let feeToStakers = $t06511265171._1
1332- let feeToVault = $t06511265171._2
1332+ let $t06520165260 = distributeFee(rolloverFee)
1333+ let feeToStakers = $t06520165260._1
1334+ let feeToVault = $t06520165260._2
13331335 let unstake = invoke(vaultAddress(), "withdrawLocked", [feeToStakers], nil)
13341336 if ((unstake == unstake))
13351337 then {
13801382 else isMarketClosed())
13811383 then throw("Invalid removeMargin parameters")
13821384 else {
1383- let $t06667266852 = getPosition(_trader, _direction)
1384- let oldPositionSize = $t06667266852._1
1385- let oldPositionMargin = $t06667266852._2
1386- let oldPositionOpenNotional = $t06667266852._3
1387- let oldPositionLstUpdCPF = $t06667266852._4
1388- let oldPositionTimestamp = $t06667266852._5
1389- let $t06685867107 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, -(_amount))
1390- let remainMargin = $t06685867107._1
1391- let badDebt = $t06685867107._2
1392- let fundingPayment = $t06685867107._3
1393- let rolloverFee = $t06685867107._4
1385+ let $t06676166941 = getPosition(_trader, _direction)
1386+ let oldPositionSize = $t06676166941._1
1387+ let oldPositionMargin = $t06676166941._2
1388+ let oldPositionOpenNotional = $t06676166941._3
1389+ let oldPositionLstUpdCPF = $t06676166941._4
1390+ let oldPositionTimestamp = $t06676166941._5
1391+ let $t06694767196 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, -(_amount))
1392+ let remainMargin = $t06694767196._1
1393+ let badDebt = $t06694767196._2
1394+ let fundingPayment = $t06694767196._3
1395+ let rolloverFee = $t06694767196._4
13941396 if ((badDebt != 0))
13951397 then throw("Invalid removed margin amount")
13961398 else {
13981400 if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
13991401 then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
14001402 else {
1401- let $t06749367552 = distributeFee(rolloverFee)
1402- let feeToStakers = $t06749367552._1
1403- let feeToVault = $t06749367552._2
1403+ let $t06758267641 = distributeFee(rolloverFee)
1404+ let feeToStakers = $t06758267641._1
1405+ let feeToVault = $t06758267641._2
14041406 let doTransferFeeToStakers = if ((rolloverFee > 0))
14051407 then {
14061408 let lockBadDebt = invoke(vaultAddress(), "exchangeFreeAndLocked", [-(feeToVault)], nil)
14581460 then throw("Invalid closePosition parameters")
14591461 else {
14601462 let oldPositionTimestamp = getPosition(_trader, _direction)._5
1461- let $t06988770490 = internalClosePosition(_trader, _direction, _size, positionFee, _minQuoteAssetAmount, _addToMargin, true, true)
1462- let newPositionSize = $t06988770490._1
1463- let newPositionMargin = $t06988770490._2
1464- let newPositionOpenNotional = $t06988770490._3
1465- let newPositionLstUpdCPF = $t06988770490._4
1466- let positionBadDebt = $t06988770490._5
1467- let realizedPnl = $t06988770490._6
1468- let marginToTrader = $t06988770490._7
1469- let quoteAssetReserveAfter = $t06988770490._8
1470- let baseAssetReserveAfter = $t06988770490._9
1471- let totalPositionSizeAfter = $t06988770490._10
1472- let openInterestNotionalAfter = $t06988770490._11
1473- let totalLongAfter = $t06988770490._12
1474- let totalShortAfter = $t06988770490._13
1475- let totalLongOpenInterestAfter = $t06988770490._14
1476- let totalShortOpenInterestAfter = $t06988770490._15
1477- let realizedFee = $t06988770490._16
1463+ let $t06997670579 = internalClosePosition(_trader, _direction, _size, positionFee, _minQuoteAssetAmount, _addToMargin, true, true)
1464+ let newPositionSize = $t06997670579._1
1465+ let newPositionMargin = $t06997670579._2
1466+ let newPositionOpenNotional = $t06997670579._3
1467+ let newPositionLstUpdCPF = $t06997670579._4
1468+ let positionBadDebt = $t06997670579._5
1469+ let realizedPnl = $t06997670579._6
1470+ let marginToTrader = $t06997670579._7
1471+ let quoteAssetReserveAfter = $t06997670579._8
1472+ let baseAssetReserveAfter = $t06997670579._9
1473+ let totalPositionSizeAfter = $t06997670579._10
1474+ let openInterestNotionalAfter = $t06997670579._11
1475+ let totalLongAfter = $t06997670579._12
1476+ let totalShortAfter = $t06997670579._13
1477+ let totalLongOpenInterestAfter = $t06997670579._14
1478+ let totalShortOpenInterestAfter = $t06997670579._15
1479+ let realizedFee = $t06997670579._16
14781480 if ((positionBadDebt > 0))
14791481 then throw("Invalid closePosition parameters: bad debt")
14801482 else if ((oldPositionTimestamp >= lastTimestamp()))
14981500 case _ =>
14991501 throw("Invalid referrerFee")
15001502 }
1501- let $t07146271535 = distributeFee((realizedFee - referrerFee))
1502- let feeToStakers = $t07146271535._1
1503- let feeToVault = $t07146271535._2
1503+ let $t07155171624 = distributeFee((realizedFee - referrerFee))
1504+ let feeToStakers = $t07155171624._1
1505+ let feeToVault = $t07155171624._2
15041506 let depositVault = invoke(vaultAddress(), "addFree", nil, [AttachedPayment(quoteAsset(), feeToVault)])
15051507 if ((depositVault == depositVault))
15061508 then {
15681570 else false
15691571 let oldPositionSize = getPosition(_trader, _direction)._1
15701572 let positionSizeAbs = abs(oldPositionSize)
1571- let $t07412874451 = if (isPartialLiquidation)
1573+ let $t07421774540 = if (isPartialLiquidation)
15721574 then {
15731575 let liquidationSize = getPartialLiquidationAmount(_trader, oldPositionSize)
15741576 let liquidationRatio = divd(abs(liquidationSize), positionSizeAbs)
15751577 $Tuple2(liquidationRatio, abs(liquidationSize))
15761578 }
15771579 else $Tuple2(0, positionSizeAbs)
1578- let liquidationRatio = $t07412874451._1
1579- let liquidationSize = $t07412874451._2
1580- let $t07445775113 = internalClosePosition(_trader, _direction, if (isPartialLiquidation)
1580+ let liquidationRatio = $t07421774540._1
1581+ let liquidationSize = $t07421774540._2
1582+ let $t07454675202 = internalClosePosition(_trader, _direction, if (isPartialLiquidation)
15811583 then liquidationSize
15821584 else positionSizeAbs, liquidationFeeRatio(), 0, true, false, true)
1583- let newPositionSize = $t07445775113._1
1584- let newPositionMargin = $t07445775113._2
1585- let newPositionOpenNotional = $t07445775113._3
1586- let newPositionLstUpdCPF = $t07445775113._4
1587- let positionBadDebt = $t07445775113._5
1588- let realizedPnl = $t07445775113._6
1589- let marginToTrader = $t07445775113._7
1590- let quoteAssetReserveAfter = $t07445775113._8
1591- let baseAssetReserveAfter = $t07445775113._9
1592- let totalPositionSizeAfter = $t07445775113._10
1593- let openInterestNotionalAfter = $t07445775113._11
1594- let totalLongAfter = $t07445775113._12
1595- let totalShortAfter = $t07445775113._13
1596- let totalLongOpenInterestAfter = $t07445775113._14
1597- let totalShortOpenInterestAfter = $t07445775113._15
1598- let liquidationPenalty = $t07445775113._16
1585+ let newPositionSize = $t07454675202._1
1586+ let newPositionMargin = $t07454675202._2
1587+ let newPositionOpenNotional = $t07454675202._3
1588+ let newPositionLstUpdCPF = $t07454675202._4
1589+ let positionBadDebt = $t07454675202._5
1590+ let realizedPnl = $t07454675202._6
1591+ let marginToTrader = $t07454675202._7
1592+ let quoteAssetReserveAfter = $t07454675202._8
1593+ let baseAssetReserveAfter = $t07454675202._9
1594+ let totalPositionSizeAfter = $t07454675202._10
1595+ let openInterestNotionalAfter = $t07454675202._11
1596+ let totalLongAfter = $t07454675202._12
1597+ let totalShortAfter = $t07454675202._13
1598+ let totalLongOpenInterestAfter = $t07454675202._14
1599+ let totalShortOpenInterestAfter = $t07454675202._15
1600+ let liquidationPenalty = $t07454675202._16
15991601 let feeToLiquidator = (liquidationPenalty / 2)
16001602 let feeToVault = (liquidationPenalty - feeToLiquidator)
16011603 let ammBalance = (cbalance() - liquidationPenalty)
16561658 then throw(((("Invalid funding block timestamp: " + toString(lastTimestamp())) + " < ") + toString(fundingBlockTimestamp)))
16571659 else {
16581660 let underlyingPrice = getOraclePrice()
1659- let $t07733177409 = getFunding()
1660- let shortPremiumFraction = $t07733177409._1
1661- let longPremiumFraction = $t07733177409._2
1662- let premiumToVault = $t07733177409._3
1661+ let $t07742077498 = getFunding()
1662+ let shortPremiumFraction = $t07742077498._1
1663+ let longPremiumFraction = $t07742077498._2
1664+ let premiumToVault = $t07742077498._3
16631665 let doPayFundingToVault = if ((premiumToVault > 0))
16641666 then {
16651667 let doPayFundingToVault = invoke(vaultAddress(), "exchangeFreeAndLocked", [-(premiumToVault)], nil)
17141716 func syncTerminalPriceToOracle () = {
17151717 let _qtAstR = qtAstR()
17161718 let _bsAstR = bsAstR()
1717- let $t07923279598 = getSyncTerminalPrice(getOraclePrice(), _qtAstR, _bsAstR)
1718- let newQuoteAssetWeight = $t07923279598._1
1719- let newBaseAssetWeight = $t07923279598._2
1720- let marginToVault = $t07923279598._3
1719+ let $t07932179687 = getSyncTerminalPrice(getOraclePrice(), _qtAstR, _bsAstR)
1720+ let newQuoteAssetWeight = $t07932179687._1
1721+ let newBaseAssetWeight = $t07932179687._2
1722+ let marginToVault = $t07932179687._3
17211723 let marginToVaultAdj = if (if ((0 > marginToVault))
17221724 then (abs(marginToVault) > cbalance())
17231725 else false)
17821784 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
17831785 if ((sync == sync))
17841786 then {
1785- let $t08339283528 = getPosition(_trader, _direction)
1786- let positionSize = $t08339283528._1
1787- let positionMargin = $t08339283528._2
1788- let pon = $t08339283528._3
1789- let positionLstUpdCPF = $t08339283528._4
1790- let positionTimestamp = $t08339283528._5
1791- let $t08353183644 = getPositionNotionalAndUnrealizedPnl(_trader, _direction, PNL_OPTION_SPOT)
1792- let positionNotional = $t08353183644._1
1793- let unrealizedPnl = $t08353183644._2
1794- let $t08364783871 = calcRemainMarginWithFundingPaymentAndRolloverFee(positionSize, positionMargin, positionLstUpdCPF, positionTimestamp, unrealizedPnl)
1795- let remainMargin = $t08364783871._1
1796- let badDebt = $t08364783871._2
1797- let fundingPayment = $t08364783871._3
1798- let rolloverFee = $t08364783871._4
1787+ let $t08348183617 = getPosition(_trader, _direction)
1788+ let positionSize = $t08348183617._1
1789+ let positionMargin = $t08348183617._2
1790+ let pon = $t08348183617._3
1791+ let positionLstUpdCPF = $t08348183617._4
1792+ let positionTimestamp = $t08348183617._5
1793+ let $t08362083733 = getPositionNotionalAndUnrealizedPnl(_trader, _direction, PNL_OPTION_SPOT)
1794+ let positionNotional = $t08362083733._1
1795+ let unrealizedPnl = $t08362083733._2
1796+ let $t08373683960 = calcRemainMarginWithFundingPaymentAndRolloverFee(positionSize, positionMargin, positionLstUpdCPF, positionTimestamp, unrealizedPnl)
1797+ let remainMargin = $t08373683960._1
1798+ let badDebt = $t08373683960._2
1799+ let fundingPayment = $t08373683960._3
1800+ let rolloverFee = $t08373683960._4
17991801 throw(((((((s(remainMargin) + s(fundingPayment)) + s(getMarginRatio(_trader, _direction))) + s(unrealizedPnl)) + s(badDebt)) + s(positionNotional)) + s(rolloverFee)))
18001802 }
18011803 else throw("Strict value is not equal to itself.")
18171819
18181820 @Callable(i)
18191821 func view_getTerminalAmmPrice () = {
1820- let $t08460684687 = getTerminalAmmState()
1821- let terminalQuoteAssetReserve = $t08460684687._1
1822- let terminalBaseAssetReserve = $t08460684687._2
1822+ let $t08469584776 = getTerminalAmmState()
1823+ let terminalQuoteAssetReserve = $t08469584776._1
1824+ let terminalBaseAssetReserve = $t08469584776._2
18231825 let price = divd(muld(terminalQuoteAssetReserve, qtAstW()), muld(terminalBaseAssetReserve, bsAstW()))
18241826 throw(toString(price))
18251827 }
18351837 if ((sync == sync))
18361838 then {
18371839 let underlyingPrice = getOraclePrice()
1838- let $t08533985417 = getFunding()
1839- let shortPremiumFraction = $t08533985417._1
1840- let longPremiumFraction = $t08533985417._2
1841- let premiumToVault = $t08533985417._3
1840+ let $t08542885506 = getFunding()
1841+ let shortPremiumFraction = $t08542885506._1
1842+ let longPremiumFraction = $t08542885506._2
1843+ let premiumToVault = $t08542885506._3
18421844 let longFunding = divd(longPremiumFraction, underlyingPrice)
18431845 let shortFunding = divd(shortPremiumFraction, underlyingPrice)
18441846 throw(((((s(longFunding) + s(shortFunding)) + s(getSpotPrice())) + s(getOraclePrice())) + s(premiumToVault)))
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let k_baseOracle = "k_baseOracle"
55
66 let k_quoteOracle = "k_quoteOracle"
77
88 let k_balance = "k_balance"
99
1010 let k_sequence = "k_sequence"
1111
1212 let k_positionSize = "k_positionSize"
1313
1414 let k_positionMargin = "k_positionMargin"
1515
1616 let k_positionOpenNotional = "k_positionOpenNotional"
1717
1818 let k_positionLastUpdatedCumulativePremiumFraction = "k_positionFraction"
1919
2020 let k_positionSequence = "k_positionSequence"
2121
2222 let k_positionFee = "k_positionFee"
2323
2424 let k_positionLastUpdatedTimestamp = "k_positionTimestamp"
2525
2626 let k_initialized = "k_initialized"
2727
2828 let k_paused = "k_paused"
2929
3030 let k_closeOnly = "k_closeOnly"
3131
3232 let k_fee = "k_fee"
3333
3434 let k_rolloverFee = "k_rollover_fee"
3535
3636 let k_fundingPeriod = "k_fundingPeriod"
3737
3838 let k_initMarginRatio = "k_initMarginRatio"
3939
4040 let k_maintenanceMarginRatio = "k_mmr"
4141
4242 let k_liquidationFeeRatio = "k_liquidationFeeRatio"
4343
4444 let k_partialLiquidationRatio = "k_partLiquidationRatio"
4545
4646 let k_spreadLimit = "k_spreadLimit"
4747
4848 let k_maxPriceImpact = "k_maxPriceImpact"
4949
5050 let k_maxPriceSpread = "k_maxPriceSpread"
5151
5252 let k_maxOpenNotional = "k_maxOpenNotional"
5353
5454 let k_feeToStakersPercent = "k_feeToStakersPercent"
5555
5656 let k_maxOracleDelay = "k_maxOracleDelay"
5757
5858 let k_fundingMode = "k_fundingMode"
5959
6060 let k_oracleMode = "k_oracleMode"
6161
6262 let k_latestLongCumulativePremiumFraction = "k_latestLongPremiumFraction"
6363
6464 let k_latestShortCumulativePremiumFraction = "k_latestShortPremiumFraction"
6565
6666 let k_nextFundingBlock = "k_nextFundingBlockMinTimestamp"
6767
6868 let k_longFundingRate = "k_longFundingRate"
6969
7070 let k_shortFundingRate = "k_shortFundingRate"
7171
7272 let k_quoteAssetReserve = "k_qtAstR"
7373
7474 let k_baseAssetReserve = "k_bsAstR"
7575
7676 let k_quoteAssetWeight = "k_qtAstW"
7777
7878 let k_baseAssetWeight = "k_bsAstW"
7979
8080 let k_totalPositionSize = "k_totalPositionSize"
8181
8282 let k_totalLongPositionSize = "k_totalLongPositionSize"
8383
8484 let k_totalShortPositionSize = "k_totalShortPositionSize"
8585
8686 let k_openInterestNotional = "k_openInterestNotional"
8787
8888 let k_openInterestShort = "k_openInterestShort"
8989
9090 let k_openInterestLong = "k_openInterestLong"
9191
9292 let k_lastTx = "k_lastTx"
9393
9494 let k_coordinatorAddress = "k_coordinatorAddress"
9595
9696 let k_vault_address = "k_vault_address"
9797
9898 let k_admin_address = "k_admin_address"
9999
100100 let k_quote_asset = "k_quote_asset"
101101
102102 let k_staking_address = "k_staking_address"
103103
104104 let k_miner_address = "k_miner_address"
105105
106106 let k_orders_address = "k_orders_address"
107107
108108 let k_referral_address = "k_referral_address"
109109
110110 let k_nft_manager_address = "k_nft_manager_address"
111111
112112 func toCompositeKey (_key,_address) = ((_key + "_") + _address)
113113
114114
115115 func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
116116
117117
118118 func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address))
119119
120120
121121 func quoteAsset () = fromBase58String(getStringValue(coordinator(), k_quote_asset))
122122
123123
124124 func stakingAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_staking_address)), "Staking not set")
125125
126126
127127 func vaultAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_vault_address)), "Vault not set")
128128
129129
130130 func minerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_miner_address)), "Miner not set")
131131
132132
133133 func ordersAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_orders_address)), "Orders not set")
134134
135135
136136 func referralAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_referral_address)), "Referral not set")
137137
138138
139139 func nftManagerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_nft_manager_address)), "NFT Manager not set")
140140
141141
142142 let k_token_param = "k_token_param"
143143
144144 let k_token_type = "k_token_type"
145145
146146 let FEE_REDUCTION_TOKEN_TYPE = "fee_reduction"
147147
148148 let DIR_LONG = 1
149149
150150 let DIR_SHORT = 2
151151
152152 let SECONDS = 1000
153153
154154 let DECIMAL_NUMBERS = 6
155155
156156 let DECIMAL_UNIT = (1 * (((((10 * 10) * 10) * 10) * 10) * 10))
157157
158158 let MINUTES_IN_YEAR = (525600 * DECIMAL_UNIT)
159159
160160 let ONE_DAY = (86400 * DECIMAL_UNIT)
161161
162162 let PNL_OPTION_SPOT = 1
163163
164164 let PNL_OPTION_ORACLE = 2
165165
166166 let FUNDING_ASYMMETRIC = 1
167167
168168 let FUNDING_SYMMETRIC = 2
169169
170170 let ORACLE_PLAIN = 1
171171
172172 let ORACLE_JIT = 2
173173
174174 func s (_x) = (toString(_x) + ",")
175175
176176
177177 func divd (_x,_y) = fraction(_x, DECIMAL_UNIT, _y, HALFEVEN)
178178
179179
180180 func muld (_x,_y) = fraction(_x, _y, DECIMAL_UNIT, HALFEVEN)
181181
182182
183183 func bdivd (_x,_y) = fraction(_x, toBigInt(DECIMAL_UNIT), _y, HALFEVEN)
184184
185185
186186 func bmuld (_x,_y) = fraction(_x, _y, toBigInt(DECIMAL_UNIT), HALFEVEN)
187187
188188
189189 func abs (_x) = if ((_x > 0))
190190 then _x
191191 else -(_x)
192192
193193
194194 func vmax (_x,_y) = if ((_x >= _y))
195195 then _x
196196 else _y
197197
198198
199199 func int (k) = valueOrErrorMessage(getInteger(this, k), ("no value for " + k))
200200
201201
202202 func intOr (k,def) = valueOrElse(getInteger(this, k), def)
203203
204204
205205 func strA (_address,_key) = {
206206 let val = valueOrErrorMessage(getString(_address, _key), ("No value for key " + _key))
207207 val
208208 }
209209
210210
211211 func intA (_address,_key) = {
212212 let val = valueOrErrorMessage(getInteger(_address, _key), ("No value for key " + _key))
213213 val
214214 }
215215
216216
217217 func cbalance () = int(k_balance)
218218
219219
220220 func fee () = int(k_fee)
221221
222222
223223 func rolloverFeeRate () = int(k_rolloverFee)
224224
225225
226226 func initMarginRatio () = int(k_initMarginRatio)
227227
228228
229229 func qtAstR () = int(k_quoteAssetReserve)
230230
231231
232232 func bsAstR () = int(k_baseAssetReserve)
233233
234234
235235 func qtAstW () = intOr(k_quoteAssetWeight, DECIMAL_UNIT)
236236
237237
238238 func bsAstW () = intOr(k_baseAssetWeight, DECIMAL_UNIT)
239239
240240
241241 func totalPositionSize () = int(k_totalPositionSize)
242242
243243
244244 func openInterestNotional () = int(k_openInterestNotional)
245245
246246
247247 func openInterestShort () = int(k_openInterestShort)
248248
249249
250250 func openInterestLong () = int(k_openInterestLong)
251251
252252
253253 func nextFundingBlockTimestamp () = int(k_nextFundingBlock)
254254
255255
256256 func fundingPeriodRaw () = int(k_fundingPeriod)
257257
258258
259259 func fundingPeriodDecimal () = (fundingPeriodRaw() * DECIMAL_UNIT)
260260
261261
262262 func fundingPeriodSeconds () = (fundingPeriodRaw() * SECONDS)
263263
264264
265265 func maintenanceMarginRatio () = int(k_maintenanceMarginRatio)
266266
267267
268268 func liquidationFeeRatio () = int(k_liquidationFeeRatio)
269269
270270
271271 func partialLiquidationRatio () = int(k_partialLiquidationRatio)
272272
273273
274274 func spreadLimit () = int(k_spreadLimit)
275275
276276
277277 func maxPriceImpact () = int(k_maxPriceImpact)
278278
279279
280280 func maxPriceSpread () = int(k_maxPriceSpread)
281281
282282
283283 func maxOpenNotional () = int(k_maxOpenNotional)
284284
285285
286286 func latestLongCumulativePremiumFraction () = int(k_latestLongCumulativePremiumFraction)
287287
288288
289289 func latestShortCumulativePremiumFraction () = int(k_latestShortCumulativePremiumFraction)
290290
291291
292292 func totalShortPositionSize () = int(k_totalShortPositionSize)
293293
294294
295295 func totalLongPositionSize () = int(k_totalLongPositionSize)
296296
297297
298298 func lastSequence () = intOr(k_sequence, 0)
299299
300300
301301 func feeToStakersPercent () = int(k_feeToStakersPercent)
302302
303303
304304 func maxOracleDelay () = int(k_maxOracleDelay)
305305
306306
307307 func fundingMode () = intOr(k_fundingMode, FUNDING_ASYMMETRIC)
308308
309309
310310 func oracleMode () = intOr(k_oracleMode, ORACLE_PLAIN)
311311
312312
313313 func lastTimestamp () = lastBlock.timestamp
314314
315315
316316 func getActualCaller (i) = valueOrElse(getString(ordersAddress(), "k_sender"), toString(i.caller))
317317
318318
319319 func requireMoreMarginRatio (_marginRatio,_baseMarginRatio,_largerThanOrEqualTo) = {
320320 let remainingMarginRatio = (_marginRatio - _baseMarginRatio)
321321 if (if (_largerThanOrEqualTo)
322322 then (0 > remainingMarginRatio)
323323 else false)
324324 then throw(((("Invalid margin: " + toString(_marginRatio)) + " < ") + toString(_baseMarginRatio)))
325325 else if (if (!(_largerThanOrEqualTo))
326326 then (remainingMarginRatio >= 0)
327327 else false)
328328 then throw(((("Invalid margin: " + toString(_marginRatio)) + " > ") + toString(_baseMarginRatio)))
329329 else true
330330 }
331331
332332
333333 func latestCumulativePremiumFraction (_positionSize) = if ((_positionSize == 0))
334334 then throw("Should not be called with _positionSize == 0")
335335 else if ((_positionSize > 0))
336336 then latestLongCumulativePremiumFraction()
337337 else latestShortCumulativePremiumFraction()
338338
339339
340340 func getPosition (_trader,_direction) = {
341341 let positionKey = ((_trader + "_") + toString(_direction))
342342 let positionSizeOpt = getInteger(this, toCompositeKey(k_positionSize, positionKey))
343343 match positionSizeOpt {
344344 case positionSize: Int =>
345345 $Tuple5(positionSize, getIntegerValue(this, toCompositeKey(k_positionMargin, positionKey)), getIntegerValue(this, toCompositeKey(k_positionOpenNotional, positionKey)), getIntegerValue(this, toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, positionKey)), getIntegerValue(this, toCompositeKey(k_positionLastUpdatedTimestamp, positionKey)))
346346 case _ =>
347347 $Tuple5(0, 0, 0, 0, 0)
348348 }
349349 }
350350
351351
352352 func getDirection (_positionSize) = if ((0 > _positionSize))
353353 then DIR_SHORT
354354 else DIR_LONG
355355
356356
357357 func getPositionFee (_trader,_direction) = {
358358 let positionKey = ((_trader + "_") + toString(_direction))
359359 let positionFeeOpt = getInteger(this, toCompositeKey(k_positionFee, positionKey))
360360 match positionFeeOpt {
361361 case positionFee: Int =>
362362 positionFee
363363 case _ =>
364364 fee()
365365 }
366366 }
367367
368368
369369 func requireOpenPosition (_trader,_direction) = if ((getPosition(_trader, _direction)._1 == 0))
370370 then throw("No open position")
371371 else true
372372
373373
374374 func getOracleData (key) = {
375375 let oracleDataStr = getString(this, key)
376376 if (if (isDefined(oracleDataStr))
377377 then (value(oracleDataStr) != "")
378378 else false)
379379 then {
380380 let oracleData = split(value(oracleDataStr), ",")
381381 let oracleAddress = valueOrErrorMessage(addressFromString(oracleData[0]), ("Invalid oracle address in: " + value(oracleDataStr)))
382382 let priceKey = oracleData[1]
383383 let blockKey = oracleData[2]
384384 let openKey = oracleData[3]
385385 $Tuple4(oracleAddress, priceKey, blockKey, openKey)
386386 }
387387 else unit
388388 }
389389
390390
391391 func initialized () = valueOrElse(getBoolean(this, k_initialized), false)
392392
393393
394394 func paused () = valueOrElse(getBoolean(this, k_paused), false)
395395
396396
397397 func closeOnly () = valueOrElse(getBoolean(this, k_closeOnly), false)
398398
399399
400400 func updateReserve (_isAdd,_quoteAssetAmount,_baseAssetAmount) = if (_isAdd)
401401 then {
402402 let newBase = (bsAstR() - _baseAssetAmount)
403403 if ((0 >= newBase))
404404 then throw("Tx lead to base asset reserve <= 0, revert")
405405 else $Tuple3((qtAstR() + _quoteAssetAmount), newBase, (totalPositionSize() + _baseAssetAmount))
406406 }
407407 else {
408408 let newQuote = (qtAstR() - _quoteAssetAmount)
409409 if ((0 >= newQuote))
410410 then throw("Tx lead to base quote reserve <= 0, revert")
411411 else $Tuple3(newQuote, (bsAstR() + _baseAssetAmount), (totalPositionSize() - _baseAssetAmount))
412412 }
413413
414414
415415 func calcInvariant (_qtAstR,_bsAstR) = {
416416 let bqtAstR = toBigInt(_qtAstR)
417417 let bbsAstR = toBigInt(_bsAstR)
418418 bmuld(bqtAstR, bbsAstR)
419419 }
420420
421421
422422 func swapInput (_isAdd,_quoteAssetAmount) = {
423423 let _qtAstR = qtAstR()
424424 let _bsAstR = bsAstR()
425425 let _qtAstW = qtAstW()
426426 let _bsAstW = bsAstW()
427427 let quoteAssetAmountAdjusted = divd(_quoteAssetAmount, _qtAstW)
428428 let k = calcInvariant(_qtAstR, _bsAstR)
429429 let quoteAssetReserveAfter = if (_isAdd)
430430 then (_qtAstR + quoteAssetAmountAdjusted)
431431 else (_qtAstR - quoteAssetAmountAdjusted)
432432 let baseAssetReserveAfter = toInt(bdivd(k, toBigInt(quoteAssetReserveAfter)))
433433 let amountBaseAssetBoughtAbs = abs((baseAssetReserveAfter - _bsAstR))
434434 let amountBaseAssetBought = if (_isAdd)
435435 then amountBaseAssetBoughtAbs
436436 else -(amountBaseAssetBoughtAbs)
437437 let $t01578615956 = updateReserve(_isAdd, quoteAssetAmountAdjusted, amountBaseAssetBoughtAbs)
438438 let quoteAssetReserveAfter1 = $t01578615956._1
439439 let baseAssetReserveAfter1 = $t01578615956._2
440440 let totalPositionSizeAfter1 = $t01578615956._3
441441 let priceBefore = divd(muld(_qtAstR, _qtAstW), muld(_bsAstR, _bsAstW))
442442 let marketPrice = divd(_quoteAssetAmount, amountBaseAssetBoughtAbs)
443443 let priceDiff = abs((priceBefore - marketPrice))
444444 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
445445 let maxPriceImpactValue = maxPriceImpact()
446446 if ((priceImpact > maxPriceImpactValue))
447447 then throw(((((((((((((("Price impact " + toString(priceImpact)) + " > max price impact ") + toString(maxPriceImpactValue)) + " before quote asset: ") + toString(_qtAstR)) + " before base asset: ") + toString(_bsAstR)) + " quote asset amount to exchange: ") + toString(_quoteAssetAmount)) + " price before: ") + toString(priceBefore)) + " marketPrice: ") + toString(marketPrice)))
448448 else $Tuple4(amountBaseAssetBought, quoteAssetReserveAfter1, baseAssetReserveAfter1, totalPositionSizeAfter1)
449449 }
450450
451451
452452 func calcRolloverFee (_oldPositionMargin,_oldPositionLastUpdatedTimestamp) = {
453453 let positionMinutes = ((((lastTimestamp() - _oldPositionLastUpdatedTimestamp) / 1000) / 60) * DECIMAL_UNIT)
454454 let rolloverFee = divd(muld(muld(_oldPositionMargin, positionMinutes), rolloverFeeRate()), MINUTES_IN_YEAR)
455455 rolloverFee
456456 }
457457
458458
459459 func calcRemainMarginWithFundingPaymentAndRolloverFee (_oldPositionSize,_oldPositionMargin,_oldPositionCumulativePremiumFraction,_oldPositionLastUpdatedTimestamp,_marginDelta) = {
460460 let fundingPayment = if ((_oldPositionSize != 0))
461461 then {
462462 let _latestCumulativePremiumFraction = latestCumulativePremiumFraction(_oldPositionSize)
463463 muld((_latestCumulativePremiumFraction - _oldPositionCumulativePremiumFraction), _oldPositionSize)
464464 }
465465 else 0
466466 let rolloverFee = calcRolloverFee(_oldPositionMargin, _oldPositionLastUpdatedTimestamp)
467467 let signedMargin = (((_marginDelta - rolloverFee) - fundingPayment) + _oldPositionMargin)
468468 let $t01862318750 = if ((0 > signedMargin))
469469 then $Tuple2(0, abs(signedMargin))
470470 else $Tuple2(abs(signedMargin), 0)
471471 let remainMargin = $t01862318750._1
472472 let badDebt = $t01862318750._2
473473 $Tuple4(remainMargin, badDebt, fundingPayment, rolloverFee)
474474 }
475475
476476
477477 func swapOutputWithReserves (_isAdd,_baseAssetAmount,_checkMaxPriceImpact,_quoteAssetReserve,_quoteAssetWeight,_baseAssetReserve,_baseAssetWeight) = {
478478 let priceBefore = divd(muld(_quoteAssetReserve, _quoteAssetWeight), muld(_baseAssetReserve, _baseAssetWeight))
479479 if ((_baseAssetAmount == 0))
480480 then throw("Invalid base asset amount")
481481 else {
482482 let k = calcInvariant(_quoteAssetReserve, _baseAssetReserve)
483483 let baseAssetPoolAmountAfter = if (_isAdd)
484484 then (_baseAssetReserve + _baseAssetAmount)
485485 else (_baseAssetReserve - _baseAssetAmount)
486486 let quoteAssetAfter = toInt(bdivd(k, toBigInt(baseAssetPoolAmountAfter)))
487487 let quoteAssetDelta = abs((quoteAssetAfter - _quoteAssetReserve))
488488 let quoteAssetSold = muld(quoteAssetDelta, _quoteAssetWeight)
489489 let maxPriceImpactValue = maxPriceImpact()
490490 let $t02001220174 = updateReserve(!(_isAdd), quoteAssetDelta, _baseAssetAmount)
491491 let quoteAssetReserveAfter1 = $t02001220174._1
492492 let baseAssetReserveAfter1 = $t02001220174._2
493493 let totalPositionSizeAfter1 = $t02001220174._3
494494 let marketPrice = divd(quoteAssetSold, _baseAssetAmount)
495495 let priceDiff = abs((priceBefore - marketPrice))
496496 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
497497 if (if ((priceImpact > maxPriceImpactValue))
498498 then _checkMaxPriceImpact
499499 else false)
500500 then throw(((((((((((((("Price impact " + toString(priceImpact)) + " > max price impact ") + toString(maxPriceImpactValue)) + " before quote asset: ") + toString(_quoteAssetReserve)) + " before base asset: ") + toString(_baseAssetReserve)) + " base asset amount to exchange: ") + toString(_baseAssetAmount)) + " price before: ") + toString(priceBefore)) + " market price: ") + toString(marketPrice)))
501501 else $Tuple7(quoteAssetSold, quoteAssetReserveAfter1, baseAssetReserveAfter1, totalPositionSizeAfter1, (totalLongPositionSize() - (if (_isAdd)
502502 then abs(_baseAssetAmount)
503503 else 0)), (totalShortPositionSize() - (if (!(_isAdd))
504504 then abs(_baseAssetAmount)
505505 else 0)), priceImpact)
506506 }
507507 }
508508
509509
510510 func swapOutput (_isAdd,_baseAssetAmount,_checkMaxPriceImpact) = swapOutputWithReserves(_isAdd, _baseAssetAmount, _checkMaxPriceImpact, qtAstR(), qtAstW(), bsAstR(), bsAstW())
511511
512512
513513 func getOraclePriceValue (oracle,priceKey,blockKey) = {
514514 let lastValue = valueOrErrorMessage(getInteger(oracle, priceKey), ((("Can not get oracle price. Oracle: " + toString(oracle)) + " key: ") + priceKey))
515515 if ((blockKey != ""))
516516 then {
517517 let currentBlock = height
518518 let lastOracleBlock = valueOrErrorMessage(getInteger(oracle, blockKey), ((("Can not get oracle block. Oracle: " + toString(oracle)) + " key: ") + blockKey))
519519 if (((currentBlock - lastOracleBlock) > maxOracleDelay()))
520520 then throw(((("Oracle stale data. Last oracle block: " + toString(lastOracleBlock)) + " current block: ") + toString(currentBlock)))
521521 else lastValue
522522 }
523523 else lastValue
524524 }
525525
526526
527527 func getOraclePrice () = {
528528 let baseOracle = valueOrErrorMessage(getOracleData(k_baseOracle), "No base asset oracle data")
529529 let baseOraclePrice = getOraclePriceValue(baseOracle._1, baseOracle._2, baseOracle._3)
530530 let quoteOracle = getOracleData(k_quoteOracle)
531531 let quoteOraclePrice = if (isDefined(quoteOracle))
532532 then {
533533 let quoteOracleV = value(quoteOracle)
534534 getOraclePriceValue(quoteOracleV._1, quoteOracleV._2, quoteOracleV._3)
535535 }
536536 else DECIMAL_UNIT
537537 divd(baseOraclePrice, quoteOraclePrice)
538538 }
539539
540540
541541 func isMarketClosed () = {
542542 let baseOracle = valueOrErrorMessage(getOracleData(k_baseOracle), "No base asset oracle data")
543543 let oracle = baseOracle._1
544544 let openKey = baseOracle._4
545545 if ((openKey != ""))
546546 then {
547547 let isOpen = valueOrErrorMessage(getBoolean(oracle, openKey), ((("Can not get oracle is open/closed. Oracle: " + toString(oracle)) + " key: ") + openKey))
548548 !(isOpen)
549549 }
550550 else false
551551 }
552552
553553
554554 func absPriceDiff (_oraclePrice,_quoteAssetReserve,_baseAssetReserve,_qtAstW,_bsAstW) = {
555555 let priceAfter = divd(muld(_quoteAssetReserve, _qtAstW), muld(_baseAssetReserve, _bsAstW))
556556 let averagePrice = divd((_oraclePrice + priceAfter), (2 * DECIMAL_UNIT))
557557 let absPriceDiff = divd(abs((_oraclePrice - priceAfter)), averagePrice)
558558 absPriceDiff
559559 }
560560
561561
562562 func requireNotOverSpreadLimit (_quoteAssetReserve,_baseAssetReserve) = {
563563 let oraclePrice = getOraclePrice()
564564 let _qtAstW = qtAstW()
565565 let _bsAstW = bsAstW()
566566 let absPriceDiffBefore = absPriceDiff(oraclePrice, qtAstR(), bsAstR(), _qtAstW, _bsAstW)
567567 let absPriceDiffAfter = absPriceDiff(oraclePrice, _quoteAssetReserve, _baseAssetReserve, _qtAstW, _bsAstW)
568568 if (if ((absPriceDiffAfter > maxPriceSpread()))
569569 then (absPriceDiffAfter > absPriceDiffBefore)
570570 else false)
571571 then throw(((("Price spread " + toString(absPriceDiffAfter)) + " > max price spread ") + toString(maxPriceSpread())))
572572 else true
573573 }
574574
575575
576576 func requireNotOverMaxOpenNotional (_longOpenNotional,_shortOpenNotional) = {
577577 let _maxOpenNotional = maxOpenNotional()
578578 if ((_longOpenNotional > _maxOpenNotional))
579579 then throw(((("Long open notional " + toString(_longOpenNotional)) + " > max open notional ") + toString(_maxOpenNotional)))
580580 else if ((_shortOpenNotional > _maxOpenNotional))
581581 then throw(((("Short open notional " + toString(_shortOpenNotional)) + " > max open notional ") + toString(_maxOpenNotional)))
582582 else true
583583 }
584584
585585
586586 func getSpotPrice () = {
587587 let _quoteAssetReserve = qtAstR()
588588 let _baseAssetReserve = bsAstR()
589589 let _qtAstW = qtAstW()
590590 let _bsAstW = bsAstW()
591591 divd(muld(_quoteAssetReserve, _qtAstW), muld(_baseAssetReserve, _bsAstW))
592592 }
593593
594594
595595 func isOverFluctuationLimit () = {
596596 let oraclePrice = getOraclePrice()
597597 let currentPrice = getSpotPrice()
598598 (divd(abs((oraclePrice - currentPrice)), oraclePrice) > spreadLimit())
599599 }
600600
601601
602602 func getPositionAdjustedOpenNotional (_positionSize,_option,_quoteAssetReserve,_quoteAssetWeight,_baseAssetReserve,_baseAssetWeight) = {
603603 let positionSizeAbs = abs(_positionSize)
604604 let isShort = (0 > _positionSize)
605605 let positionNotional = if ((_option == PNL_OPTION_SPOT))
606606 then {
607607 let outPositionNotional = swapOutputWithReserves(!(isShort), positionSizeAbs, false, _quoteAssetReserve, _quoteAssetWeight, _baseAssetReserve, _baseAssetWeight)._1
608608 outPositionNotional
609609 }
610610 else muld(positionSizeAbs, getOraclePrice())
611611 positionNotional
612612 }
613613
614614
615615 func getPositionNotionalAndUnrealizedPnlByValues (_positionSize,_positionOpenNotional,_quoteAssetReserve,_quoteAssetWeight,_baseAssetReserve,_baseAssetWeight,_option) = if ((_positionSize == 0))
616616 then throw("Invalid position size")
617617 else {
618618 let isShort = (0 > _positionSize)
619619 let positionNotional = getPositionAdjustedOpenNotional(_positionSize, _option, _quoteAssetReserve, _quoteAssetWeight, _baseAssetReserve, _baseAssetWeight)
620620 let unrealizedPnl = if (isShort)
621621 then (_positionOpenNotional - positionNotional)
622622 else (positionNotional - _positionOpenNotional)
623623 $Tuple2(positionNotional, unrealizedPnl)
624624 }
625625
626626
627627 func getPositionNotionalAndUnrealizedPnl (_trader,_direction,_option) = {
628628 let $t02805928199 = getPosition(_trader, _direction)
629629 let positionSize = $t02805928199._1
630630 let positionMargin = $t02805928199._2
631631 let positionOpenNotional = $t02805928199._3
632632 let positionLstUpdCPF = $t02805928199._4
633633 getPositionNotionalAndUnrealizedPnlByValues(positionSize, positionOpenNotional, qtAstR(), qtAstW(), bsAstR(), bsAstW(), _option)
634634 }
635635
636636
637637 func calcMarginRatio (_remainMargin,_badDebt,_positionNotional) = divd((_remainMargin - _badDebt), _positionNotional)
638638
639639
640640 func getMarginRatioByOption (_trader,_direction,_option) = {
641641 let $t02873128884 = getPosition(_trader, _direction)
642642 let positionSize = $t02873128884._1
643643 let positionMargin = $t02873128884._2
644644 let pon = $t02873128884._3
645645 let positionLastUpdatedCPF = $t02873128884._4
646646 let positionTimestamp = $t02873128884._5
647647 let $t02889028995 = getPositionNotionalAndUnrealizedPnl(_trader, _direction, _option)
648648 let positionNotional = $t02889028995._1
649649 let unrealizedPnl = $t02889028995._2
650650 let $t02900029212 = calcRemainMarginWithFundingPaymentAndRolloverFee(positionSize, positionMargin, positionLastUpdatedCPF, positionTimestamp, unrealizedPnl)
651651 let remainMargin = $t02900029212._1
652652 let badDebt = $t02900029212._2
653653 calcMarginRatio(remainMargin, badDebt, positionNotional)
654654 }
655655
656656
657657 func getMarginRatio (_trader,_direction) = getMarginRatioByOption(_trader, _direction, PNL_OPTION_SPOT)
658658
659659
660660 func getPartialLiquidationAmount (_trader,_positionSize) = {
661661 let maximumRatio = vmax(partialLiquidationRatio(), (DECIMAL_UNIT - divd(getMarginRatio(_trader, getDirection(_positionSize)), maintenanceMarginRatio())))
662662 let maxExchangedPositionSize = muld(abs(_positionSize), maximumRatio)
663663 let swapResult = swapOutput((_positionSize > 0), maxExchangedPositionSize, false)
664664 let maxExchangedQuoteAssetAmount = swapResult._1
665665 let priceImpact = swapResult._7
666666 if ((maxPriceImpact() > priceImpact))
667667 then maxExchangedPositionSize
668668 else muld(abs(_positionSize), partialLiquidationRatio())
669669 }
670670
671671
672672 func internalClosePosition (_trader,_direction,_size,_fee,_minQuoteAssetAmount,_addToMargin,_checkMaxPriceImpact,_liquidate) = {
673673 let $t03037530543 = getPosition(_trader, _direction)
674674 let oldPositionSize = $t03037530543._1
675675 let oldPositionMargin = $t03037530543._2
676676 let oldPositionOpenNotional = $t03037530543._3
677677 let oldPositionLstUpdCPF = $t03037530543._4
678678 let oldPositionTimestamp = $t03037530543._5
679679 let isLongPosition = (oldPositionSize > 0)
680680 let absOldPositionSize = abs(oldPositionSize)
681681 if (if ((absOldPositionSize >= _size))
682682 then (_size > 0)
683683 else false)
684684 then {
685685 let isPartialClose = (absOldPositionSize > _size)
686686 let $t03083531286 = swapOutput((oldPositionSize > 0), _size, _checkMaxPriceImpact)
687687 let exchangedQuoteAssetAmount = $t03083531286._1
688688 let quoteAssetReserveAfter = $t03083531286._2
689689 let baseAssetReserveAfter = $t03083531286._3
690690 let totalPositionSizeAfter = $t03083531286._4
691691 let exchangedPositionSize = if ((oldPositionSize > 0))
692692 then -(_size)
693693 else _size
694694 let $t03150131725 = getPositionNotionalAndUnrealizedPnl(_trader, _direction, PNL_OPTION_SPOT)
695695 let oldPositionNotional = $t03150131725._1
696696 let unrealizedPnl = $t03150131725._2
697697 let realizedRatio = divd(abs(exchangedPositionSize), absOldPositionSize)
698698 let realizedPnl = muld(unrealizedPnl, realizedRatio)
699699 let $t03206632312 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, unrealizedPnl)
700700 let remainMarginBefore = $t03206632312._1
701701 let x1 = $t03206632312._2
702702 let x2 = $t03206632312._3
703703 let rolloverFee = $t03206632312._4
704704 let positionBadDebt = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, realizedPnl)._2
705705 let realizedCloseFee = muld(muld(oldPositionNotional, realizedRatio), _fee)
706706 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
707707 let remainOpenNotional = if ((oldPositionSize > 0))
708708 then ((oldPositionNotional - exchangedQuoteAssetAmount) - unrealizedPnlAfter)
709709 else ((unrealizedPnlAfter + oldPositionNotional) - exchangedQuoteAssetAmount)
710710 let newPositionSize = (oldPositionSize + exchangedPositionSize)
711711 let $t03371834104 = if ((newPositionSize == 0))
712712 then $Tuple2(0, 0)
713713 else $Tuple2(abs(remainOpenNotional), latestCumulativePremiumFraction(newPositionSize))
714714 let newPositionOpenNotional = $t03371834104._1
715715 let newPositionLstUpdCPF = $t03371834104._2
716716 let openNotionalDelta = (oldPositionOpenNotional - newPositionOpenNotional)
717717 let marginRatio = getMarginRatioByOption(_trader, _direction, PNL_OPTION_SPOT)
718718 let newPositionMarginWithSameRatio = if ((oldPositionSize > 0))
719719 then (muld((newPositionOpenNotional + unrealizedPnlAfter), marginRatio) - unrealizedPnlAfter)
720720 else (muld((newPositionOpenNotional - unrealizedPnlAfter), marginRatio) - unrealizedPnlAfter)
721721 let marginToTraderRaw = ((remainMarginBefore - (newPositionMarginWithSameRatio + unrealizedPnlAfter)) - realizedCloseFee)
722722 let marginToTrader = if ((0 > marginToTraderRaw))
723723 then if (_liquidate)
724724 then 0
725725 else throw("Invalid internalClosePosition params: unable to pay fee")
726726 else marginToTraderRaw
727727 let newPositionMargin = if (_addToMargin)
728728 then (newPositionMarginWithSameRatio + marginToTrader)
729729 else newPositionMarginWithSameRatio
730730 if (if ((_minQuoteAssetAmount != 0))
731731 then (_minQuoteAssetAmount > exchangedQuoteAssetAmount)
732732 else false)
733733 then throw(((("Limit error: " + toString(exchangedQuoteAssetAmount)) + " < ") + toString(_minQuoteAssetAmount)))
734734 else $Tuple17(newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF, positionBadDebt, realizedPnl, if (if (_addToMargin)
735735 then isPartialClose
736736 else false)
737737 then 0
738738 else marginToTrader, quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, (openInterestNotional() - openNotionalDelta), (totalLongPositionSize() - (if (isLongPosition)
739739 then abs(exchangedPositionSize)
740740 else 0)), (totalShortPositionSize() - (if (!(isLongPosition))
741741 then abs(exchangedPositionSize)
742742 else 0)), (openInterestLong() - (if (isLongPosition)
743743 then openNotionalDelta
744744 else 0)), (openInterestShort() - (if (!(isLongPosition))
745745 then openNotionalDelta
746746 else 0)), (realizedCloseFee + rolloverFee), exchangedQuoteAssetAmount)
747747 }
748748 else throw(((("Invalid internalClosePosition params: invalid position size: " + toString(_size)) + " max: ") + toString(absOldPositionSize)))
749749 }
750750
751751
752752 func getTerminalAmmState () = {
753753 let _positionSize = totalPositionSize()
754754 if ((_positionSize == 0))
755755 then $Tuple2(qtAstR(), bsAstR())
756756 else {
757757 let direction = (_positionSize > 0)
758758 let $t03735137530 = swapOutput(direction, abs(_positionSize), false)
759759 let currentNetMarketValue = $t03735137530._1
760760 let terminalQuoteAssetReserve = $t03735137530._2
761761 let terminalBaseAssetReserve = $t03735137530._3
762762 $Tuple2(terminalQuoteAssetReserve, terminalBaseAssetReserve)
763763 }
764764 }
765765
766766
767767 func getQuoteAssetWeight (baseAssetReserve,totalPositionSize,quoteAssetReserve,targetPrice) = {
768768 let b = toBigInt(baseAssetReserve)
769769 let sz = toBigInt(totalPositionSize)
770770 let q = toBigInt(quoteAssetReserve)
771771 let p = toBigInt(targetPrice)
772772 let k = bmuld(q, b)
773773 let newB = (b + sz)
774774 let newQ = bdivd(k, newB)
775775 let z = bdivd(newQ, newB)
776776 let result = bdivd(p, z)
777777 toInt(result)
778778 }
779779
780780
781781 func getSyncTerminalPrice (_terminalPrice,_qtAstR,_bsAstR) = {
782782 let _positionSize = totalPositionSize()
783783 if ((_positionSize == 0))
784784 then {
785785 let newQtAstW = divd(muld(_terminalPrice, _bsAstR), _qtAstR)
786786 $Tuple3(newQtAstW, DECIMAL_UNIT, 0)
787787 }
788788 else {
789789 let direction = (_positionSize > 0)
790790 let currentNetMarketValue = swapOutput(direction, abs(_positionSize), false)._1
791791 let newQtAstW = getQuoteAssetWeight(_bsAstR, _positionSize, _qtAstR, _terminalPrice)
792792 let newBsAstW = DECIMAL_UNIT
793793 let marginToVault = getPositionNotionalAndUnrealizedPnlByValues(_positionSize, currentNetMarketValue, _qtAstR, newQtAstW, _bsAstR, newBsAstW, PNL_OPTION_SPOT)._2
794794 $Tuple3(newQtAstW, newBsAstW, marginToVault)
795795 }
796796 }
797797
798798
799799 func getFunding () = {
800800 let underlyingPrice = getOraclePrice()
801801 let spotPrice = getSpotPrice()
802802 let premium = (spotPrice - underlyingPrice)
803- if (if (if ((totalShortPositionSize() == 0))
803+ if (if (isMarketClosed())
804804 then true
805- else (totalLongPositionSize() == 0))
806- then true
807- else isMarketClosed())
805+ else if ((fundingMode() == FUNDING_ASYMMETRIC))
806+ then if ((totalShortPositionSize() == 0))
807+ then true
808+ else (totalLongPositionSize() == 0)
809+ else false)
808810 then $Tuple3(0, 0, 0)
809811 else if ((0 > premium))
810812 then {
811813 let shortPremiumFraction = divd(muld(premium, fundingPeriodDecimal()), ONE_DAY)
812814 if ((fundingMode() == FUNDING_ASYMMETRIC))
813815 then {
814816 let longPremiumFraction = divd(muld(shortPremiumFraction, totalShortPositionSize()), totalLongPositionSize())
815817 $Tuple3(shortPremiumFraction, longPremiumFraction, 0)
816818 }
817819 else {
818820 let shortTotalPremiumFraction = abs(muld(shortPremiumFraction, totalShortPositionSize()))
819821 let longTotalPremiumFraction = abs(muld(shortPremiumFraction, totalLongPositionSize()))
820822 let premiumToVault = (shortTotalPremiumFraction - longTotalPremiumFraction)
821823 $Tuple3(shortPremiumFraction, shortPremiumFraction, premiumToVault)
822824 }
823825 }
824826 else {
825827 let longPremiumFraction = divd(muld(premium, fundingPeriodDecimal()), ONE_DAY)
826828 if ((fundingMode() == FUNDING_ASYMMETRIC))
827829 then {
828830 let shortPremiumFraction = divd(muld(longPremiumFraction, totalLongPositionSize()), totalShortPositionSize())
829831 $Tuple3(shortPremiumFraction, longPremiumFraction, 0)
830832 }
831833 else {
832834 let longTotalPremiumFraction = abs(muld(longPremiumFraction, totalLongPositionSize()))
833835 let shortTotalPremiumFraction = abs(muld(longPremiumFraction, totalShortPositionSize()))
834836 let premiumToVault = (longTotalPremiumFraction - shortTotalPremiumFraction)
835837 $Tuple3(longPremiumFraction, longPremiumFraction, premiumToVault)
836838 }
837839 }
838840 }
839841
840842
841843 func getAdjustedFee (_artifactId,_baseFeeDiscount) = {
842844 let baseFeeRaw = fee()
843845 let baseFee = muld(baseFeeRaw, _baseFeeDiscount)
844- let $t04210542600 = if ((_artifactId != ""))
846+ let $t04219442689 = if ((_artifactId != ""))
845847 then {
846848 let artifactKind = strA(nftManagerAddress(), toCompositeKey(k_token_type, _artifactId))
847849 if ((artifactKind == FEE_REDUCTION_TOKEN_TYPE))
848850 then {
849851 let reduction = intA(nftManagerAddress(), toCompositeKey(k_token_param, _artifactId))
850852 let adjustedFee = muld(baseFee, reduction)
851853 $Tuple2(adjustedFee, true)
852854 }
853855 else throw("Invalid attached artifact")
854856 }
855857 else $Tuple2(baseFee, false)
856- let adjustedFee = $t04210542600._1
857- let burnArtifact = $t04210542600._2
858+ let adjustedFee = $t04219442689._1
859+ let burnArtifact = $t04219442689._2
858860 $Tuple2(adjustedFee, burnArtifact)
859861 }
860862
861863
862864 func getForTraderWithArtifact (_trader,_artifactId) = {
863865 let doGetFeeDiscount = invoke(minerAddress(), "computeFeeDiscount", [_trader], nil)
864866 if ((doGetFeeDiscount == doGetFeeDiscount))
865867 then {
866868 let feeDiscount = match doGetFeeDiscount {
867869 case x: Int =>
868870 x
869871 case _ =>
870872 throw("Invalid computeFeeDiscount result")
871873 }
872- let $t04294643020 = getAdjustedFee(_artifactId, feeDiscount)
873- let adjustedFee = $t04294643020._1
874- let burnArtifact = $t04294643020._2
874+ let $t04303543109 = getAdjustedFee(_artifactId, feeDiscount)
875+ let adjustedFee = $t04303543109._1
876+ let burnArtifact = $t04303543109._2
875877 $Tuple2(adjustedFee, burnArtifact)
876878 }
877879 else throw("Strict value is not equal to itself.")
878880 }
879881
880882
881883 func getArtifactId (i) = {
882884 let artifactId = if ((size(i.payments) > 1))
883885 then toBase58String(valueOrErrorMessage(i.payments[1].assetId, "Invalid artifactId"))
884886 else ""
885887 artifactId
886888 }
887889
888890
889891 func distributeFee (_feeAmount) = {
890892 let feeToStakers = muld(_feeAmount, feeToStakersPercent())
891893 let feeToVault = (_feeAmount - feeToStakers)
892894 $Tuple2(feeToStakers, feeToVault)
893895 }
894896
895897
896898 func updateSettings (_initMarginRatio,_mmr,_liquidationFeeRatio,_fundingPeriod,_fee,_spreadLimit,_maxPriceImpact,_partialLiquidationRatio,_maxPriceSpread,_maxOpenNotional,_feeToStakersPercent,_maxOracleDelay,_rolloverFee,_fundingMode,_oracleMode) = [IntegerEntry(k_initMarginRatio, _initMarginRatio), IntegerEntry(k_maintenanceMarginRatio, _mmr), IntegerEntry(k_liquidationFeeRatio, _liquidationFeeRatio), IntegerEntry(k_fundingPeriod, _fundingPeriod), IntegerEntry(k_fee, _fee), IntegerEntry(k_spreadLimit, _spreadLimit), IntegerEntry(k_maxPriceImpact, _maxPriceImpact), IntegerEntry(k_partialLiquidationRatio, _partialLiquidationRatio), IntegerEntry(k_maxPriceSpread, _maxPriceSpread), IntegerEntry(k_maxOpenNotional, _maxOpenNotional), IntegerEntry(k_feeToStakersPercent, _feeToStakersPercent), IntegerEntry(k_maxOracleDelay, _maxOracleDelay), IntegerEntry(k_rolloverFee, _rolloverFee), IntegerEntry(k_fundingMode, _fundingMode), IntegerEntry(k_oracleMode, _oracleMode)]
897899
898900
899901 func updateFunding (_nextFundingBlock,_latestLongCumulativePremiumFraction,_latestShortCumulativePremiumFraction,_longFundingRate,_shortFundingRate) = [IntegerEntry(k_nextFundingBlock, _nextFundingBlock), IntegerEntry(k_latestLongCumulativePremiumFraction, _latestLongCumulativePremiumFraction), IntegerEntry(k_latestShortCumulativePremiumFraction, _latestShortCumulativePremiumFraction), IntegerEntry(k_longFundingRate, _longFundingRate), IntegerEntry(k_shortFundingRate, _shortFundingRate)]
900902
901903
902904 func incrementPositionSequenceNumber (_isNewPosition,_trader,_direction) = {
903905 let positionKey = ((_trader + "_") + toString(_direction))
904906 if (_isNewPosition)
905907 then {
906908 let currentSequence = lastSequence()
907909 [IntegerEntry(toCompositeKey(k_positionSequence, positionKey), (currentSequence + 1)), IntegerEntry(k_sequence, (currentSequence + 1))]
908910 }
909911 else nil
910912 }
911913
912914
913915 func updatePositionFee (_isNewPosition,_trader,_direction,_fee) = {
914916 let positionKey = ((_trader + "_") + toString(_direction))
915917 if (_isNewPosition)
916918 then [IntegerEntry(toCompositeKey(k_positionFee, positionKey), _fee)]
917919 else nil
918920 }
919921
920922
921923 func updatePosition (_trader,_size,_margin,_openNotional,_latestCumulativePremiumFraction,_latestTimestamp) = {
922924 let direction = getDirection(_size)
923925 let positionKey = ((_trader + "_") + toString(direction))
924926 [IntegerEntry(toCompositeKey(k_positionSize, positionKey), _size), IntegerEntry(toCompositeKey(k_positionMargin, positionKey), _margin), IntegerEntry(toCompositeKey(k_positionOpenNotional, positionKey), _openNotional), IntegerEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, positionKey), _latestCumulativePremiumFraction), IntegerEntry(toCompositeKey(k_positionLastUpdatedTimestamp, positionKey), _latestTimestamp)]
925927 }
926928
927929
928930 func updateAmmReserves (_qtAstR,_bsAstR) = if (if ((0 > _qtAstR))
929931 then true
930932 else (0 > _bsAstR))
931933 then throw("Invalid amount to update reserves")
932934 else [IntegerEntry(k_quoteAssetReserve, _qtAstR), IntegerEntry(k_baseAssetReserve, _bsAstR)]
933935
934936
935937 func updateAmmWeights (_qtAstW,_bsAstW) = [IntegerEntry(k_quoteAssetWeight, _qtAstW), IntegerEntry(k_baseAssetWeight, _bsAstW)]
936938
937939
938940 func updateAmm (_qtAstR,_bsAstR,_totalPositionSizeAfter,_openInterestNotional,_totalLongPositionSize,_totalShortPositionSize,_totalLongOpenNotional,_totalShortOpenNotional) = {
939941 let _qtAstW = qtAstW()
940942 let _bsAstW = bsAstW()
941943 if (((_totalLongPositionSize - _totalShortPositionSize) != _totalPositionSizeAfter))
942944 then throw(((((("Invalid AMM state data: " + toString(_totalLongPositionSize)) + " + ") + toString(_totalShortPositionSize)) + " != ") + toString(_totalPositionSizeAfter)))
943945 else (updateAmmReserves(_qtAstR, _bsAstR) ++ [IntegerEntry(k_totalPositionSize, _totalPositionSizeAfter), IntegerEntry(k_openInterestNotional, _openInterestNotional), IntegerEntry(k_totalLongPositionSize, _totalLongPositionSize), IntegerEntry(k_totalShortPositionSize, _totalShortPositionSize), IntegerEntry(k_openInterestLong, _totalLongOpenNotional), IntegerEntry(k_openInterestShort, _totalShortOpenNotional)])
944946 }
945947
946948
947949 func deletePosition (_trader,_direction) = {
948950 let positionKey = ((_trader + "_") + toString(_direction))
949951 [DeleteEntry(toCompositeKey(k_positionSize, positionKey)), DeleteEntry(toCompositeKey(k_positionMargin, positionKey)), DeleteEntry(toCompositeKey(k_positionOpenNotional, positionKey)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, positionKey)), DeleteEntry(toCompositeKey(k_positionFee, positionKey)), DeleteEntry(toCompositeKey(k_positionLastUpdatedTimestamp, positionKey))]
950952 }
951953
952954
953955 func withdraw (_address,_amount) = {
954956 let balance = assetBalance(this, quoteAsset())
955957 if ((_amount > balance))
956958 then throw(((("Unable to withdraw " + toString(_amount)) + " from contract balance ") + toString(balance)))
957959 else [ScriptTransfer(_address, _amount, quoteAsset())]
958960 }
959961
960962
961963 func updateBalance (i) = if ((0 > i))
962964 then throw("Balance")
963965 else [IntegerEntry(k_balance, i)]
964966
965967
966968 func transferFee (i) = [ScriptTransfer(stakingAddress(), i, quoteAsset())]
967969
968970
969971 func doBurnArtifact (_burnArtifact,i) = if (_burnArtifact)
970972 then [Burn(valueOrErrorMessage(i.payments[1].assetId, "Invalid artifact"), 1)]
971973 else nil
972974
973975
974976 @Callable(i)
975977 func pause () = if ((i.caller != adminAddress()))
976978 then throw("Invalid pause params")
977979 else [BooleanEntry(k_paused, true)]
978980
979981
980982
981983 @Callable(i)
982984 func unpause () = if ((i.caller != adminAddress()))
983985 then throw("Invalid unpause params")
984986 else [BooleanEntry(k_paused, false)]
985987
986988
987989
988990 @Callable(i)
989991 func setCloseOnly () = if ((i.caller != adminAddress()))
990992 then throw("Invalid setCloseOnly params")
991993 else [BooleanEntry(k_closeOnly, true)]
992994
993995
994996
995997 @Callable(i)
996998 func unsetCloseOnly () = if ((i.caller != adminAddress()))
997999 then throw("Invalid unsetCloseOnly params")
9981000 else [BooleanEntry(k_closeOnly, false)]
9991001
10001002
10011003
10021004 @Callable(i)
10031005 func changeLiquidity (_quoteAssetAmount) = if (if ((i.caller != adminAddress()))
10041006 then true
10051007 else (_quoteAssetAmount == 0))
10061008 then throw("Invalid changeLiquidity params")
10071009 else {
10081010 let _qtAstR = qtAstR()
10091011 let _bsAstR = bsAstR()
10101012 let _qtAstW = qtAstW()
10111013 let _bsAstW = bsAstW()
10121014 let price = divd(muld(_qtAstR, _qtAstW), muld(_bsAstR, _bsAstW))
10131015 let qtAstRAfter = (_qtAstR + _quoteAssetAmount)
10141016 let baseAssetAmountToAdd = (divd(muld(qtAstRAfter, _qtAstW), price) - _bsAstR)
10151017 let bsAstRAfter = (_bsAstR + baseAssetAmountToAdd)
1016- let $t05145451605 = getSyncTerminalPrice(getOraclePrice(), qtAstRAfter, bsAstRAfter)
1017- let newQuoteAssetWeight = $t05145451605._1
1018- let newBaseAssetWeight = $t05145451605._2
1019- let marginToVault = $t05145451605._3
1018+ let $t05154351694 = getSyncTerminalPrice(getOraclePrice(), qtAstRAfter, bsAstRAfter)
1019+ let newQuoteAssetWeight = $t05154351694._1
1020+ let newBaseAssetWeight = $t05154351694._2
1021+ let marginToVault = $t05154351694._3
10201022 let doExchangePnL = if ((marginToVault != 0))
10211023 then {
10221024 let doExchangePnL = invoke(vaultAddress(), "exchangeFreeAndLocked", [marginToVault], nil)
10231025 if ((doExchangePnL == doExchangePnL))
10241026 then nil
10251027 else throw("Strict value is not equal to itself.")
10261028 }
10271029 else nil
10281030 if ((doExchangePnL == doExchangePnL))
10291031 then (updateAmmReserves(qtAstRAfter, bsAstRAfter) ++ updateAmmWeights(newQuoteAssetWeight, newBaseAssetWeight))
10301032 else throw("Strict value is not equal to itself.")
10311033 }
10321034
10331035
10341036
10351037 @Callable(i)
10361038 func changeSettings (_initMarginRatio,_mmr,_liquidationFeeRatio,_fundingPeriod,_fee,_spreadLimit,_maxPriceImpact,_partialLiquidationRatio,_maxPriceSpread,_maxOpenNotional,_feeToStakersPercent,_maxOracleDelay,_rolloverFee,_fundingMode,_oracleMode) = if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if ((0 >= _fundingPeriod))
10371039 then true
10381040 else (0 >= _initMarginRatio))
10391041 then true
10401042 else (0 >= _mmr))
10411043 then true
10421044 else (0 >= _liquidationFeeRatio))
10431045 then true
10441046 else (0 >= _fee))
10451047 then true
10461048 else (0 >= _spreadLimit))
10471049 then true
10481050 else (0 >= _maxPriceImpact))
10491051 then true
10501052 else (0 >= _partialLiquidationRatio))
10511053 then true
10521054 else (0 >= _maxPriceSpread))
10531055 then true
10541056 else (0 >= _maxOpenNotional))
10551057 then true
10561058 else (0 >= _feeToStakersPercent))
10571059 then true
10581060 else (_feeToStakersPercent > DECIMAL_UNIT))
10591061 then true
10601062 else (0 >= _maxOracleDelay))
10611063 then true
10621064 else (0 >= _rolloverFee))
10631065 then true
10641066 else if ((_fundingMode != FUNDING_SYMMETRIC))
10651067 then (_fundingMode != FUNDING_ASYMMETRIC)
10661068 else false)
10671069 then true
10681070 else if ((_oracleMode != ORACLE_PLAIN))
10691071 then (_oracleMode != ORACLE_JIT)
10701072 else false)
10711073 then true
10721074 else !(initialized()))
10731075 then true
10741076 else (i.caller != adminAddress()))
10751077 then throw("Invalid changeSettings params")
10761078 else updateSettings(_initMarginRatio, _mmr, _liquidationFeeRatio, _fundingPeriod, _fee, _spreadLimit, _maxPriceImpact, _partialLiquidationRatio, _maxPriceSpread, _maxOpenNotional, _feeToStakersPercent, _maxOracleDelay, _rolloverFee, _fundingMode, _oracleMode)
10771079
10781080
10791081
10801082 @Callable(i)
10811083 func initialize (_qtAstR,_bsAstR,_fundingPeriod,_initMarginRatio,_mmr,_liquidationFeeRatio,_fee,_baseOracleData,_quoteOracleData,_coordinator,_spreadLimit,_maxPriceImpact,_partialLiquidationRatio,_maxPriceSpread,_maxOpenNotional,_feeToStakersPercent,_maxOracleDelay,_rolloverFee,_fundingMode,_oracleMode) = if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if (if ((0 >= _qtAstR))
10821084 then true
10831085 else (0 >= _bsAstR))
10841086 then true
10851087 else (0 >= _fundingPeriod))
10861088 then true
10871089 else (0 >= _initMarginRatio))
10881090 then true
10891091 else (0 >= _mmr))
10901092 then true
10911093 else (0 >= _liquidationFeeRatio))
10921094 then true
10931095 else (0 >= _fee))
10941096 then true
10951097 else (0 >= _spreadLimit))
10961098 then true
10971099 else (0 >= _maxPriceImpact))
10981100 then true
10991101 else (0 >= _partialLiquidationRatio))
11001102 then true
11011103 else (0 >= _maxPriceSpread))
11021104 then true
11031105 else (0 >= _maxOpenNotional))
11041106 then true
11051107 else (0 >= _feeToStakersPercent))
11061108 then true
11071109 else (_feeToStakersPercent > DECIMAL_UNIT))
11081110 then true
11091111 else (0 > _maxOracleDelay))
11101112 then true
11111113 else (0 >= _rolloverFee))
11121114 then true
11131115 else if ((_fundingMode != FUNDING_SYMMETRIC))
11141116 then (_fundingMode != FUNDING_ASYMMETRIC)
11151117 else false)
11161118 then true
11171119 else if ((_oracleMode != ORACLE_PLAIN))
11181120 then (_oracleMode != ORACLE_JIT)
11191121 else false)
11201122 then true
11211123 else initialized())
11221124 then true
11231125 else (i.caller != this))
11241126 then throw("Invalid initialize parameters")
11251127 else ((((updateAmm(_qtAstR, _bsAstR, 0, 0, 0, 0, 0, 0) ++ updateSettings(_initMarginRatio, _mmr, _liquidationFeeRatio, _fundingPeriod, _fee, _spreadLimit, _maxPriceImpact, _partialLiquidationRatio, _maxPriceSpread, _maxOpenNotional, _feeToStakersPercent, _maxOracleDelay, _rolloverFee, _fundingMode, _oracleMode)) ++ updateFunding((lastTimestamp() + _fundingPeriod), 0, 0, 0, 0)) ++ updateBalance(0)) ++ [BooleanEntry(k_initialized, true), StringEntry(k_baseOracle, _baseOracleData), StringEntry(k_quoteOracle, _quoteOracleData), StringEntry(k_coordinatorAddress, toString(addressFromStringValue(_coordinator)))])
11261128
11271129
11281130
11291131 @Callable(i)
11301132 func increasePosition (_direction,_leverage,_minBaseAssetAmount,_refLink,_priceUpdate) = {
11311133 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
11321134 if ((updateOracle == updateOracle))
11331135 then {
11341136 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
11351137 if ((sync == sync))
11361138 then {
11371139 let ensureCalledOnce = invoke(this, "ensureCalledOnce", nil, nil)
11381140 if ((ensureCalledOnce == ensureCalledOnce))
11391141 then {
11401142 let _trader = getActualCaller(i)
11411143 let _rawAmount = i.payments[0].amount
11421144 let _assetId = i.payments[0].assetId
11431145 let _assetIdStr = toBase58String(value(_assetId))
11441146 let isQuoteAsset = (_assetId == quoteAsset())
11451147 if (if (if (if (if (if (if (if (if ((_direction != DIR_LONG))
11461148 then (_direction != DIR_SHORT)
11471149 else false)
11481150 then true
11491151 else (0 >= _rawAmount))
11501152 then true
11511153 else !(initialized()))
11521154 then true
11531155 else !(isQuoteAsset))
11541156 then true
11551157 else !(requireMoreMarginRatio(divd(DECIMAL_UNIT, _leverage), initMarginRatio(), true)))
11561158 then true
11571159 else paused())
11581160 then true
11591161 else closeOnly())
11601162 then true
11611163 else isMarketClosed())
11621164 then throw("Invalid increasePosition parameters")
11631165 else {
1164- let $t05798158130 = getForTraderWithArtifact(_trader, getArtifactId(i))
1165- let adjustedFee = $t05798158130._1
1166- let burnArtifact = $t05798158130._2
1166+ let $t05807058219 = getForTraderWithArtifact(_trader, getArtifactId(i))
1167+ let adjustedFee = $t05807058219._1
1168+ let burnArtifact = $t05807058219._2
11671169 let _amount = divd(_rawAmount, (muld(adjustedFee, _leverage) + DECIMAL_UNIT))
11681170 let distributeFeeAmount = (_rawAmount - _amount)
11691171 let referrerFeeAny = invoke(referralAddress(), "acceptPaymentWithLink", [_trader, _refLink], [AttachedPayment(quoteAsset(), distributeFeeAmount)])
11701172 if ((referrerFeeAny == referrerFeeAny))
11711173 then {
11721174 let referrerFee = match referrerFeeAny {
11731175 case x: Int =>
11741176 x
11751177 case _ =>
11761178 throw("Invalid referrerFee")
11771179 }
11781180 let feeAmount = (distributeFeeAmount - referrerFee)
1179- let $t05862658806 = getPosition(_trader, _direction)
1180- let oldPositionSize = $t05862658806._1
1181- let oldPositionMargin = $t05862658806._2
1182- let oldPositionOpenNotional = $t05862658806._3
1183- let oldPositionLstUpdCPF = $t05862658806._4
1184- let oldPositionTimestamp = $t05862658806._5
1181+ let $t05871558895 = getPosition(_trader, _direction)
1182+ let oldPositionSize = $t05871558895._1
1183+ let oldPositionMargin = $t05871558895._2
1184+ let oldPositionOpenNotional = $t05871558895._3
1185+ let oldPositionLstUpdCPF = $t05871558895._4
1186+ let oldPositionTimestamp = $t05871558895._5
11851187 let isNewPosition = (oldPositionSize == 0)
11861188 let isSameDirection = if ((oldPositionSize > 0))
11871189 then (_direction == DIR_LONG)
11881190 else (_direction == DIR_SHORT)
11891191 let expandExisting = if (!(isNewPosition))
11901192 then isSameDirection
11911193 else false
11921194 let isAdd = (_direction == DIR_LONG)
1193- let $t05909562228 = if (if (isNewPosition)
1195+ let $t05918462317 = if (if (isNewPosition)
11941196 then true
11951197 else expandExisting)
11961198 then {
11971199 let openNotional = muld(_amount, _leverage)
1198- let $t05960459777 = swapInput(isAdd, openNotional)
1199- let amountBaseAssetBought = $t05960459777._1
1200- let quoteAssetReserveAfter = $t05960459777._2
1201- let baseAssetReserveAfter = $t05960459777._3
1202- let totalPositionSizeAfter = $t05960459777._4
1200+ let $t05969359866 = swapInput(isAdd, openNotional)
1201+ let amountBaseAssetBought = $t05969359866._1
1202+ let quoteAssetReserveAfter = $t05969359866._2
1203+ let baseAssetReserveAfter = $t05969359866._3
1204+ let totalPositionSizeAfter = $t05969359866._4
12031205 if (if ((_minBaseAssetAmount != 0))
12041206 then (_minBaseAssetAmount > abs(amountBaseAssetBought))
12051207 else false)
12061208 then throw(((("Limit error: " + toString(abs(amountBaseAssetBought))) + " < ") + toString(_minBaseAssetAmount)))
12071209 else {
12081210 let newPositionSize = (oldPositionSize + amountBaseAssetBought)
12091211 let totalLongOpenInterestAfter = (openInterestLong() + (if ((newPositionSize > 0))
12101212 then openNotional
12111213 else 0))
12121214 let totalShortOpenInterestAfter = (openInterestShort() + (if ((0 > newPositionSize))
12131215 then openNotional
12141216 else 0))
1215- let $t06032360598 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, _amount)
1216- let remainMargin = $t06032360598._1
1217- let x1 = $t06032360598._2
1218- let x2 = $t06032360598._3
1219- let rolloverFee = $t06032360598._4
1217+ let $t06041260687 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, _amount)
1218+ let remainMargin = $t06041260687._1
1219+ let x1 = $t06041260687._2
1220+ let x2 = $t06041260687._3
1221+ let rolloverFee = $t06041260687._4
12201222 if (!(requireNotOverSpreadLimit(quoteAssetReserveAfter, baseAssetReserveAfter)))
12211223 then throw("Over max spread limit")
12221224 else if (!(requireNotOverMaxOpenNotional(totalLongOpenInterestAfter, totalShortOpenInterestAfter)))
12231225 then throw("Over max open notional")
12241226 else $Tuple14(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), lastTimestamp(), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
12251227 then abs(amountBaseAssetBought)
12261228 else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
12271229 then abs(amountBaseAssetBought)
12281230 else 0)), totalLongOpenInterestAfter, totalShortOpenInterestAfter, rolloverFee)
12291231 }
12301232 }
12311233 else {
12321234 let openNotional = muld(_amount, _leverage)
1233- let $t06191662044 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), _direction, PNL_OPTION_SPOT)
1234- let oldPositionNotional = $t06191662044._1
1235- let unrealizedPnl = $t06191662044._2
1235+ let $t06200562133 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), _direction, PNL_OPTION_SPOT)
1236+ let oldPositionNotional = $t06200562133._1
1237+ let unrealizedPnl = $t06200562133._2
12361238 if ((oldPositionNotional > openNotional))
12371239 then throw("Use decreasePosition to decrease position size")
12381240 else throw("Close position first")
12391241 }
1240- let newPositionSize = $t05909562228._1
1241- let newPositionRemainMargin = $t05909562228._2
1242- let newPositionOpenNotional = $t05909562228._3
1243- let newPositionLatestCPF = $t05909562228._4
1244- let newPositionTimestamp = $t05909562228._5
1245- let baseAssetReserveAfter = $t05909562228._6
1246- let quoteAssetReserveAfter = $t05909562228._7
1247- let totalPositionSizeAfter = $t05909562228._8
1248- let openInterestNotionalAfter = $t05909562228._9
1249- let totalLongAfter = $t05909562228._10
1250- let totalShortAfter = $t05909562228._11
1251- let totalLongOpenInterestAfter = $t05909562228._12
1252- let totalShortOpenInterestAfter = $t05909562228._13
1253- let rolloverFee = $t05909562228._14
1254- let $t06223462305 = distributeFee((feeAmount + rolloverFee))
1255- let feeToStakers = $t06223462305._1
1256- let feeToVault = $t06223462305._2
1242+ let newPositionSize = $t05918462317._1
1243+ let newPositionRemainMargin = $t05918462317._2
1244+ let newPositionOpenNotional = $t05918462317._3
1245+ let newPositionLatestCPF = $t05918462317._4
1246+ let newPositionTimestamp = $t05918462317._5
1247+ let baseAssetReserveAfter = $t05918462317._6
1248+ let quoteAssetReserveAfter = $t05918462317._7
1249+ let totalPositionSizeAfter = $t05918462317._8
1250+ let openInterestNotionalAfter = $t05918462317._9
1251+ let totalLongAfter = $t05918462317._10
1252+ let totalShortAfter = $t05918462317._11
1253+ let totalLongOpenInterestAfter = $t05918462317._12
1254+ let totalShortOpenInterestAfter = $t05918462317._13
1255+ let rolloverFee = $t05918462317._14
1256+ let $t06232362394 = distributeFee((feeAmount + rolloverFee))
1257+ let feeToStakers = $t06232362394._1
1258+ let feeToVault = $t06232362394._2
12571259 let stake = if ((_amount >= rolloverFee))
12581260 then invoke(vaultAddress(), "addLocked", nil, [AttachedPayment(quoteAsset(), (_amount - rolloverFee))])
12591261 else invoke(vaultAddress(), "withdrawLocked", [(rolloverFee - _amount)], nil)
12601262 if ((stake == stake))
12611263 then {
12621264 let depositVault = invoke(vaultAddress(), "addFree", nil, [AttachedPayment(quoteAsset(), feeToVault)])
12631265 if ((depositVault == depositVault))
12641266 then {
12651267 let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
12661268 if ((notifyFee == notifyFee))
12671269 then {
12681270 let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
12691271 if ((notifyNotional == notifyNotional))
12701272 then ((((((updatePosition(_trader, newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF, newPositionTimestamp) ++ incrementPositionSequenceNumber(isNewPosition, _trader, _direction)) ++ updatePositionFee(isNewPosition, _trader, _direction, adjustedFee)) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter, totalLongOpenInterestAfter, totalShortOpenInterestAfter)) ++ transferFee(feeToStakers)) ++ updateBalance(((cbalance() + _amount) - rolloverFee))) ++ doBurnArtifact(burnArtifact, i))
12711273 else throw("Strict value is not equal to itself.")
12721274 }
12731275 else throw("Strict value is not equal to itself.")
12741276 }
12751277 else throw("Strict value is not equal to itself.")
12761278 }
12771279 else throw("Strict value is not equal to itself.")
12781280 }
12791281 else throw("Strict value is not equal to itself.")
12801282 }
12811283 }
12821284 else throw("Strict value is not equal to itself.")
12831285 }
12841286 else throw("Strict value is not equal to itself.")
12851287 }
12861288 else throw("Strict value is not equal to itself.")
12871289 }
12881290
12891291
12901292
12911293 @Callable(i)
12921294 func addMargin (_direction,_priceUpdate) = {
12931295 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
12941296 if ((updateOracle == updateOracle))
12951297 then {
12961298 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
12971299 if ((sync == sync))
12981300 then {
12991301 let ensureCalledOnce = invoke(this, "ensureCalledOnce", nil, nil)
13001302 if ((ensureCalledOnce == ensureCalledOnce))
13011303 then {
13021304 let _trader = toString(i.caller)
13031305 let _amount = i.payments[0].amount
13041306 let _assetId = i.payments[0].assetId
13051307 let _assetIdStr = toBase58String(value(_assetId))
13061308 let isQuoteAsset = (_assetId == quoteAsset())
13071309 if (if (if (if (if (!(isQuoteAsset))
13081310 then true
13091311 else !(requireOpenPosition(toString(i.caller), _direction)))
13101312 then true
13111313 else !(initialized()))
13121314 then true
13131315 else paused())
13141316 then true
13151317 else isMarketClosed())
13161318 then throw("Invalid addMargin parameters")
13171319 else {
1318- let $t06464764827 = getPosition(_trader, _direction)
1319- let oldPositionSize = $t06464764827._1
1320- let oldPositionMargin = $t06464764827._2
1321- let oldPositionOpenNotional = $t06464764827._3
1322- let oldPositionLstUpdCPF = $t06464764827._4
1323- let oldPositionTimestamp = $t06464764827._5
1320+ let $t06473664916 = getPosition(_trader, _direction)
1321+ let oldPositionSize = $t06473664916._1
1322+ let oldPositionMargin = $t06473664916._2
1323+ let oldPositionOpenNotional = $t06473664916._3
1324+ let oldPositionLstUpdCPF = $t06473664916._4
1325+ let oldPositionTimestamp = $t06473664916._5
13241326 let stake = invoke(vaultAddress(), "addLocked", nil, [AttachedPayment(quoteAsset(), _amount)])
13251327 if ((stake == stake))
13261328 then {
13271329 let rolloverFee = calcRolloverFee(oldPositionMargin, oldPositionTimestamp)
13281330 let doTransferFeeToStakers = if ((rolloverFee > 0))
13291331 then {
1330- let $t06511265171 = distributeFee(rolloverFee)
1331- let feeToStakers = $t06511265171._1
1332- let feeToVault = $t06511265171._2
1332+ let $t06520165260 = distributeFee(rolloverFee)
1333+ let feeToStakers = $t06520165260._1
1334+ let feeToVault = $t06520165260._2
13331335 let unstake = invoke(vaultAddress(), "withdrawLocked", [feeToStakers], nil)
13341336 if ((unstake == unstake))
13351337 then {
13361338 let lockBadDebt = invoke(vaultAddress(), "exchangeFreeAndLocked", [-(feeToVault)], nil)
13371339 if ((lockBadDebt == lockBadDebt))
13381340 then transferFee(feeToStakers)
13391341 else throw("Strict value is not equal to itself.")
13401342 }
13411343 else throw("Strict value is not equal to itself.")
13421344 }
13431345 else nil
13441346 if ((doTransferFeeToStakers == doTransferFeeToStakers))
13451347 then ((updatePosition(_trader, oldPositionSize, ((oldPositionMargin - rolloverFee) + _amount), oldPositionOpenNotional, oldPositionLstUpdCPF, lastTimestamp()) ++ updateBalance(((cbalance() + _amount) - rolloverFee))) ++ doTransferFeeToStakers)
13461348 else throw("Strict value is not equal to itself.")
13471349 }
13481350 else throw("Strict value is not equal to itself.")
13491351 }
13501352 }
13511353 else throw("Strict value is not equal to itself.")
13521354 }
13531355 else throw("Strict value is not equal to itself.")
13541356 }
13551357 else throw("Strict value is not equal to itself.")
13561358 }
13571359
13581360
13591361
13601362 @Callable(i)
13611363 func removeMargin (_amount,_direction,_priceUpdate) = {
13621364 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
13631365 if ((updateOracle == updateOracle))
13641366 then {
13651367 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
13661368 if ((sync == sync))
13671369 then {
13681370 let ensureCalledOnce = invoke(this, "ensureCalledOnce", nil, nil)
13691371 if ((ensureCalledOnce == ensureCalledOnce))
13701372 then {
13711373 let _trader = toString(i.caller)
13721374 if (if (if (if (if ((0 >= _amount))
13731375 then true
13741376 else !(requireOpenPosition(_trader, _direction)))
13751377 then true
13761378 else !(initialized()))
13771379 then true
13781380 else paused())
13791381 then true
13801382 else isMarketClosed())
13811383 then throw("Invalid removeMargin parameters")
13821384 else {
1383- let $t06667266852 = getPosition(_trader, _direction)
1384- let oldPositionSize = $t06667266852._1
1385- let oldPositionMargin = $t06667266852._2
1386- let oldPositionOpenNotional = $t06667266852._3
1387- let oldPositionLstUpdCPF = $t06667266852._4
1388- let oldPositionTimestamp = $t06667266852._5
1389- let $t06685867107 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, -(_amount))
1390- let remainMargin = $t06685867107._1
1391- let badDebt = $t06685867107._2
1392- let fundingPayment = $t06685867107._3
1393- let rolloverFee = $t06685867107._4
1385+ let $t06676166941 = getPosition(_trader, _direction)
1386+ let oldPositionSize = $t06676166941._1
1387+ let oldPositionMargin = $t06676166941._2
1388+ let oldPositionOpenNotional = $t06676166941._3
1389+ let oldPositionLstUpdCPF = $t06676166941._4
1390+ let oldPositionTimestamp = $t06676166941._5
1391+ let $t06694767196 = calcRemainMarginWithFundingPaymentAndRolloverFee(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, oldPositionTimestamp, -(_amount))
1392+ let remainMargin = $t06694767196._1
1393+ let badDebt = $t06694767196._2
1394+ let fundingPayment = $t06694767196._3
1395+ let rolloverFee = $t06694767196._4
13941396 if ((badDebt != 0))
13951397 then throw("Invalid removed margin amount")
13961398 else {
13971399 let marginRatio = calcMarginRatio(remainMargin, badDebt, oldPositionOpenNotional)
13981400 if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
13991401 then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
14001402 else {
1401- let $t06749367552 = distributeFee(rolloverFee)
1402- let feeToStakers = $t06749367552._1
1403- let feeToVault = $t06749367552._2
1403+ let $t06758267641 = distributeFee(rolloverFee)
1404+ let feeToStakers = $t06758267641._1
1405+ let feeToVault = $t06758267641._2
14041406 let doTransferFeeToStakers = if ((rolloverFee > 0))
14051407 then {
14061408 let lockBadDebt = invoke(vaultAddress(), "exchangeFreeAndLocked", [-(feeToVault)], nil)
14071409 if ((lockBadDebt == lockBadDebt))
14081410 then transferFee(feeToStakers)
14091411 else throw("Strict value is not equal to itself.")
14101412 }
14111413 else nil
14121414 if ((doTransferFeeToStakers == doTransferFeeToStakers))
14131415 then {
14141416 let unstake = invoke(vaultAddress(), "withdrawLocked", [(_amount + feeToStakers)], nil)
14151417 if ((unstake == unstake))
14161418 then (((updatePosition(_trader, oldPositionSize, remainMargin, oldPositionOpenNotional, latestCumulativePremiumFraction(oldPositionSize), lastTimestamp()) ++ withdraw(i.caller, _amount)) ++ updateBalance(((cbalance() - _amount) - rolloverFee))) ++ doTransferFeeToStakers)
14171419 else throw("Strict value is not equal to itself.")
14181420 }
14191421 else throw("Strict value is not equal to itself.")
14201422 }
14211423 }
14221424 }
14231425 }
14241426 else throw("Strict value is not equal to itself.")
14251427 }
14261428 else throw("Strict value is not equal to itself.")
14271429 }
14281430 else throw("Strict value is not equal to itself.")
14291431 }
14301432
14311433
14321434
14331435 @Callable(i)
14341436 func closePosition (_size,_direction,_minQuoteAssetAmount,_addToMargin,_priceUpdate) = {
14351437 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
14361438 if ((updateOracle == updateOracle))
14371439 then {
14381440 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
14391441 if ((sync == sync))
14401442 then {
14411443 let ensureCalledOnce = invoke(this, "ensureCalledOnce", nil, nil)
14421444 if ((ensureCalledOnce == ensureCalledOnce))
14431445 then {
14441446 let _trader = getActualCaller(i)
14451447 let _traderAddress = valueOrErrorMessage(addressFromString(_trader), "Invalid caller")
14461448 let positionFee = getPositionFee(_trader, _direction)
14471449 if (if (if (if (if (if (!(requireOpenPosition(_trader, _direction)))
14481450 then true
14491451 else !(initialized()))
14501452 then true
14511453 else paused())
14521454 then true
14531455 else (0 >= _size))
14541456 then true
14551457 else (0 > _minQuoteAssetAmount))
14561458 then true
14571459 else isMarketClosed())
14581460 then throw("Invalid closePosition parameters")
14591461 else {
14601462 let oldPositionTimestamp = getPosition(_trader, _direction)._5
1461- let $t06988770490 = internalClosePosition(_trader, _direction, _size, positionFee, _minQuoteAssetAmount, _addToMargin, true, true)
1462- let newPositionSize = $t06988770490._1
1463- let newPositionMargin = $t06988770490._2
1464- let newPositionOpenNotional = $t06988770490._3
1465- let newPositionLstUpdCPF = $t06988770490._4
1466- let positionBadDebt = $t06988770490._5
1467- let realizedPnl = $t06988770490._6
1468- let marginToTrader = $t06988770490._7
1469- let quoteAssetReserveAfter = $t06988770490._8
1470- let baseAssetReserveAfter = $t06988770490._9
1471- let totalPositionSizeAfter = $t06988770490._10
1472- let openInterestNotionalAfter = $t06988770490._11
1473- let totalLongAfter = $t06988770490._12
1474- let totalShortAfter = $t06988770490._13
1475- let totalLongOpenInterestAfter = $t06988770490._14
1476- let totalShortOpenInterestAfter = $t06988770490._15
1477- let realizedFee = $t06988770490._16
1463+ let $t06997670579 = internalClosePosition(_trader, _direction, _size, positionFee, _minQuoteAssetAmount, _addToMargin, true, true)
1464+ let newPositionSize = $t06997670579._1
1465+ let newPositionMargin = $t06997670579._2
1466+ let newPositionOpenNotional = $t06997670579._3
1467+ let newPositionLstUpdCPF = $t06997670579._4
1468+ let positionBadDebt = $t06997670579._5
1469+ let realizedPnl = $t06997670579._6
1470+ let marginToTrader = $t06997670579._7
1471+ let quoteAssetReserveAfter = $t06997670579._8
1472+ let baseAssetReserveAfter = $t06997670579._9
1473+ let totalPositionSizeAfter = $t06997670579._10
1474+ let openInterestNotionalAfter = $t06997670579._11
1475+ let totalLongAfter = $t06997670579._12
1476+ let totalShortAfter = $t06997670579._13
1477+ let totalLongOpenInterestAfter = $t06997670579._14
1478+ let totalShortOpenInterestAfter = $t06997670579._15
1479+ let realizedFee = $t06997670579._16
14781480 if ((positionBadDebt > 0))
14791481 then throw("Invalid closePosition parameters: bad debt")
14801482 else if ((oldPositionTimestamp >= lastTimestamp()))
14811483 then throw("Invalid closePosition parameters: wait at least 1 block before closing the position")
14821484 else {
14831485 let isPartialClose = (newPositionSize != 0)
14841486 let withdrawAmount = (marginToTrader + realizedFee)
14851487 let ammBalance = (cbalance() - withdrawAmount)
14861488 let ammNewBalance = if ((0 > ammBalance))
14871489 then 0
14881490 else ammBalance
14891491 let unstake = invoke(vaultAddress(), "withdrawLocked", [withdrawAmount], nil)
14901492 if ((unstake == unstake))
14911493 then {
14921494 let referrerFeeAny = invoke(referralAddress(), "acceptPayment", [_trader], [AttachedPayment(quoteAsset(), realizedFee)])
14931495 if ((referrerFeeAny == referrerFeeAny))
14941496 then {
14951497 let referrerFee = match referrerFeeAny {
14961498 case x: Int =>
14971499 x
14981500 case _ =>
14991501 throw("Invalid referrerFee")
15001502 }
1501- let $t07146271535 = distributeFee((realizedFee - referrerFee))
1502- let feeToStakers = $t07146271535._1
1503- let feeToVault = $t07146271535._2
1503+ let $t07155171624 = distributeFee((realizedFee - referrerFee))
1504+ let feeToStakers = $t07155171624._1
1505+ let feeToVault = $t07155171624._2
15041506 let depositVault = invoke(vaultAddress(), "addFree", nil, [AttachedPayment(quoteAsset(), feeToVault)])
15051507 if ((depositVault == depositVault))
15061508 then {
15071509 let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, realizedFee], nil)
15081510 if ((notifyFee == notifyFee))
15091511 then {
15101512 let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
15111513 if ((notifyNotional == notifyNotional))
15121514 then (((((if (isPartialClose)
15131515 then updatePosition(_trader, newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF, lastTimestamp())
15141516 else deletePosition(_trader, _direction)) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter, totalLongOpenInterestAfter, totalShortOpenInterestAfter)) ++ (if ((marginToTrader > 0))
15151517 then withdraw(_traderAddress, marginToTrader)
15161518 else nil)) ++ updateBalance(ammNewBalance)) ++ transferFee(feeToStakers))
15171519 else throw("Strict value is not equal to itself.")
15181520 }
15191521 else throw("Strict value is not equal to itself.")
15201522 }
15211523 else throw("Strict value is not equal to itself.")
15221524 }
15231525 else throw("Strict value is not equal to itself.")
15241526 }
15251527 else throw("Strict value is not equal to itself.")
15261528 }
15271529 }
15281530 }
15291531 else throw("Strict value is not equal to itself.")
15301532 }
15311533 else throw("Strict value is not equal to itself.")
15321534 }
15331535 else throw("Strict value is not equal to itself.")
15341536 }
15351537
15361538
15371539
15381540 @Callable(i)
15391541 func liquidate (_trader,_direction,_priceUpdate) = {
15401542 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
15411543 if ((updateOracle == updateOracle))
15421544 then {
15431545 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
15441546 if ((sync == sync))
15451547 then {
15461548 let spotMarginRatio = getMarginRatioByOption(_trader, _direction, PNL_OPTION_SPOT)
15471549 let liquidationMarginRatio = if (isOverFluctuationLimit())
15481550 then {
15491551 let oracleMarginRatio = getMarginRatioByOption(_trader, _direction, PNL_OPTION_ORACLE)
15501552 vmax(spotMarginRatio, oracleMarginRatio)
15511553 }
15521554 else spotMarginRatio
15531555 if (if (if (if (if (!(requireMoreMarginRatio(liquidationMarginRatio, maintenanceMarginRatio(), false)))
15541556 then true
15551557 else !(requireOpenPosition(_trader, _direction)))
15561558 then true
15571559 else !(initialized()))
15581560 then true
15591561 else paused())
15601562 then true
15611563 else isMarketClosed())
15621564 then throw("Unable to liquidate")
15631565 else {
15641566 let isPartialLiquidation = if (if ((spotMarginRatio > liquidationFeeRatio()))
15651567 then (partialLiquidationRatio() > 0)
15661568 else false)
15671569 then (DECIMAL_UNIT > partialLiquidationRatio())
15681570 else false
15691571 let oldPositionSize = getPosition(_trader, _direction)._1
15701572 let positionSizeAbs = abs(oldPositionSize)
1571- let $t07412874451 = if (isPartialLiquidation)
1573+ let $t07421774540 = if (isPartialLiquidation)
15721574 then {
15731575 let liquidationSize = getPartialLiquidationAmount(_trader, oldPositionSize)
15741576 let liquidationRatio = divd(abs(liquidationSize), positionSizeAbs)
15751577 $Tuple2(liquidationRatio, abs(liquidationSize))
15761578 }
15771579 else $Tuple2(0, positionSizeAbs)
1578- let liquidationRatio = $t07412874451._1
1579- let liquidationSize = $t07412874451._2
1580- let $t07445775113 = internalClosePosition(_trader, _direction, if (isPartialLiquidation)
1580+ let liquidationRatio = $t07421774540._1
1581+ let liquidationSize = $t07421774540._2
1582+ let $t07454675202 = internalClosePosition(_trader, _direction, if (isPartialLiquidation)
15811583 then liquidationSize
15821584 else positionSizeAbs, liquidationFeeRatio(), 0, true, false, true)
1583- let newPositionSize = $t07445775113._1
1584- let newPositionMargin = $t07445775113._2
1585- let newPositionOpenNotional = $t07445775113._3
1586- let newPositionLstUpdCPF = $t07445775113._4
1587- let positionBadDebt = $t07445775113._5
1588- let realizedPnl = $t07445775113._6
1589- let marginToTrader = $t07445775113._7
1590- let quoteAssetReserveAfter = $t07445775113._8
1591- let baseAssetReserveAfter = $t07445775113._9
1592- let totalPositionSizeAfter = $t07445775113._10
1593- let openInterestNotionalAfter = $t07445775113._11
1594- let totalLongAfter = $t07445775113._12
1595- let totalShortAfter = $t07445775113._13
1596- let totalLongOpenInterestAfter = $t07445775113._14
1597- let totalShortOpenInterestAfter = $t07445775113._15
1598- let liquidationPenalty = $t07445775113._16
1585+ let newPositionSize = $t07454675202._1
1586+ let newPositionMargin = $t07454675202._2
1587+ let newPositionOpenNotional = $t07454675202._3
1588+ let newPositionLstUpdCPF = $t07454675202._4
1589+ let positionBadDebt = $t07454675202._5
1590+ let realizedPnl = $t07454675202._6
1591+ let marginToTrader = $t07454675202._7
1592+ let quoteAssetReserveAfter = $t07454675202._8
1593+ let baseAssetReserveAfter = $t07454675202._9
1594+ let totalPositionSizeAfter = $t07454675202._10
1595+ let openInterestNotionalAfter = $t07454675202._11
1596+ let totalLongAfter = $t07454675202._12
1597+ let totalShortAfter = $t07454675202._13
1598+ let totalLongOpenInterestAfter = $t07454675202._14
1599+ let totalShortOpenInterestAfter = $t07454675202._15
1600+ let liquidationPenalty = $t07454675202._16
15991601 let feeToLiquidator = (liquidationPenalty / 2)
16001602 let feeToVault = (liquidationPenalty - feeToLiquidator)
16011603 let ammBalance = (cbalance() - liquidationPenalty)
16021604 let newAmmBalance = if ((0 > ammBalance))
16031605 then 0
16041606 else ammBalance
16051607 let lockBadDebt = if ((positionBadDebt > 0))
16061608 then {
16071609 let lockBadDebt = invoke(vaultAddress(), "exchangeFreeAndLocked", [(positionBadDebt + liquidationPenalty)], nil)
16081610 if ((lockBadDebt == lockBadDebt))
16091611 then nil
16101612 else throw("Strict value is not equal to itself.")
16111613 }
16121614 else nil
16131615 if ((lockBadDebt == lockBadDebt))
16141616 then {
16151617 let unstake = invoke(vaultAddress(), "withdrawLocked", [liquidationPenalty], nil)
16161618 if ((unstake == unstake))
16171619 then {
16181620 let depositInsurance = invoke(vaultAddress(), "addFree", nil, [AttachedPayment(quoteAsset(), feeToVault)])
16191621 if ((depositInsurance == depositInsurance))
16201622 then {
16211623 let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
16221624 if ((notifyNotional == notifyNotional))
16231625 then ((((if (isPartialLiquidation)
16241626 then updatePosition(_trader, newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF, lastTimestamp())
16251627 else deletePosition(_trader, _direction)) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter, totalLongOpenInterestAfter, totalShortOpenInterestAfter)) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
16261628 else throw("Strict value is not equal to itself.")
16271629 }
16281630 else throw("Strict value is not equal to itself.")
16291631 }
16301632 else throw("Strict value is not equal to itself.")
16311633 }
16321634 else throw("Strict value is not equal to itself.")
16331635 }
16341636 }
16351637 else throw("Strict value is not equal to itself.")
16361638 }
16371639 else throw("Strict value is not equal to itself.")
16381640 }
16391641
16401642
16411643
16421644 @Callable(i)
16431645 func payFunding (_priceUpdate) = {
16441646 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
16451647 if ((updateOracle == updateOracle))
16461648 then {
16471649 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
16481650 if ((sync == sync))
16491651 then {
16501652 let fundingBlockTimestamp = nextFundingBlockTimestamp()
16511653 if (if (if ((fundingBlockTimestamp > lastTimestamp()))
16521654 then true
16531655 else !(initialized()))
16541656 then true
16551657 else paused())
16561658 then throw(((("Invalid funding block timestamp: " + toString(lastTimestamp())) + " < ") + toString(fundingBlockTimestamp)))
16571659 else {
16581660 let underlyingPrice = getOraclePrice()
1659- let $t07733177409 = getFunding()
1660- let shortPremiumFraction = $t07733177409._1
1661- let longPremiumFraction = $t07733177409._2
1662- let premiumToVault = $t07733177409._3
1661+ let $t07742077498 = getFunding()
1662+ let shortPremiumFraction = $t07742077498._1
1663+ let longPremiumFraction = $t07742077498._2
1664+ let premiumToVault = $t07742077498._3
16631665 let doPayFundingToVault = if ((premiumToVault > 0))
16641666 then {
16651667 let doPayFundingToVault = invoke(vaultAddress(), "exchangeFreeAndLocked", [-(premiumToVault)], nil)
16661668 if ((doPayFundingToVault == doPayFundingToVault))
16671669 then nil
16681670 else throw("Strict value is not equal to itself.")
16691671 }
16701672 else nil
16711673 if ((doPayFundingToVault == doPayFundingToVault))
16721674 then updateFunding((fundingBlockTimestamp + fundingPeriodSeconds()), (latestLongCumulativePremiumFraction() + longPremiumFraction), (latestShortCumulativePremiumFraction() + shortPremiumFraction), divd(longPremiumFraction, underlyingPrice), divd(shortPremiumFraction, underlyingPrice))
16731675 else throw("Strict value is not equal to itself.")
16741676 }
16751677 }
16761678 else throw("Strict value is not equal to itself.")
16771679 }
16781680 else throw("Strict value is not equal to itself.")
16791681 }
16801682
16811683
16821684
16831685 @Callable(i)
16841686 func updateOracle (_priceUpdate) = if ((oracleMode() == ORACLE_PLAIN))
16851687 then nil
16861688 else {
16871689 let priceUpdates = split_4C(_priceUpdate, "::")
16881690 let baseOracle = valueOrErrorMessage(getOracleData(k_baseOracle), "No base asset oracle data")
16891691 let baseOracleAddress = baseOracle._1
16901692 let doUpdateBaseOracle = invoke(baseOracleAddress, "updateData", [priceUpdates[0]], nil)
16911693 if ((doUpdateBaseOracle == doUpdateBaseOracle))
16921694 then {
16931695 let quoteOracle = getOracleData(k_quoteOracle)
16941696 let doUpdateQuoteOracle = if (isDefined(quoteOracle))
16951697 then {
16961698 let quoteOracleV = value(quoteOracle)
16971699 let quoteOracleAddress = quoteOracleV._1
16981700 let doUpdateQuoteOracle = invoke(quoteOracleAddress, "updateData", [priceUpdates[1]], nil)
16991701 if ((doUpdateQuoteOracle == doUpdateQuoteOracle))
17001702 then nil
17011703 else throw("Strict value is not equal to itself.")
17021704 }
17031705 else nil
17041706 if ((doUpdateQuoteOracle == doUpdateQuoteOracle))
17051707 then nil
17061708 else throw("Strict value is not equal to itself.")
17071709 }
17081710 else throw("Strict value is not equal to itself.")
17091711 }
17101712
17111713
17121714
17131715 @Callable(i)
17141716 func syncTerminalPriceToOracle () = {
17151717 let _qtAstR = qtAstR()
17161718 let _bsAstR = bsAstR()
1717- let $t07923279598 = getSyncTerminalPrice(getOraclePrice(), _qtAstR, _bsAstR)
1718- let newQuoteAssetWeight = $t07923279598._1
1719- let newBaseAssetWeight = $t07923279598._2
1720- let marginToVault = $t07923279598._3
1719+ let $t07932179687 = getSyncTerminalPrice(getOraclePrice(), _qtAstR, _bsAstR)
1720+ let newQuoteAssetWeight = $t07932179687._1
1721+ let newBaseAssetWeight = $t07932179687._2
1722+ let marginToVault = $t07932179687._3
17211723 let marginToVaultAdj = if (if ((0 > marginToVault))
17221724 then (abs(marginToVault) > cbalance())
17231725 else false)
17241726 then -(cbalance())
17251727 else marginToVault
17261728 let doExchangePnL = if ((marginToVaultAdj != 0))
17271729 then {
17281730 let doExchangePnL = invoke(vaultAddress(), "exchangeFreeAndLocked", [marginToVaultAdj], nil)
17291731 if ((doExchangePnL == doExchangePnL))
17301732 then nil
17311733 else throw("Strict value is not equal to itself.")
17321734 }
17331735 else nil
17341736 if ((doExchangePnL == doExchangePnL))
17351737 then (updateBalance((cbalance() + marginToVaultAdj)) ++ updateAmmWeights(newQuoteAssetWeight, newBaseAssetWeight))
17361738 else throw("Strict value is not equal to itself.")
17371739 }
17381740
17391741
17401742
17411743 @Callable(i)
17421744 func ensureCalledOnce () = if ((i.caller != this))
17431745 then throw("Invalid saveCurrentTxId parameters")
17441746 else {
17451747 let txId = toBase58String(i.transactionId)
17461748 let lastTx = valueOrElse(getString(this, k_lastTx), "")
17471749 if ((lastTx != txId))
17481750 then [StringEntry(k_lastTx, txId)]
17491751 else throw("Can not call vAMM methods twice in one tx")
17501752 }
17511753
17521754
17531755
17541756 @Callable(i)
17551757 func migratePosition (_trader) = {
17561758 let positionSizeOpt = getInteger(this, toCompositeKey(k_positionSize, _trader))
17571759 if (if (isDefined(positionSizeOpt))
17581760 then isDefined(addressFromString(_trader))
17591761 else false)
17601762 then {
17611763 let pSize = getIntegerValue(this, toCompositeKey(k_positionSize, _trader))
17621764 let pMargin = getIntegerValue(this, toCompositeKey(k_positionMargin, _trader))
17631765 let pNotional = getIntegerValue(this, toCompositeKey(k_positionOpenNotional, _trader))
17641766 let pFraction = getIntegerValue(this, toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _trader))
17651767 let pTimestamp = valueOrElse(getInteger(this, toCompositeKey(k_positionLastUpdatedTimestamp, _trader)), lastBlock.timestamp)
17661768 let pFee = valueOrElse(getInteger(this, toCompositeKey(k_positionFee, _trader)), fee())
17671769 let pSequence = getIntegerValue(this, toCompositeKey(k_positionSequence, _trader))
17681770 let pDirection = getDirection(pSize)
17691771 let positionKey = ((_trader + "_") + toString(pDirection))
17701772 [DeleteEntry(toCompositeKey(k_positionSize, _trader)), DeleteEntry(toCompositeKey(k_positionMargin, _trader)), DeleteEntry(toCompositeKey(k_positionOpenNotional, _trader)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _trader)), DeleteEntry(toCompositeKey(k_positionLastUpdatedTimestamp, _trader)), DeleteEntry(toCompositeKey(k_positionFee, _trader)), DeleteEntry(toCompositeKey(k_positionSequence, _trader)), IntegerEntry(toCompositeKey(k_positionSize, positionKey), pSize), IntegerEntry(toCompositeKey(k_positionMargin, positionKey), pMargin), IntegerEntry(toCompositeKey(k_positionOpenNotional, positionKey), pNotional), IntegerEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, positionKey), pFraction), IntegerEntry(toCompositeKey(k_positionLastUpdatedTimestamp, positionKey), pTimestamp), IntegerEntry(toCompositeKey(k_positionFee, positionKey), pFee), IntegerEntry(toCompositeKey(k_positionSequence, positionKey), pSequence)]
17711773 }
17721774 else throw(("Nothing to migrate for " + _trader))
17731775 }
17741776
17751777
17761778
17771779 @Callable(i)
17781780 func view_calcRemainMarginWithFundingPayment (_trader,_direction,_priceUpdate) = {
17791781 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
17801782 if ((updateOracle == updateOracle))
17811783 then {
17821784 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
17831785 if ((sync == sync))
17841786 then {
1785- let $t08339283528 = getPosition(_trader, _direction)
1786- let positionSize = $t08339283528._1
1787- let positionMargin = $t08339283528._2
1788- let pon = $t08339283528._3
1789- let positionLstUpdCPF = $t08339283528._4
1790- let positionTimestamp = $t08339283528._5
1791- let $t08353183644 = getPositionNotionalAndUnrealizedPnl(_trader, _direction, PNL_OPTION_SPOT)
1792- let positionNotional = $t08353183644._1
1793- let unrealizedPnl = $t08353183644._2
1794- let $t08364783871 = calcRemainMarginWithFundingPaymentAndRolloverFee(positionSize, positionMargin, positionLstUpdCPF, positionTimestamp, unrealizedPnl)
1795- let remainMargin = $t08364783871._1
1796- let badDebt = $t08364783871._2
1797- let fundingPayment = $t08364783871._3
1798- let rolloverFee = $t08364783871._4
1787+ let $t08348183617 = getPosition(_trader, _direction)
1788+ let positionSize = $t08348183617._1
1789+ let positionMargin = $t08348183617._2
1790+ let pon = $t08348183617._3
1791+ let positionLstUpdCPF = $t08348183617._4
1792+ let positionTimestamp = $t08348183617._5
1793+ let $t08362083733 = getPositionNotionalAndUnrealizedPnl(_trader, _direction, PNL_OPTION_SPOT)
1794+ let positionNotional = $t08362083733._1
1795+ let unrealizedPnl = $t08362083733._2
1796+ let $t08373683960 = calcRemainMarginWithFundingPaymentAndRolloverFee(positionSize, positionMargin, positionLstUpdCPF, positionTimestamp, unrealizedPnl)
1797+ let remainMargin = $t08373683960._1
1798+ let badDebt = $t08373683960._2
1799+ let fundingPayment = $t08373683960._3
1800+ let rolloverFee = $t08373683960._4
17991801 throw(((((((s(remainMargin) + s(fundingPayment)) + s(getMarginRatio(_trader, _direction))) + s(unrealizedPnl)) + s(badDebt)) + s(positionNotional)) + s(rolloverFee)))
18001802 }
18011803 else throw("Strict value is not equal to itself.")
18021804 }
18031805 else throw("Strict value is not equal to itself.")
18041806 }
18051807
18061808
18071809
18081810 @Callable(i)
18091811 func view_getPegAdjustCost (_price) = {
18101812 let _qtAstR = qtAstR()
18111813 let _bsAstR = bsAstR()
18121814 let result = getSyncTerminalPrice(_price, _qtAstR, _bsAstR)
18131815 throw(toString(result._3))
18141816 }
18151817
18161818
18171819
18181820 @Callable(i)
18191821 func view_getTerminalAmmPrice () = {
1820- let $t08460684687 = getTerminalAmmState()
1821- let terminalQuoteAssetReserve = $t08460684687._1
1822- let terminalBaseAssetReserve = $t08460684687._2
1822+ let $t08469584776 = getTerminalAmmState()
1823+ let terminalQuoteAssetReserve = $t08469584776._1
1824+ let terminalBaseAssetReserve = $t08469584776._2
18231825 let price = divd(muld(terminalQuoteAssetReserve, qtAstW()), muld(terminalBaseAssetReserve, bsAstW()))
18241826 throw(toString(price))
18251827 }
18261828
18271829
18281830
18291831 @Callable(i)
18301832 func view_getFunding (_priceUpdate) = {
18311833 let updateOracle = invoke(this, "updateOracle", [_priceUpdate], nil)
18321834 if ((updateOracle == updateOracle))
18331835 then {
18341836 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
18351837 if ((sync == sync))
18361838 then {
18371839 let underlyingPrice = getOraclePrice()
1838- let $t08533985417 = getFunding()
1839- let shortPremiumFraction = $t08533985417._1
1840- let longPremiumFraction = $t08533985417._2
1841- let premiumToVault = $t08533985417._3
1840+ let $t08542885506 = getFunding()
1841+ let shortPremiumFraction = $t08542885506._1
1842+ let longPremiumFraction = $t08542885506._2
1843+ let premiumToVault = $t08542885506._3
18421844 let longFunding = divd(longPremiumFraction, underlyingPrice)
18431845 let shortFunding = divd(shortPremiumFraction, underlyingPrice)
18441846 throw(((((s(longFunding) + s(shortFunding)) + s(getSpotPrice())) + s(getOraclePrice())) + s(premiumToVault)))
18451847 }
18461848 else throw("Strict value is not equal to itself.")
18471849 }
18481850 else throw("Strict value is not equal to itself.")
18491851 }
18501852
18511853
18521854
18531855 @Callable(i)
18541856 func computeSpotPrice () = {
18551857 let sync = invoke(this, "syncTerminalPriceToOracle", nil, nil)
18561858 if ((sync == sync))
18571859 then {
18581860 let result = getSpotPrice()
18591861 $Tuple2(nil, result)
18601862 }
18611863 else throw("Strict value is not equal to itself.")
18621864 }
18631865
18641866
18651867
18661868 @Callable(i)
18671869 func computeFeeForTraderWithArtifact (_trader,_artifactId) = {
18681870 let result = getForTraderWithArtifact(_trader, _artifactId)
18691871 $Tuple2(nil, result)
18701872 }
18711873
18721874
18731875 @Verifier(tx)
18741876 func verify () = {
18751877 let coordinatorStr = getString(this, k_coordinatorAddress)
18761878 if (isDefined(coordinatorStr))
18771879 then {
18781880 let admin = getString(addressFromStringValue(value(coordinatorStr)), k_admin_address)
18791881 if (isDefined(admin))
18801882 then valueOrElse(getBoolean(addressFromStringValue(value(admin)), ((("status_" + toString(this)) + "_") + toBase58String(tx.id))), false)
18811883 else throw("unable to verify: admin not set in coordinator")
18821884 }
18831885 else sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
18841886 }
18851887

github/deemru/w8io/026f985 
258.81 ms