tx · Ct51rE8dRpvWWMg1VvquHAUiYfUEBeHQ2xkuJnQB5fJe

3PNeboTZQz1XhwDY9V9N3K7xYJrq9XbaS5Z:  -0.05200000 Waves

2022.08.18 17:39 [3255540] smart account 3PNeboTZQz1XhwDY9V9N3K7xYJrq9XbaS5Z > SELF 0.00000000 Waves

{ "type": 13, "id": "Ct51rE8dRpvWWMg1VvquHAUiYfUEBeHQ2xkuJnQB5fJe", "fee": 5200000, "feeAssetId": null, "timestamp": 1660833575643, "version": 2, "chainId": 87, "sender": "3PNeboTZQz1XhwDY9V9N3K7xYJrq9XbaS5Z", "senderPublicKey": "33S2QSUiVtgmGu7pTKSbJZzNDHABx6edU6YxUK49bE4w", "proofs": [ "3dp1xanqFXdWmPzLzX7uNAJDYnVeFwQg4Z8BmNzmVCjtfMTgkbNqs6XVTgtLoSZmcL4bjXR7eyLhK3cZRH5uCgRY" ], "script": "base64:BgJeCAISABIAEgMKAQESAwoBARIJCgcBAQEBAQEBEg4KDAEBAQEBAQEICAgBARIDCgEBEgUKAwEBARIFCgMBAQESABIDCgEBEgASAwoBCBIAEgMKAQgSAwoBCBIECgIIAXcACWtfb3JhX2tleQIJa19vcmFfa2V5AA9rX29yYV9ibG9ja19rZXkCD2tfb3JhX2Jsb2NrX2tleQAFa19vcmECBWtfb3JhAAlrX2JhbGFuY2UCCWtfYmFsYW5jZQAOa19wb3NpdGlvblNpemUCDmtfcG9zaXRpb25TaXplABBrX3Bvc2l0aW9uTWFyZ2luAhBrX3Bvc2l0aW9uTWFyZ2luABZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsAhZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsAC5rX3Bvc2l0aW9uTGFzdFVwZGF0ZWRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAhJrX3Bvc2l0aW9uRnJhY3Rpb24ADWtfaW5pdGlhbGl6ZWQCDWtfaW5pdGlhbGl6ZWQACGtfcGF1c2VkAghrX3BhdXNlZAAFa19mZWUCBWtfZmVlAA9rX2Z1bmRpbmdQZXJpb2QCD2tfZnVuZGluZ1BlcmlvZAARa19pbml0TWFyZ2luUmF0aW8CEWtfaW5pdE1hcmdpblJhdGlvABhrX21haW50ZW5hbmNlTWFyZ2luUmF0aW8CBWtfbW1yABVrX2xpcXVpZGF0aW9uRmVlUmF0aW8CFWtfbGlxdWlkYXRpb25GZWVSYXRpbwANa19zcHJlYWRMaW1pdAINa19zcHJlYWRMaW1pdAAQa19tYXhQcmljZUltcGFjdAIQa19tYXhQcmljZUltcGFjdAANa19sYXN0RGF0YVN0cgINa19sYXN0RGF0YVN0cgAOa19sYXN0TWludXRlSWQCDmtfbGFzdE1pbnV0ZUlkAB1rX3R3YXBEYXRhTGFzdEN1bXVsYXRpdmVQcmljZQIda190d2FwRGF0YUxhc3RDdW11bGF0aXZlUHJpY2UAE2tfdHdhcERhdGFMYXN0UHJpY2UCE2tfdHdhcERhdGFMYXN0UHJpY2UAGmtfdHdhcERhdGFQcmV2aW91c01pbnV0ZUlkAhprX3R3YXBEYXRhUHJldmlvdXNNaW51dGVJZAAla19sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgIba19sYXRlc3RMb25nUHJlbWl1bUZyYWN0aW9uACZrX2xhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgIca19sYXRlc3RTaG9ydFByZW1pdW1GcmFjdGlvbgASa19uZXh0RnVuZGluZ0Jsb2NrAh5rX25leHRGdW5kaW5nQmxvY2tNaW5UaW1lc3RhbXAAEWtfbG9uZ0Z1bmRpbmdSYXRlAhFrX2xvbmdGdW5kaW5nUmF0ZQASa19zaG9ydEZ1bmRpbmdSYXRlAhJrX3Nob3J0RnVuZGluZ1JhdGUAE2tfcXVvdGVBc3NldFJlc2VydmUCCGtfcXRBc3RSABJrX2Jhc2VBc3NldFJlc2VydmUCCGtfYnNBc3RSABNrX3RvdGFsUG9zaXRpb25TaXplAhNrX3RvdGFsUG9zaXRpb25TaXplABdrX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQIXa190b3RhbExvbmdQb3NpdGlvblNpemUAGGtfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQIYa190b3RhbFNob3J0UG9zaXRpb25TaXplABRrX2N1bXVsYXRpdmVOb3Rpb25hbAIUa19jdW11bGF0aXZlTm90aW9uYWwAFmtfb3BlbkludGVyZXN0Tm90aW9uYWwCFmtfb3BlbkludGVyZXN0Tm90aW9uYWwAFGtfY29vcmRpbmF0b3JBZGRyZXNzAhRrX2Nvb3JkaW5hdG9yQWRkcmVzcwATa19pbnN1cmFuY2VfYWRkcmVzcwITa19pbnN1cmFuY2VfYWRkcmVzcwAPa19hZG1pbl9hZGRyZXNzAg9rX2FkbWluX2FkZHJlc3MAEmtfYWRtaW5fcHVibGljX2tleQISa19hZG1pbl9wdWJsaWNfa2V5AA1rX3F1b3RlX2Fzc2V0Ag1rX3F1b3RlX2Fzc2V0AA9rX3F1b3RlX3N0YWtpbmcCD2tfcXVvdGVfc3Rha2luZwARa19zdGFraW5nX2FkZHJlc3MCEWtfc3Rha2luZ19hZGRyZXNzAA9rX21pbmVyX2FkZHJlc3MCD2tfbWluZXJfYWRkcmVzcwELY29vcmRpbmF0b3IACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhpcwUUa19jb29yZGluYXRvckFkZHJlc3MCE0Nvb3JkaW5hdG9yIG5vdCBzZXQBDGFkbWluQWRkcmVzcwAJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUPa19hZG1pbl9hZGRyZXNzAQ5hZG1pblB1YmxpY0tleQAJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUSa19hZG1pbl9wdWJsaWNfa2V5AQpxdW90ZUFzc2V0AAkA2QQBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABQ1rX3F1b3RlX2Fzc2V0ARFxdW90ZUFzc2V0U3Rha2luZwAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABQ9rX3F1b3RlX3N0YWtpbmcCG1F1b3RlIGFzc2V0IHN0YWtpbmcgbm90IHNldAEOc3Rha2luZ0FkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAURa19zdGFraW5nX2FkZHJlc3MCEUluc3VyYW5jZSBub3Qgc2V0ARBpbnN1cmFuY2VBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFE2tfaW5zdXJhbmNlX2FkZHJlc3MCEUluc3VyYW5jZSBub3Qgc2V0AQxtaW5lckFkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUPa19taW5lcl9hZGRyZXNzAhFJbnN1cmFuY2Ugbm90IHNldAAIRElSX0xPTkcAAQAJRElSX1NIT1JUAAIADVRXQVBfSU5URVJWQUwADwAPT1JBQ0xFX0lOVEVSVkFMAA8AB1NFQ09ORFMA6AcADERFQ0lNQUxfVU5JVAkAaAIAAQkAaAIJAGgCCQBoAgkAaAIJAGgCAAoACgAKAAoACgAKAAdPTkVfREFZCQBoAgCAowUFDERFQ0lNQUxfVU5JVAAIQUxMX0ZFRVMAZAAPUE5MX09QVElPTl9TUE9UAAEAEVBOTF9PUFRJT05fT1JBQ0xFAAIBAXMBAl94CQCsAgIJAKQDAQUCX3gCASwBBGRpdmQCAl94Al95CQBuBAUCX3gFDERFQ0lNQUxfVU5JVAUCX3kFCEhBTEZFVkVOAQRtdWxkAgJfeAJfeQkAbgQFAl94BQJfeQUMREVDSU1BTF9VTklUBQhIQUxGRVZFTgEDYWJzAQJfeAMJAGYCBQJfeAAABQJfeAkBAS0BBQJfeAEEdm1heAICX3gCX3kDCQBnAgUCX3gFAl95BQJfeAUCX3kBDnRvQ29tcG9zaXRlS2V5AgRfa2V5CF9hZGRyZXNzCQCsAgIJAKwCAgUEX2tleQIBXwUIX2FkZHJlc3MBCWxpc3RUb1N0cgEFX2xpc3QKAQVfam9pbgILYWNjdW11bGF0b3IDdmFsCQCsAgIJAKwCAgULYWNjdW11bGF0b3IFA3ZhbAIBLAQKbmV3TGlzdFN0cgoAAiRsBQVfbGlzdAoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEFX2pvaW4CBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDIwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAQLbmV3TGlzdFN0clUJALMCAgUKbmV3TGlzdFN0cgABBAtuZXdMaXN0U3RyUgMJAAACCQCvAgIFC25ld0xpc3RTdHJVAAECASwJALACAgULbmV3TGlzdFN0clUAAQULbmV3TGlzdFN0clUFC25ld0xpc3RTdHJSAQlzdHJUb0xpc3QBBF9zdHIJALUJAgUEX3N0cgIBLAELcHVzaFRvUXVldWUDBV9saXN0CF9tYXhTaXplBl92YWx1ZQMJAGYCCQCQAwEFBV9saXN0BQhfbWF4U2l6ZQkAzQgCCQDRCAIFBV9saXN0AAAFBl92YWx1ZQkAzQgCBQVfbGlzdAUGX3ZhbHVlAQNpbnQBAWsJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQR0aGlzBQFrCQCsAgICDW5vIHZhbHVlIGZvciAFAWsBCGNiYWxhbmNlAAkBA2ludAEFCWtfYmFsYW5jZQEDZmVlAAkBA2ludAEFBWtfZmVlAQ9pbml0TWFyZ2luUmF0aW8ACQEDaW50AQURa19pbml0TWFyZ2luUmF0aW8BBnF0QXN0UgAJAQNpbnQBBRNrX3F1b3RlQXNzZXRSZXNlcnZlAQZic0FzdFIACQEDaW50AQUSa19iYXNlQXNzZXRSZXNlcnZlARF0b3RhbFBvc2l0aW9uU2l6ZQAJAQNpbnQBBRNrX3RvdGFsUG9zaXRpb25TaXplARJjdW11bGF0aXZlTm90aW9uYWwACQEDaW50AQUUa19jdW11bGF0aXZlTm90aW9uYWwBFG9wZW5JbnRlcmVzdE5vdGlvbmFsAAkBA2ludAEFFmtfb3BlbkludGVyZXN0Tm90aW9uYWwBGW5leHRGdW5kaW5nQmxvY2tUaW1lc3RhbXAACQEDaW50AQUSa19uZXh0RnVuZGluZ0Jsb2NrARBmdW5kaW5nUGVyaW9kUmF3AAkBA2ludAEFD2tfZnVuZGluZ1BlcmlvZAEUZnVuZGluZ1BlcmlvZERlY2ltYWwACQBoAgkBEGZ1bmRpbmdQZXJpb2RSYXcABQxERUNJTUFMX1VOSVQBFGZ1bmRpbmdQZXJpb2RTZWNvbmRzAAkAaAIJARBmdW5kaW5nUGVyaW9kUmF3AAUHU0VDT05EUwEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAJAQNpbnQBBRhrX21haW50ZW5hbmNlTWFyZ2luUmF0aW8BE2xpcXVpZGF0aW9uRmVlUmF0aW8ACQEDaW50AQUVa19saXF1aWRhdGlvbkZlZVJhdGlvAQtzcHJlYWRMaW1pdAAJAQNpbnQBBQ1rX3NwcmVhZExpbWl0AQ5tYXhQcmljZUltcGFjdAAJAQNpbnQBBRBrX21heFByaWNlSW1wYWN0ASNsYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgAJAQNpbnQBBSVrX2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uASRsYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24ACQEDaW50AQUma19sYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BFnRvdGFsU2hvcnRQb3NpdGlvblNpemUACQEDaW50AQUYa190b3RhbFNob3J0UG9zaXRpb25TaXplARV0b3RhbExvbmdQb3NpdGlvblNpemUACQEDaW50AQUXa190b3RhbExvbmdQb3NpdGlvblNpemUBFnJlcXVpcmVNb3JlTWFyZ2luUmF0aW8DDF9tYXJnaW5SYXRpbxBfYmFzZU1hcmdpblJhdGlvFF9sYXJnZXJUaGFuT3JFcXVhbFRvBBRyZW1haW5pbmdNYXJnaW5SYXRpbwkAZQIFDF9tYXJnaW5SYXRpbwUQX2Jhc2VNYXJnaW5SYXRpbwMDBRRfbGFyZ2VyVGhhbk9yRXF1YWxUbwkAZgIAAAUUcmVtYWluaW5nTWFyZ2luUmF0aW8HCQACAQIOSW52YWxpZCBtYXJnaW4DAwkBASEBBRRfbGFyZ2VyVGhhbk9yRXF1YWxUbwkAZwIFFHJlbWFpbmluZ01hcmdpblJhdGlvAAAHCQACAQIOSW52YWxpZCBtYXJnaW4GAR9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAQ1fcG9zaXRpb25TaXplAwkAAAIFDV9wb3NpdGlvblNpemUAAAkAAgECLFNob3VsZCBub3QgYmUgY2FsbGVkIHdpdGggX3Bvc2l0aW9uU2l6ZSA9PSAwAwkAZgIFDV9wb3NpdGlvblNpemUAAAkBI2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAAkBJGxhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgABC2dldFBvc2l0aW9uAQdfdHJhZGVyBA9wb3NpdGlvblNpemVPcHQJAJoIAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUOa19wb3NpdGlvblNpemUFB190cmFkZXIEByRtYXRjaDAFD3Bvc2l0aW9uU2l6ZU9wdAMJAAECBQckbWF0Y2gwAgNJbnQEDHBvc2l0aW9uU2l6ZQUHJG1hdGNoMAkAlgoEBQxwb3NpdGlvblNpemUJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFEGtfcG9zaXRpb25NYXJnaW4FB190cmFkZXIJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFFmtfcG9zaXRpb25PcGVuTm90aW9uYWwFB190cmFkZXIJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFLmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FB190cmFkZXIJAJYKBAAAAAAAAAAAARNyZXF1aXJlT3BlblBvc2l0aW9uAQdfdHJhZGVyAwkAAAIICQELZ2V0UG9zaXRpb24BBQdfdHJhZGVyAl8xAAAJAAIBAhBObyBvcGVuIHBvc2l0aW9uBgELaW5pdGlhbGl6ZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFDWtfaW5pdGlhbGl6ZWQHAQZwYXVzZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFCGtfcGF1c2VkBwENdXBkYXRlUmVzZXJ2ZQMGX2lzQWRkEV9xdW90ZUFzc2V0QW1vdW50EF9iYXNlQXNzZXRBbW91bnQDBQZfaXNBZGQEB25ld0Jhc2UJAGUCCQEGYnNBc3RSAAUQX2Jhc2VBc3NldEFtb3VudAMJAGcCAAAFB25ld0Jhc2UJAAIBAipUeCBsZWFkIHRvIGJhc2UgYXNzZXQgcmVzZXJ2ZSA8PSAwLCByZXZlcnQJAJYKBAkAZAIJAQZxdEFzdFIABRFfcXVvdGVBc3NldEFtb3VudAUHbmV3QmFzZQkAZAIJARF0b3RhbFBvc2l0aW9uU2l6ZQAFEF9iYXNlQXNzZXRBbW91bnQJAGQCCQESY3VtdWxhdGl2ZU5vdGlvbmFsAAURX3F1b3RlQXNzZXRBbW91bnQECG5ld1F1b3RlCQBlAgkBBnF0QXN0UgAFEV9xdW90ZUFzc2V0QW1vdW50AwkAZwIAAAUIbmV3UXVvdGUJAAIBAipUeCBsZWFkIHRvIGJhc2UgcXVvdGUgcmVzZXJ2ZSA8PSAwLCByZXZlcnQJAJYKBAUIbmV3UXVvdGUJAGQCCQEGYnNBc3RSAAUQX2Jhc2VBc3NldEFtb3VudAkAZQIJARF0b3RhbFBvc2l0aW9uU2l6ZQAFEF9iYXNlQXNzZXRBbW91bnQJAGUCCQESY3VtdWxhdGl2ZU5vdGlvbmFsAAURX3F1b3RlQXNzZXRBbW91bnQBCXN3YXBJbnB1dAIGX2lzQWRkEV9xdW90ZUFzc2V0QW1vdW50BAdfcXRBc3RSCQEGcXRBc3RSAAQHX2JzQXN0UgkBBmJzQXN0UgAEAWsJAQRtdWxkAgUHX3F0QXN0UgUHX2JzQXN0UgQWcXVvdGVBc3NldFJlc2VydmVBZnRlcgMFBl9pc0FkZAkAZAIFB19xdEFzdFIFEV9xdW90ZUFzc2V0QW1vdW50CQBlAgUHX3F0QXN0UgURX3F1b3RlQXNzZXRBbW91bnQEFWJhc2VBc3NldFJlc2VydmVBZnRlcgkBBGRpdmQCBQFrBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBBhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMJAQNhYnMBCQBlAgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBQdfYnNBc3RSBBVhbW91bnRCYXNlQXNzZXRCb3VnaHQDBQZfaXNBZGQFGGFtb3VudEJhc2VBc3NldEJvdWdodEFicwkBAS0BBRhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMEDSR0MDExODM3MTIwMzAJAQ11cGRhdGVSZXNlcnZlAwUGX2lzQWRkBRFfcXVvdGVBc3NldEFtb3VudAUYYW1vdW50QmFzZUFzc2V0Qm91Z2h0QWJzBBdxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyMQgFDSR0MDExODM3MTIwMzACXzEEFmJhc2VBc3NldFJlc2VydmVBZnRlcjEIBQ0kdDAxMTgzNzEyMDMwAl8yBBd0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyMQgFDSR0MDExODM3MTIwMzACXzMEGGN1bXVsYXRpdmVOb3Rpb25hbEFmdGVyMQgFDSR0MDExODM3MTIwMzACXzQEC3ByaWNlQmVmb3JlCQEEZGl2ZAIFB19xdEFzdFIFB19ic0FzdFIEC21hcmtldFByaWNlCQEEZGl2ZAIFEV9xdW90ZUFzc2V0QW1vdW50BRhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMECXByaWNlRGlmZgkBA2FicwEJAGUCBQtwcmljZUJlZm9yZQULbWFya2V0UHJpY2UEC3ByaWNlSW1wYWN0CQBlAgUMREVDSU1BTF9VTklUCQEEZGl2ZAIFC3ByaWNlQmVmb3JlCQBkAgULcHJpY2VCZWZvcmUFCXByaWNlRGlmZgQTbWF4UHJpY2VJbXBhY3RWYWx1ZQkBDm1heFByaWNlSW1wYWN0AAMJAGYCBQtwcmljZUltcGFjdAUTbWF4UHJpY2VJbXBhY3RWYWx1ZQkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgINUHJpY2UgaW1wYWN0IAkApAMBBQtwcmljZUltcGFjdAIUID4gbWF4IHByaWNlIGltcGFjdCAJAKQDAQUTbWF4UHJpY2VJbXBhY3RWYWx1ZQIVIGJlZm9yZSBxdW90ZSBhc3NldDogCQCkAwEFB19xdEFzdFICFCBiZWZvcmUgYmFzZSBhc3NldDogCQCkAwEFB19ic0FzdFICISBxdW90ZSBhc3NldCBhbW91bnQgdG8gZXhjaGFuZ2U6IAkApAMBBRFfcXVvdGVBc3NldEFtb3VudAIPIHByaWNlIGJlZm9yZTogCQCkAwEFC3ByaWNlQmVmb3JlAg4gbWFya2V0UHJpY2U6IAkApAMBBQttYXJrZXRQcmljZQkAlwoFBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQFF3F1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIxBRZiYXNlQXNzZXRSZXNlcnZlQWZ0ZXIxBRd0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyMQUYY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIxASJjYWxjUmVtYWluTWFyZ2luV2l0aEZ1bmRpbmdQYXltZW50BBBfb2xkUG9zaXRpb25TaXplEl9vbGRQb3NpdGlvbk1hcmdpbiVfb2xkUG9zaXRpb25DdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uDF9tYXJnaW5EZWx0YQQOZnVuZGluZ1BheW1lbnQDCQECIT0CBRBfb2xkUG9zaXRpb25TaXplAAAEIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uCQEfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgEFEF9vbGRQb3NpdGlvblNpemUJAQRtdWxkAgkAZQIFIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBSVfb2xkUG9zaXRpb25DdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBRBfb2xkUG9zaXRpb25TaXplAAAEDHNpZ25lZE1hcmdpbgkAZAIJAGUCBQxfbWFyZ2luRGVsdGEFDmZ1bmRpbmdQYXltZW50BRJfb2xkUG9zaXRpb25NYXJnaW4EDSR0MDEzNTIxMTM2NDgDCQBmAgAABQxzaWduZWRNYXJnaW4JAJQKAgAACQEDYWJzAQUMc2lnbmVkTWFyZ2luCQCUCgIJAQNhYnMBBQxzaWduZWRNYXJnaW4AAAQMcmVtYWluTWFyZ2luCAUNJHQwMTM1MjExMzY0OAJfMQQHYmFkRGVidAgFDSR0MDEzNTIxMTM2NDgCXzIJAJUKAwUMcmVtYWluTWFyZ2luBQdiYWREZWJ0BQ5mdW5kaW5nUGF5bWVudAEKc3dhcE91dHB1dAMGX2lzQWRkEF9iYXNlQXNzZXRBbW91bnQUX2NoZWNrTWF4UHJpY2VJbXBhY3QEEl9xdW90ZUFzc2V0UmVzZXJ2ZQkBBnF0QXN0UgAEEV9iYXNlQXNzZXRSZXNlcnZlCQEGYnNBc3RSAAQLcHJpY2VCZWZvcmUJAQRkaXZkAgUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfYmFzZUFzc2V0UmVzZXJ2ZQMJAAACBRBfYmFzZUFzc2V0QW1vdW50AAAJAAIBAhlJbnZhbGlkIGJhc2UgYXNzZXQgYW1vdW50BAFrCQEEbXVsZAIFEl9xdW90ZUFzc2V0UmVzZXJ2ZQURX2Jhc2VBc3NldFJlc2VydmUEGGJhc2VBc3NldFBvb2xBbW91bnRBZnRlcgMFBl9pc0FkZAkAZAIFEV9iYXNlQXNzZXRSZXNlcnZlBRBfYmFzZUFzc2V0QW1vdW50CQBlAgURX2Jhc2VBc3NldFJlc2VydmUFEF9iYXNlQXNzZXRBbW91bnQED3F1b3RlQXNzZXRBZnRlcgkBBGRpdmQCBQFrBRhiYXNlQXNzZXRQb29sQW1vdW50QWZ0ZXIEDnF1b3RlQXNzZXRTb2xkCQEDYWJzAQkAZQIFD3F1b3RlQXNzZXRBZnRlcgUSX3F1b3RlQXNzZXRSZXNlcnZlBBNtYXhQcmljZUltcGFjdFZhbHVlCQEObWF4UHJpY2VJbXBhY3QABA0kdDAxNDYzMjE0ODI1CQENdXBkYXRlUmVzZXJ2ZQMJAQEhAQUGX2lzQWRkBQ5xdW90ZUFzc2V0U29sZAUQX2Jhc2VBc3NldEFtb3VudAQXcXVvdGVBc3NldFJlc2VydmVBZnRlcjEIBQ0kdDAxNDYzMjE0ODI1Al8xBBZiYXNlQXNzZXRSZXNlcnZlQWZ0ZXIxCAUNJHQwMTQ2MzIxNDgyNQJfMgQXdG90YWxQb3NpdGlvblNpemVBZnRlcjEIBQ0kdDAxNDYzMjE0ODI1Al8zBBhjdW11bGF0aXZlTm90aW9uYWxBZnRlcjEIBQ0kdDAxNDYzMjE0ODI1Al80BAttYXJrZXRQcmljZQkBBGRpdmQCBQ5xdW90ZUFzc2V0U29sZAUQX2Jhc2VBc3NldEFtb3VudAQJcHJpY2VEaWZmCQEDYWJzAQkAZQIFC3ByaWNlQmVmb3JlBQttYXJrZXRQcmljZQQLcHJpY2VJbXBhY3QJAGUCBQxERUNJTUFMX1VOSVQJAQRkaXZkAgULcHJpY2VCZWZvcmUJAGQCBQtwcmljZUJlZm9yZQUJcHJpY2VEaWZmAwMJAGYCBQtwcmljZUltcGFjdAUTbWF4UHJpY2VJbXBhY3RWYWx1ZQUUX2NoZWNrTWF4UHJpY2VJbXBhY3QHCQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAg1QcmljZSBpbXBhY3QgCQCkAwEFC3ByaWNlSW1wYWN0AhQgPiBtYXggcHJpY2UgaW1wYWN0IAkApAMBBRNtYXhQcmljZUltcGFjdFZhbHVlAhUgYmVmb3JlIHF1b3RlIGFzc2V0OiAJAKQDAQUSX3F1b3RlQXNzZXRSZXNlcnZlAhQgYmVmb3JlIGJhc2UgYXNzZXQ6IAkApAMBBRFfYmFzZUFzc2V0UmVzZXJ2ZQIgIGJhc2UgYXNzZXQgYW1vdW50IHRvIGV4Y2hhbmdlOiAJAKQDAQUQX2Jhc2VBc3NldEFtb3VudAIPIHByaWNlIGJlZm9yZTogCQCkAwEFC3ByaWNlQmVmb3JlAg8gbWFya2V0IHByaWNlOiAJAKQDAQULbWFya2V0UHJpY2UJAJkKBwUOcXVvdGVBc3NldFNvbGQFF3F1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIxBRZiYXNlQXNzZXRSZXNlcnZlQWZ0ZXIxBRd0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyMQUYY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIxCQBlAgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQADBQZfaXNBZGQJAQNhYnMBBRBfYmFzZUFzc2V0QW1vdW50AAAJAGUCCQEWdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQADCQEBIQEFBl9pc0FkZAkBA2FicwEFEF9iYXNlQXNzZXRBbW91bnQAAAESZ2V0T3JhY2xlVHdhcFByaWNlAAQGb3JhY2xlCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhpcwUFa19vcmECAAQIcHJpY2VLZXkJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMFCWtfb3JhX2tleQQIYmxvY2tLZXkJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMFD2tfb3JhX2Jsb2NrX2tleQQJbGFzdFZhbHVlCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUGb3JhY2xlBQhwcmljZUtleQkArAICCQCsAgIJAKwCAgIiQ2FuIG5vdCBnZXQgb3JhY2xlIHByaWNlLiBPcmFjbGU6IAkApQgBBQZvcmFjbGUCBiBrZXk6IAUIcHJpY2VLZXkFCWxhc3RWYWx1ZQEMZ2V0U3BvdFByaWNlAAQSX3F1b3RlQXNzZXRSZXNlcnZlCQEGcXRBc3RSAAQRX2Jhc2VBc3NldFJlc2VydmUJAQZic0FzdFIACQEEZGl2ZAIFEl9xdW90ZUFzc2V0UmVzZXJ2ZQURX2Jhc2VBc3NldFJlc2VydmUBFmlzT3ZlckZsdWN0dWF0aW9uTGltaXQABAtvcmFjbGVQcmljZQkBEmdldE9yYWNsZVR3YXBQcmljZQAEDGN1cnJlbnRQcmljZQkBDGdldFNwb3RQcmljZQAJAGYCCQEEZGl2ZAIJAQNhYnMBCQBlAgULb3JhY2xlUHJpY2UFDGN1cnJlbnRQcmljZQULb3JhY2xlUHJpY2UJAQtzcHJlYWRMaW1pdAABI2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sAgdfdHJhZGVyB19vcHRpb24EDSR0MDE3MjkzMTc0MjEJAQtnZXRQb3NpdGlvbgEFB190cmFkZXIEDHBvc2l0aW9uU2l6ZQgFDSR0MDE3MjkzMTc0MjECXzEEDnBvc2l0aW9uTWFyZ2luCAUNJHQwMTcyOTMxNzQyMQJfMgQUcG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDAxNzI5MzE3NDIxAl8zBBFwb3NpdGlvbkxzdFVwZENQRggFDSR0MDE3MjkzMTc0MjECXzQED3Bvc2l0aW9uU2l6ZUFicwkBA2FicwEFDHBvc2l0aW9uU2l6ZQMJAAACBQ9wb3NpdGlvblNpemVBYnMAAAkAAgECFUludmFsaWQgcG9zaXRpb24gc2l6ZQQHaXNTaG9ydAkAZgIAAAUMcG9zaXRpb25TaXplBBBwb3NpdGlvbk5vdGlvbmFsAwkAAAIFB19vcHRpb24FD1BOTF9PUFRJT05fU1BPVAQNJHQwMTc2NjgxNzc5MgkBCnN3YXBPdXRwdXQDCQEBIQEFB2lzU2hvcnQFD3Bvc2l0aW9uU2l6ZUFicwcEE291dFBvc2l0aW9uTm90aW9uYWwIBQ0kdDAxNzY2ODE3NzkyAl8xBAJ4MQgFDSR0MDE3NjY4MTc3OTICXzIEAngyCAUNJHQwMTc2NjgxNzc5MgJfMwQCeDMIBQ0kdDAxNzY2ODE3NzkyAl80BRNvdXRQb3NpdGlvbk5vdGlvbmFsCQEEbXVsZAIFD3Bvc2l0aW9uU2l6ZUFicwkBEmdldE9yYWNsZVR3YXBQcmljZQAEDXVucmVhbGl6ZWRQbmwDBQdpc1Nob3J0CQBlAgUUcG9zaXRpb25PcGVuTm90aW9uYWwFEHBvc2l0aW9uTm90aW9uYWwJAGUCBRBwb3NpdGlvbk5vdGlvbmFsBRRwb3NpdGlvbk9wZW5Ob3Rpb25hbAkAlAoCBRBwb3NpdGlvbk5vdGlvbmFsBQ11bnJlYWxpemVkUG5sAQ9jYWxjTWFyZ2luUmF0aW8DDV9yZW1haW5NYXJnaW4IX2JhZERlYnQRX3Bvc2l0aW9uTm90aW9uYWwJAQRkaXZkAgkAZQIFDV9yZW1haW5NYXJnaW4FCF9iYWREZWJ0BRFfcG9zaXRpb25Ob3Rpb25hbAEWZ2V0TWFyZ2luUmF0aW9CeU9wdGlvbgIHX3RyYWRlcgdfb3B0aW9uBA0kdDAxODQ2MjE4NTczCQELZ2V0UG9zaXRpb24BBQdfdHJhZGVyBAxwb3NpdGlvblNpemUIBQ0kdDAxODQ2MjE4NTczAl8xBA5wb3NpdGlvbk1hcmdpbggFDSR0MDE4NDYyMTg1NzMCXzIEA3BvbggFDSR0MDE4NDYyMTg1NzMCXzMEEXBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwMTg0NjIxODU3MwJfNAQNJHQwMTg1NzkxODY3MgkBI2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sAgUHX3RyYWRlcgUHX29wdGlvbgQQcG9zaXRpb25Ob3Rpb25hbAgFDSR0MDE4NTc5MTg2NzICXzEEDXVucmVhbGl6ZWRQbmwIBQ0kdDAxODU3OTE4NjcyAl8yBA0kdDAxODY3NzE4ODQzCQEiY2FsY1JlbWFpbk1hcmdpbldpdGhGdW5kaW5nUGF5bWVudAQFDHBvc2l0aW9uU2l6ZQUOcG9zaXRpb25NYXJnaW4FEXBvc2l0aW9uTHN0VXBkQ1BGBQ11bnJlYWxpemVkUG5sBAxyZW1haW5NYXJnaW4IBQ0kdDAxODY3NzE4ODQzAl8xBAdiYWREZWJ0CAUNJHQwMTg2NzcxODg0MwJfMgkBD2NhbGNNYXJnaW5SYXRpbwMFDHJlbWFpbk1hcmdpbgUHYmFkRGVidAUQcG9zaXRpb25Ob3Rpb25hbAEOZ2V0TWFyZ2luUmF0aW8BB190cmFkZXIJARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAgUHX3RyYWRlcgUPUE5MX09QVElPTl9TUE9UARVpbnRlcm5hbENsb3NlUG9zaXRpb24CB190cmFkZXIUX2NoZWNrTWF4UHJpY2VJbXBhY3QEDSR0MDE5MTkwMTkzMTgJAQtnZXRQb3NpdGlvbgEFB190cmFkZXIEDHBvc2l0aW9uU2l6ZQgFDSR0MDE5MTkwMTkzMTgCXzEEDnBvc2l0aW9uTWFyZ2luCAUNJHQwMTkxOTAxOTMxOAJfMgQUcG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDAxOTE5MDE5MzE4Al8zBBFwb3NpdGlvbkxzdFVwZENQRggFDSR0MDE5MTkwMTkzMTgCXzQEDSR0MDE5MzI0MTk0MTEJASNnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubAIFB190cmFkZXIFD1BOTF9PUFRJT05fU1BPVAQCeDEIBQ0kdDAxOTMyNDE5NDExAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwMTkzMjQxOTQxMQJfMgQNJHQwMTk0MTYxOTU4NAkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQxwb3NpdGlvblNpemUFDnBvc2l0aW9uTWFyZ2luBRFwb3NpdGlvbkxzdFVwZENQRgUNdW5yZWFsaXplZFBubAQMcmVtYWluTWFyZ2luCAUNJHQwMTk0MTYxOTU4NAJfMQQHYmFkRGVidAgFDSR0MDE5NDE2MTk1ODQCXzIEFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQkBAS0BBQxwb3NpdGlvblNpemUEC3JlYWxpemVkUG5sBQ11bnJlYWxpemVkUG5sBA1tYXJnaW5Ub1ZhdWx0CQEBLQEFDHJlbWFpbk1hcmdpbgQNJHQwMTk3MTEyMDAyMgkBCnN3YXBPdXRwdXQDCQBmAgUMcG9zaXRpb25TaXplAAAJAQNhYnMBBQxwb3NpdGlvblNpemUFFF9jaGVja01heFByaWNlSW1wYWN0BBlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50CAUNJHQwMTk3MTEyMDAyMgJfMQQWcXVvdGVBc3NldFJlc2VydmVBZnRlcggFDSR0MDE5NzExMjAwMjICXzIEFWJhc2VBc3NldFJlc2VydmVBZnRlcggFDSR0MDE5NzExMjAwMjICXzMEFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIIBQ0kdDAxOTcxMTIwMDIyAl80BBdjdW11bGF0aXZlTm90aW9uYWxBZnRlcggFDSR0MDE5NzExMjAwMjICXzUEDnRvdGFsTG9uZ0FmdGVyCAUNJHQwMTk3MTEyMDAyMgJfNgQPdG90YWxTaG9ydEFmdGVyCAUNJHQwMTk3MTEyMDAyMgJfNwQZb3BlbkludGVyZXN0Tm90aW9uYWxBZnRlcgkAZQIJARRvcGVuSW50ZXJlc3ROb3Rpb25hbAAFFHBvc2l0aW9uT3Blbk5vdGlvbmFsCQCeCgwFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQUHYmFkRGVidAULcmVhbGl6ZWRQbmwFDW1hcmdpblRvVmF1bHQFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIFGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIFGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQFDnRvdGFsTG9uZ0FmdGVyBQ90b3RhbFNob3J0QWZ0ZXIBEGdldFR3YXBTcG90UHJpY2UABAhtaW51dGVJZAkAaQIJAGkCCAUJbGFzdEJsb2NrCXRpbWVzdGFtcADoBwA8BA1zdGFydE1pbnV0ZUlkCQBlAgUIbWludXRlSWQFDVRXQVBfSU5URVJWQUwEB2xpc3RTdHIJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwUNa19sYXN0RGF0YVN0cgIABARsaXN0CQC1CQIFB2xpc3RTdHICASwKAQhmaWx0ZXJGbgILYWNjdW11bGF0b3IEbmV4dAMJAGcCBQ1zdGFydE1pbnV0ZUlkCQENcGFyc2VJbnRWYWx1ZQEFBG5leHQJAM0IAgULYWNjdW11bGF0b3IJAQ1wYXJzZUludFZhbHVlAQUEbmV4dAULYWNjdW11bGF0b3IEBWxpc3RGCgACJGwFBGxpc3QKAAIkcwkAkAMBBQIkbAoABSRhY2MwBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBCGZpbHRlckZuAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAyMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQECG1heEluZGV4AwkAZgIJAJADAQUFbGlzdEYAAAkAlgMBBQVsaXN0RgkBDXBhcnNlSW50VmFsdWUBCQCRAwIFBGxpc3QAAAQMbGFzdE1pbnV0ZUlkCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMFDmtfbGFzdE1pbnV0ZUlkAAAEFmVuZExhc3RDdW11bGF0aXZlUHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkArAICCQCsAgIFHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlAgFfCQCkAwEFDGxhc3RNaW51dGVJZAAABAxlbmRMYXN0UHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkArAICCQCsAgIFE2tfdHdhcERhdGFMYXN0UHJpY2UCAV8JAKQDAQUMbGFzdE1pbnV0ZUlkAAAEEm5vd0N1bXVsYXRpdmVQcmljZQkAZAIFFmVuZExhc3RDdW11bGF0aXZlUHJpY2UJAGgCCQBlAgUIbWludXRlSWQFDGxhc3RNaW51dGVJZAUMZW5kTGFzdFByaWNlBBhzdGFydExhc3RDdW11bGF0aXZlUHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkArAICCQCsAgIFHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlAgFfCQCkAwEFCG1heEluZGV4AAAEDnN0YXJ0TGFzdFByaWNlCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAKwCAgkArAICBRNrX3R3YXBEYXRhTGFzdFByaWNlAgFfCQCkAwEFCG1heEluZGV4AAAEFHN0YXJ0Q3VtdWxhdGl2ZVByaWNlCQBkAgUYc3RhcnRMYXN0Q3VtdWxhdGl2ZVByaWNlCQBoAgkAZQIFDXN0YXJ0TWludXRlSWQFCG1heEluZGV4BQ5zdGFydExhc3RQcmljZQkAaQIJAGUCBRJub3dDdW11bGF0aXZlUHJpY2UFFHN0YXJ0Q3VtdWxhdGl2ZVByaWNlBQ1UV0FQX0lOVEVSVkFMAQ51cGRhdGVTZXR0aW5ncwcQX2luaXRNYXJnaW5SYXRpbwRfbW1yFF9saXF1aWRhdGlvbkZlZVJhdGlvDl9mdW5kaW5nUGVyaW9kBF9mZWUMX3NwcmVhZExpbWl0D19tYXhQcmljZUltcGFjdAkAzAgCCQEMSW50ZWdlckVudHJ5AgURa19pbml0TWFyZ2luUmF0aW8FEF9pbml0TWFyZ2luUmF0aW8JAMwIAgkBDEludGVnZXJFbnRyeQIFGGtfbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwUEX21tcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUVa19saXF1aWRhdGlvbkZlZVJhdGlvBRRfbGlxdWlkYXRpb25GZWVSYXRpbwkAzAgCCQEMSW50ZWdlckVudHJ5AgUPa19mdW5kaW5nUGVyaW9kBQ5fZnVuZGluZ1BlcmlvZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUFa19mZWUFBF9mZWUJAMwIAgkBDEludGVnZXJFbnRyeQIFDWtfc3ByZWFkTGltaXQFDF9zcHJlYWRMaW1pdAkAzAgCCQEMSW50ZWdlckVudHJ5AgUQa19tYXhQcmljZUltcGFjdAUPX21heFByaWNlSW1wYWN0BQNuaWwBDXVwZGF0ZUZ1bmRpbmcFEV9uZXh0RnVuZGluZ0Jsb2NrJF9sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbiVfbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uEF9sb25nRnVuZGluZ1JhdGURX3Nob3J0RnVuZGluZ1JhdGUJAMwIAgkBDEludGVnZXJFbnRyeQIFEmtfbmV4dEZ1bmRpbmdCbG9jawURX25leHRGdW5kaW5nQmxvY2sJAMwIAgkBDEludGVnZXJFbnRyeQIFJWtfbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FJF9sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgkAzAgCCQEMSW50ZWdlckVudHJ5AgUma19sYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FJV9sYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24JAMwIAgkBDEludGVnZXJFbnRyeQIFEWtfbG9uZ0Z1bmRpbmdSYXRlBRBfbG9uZ0Z1bmRpbmdSYXRlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRJrX3Nob3J0RnVuZGluZ1JhdGUFEV9zaG9ydEZ1bmRpbmdSYXRlBQNuaWwBDnVwZGF0ZVBvc2l0aW9uBQhfYWRkcmVzcwVfc2l6ZQdfbWFyZ2luDV9vcGVuTm90aW9uYWwgX2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24JAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFDmtfcG9zaXRpb25TaXplBQhfYWRkcmVzcwUFX3NpemUJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFEGtfcG9zaXRpb25NYXJnaW4FCF9hZGRyZXNzBQdfbWFyZ2luCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsBQhfYWRkcmVzcwUNX29wZW5Ob3Rpb25hbAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUua19wb3NpdGlvbkxhc3RVcGRhdGVkQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgUIX2FkZHJlc3MFIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBQNuaWwBCmFwcGVuZFR3YXABBXByaWNlBAhtaW51dGVJZAkAaQIJAGkCCAUJbGFzdEJsb2NrCXRpbWVzdGFtcADoBwA8BBBwcmV2aW91c01pbnV0ZUlkCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMFDmtfbGFzdE1pbnV0ZUlkAAADCQBmAgUQcHJldmlvdXNNaW51dGVJZAUIbWludXRlSWQJAAIBAhFUV0FQIG91dC1vZi1vcmRlcgQMbGFzdE1pbnV0ZUlkAwkAAAIFEHByZXZpb3VzTWludXRlSWQAAAUIbWludXRlSWQFEHByZXZpb3VzTWludXRlSWQDCQBmAgUIbWludXRlSWQFEHByZXZpb3VzTWludXRlSWQEE3ByZXZDdW11bGF0aXZlUHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkArAICCQCsAgIFHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlAgFfCQCkAwEFEHByZXZpb3VzTWludXRlSWQAAAQJcHJldlByaWNlCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAKwCAgkArAICBRNrX3R3YXBEYXRhTGFzdFByaWNlAgFfCQCkAwEFEHByZXZpb3VzTWludXRlSWQFBXByaWNlBBNsYXN0Q3VtdWxhdGl2ZVByaWNlCQBkAgUTcHJldkN1bXVsYXRpdmVQcmljZQkAaAIJAGUCBQhtaW51dGVJZAUMbGFzdE1pbnV0ZUlkBQlwcmV2UHJpY2UEBGxpc3QJAQtwdXNoVG9RdWV1ZQMJAQlzdHJUb0xpc3QBCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoaXMFDWtfbGFzdERhdGFTdHICAAUNVFdBUF9JTlRFUlZBTAkApAMBBQhtaW51dGVJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUda190d2FwRGF0YUxhc3RDdW11bGF0aXZlUHJpY2UJAKQDAQUIbWludXRlSWQFE2xhc3RDdW11bGF0aXZlUHJpY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFE2tfdHdhcERhdGFMYXN0UHJpY2UJAKQDAQUIbWludXRlSWQFBXByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRprX3R3YXBEYXRhUHJldmlvdXNNaW51dGVJZAkApAMBBQhtaW51dGVJZAUQcHJldmlvdXNNaW51dGVJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUOa19sYXN0TWludXRlSWQFCG1pbnV0ZUlkCQDMCAIJAQtTdHJpbmdFbnRyeQIFDWtfbGFzdERhdGFTdHIJAQlsaXN0VG9TdHIBBQRsaXN0BQNuaWwEGHR3YXBEYXRhUHJldmlvdXNNaW51dGVJZAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBRprX3R3YXBEYXRhUHJldmlvdXNNaW51dGVJZAkApAMBBQhtaW51dGVJZAAABBNwcmV2Q3VtdWxhdGl2ZVByaWNlCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlCQCkAwEFGHR3YXBEYXRhUHJldmlvdXNNaW51dGVJZAAABAlwcmV2UHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUTa190d2FwRGF0YUxhc3RQcmljZQkApAMBBRh0d2FwRGF0YVByZXZpb3VzTWludXRlSWQFBXByaWNlBBNsYXN0Q3VtdWxhdGl2ZVByaWNlCQBkAgUTcHJldkN1bXVsYXRpdmVQcmljZQkAaAIJAGUCBQhtaW51dGVJZAUYdHdhcERhdGFQcmV2aW91c01pbnV0ZUlkBQlwcmV2UHJpY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlCQCkAwEFCG1pbnV0ZUlkBRNsYXN0Q3VtdWxhdGl2ZVByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRNrX3R3YXBEYXRhTGFzdFByaWNlCQCkAwEFCG1pbnV0ZUlkBQVwcmljZQUDbmlsARF1cGRhdGVBbW1SZXNlcnZlcwIHX3F0QXN0UgdfYnNBc3RSCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRNrX3F1b3RlQXNzZXRSZXNlcnZlBQdfcXRBc3RSCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRJrX2Jhc2VBc3NldFJlc2VydmUFB19ic0FzdFIFA25pbAEJdXBkYXRlQW1tBwdfcXRBc3RSB19ic0FzdFIXX3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIYX2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyFV9vcGVuSW50ZXJlc3ROb3Rpb25hbBZfdG90YWxMb25nUG9zaXRpb25TaXplF190b3RhbFNob3J0UG9zaXRpb25TaXplAwkBAiE9AgkAZQIFFl90b3RhbExvbmdQb3NpdGlvblNpemUFF190b3RhbFNob3J0UG9zaXRpb25TaXplBRdfdG90YWxQb3NpdGlvblNpemVBZnRlcgkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAhhJbnZhbGlkIEFNTSBzdGF0ZSBkYXRhOiAJAKQDAQUWX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQIEICsgIAkApAMBBRdfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQIEICE9IAkApAMBBRdfdG90YWxQb3NpdGlvblNpemVBZnRlcgkAzggCCQDOCAIJARF1cGRhdGVBbW1SZXNlcnZlcwIFB19xdEFzdFIFB19ic0FzdFIJAMwIAgkBDEludGVnZXJFbnRyeQIFE2tfdG90YWxQb3NpdGlvblNpemUFF190b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRRrX2N1bXVsYXRpdmVOb3Rpb25hbAUYX2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRZrX29wZW5JbnRlcmVzdE5vdGlvbmFsBRVfb3BlbkludGVyZXN0Tm90aW9uYWwJAMwIAgkBDEludGVnZXJFbnRyeQIFF2tfdG90YWxMb25nUG9zaXRpb25TaXplBRZfdG90YWxMb25nUG9zaXRpb25TaXplCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRhrX3RvdGFsU2hvcnRQb3NpdGlvblNpemUFF190b3RhbFNob3J0UG9zaXRpb25TaXplBQNuaWwJAQphcHBlbmRUd2FwAQkBBGRpdmQCBQdfcXRBc3RSBQdfYnNBc3RSAQ5kZWxldGVQb3NpdGlvbgEIX2FkZHJlc3MJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBDnRvQ29tcG9zaXRlS2V5AgUOa19wb3NpdGlvblNpemUFCF9hZGRyZXNzCQDMCAIJAQtEZWxldGVFbnRyeQEJAQ50b0NvbXBvc2l0ZUtleQIFEGtfcG9zaXRpb25NYXJnaW4FCF9hZGRyZXNzCQDMCAIJAQtEZWxldGVFbnRyeQEJAQ50b0NvbXBvc2l0ZUtleQIFFmtfcG9zaXRpb25PcGVuTm90aW9uYWwFCF9hZGRyZXNzCQDMCAIJAQtEZWxldGVFbnRyeQEJAQ50b0NvbXBvc2l0ZUtleQIFLmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FCF9hZGRyZXNzBQNuaWwBCHdpdGhkcmF3AghfYWRkcmVzcwdfYW1vdW50BAdiYWxhbmNlCQDwBwIFBHRoaXMJAQpxdW90ZUFzc2V0AAMJAGYCBQdfYW1vdW50BQdiYWxhbmNlCQACAQkArAICCQCsAgIJAKwCAgITVW5hYmxlIHRvIHdpdGhkcmF3IAkApAMBBQdfYW1vdW50AhcgZnJvbSBjb250cmFjdCBiYWxhbmNlIAkApAMBBQdiYWxhbmNlCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMFCF9hZGRyZXNzBQdfYW1vdW50CQEKcXVvdGVBc3NldAAFA25pbAENdXBkYXRlQmFsYW5jZQEBaQMJAGYCAAAFAWkJAAIBAgdCYWxhbmNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQlrX2JhbGFuY2UFAWkFA25pbAELdHJhbnNmZXJGZWUBAWkJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwkBDnN0YWtpbmdBZGRyZXNzAAUBaQkBCnF1b3RlQXNzZXQABQNuaWwRAWkBBXBhdXNlAAMJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAJAAIBAhpJbnZhbGlkIHRvZ2dsZVBhdXNlIHBhcmFtcwkAzAgCCQEMQm9vbGVhbkVudHJ5AgUIa19wYXVzZWQGBQNuaWwBaQEHdW5wYXVzZQADCQECIT0CCAUBaQZjYWxsZXIJAQxhZG1pbkFkZHJlc3MACQACAQIaSW52YWxpZCB0b2dnbGVQYXVzZSBwYXJhbXMJAMwIAgkBDEJvb2xlYW5FbnRyeQIFCGtfcGF1c2VkBwUDbmlsAWkBDGFkZExpcXVpZGl0eQERX3F1b3RlQXNzZXRBbW91bnQDAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAYJAGcCAAAFEV9xdW90ZUFzc2V0QW1vdW50CQACAQIbSW52YWxpZCBhZGRMaXF1aWRpdHkgcGFyYW1zBAdfcXRBc3RSCQEGcXRBc3RSAAQHX2JzQXN0UgkBBmJzQXN0UgAEBXByaWNlCQEEZGl2ZAIFB19xdEFzdFIFB19ic0FzdFIEFGJhc2VBc3NldEFtb3VudFRvQWRkCQEEZGl2ZAIFEV9xdW90ZUFzc2V0QW1vdW50BQVwcmljZQQLcXRBc3RSQWZ0ZXIJAGQCBQdfcXRBc3RSBRFfcXVvdGVBc3NldEFtb3VudAQLYnNBc3RSQWZ0ZXIJAGQCBQdfYnNBc3RSBRRiYXNlQXNzZXRBbW91bnRUb0FkZAkBEXVwZGF0ZUFtbVJlc2VydmVzAgULcXRBc3RSQWZ0ZXIFC2JzQXN0UkFmdGVyAWkBD3JlbW92ZUxpcXVpZGl0eQERX3F1b3RlQXNzZXRBbW91bnQDAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAYJAGcCAAAFEV9xdW90ZUFzc2V0QW1vdW50CQACAQIeSW52YWxpZCByZW1vdmVMaXF1aWRpdHkgcGFyYW1zBAdfcXRBc3RSCQEGcXRBc3RSAAQHX2JzQXN0UgkBBmJzQXN0UgAEBXByaWNlCQEEZGl2ZAIFB19xdEFzdFIFB19ic0FzdFIEF2Jhc2VBc3NldEFtb3VudFRvUmVtb3ZlCQEEZGl2ZAIFEV9xdW90ZUFzc2V0QW1vdW50BQVwcmljZQQLcXRBc3RSQWZ0ZXIJAGUCBQdfcXRBc3RSBRFfcXVvdGVBc3NldEFtb3VudAQLYnNBc3RSQWZ0ZXIJAGUCBQdfYnNBc3RSBRdiYXNlQXNzZXRBbW91bnRUb1JlbW92ZQkBEXVwZGF0ZUFtbVJlc2VydmVzAgULcXRBc3RSQWZ0ZXIFC2JzQXN0UkFmdGVyAWkBDmNoYW5nZVNldHRpbmdzBxBfaW5pdE1hcmdpblJhdGlvBF9tbXIUX2xpcXVpZGF0aW9uRmVlUmF0aW8OX2Z1bmRpbmdQZXJpb2QEX2ZlZQxfc3ByZWFkTGltaXQPX21heFByaWNlSW1wYWN0AwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAkAAgECHUludmFsaWQgY2hhbmdlU2V0dGluZ3MgcGFyYW1zCQEOdXBkYXRlU2V0dGluZ3MHBRBfaW5pdE1hcmdpblJhdGlvBQRfbW1yBRRfbGlxdWlkYXRpb25GZWVSYXRpbwUOX2Z1bmRpbmdQZXJpb2QFBF9mZWUFDF9zcHJlYWRMaW1pdAUPX21heFByaWNlSW1wYWN0AWkBCmluaXRpYWxpemUMB19xdEFzdFIHX2JzQXN0Ug5fZnVuZGluZ1BlcmlvZBBfaW5pdE1hcmdpblJhdGlvBF9tbXIUX2xpcXVpZGF0aW9uRmVlUmF0aW8EX2ZlZQdfb3JhY2xlCl9vcmFjbGVLZXkMX2Nvb3JkaW5hdG9yDF9zcHJlYWRMaW1pdA9fbWF4UHJpY2VJbXBhY3QDAwMDAwMDAwMDCQBnAgAABQdfcXRBc3RSBgkAZwIAAAUHX2JzQXN0UgYJAGcCAAAFDl9mdW5kaW5nUGVyaW9kBgkAZwIAAAUQX2luaXRNYXJnaW5SYXRpbwYJAGcCAAAFBF9tbXIGCQBnAgAABRRfbGlxdWlkYXRpb25GZWVSYXRpbwYJAGcCAAAFBF9mZWUGCQBnAgAABQxfc3ByZWFkTGltaXQGCQBnAgAABQ9fbWF4UHJpY2VJbXBhY3QGCQELaW5pdGlhbGl6ZWQACQACAQIdSW52YWxpZCBpbml0aWFsaXplIHBhcmFtZXRlcnMJAM4IAgkAzggCCQDOCAIJAM4IAgkBCXVwZGF0ZUFtbQcFB19xdEFzdFIFB19ic0FzdFIAAAAAAAAAAAAACQEOdXBkYXRlU2V0dGluZ3MHBRBfaW5pdE1hcmdpblJhdGlvBQRfbW1yBRRfbGlxdWlkYXRpb25GZWVSYXRpbwUOX2Z1bmRpbmdQZXJpb2QFBF9mZWUFDF9zcHJlYWRMaW1pdAUPX21heFByaWNlSW1wYWN0CQENdXBkYXRlRnVuZGluZwUJAGQCCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAUOX2Z1bmRpbmdQZXJpb2QAAAAAAAAAAAkBDXVwZGF0ZUJhbGFuY2UBAAAJAMwIAgkBDEJvb2xlYW5FbnRyeQIFDWtfaW5pdGlhbGl6ZWQGCQDMCAIJAQtTdHJpbmdFbnRyeQIFBWtfb3JhBQdfb3JhY2xlCQDMCAIJAQtTdHJpbmdFbnRyeQIFCWtfb3JhX2tleQUKX29yYWNsZUtleQkAzAgCCQELU3RyaW5nRW50cnkCBRRrX2Nvb3JkaW5hdG9yQWRkcmVzcwUMX2Nvb3JkaW5hdG9yBQNuaWwBaQESc2V0SW5pdE1hcmdpblJhdGlvARBfaW5pdE1hcmdpblJhdGlvAwMJAGcCAAAFEF9pbml0TWFyZ2luUmF0aW8GCQEBIQEJAQtpbml0aWFsaXplZAAJAAIBAiVJbnZhbGlkIHNldEluaXRNYXJnaW5SYXRpbyBwYXJhbWV0ZXJzCQEOdXBkYXRlU2V0dGluZ3MHBRBfaW5pdE1hcmdpblJhdGlvCQEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAJARNsaXF1aWRhdGlvbkZlZVJhdGlvAAkBEGZ1bmRpbmdQZXJpb2RSYXcACQEDZmVlAAkBC3NwcmVhZExpbWl0AAkBDm1heFByaWNlSW1wYWN0AAFpARBkZWNyZWFzZVBvc2l0aW9uAwdfYW1vdW50CV9sZXZlcmFnZRNfbWluQmFzZUFzc2V0QW1vdW50AwMDAwMJAGcCAAAFB19hbW91bnQGCQEBIQEJAQtpbml0aWFsaXplZAAGCQEBIQEJARZyZXF1aXJlTW9yZU1hcmdpblJhdGlvAwkBBGRpdmQCBQxERUNJTUFMX1VOSVQFCV9sZXZlcmFnZQkBD2luaXRNYXJnaW5SYXRpbwAGBgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBgkBBnBhdXNlZAAJAAIBAiNJbnZhbGlkIGRlY3JlYXNlUG9zaXRpb24gcGFyYW1ldGVycwQNJHQwMzEzMzQzMTQ4NgkBC2dldFBvc2l0aW9uAQkApQgBCAUBaQZjYWxsZXIED29sZFBvc2l0aW9uU2l6ZQgFDSR0MDMxMzM0MzE0ODYCXzEEEW9sZFBvc2l0aW9uTWFyZ2luCAUNJHQwMzEzMzQzMTQ4NgJfMgQXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDAzMTMzNDMxNDg2Al8zBBRvbGRQb3NpdGlvbkxzdFVwZENQRggFDSR0MDMxMzM0MzE0ODYCXzQECl9kaXJlY3Rpb24DCQBmAgUPb2xkUG9zaXRpb25TaXplAAAFCURJUl9TSE9SVAUIRElSX0xPTkcEBWlzQWRkCQAAAgUKX2RpcmVjdGlvbgUIRElSX0xPTkcEDG9wZW5Ob3Rpb25hbAkBBG11bGQCBQdfYW1vdW50BQlfbGV2ZXJhZ2UEDSR0MDMxNjU5MzE3NzUJASNnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubAIJAKUIAQgFAWkGY2FsbGVyBQ9QTkxfT1BUSU9OX1NQT1QEE29sZFBvc2l0aW9uTm90aW9uYWwIBQ0kdDAzMTY1OTMxNzc1Al8xBA11bnJlYWxpemVkUG5sCAUNJHQwMzE2NTkzMTc3NQJfMgQNJHQwMzE3ODEzNDMzMAMJAGYCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBQxvcGVuTm90aW9uYWwEDSR0MDMyMTU4MzIzNzcJAQlzd2FwSW5wdXQCBQVpc0FkZAUMb3Blbk5vdGlvbmFsBBVleGNoYW5nZWRQb3NpdGlvblNpemUIBQ0kdDAzMjE1ODMyMzc3Al8xBBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMzIxNTgzMjM3NwJfMgQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMzIxNTgzMjM3NwJfMwQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDMyMTU4MzIzNzcCXzQEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwMzIxNTgzMjM3NwJfNQQYZXhjaGFuZ2VkUG9zaXRpb25TaXplQWJzCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplAwMJAQIhPQIFE19taW5CYXNlQXNzZXRBbW91bnQAAAkAZgIFE19taW5CYXNlQXNzZXRBbW91bnQFGGV4Y2hhbmdlZFBvc2l0aW9uU2l6ZUFicwcJAAIBCQCsAgIJAKwCAgkArAICAiVUb28gbGl0dGxlIGJhc2UgYXNzZXQgZXhjaGFuZ2VkLCBnb3QgCQCkAwEFGGV4Y2hhbmdlZFBvc2l0aW9uU2l6ZUFicwIKIGV4cGVjdGVkIAkApAMBBRNfbWluQmFzZUFzc2V0QW1vdW50BAtyZWFsaXplZFBubAkBBGRpdmQCCQEEbXVsZAIFDXVucmVhbGl6ZWRQbmwFGGV4Y2hhbmdlZFBvc2l0aW9uU2l6ZUFicwkBA2FicwEFD29sZFBvc2l0aW9uU2l6ZQQNJHQwMzI4MTQzMzA1OQkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQ9vbGRQb3NpdGlvblNpemUFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvbkxzdFVwZENQRgULcmVhbGl6ZWRQbmwEDHJlbWFpbk1hcmdpbggFDSR0MDMyODE0MzMwNTkCXzEEB2JhZERlYnQIBQ0kdDAzMjgxNDMzMDU5Al8yBA5mdW5kaW5nUGF5bWVudAgFDSR0MDMyODE0MzMwNTkCXzMEGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQFDG9wZW5Ob3Rpb25hbAQSdW5yZWFsaXplZFBubEFmdGVyCQBlAgUNdW5yZWFsaXplZFBubAULcmVhbGl6ZWRQbmwEEnJlbWFpbk9wZW5Ob3Rpb25hbAMJAGYCBQ9vbGRQb3NpdGlvblNpemUAAAkAZQIJAGUCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BRJ1bnJlYWxpemVkUG5sQWZ0ZXIJAGUCCQBkAgUSdW5yZWFsaXplZFBubEFmdGVyBRNvbGRQb3NpdGlvbk5vdGlvbmFsBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BA9uZXdQb3NpdGlvblNpemUJAGQCBQ9vbGRQb3NpdGlvblNpemUFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQkAnQoLBQ9uZXdQb3NpdGlvblNpemUFDHJlbWFpbk1hcmdpbgkBA2FicwEFEnJlbWFpbk9wZW5Ob3Rpb25hbAkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBQ9uZXdQb3NpdGlvblNpemUFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIJAGUCCQEUb3BlbkludGVyZXN0Tm90aW9uYWwABQxvcGVuTm90aW9uYWwJAGUCCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAMJAGYCBQ9uZXdQb3NpdGlvblNpemUAAAkBA2FicwEFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQAACQBlAgkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUAAwkAZgIAAAUPbmV3UG9zaXRpb25TaXplCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplAAAJAAIBAhRDbG9zZSBwb3NpdGlvbiBmaXJzdAQPbmV3UG9zaXRpb25TaXplCAUNJHQwMzE3ODEzNDMzMAJfMQQXbmV3UG9zaXRpb25SZW1haW5NYXJnaW4IBQ0kdDAzMTc4MTM0MzMwAl8yBBduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDMxNzgxMzQzMzACXzMEFG5ld1Bvc2l0aW9uTGF0ZXN0Q1BGCAUNJHQwMzE3ODEzNDMzMAJfNAQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMzE3ODEzNDMzMAJfNQQWcXVvdGVBc3NldFJlc2VydmVBZnRlcggFDSR0MDMxNzgxMzQzMzACXzYEFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIIBQ0kdDAzMTc4MTM0MzMwAl83BBdjdW11bGF0aXZlTm90aW9uYWxBZnRlcggFDSR0MDMxNzgxMzQzMzACXzgEGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIIBQ0kdDAzMTc4MTM0MzMwAl85BA50b3RhbExvbmdBZnRlcggFDSR0MDMxNzgxMzQzMzADXzEwBA90b3RhbFNob3J0QWZ0ZXIIBQ0kdDAzMTc4MTM0MzMwA18xMQQObm90aWZ5Tm90aW9uYWwJAPwHBAkBDG1pbmVyQWRkcmVzcwACDm5vdGlmeU5vdGlvbmFsCQDMCAIJAKUIAQgFAWkGY2FsbGVyCQDMCAIFF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBQNuaWwFA25pbAMJAAACBQ5ub3RpZnlOb3Rpb25hbAUObm90aWZ5Tm90aW9uYWwJAM4IAgkBDnVwZGF0ZVBvc2l0aW9uBQkApQgBCAUBaQZjYWxsZXIFD25ld1Bvc2l0aW9uU2l6ZQUXbmV3UG9zaXRpb25SZW1haW5NYXJnaW4FF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBRRuZXdQb3NpdGlvbkxhdGVzdENQRgkBCXVwZGF0ZUFtbQcFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIFGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIFDnRvdGFsTG9uZ0FmdGVyBQ90b3RhbFNob3J0QWZ0ZXIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEQaW5jcmVhc2VQb3NpdGlvbgMKX2RpcmVjdGlvbglfbGV2ZXJhZ2UTX21pbkJhc2VBc3NldEFtb3VudAQKX3Jhd0Ftb3VudAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQDAwMDAwMDCQECIT0CBQpfZGlyZWN0aW9uBQhESVJfTE9ORwkBAiE9AgUKX2RpcmVjdGlvbgUJRElSX1NIT1JUBwYJAGcCAAAFCl9yYXdBbW91bnQGCQEBIQEJAQtpbml0aWFsaXplZAAGCQECIT0CCAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQJAQpxdW90ZUFzc2V0AAYJAQEhAQkBFnJlcXVpcmVNb3JlTWFyZ2luUmF0aW8DCQEEZGl2ZAIFDERFQ0lNQUxfVU5JVAUJX2xldmVyYWdlCQEPaW5pdE1hcmdpblJhdGlvAAYGCQEGcGF1c2VkAAkAAgECI0ludmFsaWQgaW5jcmVhc2VQb3NpdGlvbiBwYXJhbWV0ZXJzBAlmZWVBbW91bnQJAQRtdWxkAgUKX3Jhd0Ftb3VudAkBA2ZlZQAEB19hbW91bnQJAGUCBQpfcmF3QW1vdW50BQlmZWVBbW91bnQEDSR0MDM1NDQ4MzU2MDAJAQtnZXRQb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBA9vbGRQb3NpdGlvblNpemUIBQ0kdDAzNTQ0ODM1NjAwAl8xBBFvbGRQb3NpdGlvbk1hcmdpbggFDSR0MDM1NDQ4MzU2MDACXzIEF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwMzU0NDgzNTYwMAJfMwQUb2xkUG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDAzNTQ0ODM1NjAwAl80BA1pc05ld1Bvc2l0aW9uCQAAAgUPb2xkUG9zaXRpb25TaXplAAAED2lzU2FtZURpcmVjdGlvbgMJAGYCBQ9vbGRQb3NpdGlvblNpemUAAAkAAAIFCl9kaXJlY3Rpb24FCERJUl9MT05HCQAAAgUKX2RpcmVjdGlvbgUJRElSX1NIT1JUBA5leHBhbmRFeGlzdGluZwMJAQEhAQUNaXNOZXdQb3NpdGlvbgUPaXNTYW1lRGlyZWN0aW9uBwQFaXNBZGQJAAACBQpfZGlyZWN0aW9uBQhESVJfTE9ORwQNJHQwMzU4ODkzODI3OAMDBQ1pc05ld1Bvc2l0aW9uBgUOZXhwYW5kRXhpc3RpbmcEDG9wZW5Ob3Rpb25hbAkBBG11bGQCBQdfYW1vdW50BQlfbGV2ZXJhZ2UEDSR0MDM2MzEzMzY1MTkJAQlzd2FwSW5wdXQCBQVpc0FkZAUMb3Blbk5vdGlvbmFsBBVhbW91bnRCYXNlQXNzZXRCb3VnaHQIBQ0kdDAzNjMxMzM2NTE5Al8xBBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMzYzMTMzNjUxOQJfMgQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMzYzMTMzNjUxOQJfMwQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDM2MzEzMzY1MTkCXzQEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwMzYzMTMzNjUxOQJfNQMDCQECIT0CBRNfbWluQmFzZUFzc2V0QW1vdW50AAAJAGYCBRNfbWluQmFzZUFzc2V0QW1vdW50CQEDYWJzAQUVYW1vdW50QmFzZUFzc2V0Qm91Z2h0BwkAAgEJAKwCAgkArAICCQCsAgICDUxpbWl0IGVycm9yOiAJAKQDAQkBA2FicwEFFWFtb3VudEJhc2VBc3NldEJvdWdodAIDIDwgCQCkAwEFE19taW5CYXNlQXNzZXRBbW91bnQED25ld1Bvc2l0aW9uU2l6ZQkAZAIFD29sZFBvc2l0aW9uU2l6ZQUVYW1vdW50QmFzZUFzc2V0Qm91Z2h0BBlpbmNyZWFzZU1hcmdpblJlcXVpcmVtZW50CQEEZGl2ZAIFDG9wZW5Ob3Rpb25hbAUJX2xldmVyYWdlBA0kdDAzNjkwMDM3MTM5CQEiY2FsY1JlbWFpbk1hcmdpbldpdGhGdW5kaW5nUGF5bWVudAQFD29sZFBvc2l0aW9uU2l6ZQURb2xkUG9zaXRpb25NYXJnaW4FFG9sZFBvc2l0aW9uTHN0VXBkQ1BGBRlpbmNyZWFzZU1hcmdpblJlcXVpcmVtZW50BAxyZW1haW5NYXJnaW4IBQ0kdDAzNjkwMDM3MTM5Al8xBAJ4MQgFDSR0MDM2OTAwMzcxMzkCXzIEAngyCAUNJHQwMzY5MDAzNzEzOQJfMwkAnQoLBQ9uZXdQb3NpdGlvblNpemUFDHJlbWFpbk1hcmdpbgkAZAIFF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsBQxvcGVuTm90aW9uYWwJAR9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAQUPbmV3UG9zaXRpb25TaXplBRViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIFF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCQBkAgkBFG9wZW5JbnRlcmVzdE5vdGlvbmFsAAUMb3Blbk5vdGlvbmFsCQBkAgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQADCQBmAgUPbmV3UG9zaXRpb25TaXplAAAJAQNhYnMBBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQAAAkAZAIJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAMJAGYCAAAFD25ld1Bvc2l0aW9uU2l6ZQkBA2FicwEFFWFtb3VudEJhc2VBc3NldEJvdWdodAAABAxvcGVuTm90aW9uYWwJAQRtdWxkAgUHX2Ftb3VudAUJX2xldmVyYWdlBA0kdDAzNzk3MTM4MDg3CQEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwCCQClCAEIBQFpBmNhbGxlcgUPUE5MX09QVElPTl9TUE9UBBNvbGRQb3NpdGlvbk5vdGlvbmFsCAUNJHQwMzc5NzEzODA4NwJfMQQNdW5yZWFsaXplZFBubAgFDSR0MDM3OTcxMzgwODcCXzIDCQBmAgUTb2xkUG9zaXRpb25Ob3Rpb25hbAUMb3Blbk5vdGlvbmFsCQACAQIuVXNlIGRlY3JlYXNlUG9zaXRpb24gdG8gZGVjcmVhc2UgcG9zaXRpb24gc2l6ZQkAAgECFENsb3NlIHBvc2l0aW9uIGZpcnN0BA9uZXdQb3NpdGlvblNpemUIBQ0kdDAzNTg4OTM4Mjc4Al8xBBduZXdQb3NpdGlvblJlbWFpbk1hcmdpbggFDSR0MDM1ODg5MzgyNzgCXzIEF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwMzU4ODkzODI3OAJfMwQUbmV3UG9zaXRpb25MYXRlc3RDUEYIBQ0kdDAzNTg4OTM4Mjc4Al80BBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDAzNTg4OTM4Mjc4Al81BBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMzU4ODkzODI3OAJfNgQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDM1ODg5MzgyNzgCXzcEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwMzU4ODkzODI3OAJfOAQZb3BlbkludGVyZXN0Tm90aW9uYWxBZnRlcggFDSR0MDM1ODg5MzgyNzgCXzkEDnRvdGFsTG9uZ0FmdGVyCAUNJHQwMzU4ODkzODI3OANfMTAED3RvdGFsU2hvcnRBZnRlcggFDSR0MDM1ODg5MzgyNzgDXzExBAxmZWVUb1N0YWtlcnMJAGkCBQlmZWVBbW91bnQAAgQOZmVlVG9JbnN1cmFuY2UJAGUCBQlmZWVBbW91bnQFDGZlZVRvU3Rha2VycwQFc3Rha2UJAPwHBAkBEXF1b3RlQXNzZXRTdGFraW5nAAIObG9ja05ldXRyaW5vU1AJAMwIAgkApQgBCQEOc3Rha2luZ0FkZHJlc3MACQDMCAIFCEFMTF9GRUVTBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUHX2Ftb3VudAUDbmlsAwkAAAIFBXN0YWtlBQVzdGFrZQQQZGVwb3NpdEluc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQ5mZWVUb0luc3VyYW5jZQUDbmlsAwkAAAIFEGRlcG9zaXRJbnN1cmFuY2UFEGRlcG9zaXRJbnN1cmFuY2UECW5vdGlmeUZlZQkA/AcECQEMbWluZXJBZGRyZXNzAAIKbm90aWZ5RmVlcwkAzAgCCQClCAEIBQFpBmNhbGxlcgkAzAgCBQlmZWVBbW91bnQFA25pbAUDbmlsAwkAAAIFCW5vdGlmeUZlZQUJbm90aWZ5RmVlBA5ub3RpZnlOb3Rpb25hbAkA/AcECQEMbWluZXJBZGRyZXNzAAIObm90aWZ5Tm90aW9uYWwJAMwIAgkApQgBCAUBaQZjYWxsZXIJAMwIAgUXbmV3UG9zaXRpb25PcGVuTm90aW9uYWwFA25pbAUDbmlsAwkAAAIFDm5vdGlmeU5vdGlvbmFsBQ5ub3RpZnlOb3Rpb25hbAkAzggCCQDOCAIJAM4IAgkBDnVwZGF0ZVBvc2l0aW9uBQkApQgBCAUBaQZjYWxsZXIFD25ld1Bvc2l0aW9uU2l6ZQUXbmV3UG9zaXRpb25SZW1haW5NYXJnaW4FF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBRRuZXdQb3NpdGlvbkxhdGVzdENQRgkBCXVwZGF0ZUFtbQcFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIFGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIFDnRvdGFsTG9uZ0FmdGVyBQ90b3RhbFNob3J0QWZ0ZXIJAQt0cmFuc2ZlckZlZQEFDGZlZVRvU3Rha2VycwkBDXVwZGF0ZUJhbGFuY2UBCQBkAgkBCGNiYWxhbmNlAAUHX2Ftb3VudAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQlhZGRNYXJnaW4ABApfcmF3QW1vdW50CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAMDAwMJAQIhPQIICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAkBCnF1b3RlQXNzZXQABgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBgkBASEBCQELaW5pdGlhbGl6ZWQABgkBBnBhdXNlZAAJAAIBAhxJbnZhbGlkIGFkZE1hcmdpbiBwYXJhbWV0ZXJzBAlmZWVBbW91bnQJAQRtdWxkAgUKX3Jhd0Ftb3VudAkBA2ZlZQAEB19hbW91bnQJAGUCBQpfcmF3QW1vdW50BQlmZWVBbW91bnQEDSR0MDM5ODM0Mzk5ODYJAQtnZXRQb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBA9vbGRQb3NpdGlvblNpemUIBQ0kdDAzOTgzNDM5OTg2Al8xBBFvbGRQb3NpdGlvbk1hcmdpbggFDSR0MDM5ODM0Mzk5ODYCXzIEF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwMzk4MzQzOTk4NgJfMwQUb2xkUG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDAzOTgzNDM5OTg2Al80BAxmZWVUb1N0YWtlcnMJAGkCBQlmZWVBbW91bnQAAgQOZmVlVG9JbnN1cmFuY2UJAGUCBQlmZWVBbW91bnQFDGZlZVRvU3Rha2VycwQFc3Rha2UJAPwHBAkBEXF1b3RlQXNzZXRTdGFraW5nAAIObG9ja05ldXRyaW5vU1AJAMwIAgkApQgBCQEOc3Rha2luZ0FkZHJlc3MACQDMCAIFCEFMTF9GRUVTBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUHX2Ftb3VudAUDbmlsAwkAAAIFBXN0YWtlBQVzdGFrZQQQZGVwb3NpdEluc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQ5mZWVUb0luc3VyYW5jZQUDbmlsAwkAAAIFEGRlcG9zaXRJbnN1cmFuY2UFEGRlcG9zaXRJbnN1cmFuY2UECW5vdGlmeUZlZQkA/AcECQEMbWluZXJBZGRyZXNzAAIKbm90aWZ5RmVlcwkAzAgCCQClCAEIBQFpBmNhbGxlcgkAzAgCBQlmZWVBbW91bnQFA25pbAUDbmlsAwkAAAIFCW5vdGlmeUZlZQUJbm90aWZ5RmVlCQDOCAIJAM4IAgkBDnVwZGF0ZVBvc2l0aW9uBQkApQgBCAUBaQZjYWxsZXIFD29sZFBvc2l0aW9uU2l6ZQkAZAIFEW9sZFBvc2l0aW9uTWFyZ2luBQdfYW1vdW50BRdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAUUb2xkUG9zaXRpb25Mc3RVcGRDUEYJAQt0cmFuc2ZlckZlZQEFDGZlZVRvU3Rha2VycwkBDXVwZGF0ZUJhbGFuY2UBCQBkAgkBCGNiYWxhbmNlAAUHX2Ftb3VudAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQxyZW1vdmVNYXJnaW4BB19hbW91bnQDAwMDCQBnAgAABQdfYW1vdW50BgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBgkBASEBCQELaW5pdGlhbGl6ZWQABgkBBnBhdXNlZAAJAAIBAh9JbnZhbGlkIHJlbW92ZU1hcmdpbiBwYXJhbWV0ZXJzBA0kdDA0MTAzNTQxMTg3CQELZ2V0UG9zaXRpb24BCQClCAEIBQFpBmNhbGxlcgQPb2xkUG9zaXRpb25TaXplCAUNJHQwNDEwMzU0MTE4NwJfMQQRb2xkUG9zaXRpb25NYXJnaW4IBQ0kdDA0MTAzNTQxMTg3Al8yBBdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDQxMDM1NDExODcCXzMEFG9sZFBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwNDEwMzU0MTE4NwJfNAQLbWFyZ2luRGVsdGEJAQEtAQUHX2Ftb3VudAQNJHQwNDEyMjQ0MTQwMwkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQ9vbGRQb3NpdGlvblNpemUFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvbkxzdFVwZENQRgULbWFyZ2luRGVsdGEEDHJlbWFpbk1hcmdpbggFDSR0MDQxMjI0NDE0MDMCXzEEB2JhZERlYnQIBQ0kdDA0MTIyNDQxNDAzAl8yAwkBAiE9AgUHYmFkRGVidAAACQACAQIdSW52YWxpZCByZW1vdmVkIG1hcmdpbiBhbW91bnQEC21hcmdpblJhdGlvCQEPY2FsY01hcmdpblJhdGlvAwUMcmVtYWluTWFyZ2luBQdiYWREZWJ0BRdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAMJAQEhAQkBFnJlcXVpcmVNb3JlTWFyZ2luUmF0aW8DBQttYXJnaW5SYXRpbwkBD2luaXRNYXJnaW5SYXRpbwAGCQACAQkArAICCQCsAgIJAKwCAgIZVG9vIG11Y2ggbWFyZ2luIHJlbW92ZWQ6IAkApAMBBQttYXJnaW5SYXRpbwIDIDwgCQCkAwEJAQ9pbml0TWFyZ2luUmF0aW8ABAd1bnN0YWtlCQD8BwQJARFxdW90ZUFzc2V0U3Rha2luZwACDnVubG9ja05ldXRyaW5vCQDMCAIFB19hbW91bnQJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UJAM4IAgkAzggCCQEOdXBkYXRlUG9zaXRpb24FCQClCAEIBQFpBmNhbGxlcgUPb2xkUG9zaXRpb25TaXplBQxyZW1haW5NYXJnaW4FF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsCQEfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgEFD29sZFBvc2l0aW9uU2l6ZQkBCHdpdGhkcmF3AggFAWkGY2FsbGVyBQdfYW1vdW50CQENdXBkYXRlQmFsYW5jZQEJAGUCCQEIY2JhbGFuY2UABQdfYW1vdW50CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBDWNsb3NlUG9zaXRpb24AAwMDCQEBIQEJARNyZXF1aXJlT3BlblBvc2l0aW9uAQkApQgBCAUBaQZjYWxsZXIGCQEBIQEJAQtpbml0aWFsaXplZAAGCQEGcGF1c2VkAAkAAgECIEludmFsaWQgY2xvc2VQb3NpdGlvbiBwYXJhbWV0ZXJzBA0kdDA0MjQ5OTQyODg5CQEVaW50ZXJuYWxDbG9zZVBvc2l0aW9uAgkApQgBCAUBaQZjYWxsZXIGBAJ4MQgFDSR0MDQyNDk5NDI4ODkCXzEED3Bvc2l0aW9uQmFkRGVidAgFDSR0MDQyNDk5NDI4ODkCXzIEC3JlYWxpemVkUG5sCAUNJHQwNDI0OTk0Mjg4OQJfMwQNbWFyZ2luVG9WYXVsdAgFDSR0MDQyNDk5NDI4ODkCXzQEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA0MjQ5OTQyODg5Al81BBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA0MjQ5OTQyODg5Al82BBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwNDI0OTk0Mjg4OQJfNwQXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIIBQ0kdDA0MjQ5OTQyODg5Al84BBlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCAUNJHQwNDI0OTk0Mjg4OQJfOQQCeDIIBQ0kdDA0MjQ5OTQyODg5A18xMAQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA0MjQ5OTQyODg5A18xMQQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNDI0OTk0Mjg4OQNfMTIDCQBmAgUPcG9zaXRpb25CYWREZWJ0AAAJAAIBAiZVbmFibGUgdG8gY2xvc2UgcG9zaXRpb24gd2l0aCBiYWQgZGVidAQOd2l0aGRyYXdBbW91bnQJAQNhYnMBBQ1tYXJnaW5Ub1ZhdWx0BAphbW1CYWxhbmNlCQBlAgkBCGNiYWxhbmNlAAUOd2l0aGRyYXdBbW91bnQEDSR0MDQzMDk4NDMyNDADCQBmAgAABQphbW1CYWxhbmNlCQCUCgIAAAkBA2FicwEFCmFtbUJhbGFuY2UJAJQKAgUKYW1tQmFsYW5jZQAABA1hbW1OZXdCYWxhbmNlCAUNJHQwNDMwOTg0MzI0MAJfMQQQZ2V0RnJvbUluc3VyYW5jZQgFDSR0MDQzMDk4NDMyNDACXzIEAXgDCQBmAgUQZ2V0RnJvbUluc3VyYW5jZQAABBF3aXRoZHJhd0luc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACCHdpdGhkcmF3CQDMCAIFEGdldEZyb21JbnN1cmFuY2UFA25pbAUDbmlsAwkAAAIFEXdpdGhkcmF3SW5zdXJhbmNlBRF3aXRoZHJhd0luc3VyYW5jZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUBeAUBeAQHdW5zdGFrZQkA/AcECQERcXVvdGVBc3NldFN0YWtpbmcAAg51bmxvY2tOZXV0cmlubwkAzAgCCQBlAgUOd2l0aGRyYXdBbW91bnQFEGdldEZyb21JbnN1cmFuY2UJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UEDm5vdGlmeU5vdGlvbmFsCQD8BwQJAQxtaW5lckFkZHJlc3MAAg5ub3RpZnlOb3Rpb25hbAkAzAgCCQClCAEIBQFpBmNhbGxlcgkAzAgCAAAFA25pbAUDbmlsAwkAAAIFDm5vdGlmeU5vdGlvbmFsBQ5ub3RpZnlOb3Rpb25hbAkAzggCCQDOCAIJAM4IAgkBDmRlbGV0ZVBvc2l0aW9uAQkApQgBCAUBaQZjYWxsZXIJAQl1cGRhdGVBbW0HBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBRViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIFFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIFF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyBRlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyBQ50b3RhbExvbmdBZnRlcgUPdG90YWxTaG9ydEFmdGVyCQEId2l0aGRyYXcCCAUBaQZjYWxsZXIFDndpdGhkcmF3QW1vdW50CQENdXBkYXRlQmFsYW5jZQEFDWFtbU5ld0JhbGFuY2UJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEJbGlxdWlkYXRlAQdfdHJhZGVyBA9zcG90TWFyZ2luUmF0aW8JARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAgUHX3RyYWRlcgUPUE5MX09QVElPTl9TUE9UBAttYXJnaW5SYXRpbwMJARZpc092ZXJGbHVjdHVhdGlvbkxpbWl0AAQRb3JhY2xlTWFyZ2luUmF0aW8JARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAgUHX3RyYWRlcgURUE5MX09QVElPTl9PUkFDTEUJAQR2bWF4AgUPc3BvdE1hcmdpblJhdGlvBRFvcmFjbGVNYXJnaW5SYXRpbwUPc3BvdE1hcmdpblJhdGlvAwMDAwkBASEBCQEWcmVxdWlyZU1vcmVNYXJnaW5SYXRpbwMFC21hcmdpblJhdGlvCQEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAHBgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEFB190cmFkZXIGCQEBIQEJAQtpbml0aWFsaXplZAAGCQEGcGF1c2VkAAkAAgECE1VuYWJsZSB0byBsaXF1aWRhdGUEDSR0MDQ0OTMwNDUzNTMJARVpbnRlcm5hbENsb3NlUG9zaXRpb24CBQdfdHJhZGVyBwQCeDEIBQ0kdDA0NDkzMDQ1MzUzAl8xBAdiYWREZWJ0CAUNJHQwNDQ5MzA0NTM1MwJfMgQCeDIIBQ0kdDA0NDkzMDQ1MzUzAl8zBAJ4MwgFDSR0MDQ0OTMwNDUzNTMCXzQEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA0NDkzMDQ1MzUzAl81BBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA0NDkzMDQ1MzUzAl82BBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwNDQ5MzA0NTM1MwJfNwQXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIIBQ0kdDA0NDkzMDQ1MzUzAl84BBlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCAUNJHQwNDQ5MzA0NTM1MwJfOQQZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAgFDSR0MDQ0OTMwNDUzNTMDXzEwBA50b3RhbExvbmdBZnRlcggFDSR0MDQ0OTMwNDUzNTMDXzExBA90b3RhbFNob3J0QWZ0ZXIIBQ0kdDA0NDkzMDQ1MzUzA18xMgQSbGlxdWlkYXRpb25QZW5hbHR5CQEEbXVsZAIFGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQJARNsaXF1aWRhdGlvbkZlZVJhdGlvAAQPZmVlVG9MaXF1aWRhdG9yCQBpAgUSbGlxdWlkYXRpb25QZW5hbHR5AAIEDmZlZVRvSW5zdXJhbmNlCQBlAgUSbGlxdWlkYXRpb25QZW5hbHR5BQ9mZWVUb0xpcXVpZGF0b3IECmFtbUJhbGFuY2UJAGUCCQEIY2JhbGFuY2UABRJsaXF1aWRhdGlvblBlbmFsdHkEDSR0MDQ1NzUxNDU4ODYDCQBmAgAABQphbW1CYWxhbmNlCQCUCgIAAAkBA2FicwEFCmFtbUJhbGFuY2UJAJQKAgUKYW1tQmFsYW5jZQAABA1uZXdBbW1CYWxhbmNlCAUNJHQwNDU3NTE0NTg4NgJfMQQRdGFrZUZyb21JbnN1cmFuY2UIBQ0kdDA0NTc1MTQ1ODg2Al8yBAF4AwkAZgIFEXRha2VGcm9tSW5zdXJhbmNlAAAEEXdpdGhkcmF3SW5zdXJhbmNlCQD8BwQJARBpbnN1cmFuY2VBZGRyZXNzAAIId2l0aGRyYXcJAMwIAgURdGFrZUZyb21JbnN1cmFuY2UFA25pbAUDbmlsAwkAAAIFEXdpdGhkcmF3SW5zdXJhbmNlBRF3aXRoZHJhd0luc3VyYW5jZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUBeAUBeAQHdW5zdGFrZQkA/AcECQERcXVvdGVBc3NldFN0YWtpbmcAAg51bmxvY2tOZXV0cmlubwkAzAgCCQBlAgUSbGlxdWlkYXRpb25QZW5hbHR5BRF0YWtlRnJvbUluc3VyYW5jZQkAzAgCCQDYBAEJAQpxdW90ZUFzc2V0AAUDbmlsBQNuaWwDCQAAAgUHdW5zdGFrZQUHdW5zdGFrZQQQZGVwb3NpdEluc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQ5mZWVUb0luc3VyYW5jZQUDbmlsAwkAAAIFEGRlcG9zaXRJbnN1cmFuY2UFEGRlcG9zaXRJbnN1cmFuY2UEDm5vdGlmeU5vdGlvbmFsCQD8BwQJAQxtaW5lckFkZHJlc3MAAg5ub3RpZnlOb3Rpb25hbAkAzAgCBQdfdHJhZGVyCQDMCAIAAAUDbmlsBQNuaWwDCQAAAgUObm90aWZ5Tm90aW9uYWwFDm5vdGlmeU5vdGlvbmFsCQDOCAIJAM4IAgkAzggCCQEOZGVsZXRlUG9zaXRpb24BBQdfdHJhZGVyCQEJdXBkYXRlQW1tBwUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBRZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyBRdjdW11bGF0aXZlTm90aW9uYWxBZnRlcgUZb3BlbkludGVyZXN0Tm90aW9uYWxBZnRlcgUOdG90YWxMb25nQWZ0ZXIFD3RvdGFsU2hvcnRBZnRlcgkBCHdpdGhkcmF3AggFAWkGY2FsbGVyBQ9mZWVUb0xpcXVpZGF0b3IJAQ11cGRhdGVCYWxhbmNlAQUNbmV3QW1tQmFsYW5jZQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQpwYXlGdW5kaW5nAAQVZnVuZGluZ0Jsb2NrVGltZXN0YW1wCQEZbmV4dEZ1bmRpbmdCbG9ja1RpbWVzdGFtcAADAwMJAGYCBRVmdW5kaW5nQmxvY2tUaW1lc3RhbXAIBQlsYXN0QmxvY2sJdGltZXN0YW1wBgkBASEBCQELaW5pdGlhbGl6ZWQABgkBBnBhdXNlZAAJAAIBCQCsAgIJAKwCAgkArAICAiFJbnZhbGlkIGZ1bmRpbmcgYmxvY2sgdGltZXN0YW1wOiAJAKQDAQgFCWxhc3RCbG9jawl0aW1lc3RhbXACAyA8IAkApAMBBRVmdW5kaW5nQmxvY2tUaW1lc3RhbXAED3VuZGVybHlpbmdQcmljZQkBEmdldE9yYWNsZVR3YXBQcmljZQAEDXNwb3RUd2FwUHJpY2UJARBnZXRUd2FwU3BvdFByaWNlAAQHcHJlbWl1bQkAZQIFDXNwb3RUd2FwUHJpY2UFD3VuZGVybHlpbmdQcmljZQQNJHQwNDczODA0ODcxNQMDCQAAAgkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUAAAAGCQAAAgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQAAAAkAlAoCAAAAAAMJAGYCAAAFB3ByZW1pdW0EFHNob3J0UHJlbWl1bUZyYWN0aW9uCQEEZGl2ZAIJAQRtdWxkAgUHcHJlbWl1bQkBFGZ1bmRpbmdQZXJpb2REZWNpbWFsAAUHT05FX0RBWQQTbG9uZ1ByZW1pdW1GcmFjdGlvbgkBBGRpdmQCCQEEbXVsZAIFFHNob3J0UHJlbWl1bUZyYWN0aW9uCQEWdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQAJARV0b3RhbExvbmdQb3NpdGlvblNpemUACQCUCgIFFHNob3J0UHJlbWl1bUZyYWN0aW9uBRNsb25nUHJlbWl1bUZyYWN0aW9uBBNsb25nUHJlbWl1bUZyYWN0aW9uCQEEZGl2ZAIJAQRtdWxkAgUHcHJlbWl1bQkBFGZ1bmRpbmdQZXJpb2REZWNpbWFsAAUHT05FX0RBWQQUc2hvcnRQcmVtaXVtRnJhY3Rpb24JAQRkaXZkAgkBBG11bGQCBRNsb25nUHJlbWl1bUZyYWN0aW9uCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUACQCUCgIFFHNob3J0UHJlbWl1bUZyYWN0aW9uBRNsb25nUHJlbWl1bUZyYWN0aW9uBBRzaG9ydFByZW1pdW1GcmFjdGlvbggFDSR0MDQ3MzgwNDg3MTUCXzEEE2xvbmdQcmVtaXVtRnJhY3Rpb24IBQ0kdDA0NzM4MDQ4NzE1Al8yCQENdXBkYXRlRnVuZGluZwUJAGQCBRVmdW5kaW5nQmxvY2tUaW1lc3RhbXAJARRmdW5kaW5nUGVyaW9kU2Vjb25kcwAJAGQCCQEjbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24ABRNsb25nUHJlbWl1bUZyYWN0aW9uCQBkAgkBJGxhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgAFFHNob3J0UHJlbWl1bUZyYWN0aW9uCQEEZGl2ZAIFE2xvbmdQcmVtaXVtRnJhY3Rpb24FD3VuZGVybHlpbmdQcmljZQkBBGRpdmQCBRRzaG9ydFByZW1pdW1GcmFjdGlvbgUPdW5kZXJseWluZ1ByaWNlAWkBBXZfZ2V0AQdfdHJhZGVyBA0kdDA0OTA5MDQ5MTUwCQEVaW50ZXJuYWxDbG9zZVBvc2l0aW9uAgUHX3RyYWRlcgcEAngxCAUNJHQwNDkwOTA0OTE1MAJfMQQCeDIIBQ0kdDA0OTA5MDQ5MTUwAl8yBAJ4MwgFDSR0MDQ5MDkwNDkxNTACXzMEAng0CAUNJHQwNDkwOTA0OTE1MAJfNAkAAgEJAKwCAgkArAICCQCsAgIJAQFzAQUCeDIJAQFzAQUCeDMJAQFzAQUCeDQJAQFzAQkBDmdldE1hcmdpblJhdGlvAQUHX3RyYWRlcgFpASd2aWV3X2NhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQBB190cmFkZXIEDSR0MDQ5Mjk3NDk0MDgJAQtnZXRQb3NpdGlvbgEFB190cmFkZXIEDHBvc2l0aW9uU2l6ZQgFDSR0MDQ5Mjk3NDk0MDgCXzEEDnBvc2l0aW9uTWFyZ2luCAUNJHQwNDkyOTc0OTQwOAJfMgQDcG9uCAUNJHQwNDkyOTc0OTQwOAJfMwQRcG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDA0OTI5NzQ5NDA4Al80BA0kdDA0OTQxMzQ5NTE0CQEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwCBQdfdHJhZGVyBQ9QTkxfT1BUSU9OX1NQT1QEEHBvc2l0aW9uTm90aW9uYWwIBQ0kdDA0OTQxMzQ5NTE0Al8xBA11bnJlYWxpemVkUG5sCAUNJHQwNDk0MTM0OTUxNAJfMgQNJHQwNDk1MTk0OTcwMQkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQxwb3NpdGlvblNpemUFDnBvc2l0aW9uTWFyZ2luBRFwb3NpdGlvbkxzdFVwZENQRgUNdW5yZWFsaXplZFBubAQMcmVtYWluTWFyZ2luCAUNJHQwNDk1MTk0OTcwMQJfMQQHYmFkRGVidAgFDSR0MDQ5NTE5NDk3MDECXzIEDmZ1bmRpbmdQYXltZW50CAUNJHQwNDk1MTk0OTcwMQJfMwkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkBAXMBBQxyZW1haW5NYXJnaW4JAQFzAQUOZnVuZGluZ1BheW1lbnQJAQFzAQkBDmdldE1hcmdpblJhdGlvAQUHX3RyYWRlcgkBAXMBBQ11bnJlYWxpemVkUG5sCQEBcwEFB2JhZERlYnQBaQEOZm9yY2VNb3ZlQXNzZXQCB190cmFkZXIHX2Ftb3VudAMDCQECIT0CCQCnCAEJAQ5hZG1pblB1YmxpY0tleQAIBQFpBmNhbGxlcgYJAGYCAAAFB19hbW91bnQJAAIBAiFJbnZhbGlkIGZvcmNlTW92ZUFzc2V0IHBhcmFtZXRlcnMEB3Vuc3Rha2UJAPwHBAkBEXF1b3RlQXNzZXRTdGFraW5nAAIOdW5sb2NrTmV1dHJpbm8JAMwIAgUHX2Ftb3VudAkAzAgCCQDYBAEJAQpxdW90ZUFzc2V0AAUDbmlsBQNuaWwDCQAAAgUHdW5zdGFrZQUHdW5zdGFrZQkAzggCCQEId2l0aGRyYXcCCQERQGV4dHJOYXRpdmUoMTA2MikBBQdfdHJhZGVyBQdfYW1vdW50CQENdXBkYXRlQmFsYW5jZQEJAGUCCQEIY2JhbGFuY2UABQdfYW1vdW50CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQJ0eAEGdmVyaWZ5AAkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAkBDmFkbWluUHVibGljS2V5ACWfQ1I=", "height": 3255540, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 56HuXJrDTqXy3xVAucGTANZSH8VqMKwZLWbfsKVeoH76 Next: Dn2Cy1nyovEsasaKgw9xCd6fubgynPziN37BP4h98Xg3 Diff:
OldNewDifferences
669669
670670
671671 @Callable(i)
672+func setInitMarginRatio (_initMarginRatio) = if (if ((0 >= _initMarginRatio))
673+ then true
674+ else !(initialized()))
675+ then throw("Invalid setInitMarginRatio parameters")
676+ else updateSettings(_initMarginRatio, maintenanceMarginRatio(), liquidationFeeRatio(), fundingPeriodRaw(), fee(), spreadLimit(), maxPriceImpact())
677+
678+
679+
680+@Callable(i)
672681 func decreasePosition (_amount,_leverage,_minBaseAssetAmount) = if (if (if (if (if ((0 >= _amount))
673682 then true
674683 else !(initialized()))
680689 else paused())
681690 then throw("Invalid decreasePosition parameters")
682691 else {
683- let $t03093131083 = getPosition(toString(i.caller))
684- let oldPositionSize = $t03093131083._1
685- let oldPositionMargin = $t03093131083._2
686- let oldPositionOpenNotional = $t03093131083._3
687- let oldPositionLstUpdCPF = $t03093131083._4
692+ let $t03133431486 = getPosition(toString(i.caller))
693+ let oldPositionSize = $t03133431486._1
694+ let oldPositionMargin = $t03133431486._2
695+ let oldPositionOpenNotional = $t03133431486._3
696+ let oldPositionLstUpdCPF = $t03133431486._4
688697 let _direction = if ((oldPositionSize > 0))
689698 then DIR_SHORT
690699 else DIR_LONG
691700 let isAdd = (_direction == DIR_LONG)
692701 let openNotional = muld(_amount, _leverage)
693- let $t03125631372 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
694- let oldPositionNotional = $t03125631372._1
695- let unrealizedPnl = $t03125631372._2
696- let $t03137833927 = if ((oldPositionNotional > openNotional))
702+ let $t03165931775 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
703+ let oldPositionNotional = $t03165931775._1
704+ let unrealizedPnl = $t03165931775._2
705+ let $t03178134330 = if ((oldPositionNotional > openNotional))
697706 then {
698- let $t03175531974 = swapInput(isAdd, openNotional)
699- let exchangedPositionSize = $t03175531974._1
700- let quoteAssetReserveAfter = $t03175531974._2
701- let baseAssetReserveAfter = $t03175531974._3
702- let totalPositionSizeAfter = $t03175531974._4
703- let cumulativeNotionalAfter = $t03175531974._5
707+ let $t03215832377 = swapInput(isAdd, openNotional)
708+ let exchangedPositionSize = $t03215832377._1
709+ let quoteAssetReserveAfter = $t03215832377._2
710+ let baseAssetReserveAfter = $t03215832377._3
711+ let totalPositionSizeAfter = $t03215832377._4
712+ let cumulativeNotionalAfter = $t03215832377._5
704713 let exchangedPositionSizeAbs = abs(exchangedPositionSize)
705714 if (if ((_minBaseAssetAmount != 0))
706715 then (_minBaseAssetAmount > exchangedPositionSizeAbs)
708717 then throw(((("Too little base asset exchanged, got " + toString(exchangedPositionSizeAbs)) + " expected ") + toString(_minBaseAssetAmount)))
709718 else {
710719 let realizedPnl = divd(muld(unrealizedPnl, exchangedPositionSizeAbs), abs(oldPositionSize))
711- let $t03241132656 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
712- let remainMargin = $t03241132656._1
713- let badDebt = $t03241132656._2
714- let fundingPayment = $t03241132656._3
720+ let $t03281433059 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
721+ let remainMargin = $t03281433059._1
722+ let badDebt = $t03281433059._2
723+ let fundingPayment = $t03281433059._3
715724 let exchangedQuoteAssetAmount = openNotional
716725 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
717726 let remainOpenNotional = if ((oldPositionSize > 0))
726735 }
727736 }
728737 else throw("Close position first")
729- let newPositionSize = $t03137833927._1
730- let newPositionRemainMargin = $t03137833927._2
731- let newPositionOpenNotional = $t03137833927._3
732- let newPositionLatestCPF = $t03137833927._4
733- let baseAssetReserveAfter = $t03137833927._5
734- let quoteAssetReserveAfter = $t03137833927._6
735- let totalPositionSizeAfter = $t03137833927._7
736- let cumulativeNotionalAfter = $t03137833927._8
737- let openInterestNotionalAfter = $t03137833927._9
738- let totalLongAfter = $t03137833927._10
739- let totalShortAfter = $t03137833927._11
738+ let newPositionSize = $t03178134330._1
739+ let newPositionRemainMargin = $t03178134330._2
740+ let newPositionOpenNotional = $t03178134330._3
741+ let newPositionLatestCPF = $t03178134330._4
742+ let baseAssetReserveAfter = $t03178134330._5
743+ let quoteAssetReserveAfter = $t03178134330._6
744+ let totalPositionSizeAfter = $t03178134330._7
745+ let cumulativeNotionalAfter = $t03178134330._8
746+ let openInterestNotionalAfter = $t03178134330._9
747+ let totalLongAfter = $t03178134330._10
748+ let totalShortAfter = $t03178134330._11
740749 let notifyNotional = invoke(minerAddress(), "notifyNotional", [toString(i.caller), newPositionOpenNotional], nil)
741750 if ((notifyNotional == notifyNotional))
742751 then (updatePosition(toString(i.caller), newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter))
765774 else {
766775 let feeAmount = muld(_rawAmount, fee())
767776 let _amount = (_rawAmount - feeAmount)
768- let $t03504535197 = getPosition(toString(i.caller))
769- let oldPositionSize = $t03504535197._1
770- let oldPositionMargin = $t03504535197._2
771- let oldPositionOpenNotional = $t03504535197._3
772- let oldPositionLstUpdCPF = $t03504535197._4
777+ let $t03544835600 = getPosition(toString(i.caller))
778+ let oldPositionSize = $t03544835600._1
779+ let oldPositionMargin = $t03544835600._2
780+ let oldPositionOpenNotional = $t03544835600._3
781+ let oldPositionLstUpdCPF = $t03544835600._4
773782 let isNewPosition = (oldPositionSize == 0)
774783 let isSameDirection = if ((oldPositionSize > 0))
775784 then (_direction == DIR_LONG)
778787 then isSameDirection
779788 else false
780789 let isAdd = (_direction == DIR_LONG)
781- let $t03548637875 = if (if (isNewPosition)
790+ let $t03588938278 = if (if (isNewPosition)
782791 then true
783792 else expandExisting)
784793 then {
785794 let openNotional = muld(_amount, _leverage)
786- let $t03591036116 = swapInput(isAdd, openNotional)
787- let amountBaseAssetBought = $t03591036116._1
788- let quoteAssetReserveAfter = $t03591036116._2
789- let baseAssetReserveAfter = $t03591036116._3
790- let totalPositionSizeAfter = $t03591036116._4
791- let cumulativeNotionalAfter = $t03591036116._5
795+ let $t03631336519 = swapInput(isAdd, openNotional)
796+ let amountBaseAssetBought = $t03631336519._1
797+ let quoteAssetReserveAfter = $t03631336519._2
798+ let baseAssetReserveAfter = $t03631336519._3
799+ let totalPositionSizeAfter = $t03631336519._4
800+ let cumulativeNotionalAfter = $t03631336519._5
792801 if (if ((_minBaseAssetAmount != 0))
793802 then (_minBaseAssetAmount > abs(amountBaseAssetBought))
794803 else false)
796805 else {
797806 let newPositionSize = (oldPositionSize + amountBaseAssetBought)
798807 let increaseMarginRequirement = divd(openNotional, _leverage)
799- let $t03649736736 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
800- let remainMargin = $t03649736736._1
801- let x1 = $t03649736736._2
802- let x2 = $t03649736736._3
808+ let $t03690037139 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
809+ let remainMargin = $t03690037139._1
810+ let x1 = $t03690037139._2
811+ let x2 = $t03690037139._3
803812 $Tuple11(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
804813 then abs(amountBaseAssetBought)
805814 else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
809818 }
810819 else {
811820 let openNotional = muld(_amount, _leverage)
812- let $t03756837684 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
813- let oldPositionNotional = $t03756837684._1
814- let unrealizedPnl = $t03756837684._2
821+ let $t03797138087 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
822+ let oldPositionNotional = $t03797138087._1
823+ let unrealizedPnl = $t03797138087._2
815824 if ((oldPositionNotional > openNotional))
816825 then throw("Use decreasePosition to decrease position size")
817826 else throw("Close position first")
818827 }
819- let newPositionSize = $t03548637875._1
820- let newPositionRemainMargin = $t03548637875._2
821- let newPositionOpenNotional = $t03548637875._3
822- let newPositionLatestCPF = $t03548637875._4
823- let baseAssetReserveAfter = $t03548637875._5
824- let quoteAssetReserveAfter = $t03548637875._6
825- let totalPositionSizeAfter = $t03548637875._7
826- let cumulativeNotionalAfter = $t03548637875._8
827- let openInterestNotionalAfter = $t03548637875._9
828- let totalLongAfter = $t03548637875._10
829- let totalShortAfter = $t03548637875._11
828+ let newPositionSize = $t03588938278._1
829+ let newPositionRemainMargin = $t03588938278._2
830+ let newPositionOpenNotional = $t03588938278._3
831+ let newPositionLatestCPF = $t03588938278._4
832+ let baseAssetReserveAfter = $t03588938278._5
833+ let quoteAssetReserveAfter = $t03588938278._6
834+ let totalPositionSizeAfter = $t03588938278._7
835+ let cumulativeNotionalAfter = $t03588938278._8
836+ let openInterestNotionalAfter = $t03588938278._9
837+ let totalLongAfter = $t03588938278._10
838+ let totalShortAfter = $t03588938278._11
830839 let feeToStakers = (feeAmount / 2)
831840 let feeToInsurance = (feeAmount - feeToStakers)
832841 let stake = invoke(quoteAssetStaking(), "lockNeutrinoSP", [toString(stakingAddress()), ALL_FEES], [AttachedPayment(quoteAsset(), _amount)])
867876 else {
868877 let feeAmount = muld(_rawAmount, fee())
869878 let _amount = (_rawAmount - feeAmount)
870- let $t03943139583 = getPosition(toString(i.caller))
871- let oldPositionSize = $t03943139583._1
872- let oldPositionMargin = $t03943139583._2
873- let oldPositionOpenNotional = $t03943139583._3
874- let oldPositionLstUpdCPF = $t03943139583._4
879+ let $t03983439986 = getPosition(toString(i.caller))
880+ let oldPositionSize = $t03983439986._1
881+ let oldPositionMargin = $t03983439986._2
882+ let oldPositionOpenNotional = $t03983439986._3
883+ let oldPositionLstUpdCPF = $t03983439986._4
875884 let feeToStakers = (feeAmount / 2)
876885 let feeToInsurance = (feeAmount - feeToStakers)
877886 let stake = invoke(quoteAssetStaking(), "lockNeutrinoSP", [toString(stakingAddress()), ALL_FEES], [AttachedPayment(quoteAsset(), _amount)])
903912 else paused())
904913 then throw("Invalid removeMargin parameters")
905914 else {
906- let $t04063240784 = getPosition(toString(i.caller))
907- let oldPositionSize = $t04063240784._1
908- let oldPositionMargin = $t04063240784._2
909- let oldPositionOpenNotional = $t04063240784._3
910- let oldPositionLstUpdCPF = $t04063240784._4
915+ let $t04103541187 = getPosition(toString(i.caller))
916+ let oldPositionSize = $t04103541187._1
917+ let oldPositionMargin = $t04103541187._2
918+ let oldPositionOpenNotional = $t04103541187._3
919+ let oldPositionLstUpdCPF = $t04103541187._4
911920 let marginDelta = -(_amount)
912- let $t04082141000 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
913- let remainMargin = $t04082141000._1
914- let badDebt = $t04082141000._2
921+ let $t04122441403 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
922+ let remainMargin = $t04122441403._1
923+ let badDebt = $t04122441403._2
915924 if ((badDebt != 0))
916925 then throw("Invalid removed margin amount")
917926 else {
937946 else paused())
938947 then throw("Invalid closePosition parameters")
939948 else {
940- let $t04209642486 = internalClosePosition(toString(i.caller), true)
941- let x1 = $t04209642486._1
942- let positionBadDebt = $t04209642486._2
943- let realizedPnl = $t04209642486._3
944- let marginToVault = $t04209642486._4
945- let quoteAssetReserveAfter = $t04209642486._5
946- let baseAssetReserveAfter = $t04209642486._6
947- let totalPositionSizeAfter = $t04209642486._7
948- let cumulativeNotionalAfter = $t04209642486._8
949- let openInterestNotionalAfter = $t04209642486._9
950- let x2 = $t04209642486._10
951- let totalLongAfter = $t04209642486._11
952- let totalShortAfter = $t04209642486._12
949+ let $t04249942889 = internalClosePosition(toString(i.caller), true)
950+ let x1 = $t04249942889._1
951+ let positionBadDebt = $t04249942889._2
952+ let realizedPnl = $t04249942889._3
953+ let marginToVault = $t04249942889._4
954+ let quoteAssetReserveAfter = $t04249942889._5
955+ let baseAssetReserveAfter = $t04249942889._6
956+ let totalPositionSizeAfter = $t04249942889._7
957+ let cumulativeNotionalAfter = $t04249942889._8
958+ let openInterestNotionalAfter = $t04249942889._9
959+ let x2 = $t04249942889._10
960+ let totalLongAfter = $t04249942889._11
961+ let totalShortAfter = $t04249942889._12
953962 if ((positionBadDebt > 0))
954963 then throw("Unable to close position with bad debt")
955964 else {
956965 let withdrawAmount = abs(marginToVault)
957966 let ammBalance = (cbalance() - withdrawAmount)
958- let $t04269542837 = if ((0 > ammBalance))
967+ let $t04309843240 = if ((0 > ammBalance))
959968 then $Tuple2(0, abs(ammBalance))
960969 else $Tuple2(ammBalance, 0)
961- let ammNewBalance = $t04269542837._1
962- let getFromInsurance = $t04269542837._2
970+ let ammNewBalance = $t04309843240._1
971+ let getFromInsurance = $t04309843240._2
963972 let x = if ((getFromInsurance > 0))
964973 then {
965974 let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [getFromInsurance], nil)
10041013 else paused())
10051014 then throw("Unable to liquidate")
10061015 else {
1007- let $t04452744950 = internalClosePosition(_trader, false)
1008- let x1 = $t04452744950._1
1009- let badDebt = $t04452744950._2
1010- let x2 = $t04452744950._3
1011- let x3 = $t04452744950._4
1012- let quoteAssetReserveAfter = $t04452744950._5
1013- let baseAssetReserveAfter = $t04452744950._6
1014- let totalPositionSizeAfter = $t04452744950._7
1015- let cumulativeNotionalAfter = $t04452744950._8
1016- let openInterestNotionalAfter = $t04452744950._9
1017- let exchangedQuoteAssetAmount = $t04452744950._10
1018- let totalLongAfter = $t04452744950._11
1019- let totalShortAfter = $t04452744950._12
1016+ let $t04493045353 = internalClosePosition(_trader, false)
1017+ let x1 = $t04493045353._1
1018+ let badDebt = $t04493045353._2
1019+ let x2 = $t04493045353._3
1020+ let x3 = $t04493045353._4
1021+ let quoteAssetReserveAfter = $t04493045353._5
1022+ let baseAssetReserveAfter = $t04493045353._6
1023+ let totalPositionSizeAfter = $t04493045353._7
1024+ let cumulativeNotionalAfter = $t04493045353._8
1025+ let openInterestNotionalAfter = $t04493045353._9
1026+ let exchangedQuoteAssetAmount = $t04493045353._10
1027+ let totalLongAfter = $t04493045353._11
1028+ let totalShortAfter = $t04493045353._12
10201029 let liquidationPenalty = muld(exchangedQuoteAssetAmount, liquidationFeeRatio())
10211030 let feeToLiquidator = (liquidationPenalty / 2)
10221031 let feeToInsurance = (liquidationPenalty - feeToLiquidator)
10231032 let ammBalance = (cbalance() - liquidationPenalty)
1024- let $t04534845483 = if ((0 > ammBalance))
1033+ let $t04575145886 = if ((0 > ammBalance))
10251034 then $Tuple2(0, abs(ammBalance))
10261035 else $Tuple2(ammBalance, 0)
1027- let newAmmBalance = $t04534845483._1
1028- let takeFromInsurance = $t04534845483._2
1036+ let newAmmBalance = $t04575145886._1
1037+ let takeFromInsurance = $t04575145886._2
10291038 let x = if ((takeFromInsurance > 0))
10301039 then {
10311040 let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
10701079 let underlyingPrice = getOracleTwapPrice()
10711080 let spotTwapPrice = getTwapSpotPrice()
10721081 let premium = (spotTwapPrice - underlyingPrice)
1073- let $t04697748312 = if (if ((totalShortPositionSize() == 0))
1082+ let $t04738048715 = if (if ((totalShortPositionSize() == 0))
10741083 then true
10751084 else (totalLongPositionSize() == 0))
10761085 then $Tuple2(0, 0)
10851094 let shortPremiumFraction = divd(muld(longPremiumFraction, totalLongPositionSize()), totalShortPositionSize())
10861095 $Tuple2(shortPremiumFraction, longPremiumFraction)
10871096 }
1088- let shortPremiumFraction = $t04697748312._1
1089- let longPremiumFraction = $t04697748312._2
1097+ let shortPremiumFraction = $t04738048715._1
1098+ let longPremiumFraction = $t04738048715._2
10901099 updateFunding((fundingBlockTimestamp + fundingPeriodSeconds()), (latestLongCumulativePremiumFraction() + longPremiumFraction), (latestShortCumulativePremiumFraction() + shortPremiumFraction), divd(longPremiumFraction, underlyingPrice), divd(shortPremiumFraction, underlyingPrice))
10911100 }
10921101 }
10951104
10961105 @Callable(i)
10971106 func v_get (_trader) = {
1098- let $t04868748747 = internalClosePosition(_trader, false)
1099- let x1 = $t04868748747._1
1100- let x2 = $t04868748747._2
1101- let x3 = $t04868748747._3
1102- let x4 = $t04868748747._4
1107+ let $t04909049150 = internalClosePosition(_trader, false)
1108+ let x1 = $t04909049150._1
1109+ let x2 = $t04909049150._2
1110+ let x3 = $t04909049150._3
1111+ let x4 = $t04909049150._4
11031112 throw((((s(x2) + s(x3)) + s(x4)) + s(getMarginRatio(_trader))))
11041113 }
11051114
11071116
11081117 @Callable(i)
11091118 func view_calcRemainMarginWithFundingPayment (_trader) = {
1110- let $t04889449005 = getPosition(_trader)
1111- let positionSize = $t04889449005._1
1112- let positionMargin = $t04889449005._2
1113- let pon = $t04889449005._3
1114- let positionLstUpdCPF = $t04889449005._4
1115- let $t04901049111 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1116- let positionNotional = $t04901049111._1
1117- let unrealizedPnl = $t04901049111._2
1118- let $t04911649298 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1119- let remainMargin = $t04911649298._1
1120- let badDebt = $t04911649298._2
1121- let fundingPayment = $t04911649298._3
1119+ let $t04929749408 = getPosition(_trader)
1120+ let positionSize = $t04929749408._1
1121+ let positionMargin = $t04929749408._2
1122+ let pon = $t04929749408._3
1123+ let positionLstUpdCPF = $t04929749408._4
1124+ let $t04941349514 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1125+ let positionNotional = $t04941349514._1
1126+ let unrealizedPnl = $t04941349514._2
1127+ let $t04951949701 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1128+ let remainMargin = $t04951949701._1
1129+ let badDebt = $t04951949701._2
1130+ let fundingPayment = $t04951949701._3
11221131 throw(((((s(remainMargin) + s(fundingPayment)) + s(getMarginRatio(_trader))) + s(unrealizedPnl)) + s(badDebt)))
11231132 }
11241133
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let k_ora_key = "k_ora_key"
55
66 let k_ora_block_key = "k_ora_block_key"
77
88 let k_ora = "k_ora"
99
1010 let k_balance = "k_balance"
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_initialized = "k_initialized"
2121
2222 let k_paused = "k_paused"
2323
2424 let k_fee = "k_fee"
2525
2626 let k_fundingPeriod = "k_fundingPeriod"
2727
2828 let k_initMarginRatio = "k_initMarginRatio"
2929
3030 let k_maintenanceMarginRatio = "k_mmr"
3131
3232 let k_liquidationFeeRatio = "k_liquidationFeeRatio"
3333
3434 let k_spreadLimit = "k_spreadLimit"
3535
3636 let k_maxPriceImpact = "k_maxPriceImpact"
3737
3838 let k_lastDataStr = "k_lastDataStr"
3939
4040 let k_lastMinuteId = "k_lastMinuteId"
4141
4242 let k_twapDataLastCumulativePrice = "k_twapDataLastCumulativePrice"
4343
4444 let k_twapDataLastPrice = "k_twapDataLastPrice"
4545
4646 let k_twapDataPreviousMinuteId = "k_twapDataPreviousMinuteId"
4747
4848 let k_latestLongCumulativePremiumFraction = "k_latestLongPremiumFraction"
4949
5050 let k_latestShortCumulativePremiumFraction = "k_latestShortPremiumFraction"
5151
5252 let k_nextFundingBlock = "k_nextFundingBlockMinTimestamp"
5353
5454 let k_longFundingRate = "k_longFundingRate"
5555
5656 let k_shortFundingRate = "k_shortFundingRate"
5757
5858 let k_quoteAssetReserve = "k_qtAstR"
5959
6060 let k_baseAssetReserve = "k_bsAstR"
6161
6262 let k_totalPositionSize = "k_totalPositionSize"
6363
6464 let k_totalLongPositionSize = "k_totalLongPositionSize"
6565
6666 let k_totalShortPositionSize = "k_totalShortPositionSize"
6767
6868 let k_cumulativeNotional = "k_cumulativeNotional"
6969
7070 let k_openInterestNotional = "k_openInterestNotional"
7171
7272 let k_coordinatorAddress = "k_coordinatorAddress"
7373
7474 let k_insurance_address = "k_insurance_address"
7575
7676 let k_admin_address = "k_admin_address"
7777
7878 let k_admin_public_key = "k_admin_public_key"
7979
8080 let k_quote_asset = "k_quote_asset"
8181
8282 let k_quote_staking = "k_quote_staking"
8383
8484 let k_staking_address = "k_staking_address"
8585
8686 let k_miner_address = "k_miner_address"
8787
8888 func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
8989
9090
9191 func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address))
9292
9393
9494 func adminPublicKey () = fromBase58String(getStringValue(coordinator(), k_admin_public_key))
9595
9696
9797 func quoteAsset () = fromBase58String(getStringValue(coordinator(), k_quote_asset))
9898
9999
100100 func quoteAssetStaking () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_quote_staking)), "Quote asset staking not set")
101101
102102
103103 func stakingAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_staking_address)), "Insurance not set")
104104
105105
106106 func insuranceAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_insurance_address)), "Insurance not set")
107107
108108
109109 func minerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_miner_address)), "Insurance not set")
110110
111111
112112 let DIR_LONG = 1
113113
114114 let DIR_SHORT = 2
115115
116116 let TWAP_INTERVAL = 15
117117
118118 let ORACLE_INTERVAL = 15
119119
120120 let SECONDS = 1000
121121
122122 let DECIMAL_UNIT = (1 * (((((10 * 10) * 10) * 10) * 10) * 10))
123123
124124 let ONE_DAY = (86400 * DECIMAL_UNIT)
125125
126126 let ALL_FEES = 100
127127
128128 let PNL_OPTION_SPOT = 1
129129
130130 let PNL_OPTION_ORACLE = 2
131131
132132 func s (_x) = (toString(_x) + ",")
133133
134134
135135 func divd (_x,_y) = fraction(_x, DECIMAL_UNIT, _y, HALFEVEN)
136136
137137
138138 func muld (_x,_y) = fraction(_x, _y, DECIMAL_UNIT, HALFEVEN)
139139
140140
141141 func abs (_x) = if ((_x > 0))
142142 then _x
143143 else -(_x)
144144
145145
146146 func vmax (_x,_y) = if ((_x >= _y))
147147 then _x
148148 else _y
149149
150150
151151 func toCompositeKey (_key,_address) = ((_key + "_") + _address)
152152
153153
154154 func listToStr (_list) = {
155155 func _join (accumulator,val) = ((accumulator + val) + ",")
156156
157157 let newListStr = {
158158 let $l = _list
159159 let $s = size($l)
160160 let $acc0 = ""
161161 func $f0_1 ($a,$i) = if (($i >= $s))
162162 then $a
163163 else _join($a, $l[$i])
164164
165165 func $f0_2 ($a,$i) = if (($i >= $s))
166166 then $a
167167 else throw("List size exceeds 20")
168168
169169 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20)
170170 }
171171 let newListStrU = dropRight(newListStr, 1)
172172 let newListStrR = if ((take(newListStrU, 1) == ","))
173173 then drop(newListStrU, 1)
174174 else newListStrU
175175 newListStrR
176176 }
177177
178178
179179 func strToList (_str) = split(_str, ",")
180180
181181
182182 func pushToQueue (_list,_maxSize,_value) = if ((size(_list) > _maxSize))
183183 then (removeByIndex(_list, 0) :+ _value)
184184 else (_list :+ _value)
185185
186186
187187 func int (k) = valueOrErrorMessage(getInteger(this, k), ("no value for " + k))
188188
189189
190190 func cbalance () = int(k_balance)
191191
192192
193193 func fee () = int(k_fee)
194194
195195
196196 func initMarginRatio () = int(k_initMarginRatio)
197197
198198
199199 func qtAstR () = int(k_quoteAssetReserve)
200200
201201
202202 func bsAstR () = int(k_baseAssetReserve)
203203
204204
205205 func totalPositionSize () = int(k_totalPositionSize)
206206
207207
208208 func cumulativeNotional () = int(k_cumulativeNotional)
209209
210210
211211 func openInterestNotional () = int(k_openInterestNotional)
212212
213213
214214 func nextFundingBlockTimestamp () = int(k_nextFundingBlock)
215215
216216
217217 func fundingPeriodRaw () = int(k_fundingPeriod)
218218
219219
220220 func fundingPeriodDecimal () = (fundingPeriodRaw() * DECIMAL_UNIT)
221221
222222
223223 func fundingPeriodSeconds () = (fundingPeriodRaw() * SECONDS)
224224
225225
226226 func maintenanceMarginRatio () = int(k_maintenanceMarginRatio)
227227
228228
229229 func liquidationFeeRatio () = int(k_liquidationFeeRatio)
230230
231231
232232 func spreadLimit () = int(k_spreadLimit)
233233
234234
235235 func maxPriceImpact () = int(k_maxPriceImpact)
236236
237237
238238 func latestLongCumulativePremiumFraction () = int(k_latestLongCumulativePremiumFraction)
239239
240240
241241 func latestShortCumulativePremiumFraction () = int(k_latestShortCumulativePremiumFraction)
242242
243243
244244 func totalShortPositionSize () = int(k_totalShortPositionSize)
245245
246246
247247 func totalLongPositionSize () = int(k_totalLongPositionSize)
248248
249249
250250 func requireMoreMarginRatio (_marginRatio,_baseMarginRatio,_largerThanOrEqualTo) = {
251251 let remainingMarginRatio = (_marginRatio - _baseMarginRatio)
252252 if (if (_largerThanOrEqualTo)
253253 then (0 > remainingMarginRatio)
254254 else false)
255255 then throw("Invalid margin")
256256 else if (if (!(_largerThanOrEqualTo))
257257 then (remainingMarginRatio >= 0)
258258 else false)
259259 then throw("Invalid margin")
260260 else true
261261 }
262262
263263
264264 func latestCumulativePremiumFraction (_positionSize) = if ((_positionSize == 0))
265265 then throw("Should not be called with _positionSize == 0")
266266 else if ((_positionSize > 0))
267267 then latestLongCumulativePremiumFraction()
268268 else latestShortCumulativePremiumFraction()
269269
270270
271271 func getPosition (_trader) = {
272272 let positionSizeOpt = getInteger(this, toCompositeKey(k_positionSize, _trader))
273273 match positionSizeOpt {
274274 case positionSize: Int =>
275275 $Tuple4(positionSize, getIntegerValue(this, toCompositeKey(k_positionMargin, _trader)), getIntegerValue(this, toCompositeKey(k_positionOpenNotional, _trader)), getIntegerValue(this, toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _trader)))
276276 case _ =>
277277 $Tuple4(0, 0, 0, 0)
278278 }
279279 }
280280
281281
282282 func requireOpenPosition (_trader) = if ((getPosition(_trader)._1 == 0))
283283 then throw("No open position")
284284 else true
285285
286286
287287 func initialized () = valueOrElse(getBoolean(this, k_initialized), false)
288288
289289
290290 func paused () = valueOrElse(getBoolean(this, k_paused), false)
291291
292292
293293 func updateReserve (_isAdd,_quoteAssetAmount,_baseAssetAmount) = if (_isAdd)
294294 then {
295295 let newBase = (bsAstR() - _baseAssetAmount)
296296 if ((0 >= newBase))
297297 then throw("Tx lead to base asset reserve <= 0, revert")
298298 else $Tuple4((qtAstR() + _quoteAssetAmount), newBase, (totalPositionSize() + _baseAssetAmount), (cumulativeNotional() + _quoteAssetAmount))
299299 }
300300 else {
301301 let newQuote = (qtAstR() - _quoteAssetAmount)
302302 if ((0 >= newQuote))
303303 then throw("Tx lead to base quote reserve <= 0, revert")
304304 else $Tuple4(newQuote, (bsAstR() + _baseAssetAmount), (totalPositionSize() - _baseAssetAmount), (cumulativeNotional() - _quoteAssetAmount))
305305 }
306306
307307
308308 func swapInput (_isAdd,_quoteAssetAmount) = {
309309 let _qtAstR = qtAstR()
310310 let _bsAstR = bsAstR()
311311 let k = muld(_qtAstR, _bsAstR)
312312 let quoteAssetReserveAfter = if (_isAdd)
313313 then (_qtAstR + _quoteAssetAmount)
314314 else (_qtAstR - _quoteAssetAmount)
315315 let baseAssetReserveAfter = divd(k, quoteAssetReserveAfter)
316316 let amountBaseAssetBoughtAbs = abs((baseAssetReserveAfter - _bsAstR))
317317 let amountBaseAssetBought = if (_isAdd)
318318 then amountBaseAssetBoughtAbs
319319 else -(amountBaseAssetBoughtAbs)
320320 let $t01183712030 = updateReserve(_isAdd, _quoteAssetAmount, amountBaseAssetBoughtAbs)
321321 let quoteAssetReserveAfter1 = $t01183712030._1
322322 let baseAssetReserveAfter1 = $t01183712030._2
323323 let totalPositionSizeAfter1 = $t01183712030._3
324324 let cumulativeNotionalAfter1 = $t01183712030._4
325325 let priceBefore = divd(_qtAstR, _bsAstR)
326326 let marketPrice = divd(_quoteAssetAmount, amountBaseAssetBoughtAbs)
327327 let priceDiff = abs((priceBefore - marketPrice))
328328 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
329329 let maxPriceImpactValue = maxPriceImpact()
330330 if ((priceImpact > maxPriceImpactValue))
331331 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)))
332332 else $Tuple5(amountBaseAssetBought, quoteAssetReserveAfter1, baseAssetReserveAfter1, totalPositionSizeAfter1, cumulativeNotionalAfter1)
333333 }
334334
335335
336336 func calcRemainMarginWithFundingPayment (_oldPositionSize,_oldPositionMargin,_oldPositionCumulativePremiumFraction,_marginDelta) = {
337337 let fundingPayment = if ((_oldPositionSize != 0))
338338 then {
339339 let _latestCumulativePremiumFraction = latestCumulativePremiumFraction(_oldPositionSize)
340340 muld((_latestCumulativePremiumFraction - _oldPositionCumulativePremiumFraction), _oldPositionSize)
341341 }
342342 else 0
343343 let signedMargin = ((_marginDelta - fundingPayment) + _oldPositionMargin)
344344 let $t01352113648 = if ((0 > signedMargin))
345345 then $Tuple2(0, abs(signedMargin))
346346 else $Tuple2(abs(signedMargin), 0)
347347 let remainMargin = $t01352113648._1
348348 let badDebt = $t01352113648._2
349349 $Tuple3(remainMargin, badDebt, fundingPayment)
350350 }
351351
352352
353353 func swapOutput (_isAdd,_baseAssetAmount,_checkMaxPriceImpact) = {
354354 let _quoteAssetReserve = qtAstR()
355355 let _baseAssetReserve = bsAstR()
356356 let priceBefore = divd(_quoteAssetReserve, _baseAssetReserve)
357357 if ((_baseAssetAmount == 0))
358358 then throw("Invalid base asset amount")
359359 else {
360360 let k = muld(_quoteAssetReserve, _baseAssetReserve)
361361 let baseAssetPoolAmountAfter = if (_isAdd)
362362 then (_baseAssetReserve + _baseAssetAmount)
363363 else (_baseAssetReserve - _baseAssetAmount)
364364 let quoteAssetAfter = divd(k, baseAssetPoolAmountAfter)
365365 let quoteAssetSold = abs((quoteAssetAfter - _quoteAssetReserve))
366366 let maxPriceImpactValue = maxPriceImpact()
367367 let $t01463214825 = updateReserve(!(_isAdd), quoteAssetSold, _baseAssetAmount)
368368 let quoteAssetReserveAfter1 = $t01463214825._1
369369 let baseAssetReserveAfter1 = $t01463214825._2
370370 let totalPositionSizeAfter1 = $t01463214825._3
371371 let cumulativeNotionalAfter1 = $t01463214825._4
372372 let marketPrice = divd(quoteAssetSold, _baseAssetAmount)
373373 let priceDiff = abs((priceBefore - marketPrice))
374374 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
375375 if (if ((priceImpact > maxPriceImpactValue))
376376 then _checkMaxPriceImpact
377377 else false)
378378 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)))
379379 else $Tuple7(quoteAssetSold, quoteAssetReserveAfter1, baseAssetReserveAfter1, totalPositionSizeAfter1, cumulativeNotionalAfter1, (totalLongPositionSize() - (if (_isAdd)
380380 then abs(_baseAssetAmount)
381381 else 0)), (totalShortPositionSize() - (if (!(_isAdd))
382382 then abs(_baseAssetAmount)
383383 else 0)))
384384 }
385385 }
386386
387387
388388 func getOracleTwapPrice () = {
389389 let oracle = valueOrErrorMessage(addressFromString(getStringValue(this, k_ora)), "")
390390 let priceKey = getStringValue(this, k_ora_key)
391391 let blockKey = getStringValue(this, k_ora_block_key)
392392 let lastValue = valueOrErrorMessage(getInteger(oracle, priceKey), ((("Can not get oracle price. Oracle: " + toString(oracle)) + " key: ") + priceKey))
393393 lastValue
394394 }
395395
396396
397397 func getSpotPrice () = {
398398 let _quoteAssetReserve = qtAstR()
399399 let _baseAssetReserve = bsAstR()
400400 divd(_quoteAssetReserve, _baseAssetReserve)
401401 }
402402
403403
404404 func isOverFluctuationLimit () = {
405405 let oraclePrice = getOracleTwapPrice()
406406 let currentPrice = getSpotPrice()
407407 (divd(abs((oraclePrice - currentPrice)), oraclePrice) > spreadLimit())
408408 }
409409
410410
411411 func getPositionNotionalAndUnrealizedPnl (_trader,_option) = {
412412 let $t01729317421 = getPosition(_trader)
413413 let positionSize = $t01729317421._1
414414 let positionMargin = $t01729317421._2
415415 let positionOpenNotional = $t01729317421._3
416416 let positionLstUpdCPF = $t01729317421._4
417417 let positionSizeAbs = abs(positionSize)
418418 if ((positionSizeAbs == 0))
419419 then throw("Invalid position size")
420420 else {
421421 let isShort = (0 > positionSize)
422422 let positionNotional = if ((_option == PNL_OPTION_SPOT))
423423 then {
424424 let $t01766817792 = swapOutput(!(isShort), positionSizeAbs, false)
425425 let outPositionNotional = $t01766817792._1
426426 let x1 = $t01766817792._2
427427 let x2 = $t01766817792._3
428428 let x3 = $t01766817792._4
429429 outPositionNotional
430430 }
431431 else muld(positionSizeAbs, getOracleTwapPrice())
432432 let unrealizedPnl = if (isShort)
433433 then (positionOpenNotional - positionNotional)
434434 else (positionNotional - positionOpenNotional)
435435 $Tuple2(positionNotional, unrealizedPnl)
436436 }
437437 }
438438
439439
440440 func calcMarginRatio (_remainMargin,_badDebt,_positionNotional) = divd((_remainMargin - _badDebt), _positionNotional)
441441
442442
443443 func getMarginRatioByOption (_trader,_option) = {
444444 let $t01846218573 = getPosition(_trader)
445445 let positionSize = $t01846218573._1
446446 let positionMargin = $t01846218573._2
447447 let pon = $t01846218573._3
448448 let positionLstUpdCPF = $t01846218573._4
449449 let $t01857918672 = getPositionNotionalAndUnrealizedPnl(_trader, _option)
450450 let positionNotional = $t01857918672._1
451451 let unrealizedPnl = $t01857918672._2
452452 let $t01867718843 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
453453 let remainMargin = $t01867718843._1
454454 let badDebt = $t01867718843._2
455455 calcMarginRatio(remainMargin, badDebt, positionNotional)
456456 }
457457
458458
459459 func getMarginRatio (_trader) = getMarginRatioByOption(_trader, PNL_OPTION_SPOT)
460460
461461
462462 func internalClosePosition (_trader,_checkMaxPriceImpact) = {
463463 let $t01919019318 = getPosition(_trader)
464464 let positionSize = $t01919019318._1
465465 let positionMargin = $t01919019318._2
466466 let positionOpenNotional = $t01919019318._3
467467 let positionLstUpdCPF = $t01919019318._4
468468 let $t01932419411 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
469469 let x1 = $t01932419411._1
470470 let unrealizedPnl = $t01932419411._2
471471 let $t01941619584 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
472472 let remainMargin = $t01941619584._1
473473 let badDebt = $t01941619584._2
474474 let exchangedPositionSize = -(positionSize)
475475 let realizedPnl = unrealizedPnl
476476 let marginToVault = -(remainMargin)
477477 let $t01971120022 = swapOutput((positionSize > 0), abs(positionSize), _checkMaxPriceImpact)
478478 let exchangedQuoteAssetAmount = $t01971120022._1
479479 let quoteAssetReserveAfter = $t01971120022._2
480480 let baseAssetReserveAfter = $t01971120022._3
481481 let totalPositionSizeAfter = $t01971120022._4
482482 let cumulativeNotionalAfter = $t01971120022._5
483483 let totalLongAfter = $t01971120022._6
484484 let totalShortAfter = $t01971120022._7
485485 let openInterestNotionalAfter = (openInterestNotional() - positionOpenNotional)
486486 $Tuple12(exchangedPositionSize, badDebt, realizedPnl, marginToVault, quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, exchangedQuoteAssetAmount, totalLongAfter, totalShortAfter)
487487 }
488488
489489
490490 func getTwapSpotPrice () = {
491491 let minuteId = ((lastBlock.timestamp / 1000) / 60)
492492 let startMinuteId = (minuteId - TWAP_INTERVAL)
493493 let listStr = valueOrElse(getString(this, k_lastDataStr), "")
494494 let list = split(listStr, ",")
495495 func filterFn (accumulator,next) = if ((startMinuteId >= parseIntValue(next)))
496496 then (accumulator :+ parseIntValue(next))
497497 else accumulator
498498
499499 let listF = {
500500 let $l = list
501501 let $s = size($l)
502502 let $acc0 = nil
503503 func $f0_1 ($a,$i) = if (($i >= $s))
504504 then $a
505505 else filterFn($a, $l[$i])
506506
507507 func $f0_2 ($a,$i) = if (($i >= $s))
508508 then $a
509509 else throw("List size exceeds 20")
510510
511511 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20)
512512 }
513513 let maxIndex = if ((size(listF) > 0))
514514 then max(listF)
515515 else parseIntValue(list[0])
516516 let lastMinuteId = valueOrElse(getInteger(this, k_lastMinuteId), 0)
517517 let endLastCumulativePrice = valueOrElse(getInteger(this, ((k_twapDataLastCumulativePrice + "_") + toString(lastMinuteId))), 0)
518518 let endLastPrice = valueOrElse(getInteger(this, ((k_twapDataLastPrice + "_") + toString(lastMinuteId))), 0)
519519 let nowCumulativePrice = (endLastCumulativePrice + ((minuteId - lastMinuteId) * endLastPrice))
520520 let startLastCumulativePrice = valueOrElse(getInteger(this, ((k_twapDataLastCumulativePrice + "_") + toString(maxIndex))), 0)
521521 let startLastPrice = valueOrElse(getInteger(this, ((k_twapDataLastPrice + "_") + toString(maxIndex))), 0)
522522 let startCumulativePrice = (startLastCumulativePrice + ((startMinuteId - maxIndex) * startLastPrice))
523523 ((nowCumulativePrice - startCumulativePrice) / TWAP_INTERVAL)
524524 }
525525
526526
527527 func updateSettings (_initMarginRatio,_mmr,_liquidationFeeRatio,_fundingPeriod,_fee,_spreadLimit,_maxPriceImpact) = [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)]
528528
529529
530530 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)]
531531
532532
533533 func updatePosition (_address,_size,_margin,_openNotional,_latestCumulativePremiumFraction) = [IntegerEntry(toCompositeKey(k_positionSize, _address), _size), IntegerEntry(toCompositeKey(k_positionMargin, _address), _margin), IntegerEntry(toCompositeKey(k_positionOpenNotional, _address), _openNotional), IntegerEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _address), _latestCumulativePremiumFraction)]
534534
535535
536536 func appendTwap (price) = {
537537 let minuteId = ((lastBlock.timestamp / 1000) / 60)
538538 let previousMinuteId = valueOrElse(getInteger(this, k_lastMinuteId), 0)
539539 if ((previousMinuteId > minuteId))
540540 then throw("TWAP out-of-order")
541541 else {
542542 let lastMinuteId = if ((previousMinuteId == 0))
543543 then minuteId
544544 else previousMinuteId
545545 if ((minuteId > previousMinuteId))
546546 then {
547547 let prevCumulativePrice = valueOrElse(getInteger(this, ((k_twapDataLastCumulativePrice + "_") + toString(previousMinuteId))), 0)
548548 let prevPrice = valueOrElse(getInteger(this, ((k_twapDataLastPrice + "_") + toString(previousMinuteId))), price)
549549 let lastCumulativePrice = (prevCumulativePrice + ((minuteId - lastMinuteId) * prevPrice))
550550 let list = pushToQueue(strToList(valueOrElse(getString(this, k_lastDataStr), "")), TWAP_INTERVAL, toString(minuteId))
551551 [IntegerEntry(toCompositeKey(k_twapDataLastCumulativePrice, toString(minuteId)), lastCumulativePrice), IntegerEntry(toCompositeKey(k_twapDataLastPrice, toString(minuteId)), price), IntegerEntry(toCompositeKey(k_twapDataPreviousMinuteId, toString(minuteId)), previousMinuteId), IntegerEntry(k_lastMinuteId, minuteId), StringEntry(k_lastDataStr, listToStr(list))]
552552 }
553553 else {
554554 let twapDataPreviousMinuteId = valueOrElse(getInteger(this, toCompositeKey(k_twapDataPreviousMinuteId, toString(minuteId))), 0)
555555 let prevCumulativePrice = valueOrElse(getInteger(this, toCompositeKey(k_twapDataLastCumulativePrice, toString(twapDataPreviousMinuteId))), 0)
556556 let prevPrice = valueOrElse(getInteger(this, toCompositeKey(k_twapDataLastPrice, toString(twapDataPreviousMinuteId))), price)
557557 let lastCumulativePrice = (prevCumulativePrice + ((minuteId - twapDataPreviousMinuteId) * prevPrice))
558558 [IntegerEntry(toCompositeKey(k_twapDataLastCumulativePrice, toString(minuteId)), lastCumulativePrice), IntegerEntry(toCompositeKey(k_twapDataLastPrice, toString(minuteId)), price)]
559559 }
560560 }
561561 }
562562
563563
564564 func updateAmmReserves (_qtAstR,_bsAstR) = [IntegerEntry(k_quoteAssetReserve, _qtAstR), IntegerEntry(k_baseAssetReserve, _bsAstR)]
565565
566566
567567 func updateAmm (_qtAstR,_bsAstR,_totalPositionSizeAfter,_cumulativeNotionalAfter,_openInterestNotional,_totalLongPositionSize,_totalShortPositionSize) = if (((_totalLongPositionSize - _totalShortPositionSize) != _totalPositionSizeAfter))
568568 then throw(((((("Invalid AMM state data: " + toString(_totalLongPositionSize)) + " + ") + toString(_totalShortPositionSize)) + " != ") + toString(_totalPositionSizeAfter)))
569569 else ((updateAmmReserves(_qtAstR, _bsAstR) ++ [IntegerEntry(k_totalPositionSize, _totalPositionSizeAfter), IntegerEntry(k_cumulativeNotional, _cumulativeNotionalAfter), IntegerEntry(k_openInterestNotional, _openInterestNotional), IntegerEntry(k_totalLongPositionSize, _totalLongPositionSize), IntegerEntry(k_totalShortPositionSize, _totalShortPositionSize)]) ++ appendTwap(divd(_qtAstR, _bsAstR)))
570570
571571
572572 func deletePosition (_address) = [DeleteEntry(toCompositeKey(k_positionSize, _address)), DeleteEntry(toCompositeKey(k_positionMargin, _address)), DeleteEntry(toCompositeKey(k_positionOpenNotional, _address)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _address))]
573573
574574
575575 func withdraw (_address,_amount) = {
576576 let balance = assetBalance(this, quoteAsset())
577577 if ((_amount > balance))
578578 then throw(((("Unable to withdraw " + toString(_amount)) + " from contract balance ") + toString(balance)))
579579 else [ScriptTransfer(_address, _amount, quoteAsset())]
580580 }
581581
582582
583583 func updateBalance (i) = if ((0 > i))
584584 then throw("Balance")
585585 else [IntegerEntry(k_balance, i)]
586586
587587
588588 func transferFee (i) = [ScriptTransfer(stakingAddress(), i, quoteAsset())]
589589
590590
591591 @Callable(i)
592592 func pause () = if ((i.caller != adminAddress()))
593593 then throw("Invalid togglePause params")
594594 else [BooleanEntry(k_paused, true)]
595595
596596
597597
598598 @Callable(i)
599599 func unpause () = if ((i.caller != adminAddress()))
600600 then throw("Invalid togglePause params")
601601 else [BooleanEntry(k_paused, false)]
602602
603603
604604
605605 @Callable(i)
606606 func addLiquidity (_quoteAssetAmount) = if (if ((i.caller != adminAddress()))
607607 then true
608608 else (0 >= _quoteAssetAmount))
609609 then throw("Invalid addLiquidity params")
610610 else {
611611 let _qtAstR = qtAstR()
612612 let _bsAstR = bsAstR()
613613 let price = divd(_qtAstR, _bsAstR)
614614 let baseAssetAmountToAdd = divd(_quoteAssetAmount, price)
615615 let qtAstRAfter = (_qtAstR + _quoteAssetAmount)
616616 let bsAstRAfter = (_bsAstR + baseAssetAmountToAdd)
617617 updateAmmReserves(qtAstRAfter, bsAstRAfter)
618618 }
619619
620620
621621
622622 @Callable(i)
623623 func removeLiquidity (_quoteAssetAmount) = if (if ((i.caller != adminAddress()))
624624 then true
625625 else (0 >= _quoteAssetAmount))
626626 then throw("Invalid removeLiquidity params")
627627 else {
628628 let _qtAstR = qtAstR()
629629 let _bsAstR = bsAstR()
630630 let price = divd(_qtAstR, _bsAstR)
631631 let baseAssetAmountToRemove = divd(_quoteAssetAmount, price)
632632 let qtAstRAfter = (_qtAstR - _quoteAssetAmount)
633633 let bsAstRAfter = (_bsAstR - baseAssetAmountToRemove)
634634 updateAmmReserves(qtAstRAfter, bsAstRAfter)
635635 }
636636
637637
638638
639639 @Callable(i)
640640 func changeSettings (_initMarginRatio,_mmr,_liquidationFeeRatio,_fundingPeriod,_fee,_spreadLimit,_maxPriceImpact) = if ((i.caller != adminAddress()))
641641 then throw("Invalid changeSettings params")
642642 else updateSettings(_initMarginRatio, _mmr, _liquidationFeeRatio, _fundingPeriod, _fee, _spreadLimit, _maxPriceImpact)
643643
644644
645645
646646 @Callable(i)
647647 func initialize (_qtAstR,_bsAstR,_fundingPeriod,_initMarginRatio,_mmr,_liquidationFeeRatio,_fee,_oracle,_oracleKey,_coordinator,_spreadLimit,_maxPriceImpact) = if (if (if (if (if (if (if (if (if (if ((0 >= _qtAstR))
648648 then true
649649 else (0 >= _bsAstR))
650650 then true
651651 else (0 >= _fundingPeriod))
652652 then true
653653 else (0 >= _initMarginRatio))
654654 then true
655655 else (0 >= _mmr))
656656 then true
657657 else (0 >= _liquidationFeeRatio))
658658 then true
659659 else (0 >= _fee))
660660 then true
661661 else (0 >= _spreadLimit))
662662 then true
663663 else (0 >= _maxPriceImpact))
664664 then true
665665 else initialized())
666666 then throw("Invalid initialize parameters")
667667 else ((((updateAmm(_qtAstR, _bsAstR, 0, 0, 0, 0, 0) ++ updateSettings(_initMarginRatio, _mmr, _liquidationFeeRatio, _fundingPeriod, _fee, _spreadLimit, _maxPriceImpact)) ++ updateFunding((lastBlock.timestamp + _fundingPeriod), 0, 0, 0, 0)) ++ updateBalance(0)) ++ [BooleanEntry(k_initialized, true), StringEntry(k_ora, _oracle), StringEntry(k_ora_key, _oracleKey), StringEntry(k_coordinatorAddress, _coordinator)])
668668
669669
670670
671671 @Callable(i)
672+func setInitMarginRatio (_initMarginRatio) = if (if ((0 >= _initMarginRatio))
673+ then true
674+ else !(initialized()))
675+ then throw("Invalid setInitMarginRatio parameters")
676+ else updateSettings(_initMarginRatio, maintenanceMarginRatio(), liquidationFeeRatio(), fundingPeriodRaw(), fee(), spreadLimit(), maxPriceImpact())
677+
678+
679+
680+@Callable(i)
672681 func decreasePosition (_amount,_leverage,_minBaseAssetAmount) = if (if (if (if (if ((0 >= _amount))
673682 then true
674683 else !(initialized()))
675684 then true
676685 else !(requireMoreMarginRatio(divd(DECIMAL_UNIT, _leverage), initMarginRatio(), true)))
677686 then true
678687 else !(requireOpenPosition(toString(i.caller))))
679688 then true
680689 else paused())
681690 then throw("Invalid decreasePosition parameters")
682691 else {
683- let $t03093131083 = getPosition(toString(i.caller))
684- let oldPositionSize = $t03093131083._1
685- let oldPositionMargin = $t03093131083._2
686- let oldPositionOpenNotional = $t03093131083._3
687- let oldPositionLstUpdCPF = $t03093131083._4
692+ let $t03133431486 = getPosition(toString(i.caller))
693+ let oldPositionSize = $t03133431486._1
694+ let oldPositionMargin = $t03133431486._2
695+ let oldPositionOpenNotional = $t03133431486._3
696+ let oldPositionLstUpdCPF = $t03133431486._4
688697 let _direction = if ((oldPositionSize > 0))
689698 then DIR_SHORT
690699 else DIR_LONG
691700 let isAdd = (_direction == DIR_LONG)
692701 let openNotional = muld(_amount, _leverage)
693- let $t03125631372 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
694- let oldPositionNotional = $t03125631372._1
695- let unrealizedPnl = $t03125631372._2
696- let $t03137833927 = if ((oldPositionNotional > openNotional))
702+ let $t03165931775 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
703+ let oldPositionNotional = $t03165931775._1
704+ let unrealizedPnl = $t03165931775._2
705+ let $t03178134330 = if ((oldPositionNotional > openNotional))
697706 then {
698- let $t03175531974 = swapInput(isAdd, openNotional)
699- let exchangedPositionSize = $t03175531974._1
700- let quoteAssetReserveAfter = $t03175531974._2
701- let baseAssetReserveAfter = $t03175531974._3
702- let totalPositionSizeAfter = $t03175531974._4
703- let cumulativeNotionalAfter = $t03175531974._5
707+ let $t03215832377 = swapInput(isAdd, openNotional)
708+ let exchangedPositionSize = $t03215832377._1
709+ let quoteAssetReserveAfter = $t03215832377._2
710+ let baseAssetReserveAfter = $t03215832377._3
711+ let totalPositionSizeAfter = $t03215832377._4
712+ let cumulativeNotionalAfter = $t03215832377._5
704713 let exchangedPositionSizeAbs = abs(exchangedPositionSize)
705714 if (if ((_minBaseAssetAmount != 0))
706715 then (_minBaseAssetAmount > exchangedPositionSizeAbs)
707716 else false)
708717 then throw(((("Too little base asset exchanged, got " + toString(exchangedPositionSizeAbs)) + " expected ") + toString(_minBaseAssetAmount)))
709718 else {
710719 let realizedPnl = divd(muld(unrealizedPnl, exchangedPositionSizeAbs), abs(oldPositionSize))
711- let $t03241132656 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
712- let remainMargin = $t03241132656._1
713- let badDebt = $t03241132656._2
714- let fundingPayment = $t03241132656._3
720+ let $t03281433059 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
721+ let remainMargin = $t03281433059._1
722+ let badDebt = $t03281433059._2
723+ let fundingPayment = $t03281433059._3
715724 let exchangedQuoteAssetAmount = openNotional
716725 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
717726 let remainOpenNotional = if ((oldPositionSize > 0))
718727 then ((oldPositionNotional - exchangedQuoteAssetAmount) - unrealizedPnlAfter)
719728 else ((unrealizedPnlAfter + oldPositionNotional) - exchangedQuoteAssetAmount)
720729 let newPositionSize = (oldPositionSize + exchangedPositionSize)
721730 $Tuple11(newPositionSize, remainMargin, abs(remainOpenNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() - openNotional), (totalLongPositionSize() - (if ((newPositionSize > 0))
722731 then abs(exchangedPositionSize)
723732 else 0)), (totalShortPositionSize() - (if ((0 > newPositionSize))
724733 then abs(exchangedPositionSize)
725734 else 0)))
726735 }
727736 }
728737 else throw("Close position first")
729- let newPositionSize = $t03137833927._1
730- let newPositionRemainMargin = $t03137833927._2
731- let newPositionOpenNotional = $t03137833927._3
732- let newPositionLatestCPF = $t03137833927._4
733- let baseAssetReserveAfter = $t03137833927._5
734- let quoteAssetReserveAfter = $t03137833927._6
735- let totalPositionSizeAfter = $t03137833927._7
736- let cumulativeNotionalAfter = $t03137833927._8
737- let openInterestNotionalAfter = $t03137833927._9
738- let totalLongAfter = $t03137833927._10
739- let totalShortAfter = $t03137833927._11
738+ let newPositionSize = $t03178134330._1
739+ let newPositionRemainMargin = $t03178134330._2
740+ let newPositionOpenNotional = $t03178134330._3
741+ let newPositionLatestCPF = $t03178134330._4
742+ let baseAssetReserveAfter = $t03178134330._5
743+ let quoteAssetReserveAfter = $t03178134330._6
744+ let totalPositionSizeAfter = $t03178134330._7
745+ let cumulativeNotionalAfter = $t03178134330._8
746+ let openInterestNotionalAfter = $t03178134330._9
747+ let totalLongAfter = $t03178134330._10
748+ let totalShortAfter = $t03178134330._11
740749 let notifyNotional = invoke(minerAddress(), "notifyNotional", [toString(i.caller), newPositionOpenNotional], nil)
741750 if ((notifyNotional == notifyNotional))
742751 then (updatePosition(toString(i.caller), newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter))
743752 else throw("Strict value is not equal to itself.")
744753 }
745754
746755
747756
748757 @Callable(i)
749758 func increasePosition (_direction,_leverage,_minBaseAssetAmount) = {
750759 let _rawAmount = i.payments[0].amount
751760 if (if (if (if (if (if (if ((_direction != DIR_LONG))
752761 then (_direction != DIR_SHORT)
753762 else false)
754763 then true
755764 else (0 >= _rawAmount))
756765 then true
757766 else !(initialized()))
758767 then true
759768 else (i.payments[0].assetId != quoteAsset()))
760769 then true
761770 else !(requireMoreMarginRatio(divd(DECIMAL_UNIT, _leverage), initMarginRatio(), true)))
762771 then true
763772 else paused())
764773 then throw("Invalid increasePosition parameters")
765774 else {
766775 let feeAmount = muld(_rawAmount, fee())
767776 let _amount = (_rawAmount - feeAmount)
768- let $t03504535197 = getPosition(toString(i.caller))
769- let oldPositionSize = $t03504535197._1
770- let oldPositionMargin = $t03504535197._2
771- let oldPositionOpenNotional = $t03504535197._3
772- let oldPositionLstUpdCPF = $t03504535197._4
777+ let $t03544835600 = getPosition(toString(i.caller))
778+ let oldPositionSize = $t03544835600._1
779+ let oldPositionMargin = $t03544835600._2
780+ let oldPositionOpenNotional = $t03544835600._3
781+ let oldPositionLstUpdCPF = $t03544835600._4
773782 let isNewPosition = (oldPositionSize == 0)
774783 let isSameDirection = if ((oldPositionSize > 0))
775784 then (_direction == DIR_LONG)
776785 else (_direction == DIR_SHORT)
777786 let expandExisting = if (!(isNewPosition))
778787 then isSameDirection
779788 else false
780789 let isAdd = (_direction == DIR_LONG)
781- let $t03548637875 = if (if (isNewPosition)
790+ let $t03588938278 = if (if (isNewPosition)
782791 then true
783792 else expandExisting)
784793 then {
785794 let openNotional = muld(_amount, _leverage)
786- let $t03591036116 = swapInput(isAdd, openNotional)
787- let amountBaseAssetBought = $t03591036116._1
788- let quoteAssetReserveAfter = $t03591036116._2
789- let baseAssetReserveAfter = $t03591036116._3
790- let totalPositionSizeAfter = $t03591036116._4
791- let cumulativeNotionalAfter = $t03591036116._5
795+ let $t03631336519 = swapInput(isAdd, openNotional)
796+ let amountBaseAssetBought = $t03631336519._1
797+ let quoteAssetReserveAfter = $t03631336519._2
798+ let baseAssetReserveAfter = $t03631336519._3
799+ let totalPositionSizeAfter = $t03631336519._4
800+ let cumulativeNotionalAfter = $t03631336519._5
792801 if (if ((_minBaseAssetAmount != 0))
793802 then (_minBaseAssetAmount > abs(amountBaseAssetBought))
794803 else false)
795804 then throw(((("Limit error: " + toString(abs(amountBaseAssetBought))) + " < ") + toString(_minBaseAssetAmount)))
796805 else {
797806 let newPositionSize = (oldPositionSize + amountBaseAssetBought)
798807 let increaseMarginRequirement = divd(openNotional, _leverage)
799- let $t03649736736 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
800- let remainMargin = $t03649736736._1
801- let x1 = $t03649736736._2
802- let x2 = $t03649736736._3
808+ let $t03690037139 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
809+ let remainMargin = $t03690037139._1
810+ let x1 = $t03690037139._2
811+ let x2 = $t03690037139._3
803812 $Tuple11(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
804813 then abs(amountBaseAssetBought)
805814 else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
806815 then abs(amountBaseAssetBought)
807816 else 0)))
808817 }
809818 }
810819 else {
811820 let openNotional = muld(_amount, _leverage)
812- let $t03756837684 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
813- let oldPositionNotional = $t03756837684._1
814- let unrealizedPnl = $t03756837684._2
821+ let $t03797138087 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
822+ let oldPositionNotional = $t03797138087._1
823+ let unrealizedPnl = $t03797138087._2
815824 if ((oldPositionNotional > openNotional))
816825 then throw("Use decreasePosition to decrease position size")
817826 else throw("Close position first")
818827 }
819- let newPositionSize = $t03548637875._1
820- let newPositionRemainMargin = $t03548637875._2
821- let newPositionOpenNotional = $t03548637875._3
822- let newPositionLatestCPF = $t03548637875._4
823- let baseAssetReserveAfter = $t03548637875._5
824- let quoteAssetReserveAfter = $t03548637875._6
825- let totalPositionSizeAfter = $t03548637875._7
826- let cumulativeNotionalAfter = $t03548637875._8
827- let openInterestNotionalAfter = $t03548637875._9
828- let totalLongAfter = $t03548637875._10
829- let totalShortAfter = $t03548637875._11
828+ let newPositionSize = $t03588938278._1
829+ let newPositionRemainMargin = $t03588938278._2
830+ let newPositionOpenNotional = $t03588938278._3
831+ let newPositionLatestCPF = $t03588938278._4
832+ let baseAssetReserveAfter = $t03588938278._5
833+ let quoteAssetReserveAfter = $t03588938278._6
834+ let totalPositionSizeAfter = $t03588938278._7
835+ let cumulativeNotionalAfter = $t03588938278._8
836+ let openInterestNotionalAfter = $t03588938278._9
837+ let totalLongAfter = $t03588938278._10
838+ let totalShortAfter = $t03588938278._11
830839 let feeToStakers = (feeAmount / 2)
831840 let feeToInsurance = (feeAmount - feeToStakers)
832841 let stake = invoke(quoteAssetStaking(), "lockNeutrinoSP", [toString(stakingAddress()), ALL_FEES], [AttachedPayment(quoteAsset(), _amount)])
833842 if ((stake == stake))
834843 then {
835844 let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
836845 if ((depositInsurance == depositInsurance))
837846 then {
838847 let notifyFee = invoke(minerAddress(), "notifyFees", [toString(i.caller), feeAmount], nil)
839848 if ((notifyFee == notifyFee))
840849 then {
841850 let notifyNotional = invoke(minerAddress(), "notifyNotional", [toString(i.caller), newPositionOpenNotional], nil)
842851 if ((notifyNotional == notifyNotional))
843852 then (((updatePosition(toString(i.caller), newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount)))
844853 else throw("Strict value is not equal to itself.")
845854 }
846855 else throw("Strict value is not equal to itself.")
847856 }
848857 else throw("Strict value is not equal to itself.")
849858 }
850859 else throw("Strict value is not equal to itself.")
851860 }
852861 }
853862
854863
855864
856865 @Callable(i)
857866 func addMargin () = {
858867 let _rawAmount = i.payments[0].amount
859868 if (if (if (if ((i.payments[0].assetId != quoteAsset()))
860869 then true
861870 else !(requireOpenPosition(toString(i.caller))))
862871 then true
863872 else !(initialized()))
864873 then true
865874 else paused())
866875 then throw("Invalid addMargin parameters")
867876 else {
868877 let feeAmount = muld(_rawAmount, fee())
869878 let _amount = (_rawAmount - feeAmount)
870- let $t03943139583 = getPosition(toString(i.caller))
871- let oldPositionSize = $t03943139583._1
872- let oldPositionMargin = $t03943139583._2
873- let oldPositionOpenNotional = $t03943139583._3
874- let oldPositionLstUpdCPF = $t03943139583._4
879+ let $t03983439986 = getPosition(toString(i.caller))
880+ let oldPositionSize = $t03983439986._1
881+ let oldPositionMargin = $t03983439986._2
882+ let oldPositionOpenNotional = $t03983439986._3
883+ let oldPositionLstUpdCPF = $t03983439986._4
875884 let feeToStakers = (feeAmount / 2)
876885 let feeToInsurance = (feeAmount - feeToStakers)
877886 let stake = invoke(quoteAssetStaking(), "lockNeutrinoSP", [toString(stakingAddress()), ALL_FEES], [AttachedPayment(quoteAsset(), _amount)])
878887 if ((stake == stake))
879888 then {
880889 let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
881890 if ((depositInsurance == depositInsurance))
882891 then {
883892 let notifyFee = invoke(minerAddress(), "notifyFees", [toString(i.caller), feeAmount], nil)
884893 if ((notifyFee == notifyFee))
885894 then ((updatePosition(toString(i.caller), oldPositionSize, (oldPositionMargin + _amount), oldPositionOpenNotional, oldPositionLstUpdCPF) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount)))
886895 else throw("Strict value is not equal to itself.")
887896 }
888897 else throw("Strict value is not equal to itself.")
889898 }
890899 else throw("Strict value is not equal to itself.")
891900 }
892901 }
893902
894903
895904
896905 @Callable(i)
897906 func removeMargin (_amount) = if (if (if (if ((0 >= _amount))
898907 then true
899908 else !(requireOpenPosition(toString(i.caller))))
900909 then true
901910 else !(initialized()))
902911 then true
903912 else paused())
904913 then throw("Invalid removeMargin parameters")
905914 else {
906- let $t04063240784 = getPosition(toString(i.caller))
907- let oldPositionSize = $t04063240784._1
908- let oldPositionMargin = $t04063240784._2
909- let oldPositionOpenNotional = $t04063240784._3
910- let oldPositionLstUpdCPF = $t04063240784._4
915+ let $t04103541187 = getPosition(toString(i.caller))
916+ let oldPositionSize = $t04103541187._1
917+ let oldPositionMargin = $t04103541187._2
918+ let oldPositionOpenNotional = $t04103541187._3
919+ let oldPositionLstUpdCPF = $t04103541187._4
911920 let marginDelta = -(_amount)
912- let $t04082141000 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
913- let remainMargin = $t04082141000._1
914- let badDebt = $t04082141000._2
921+ let $t04122441403 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
922+ let remainMargin = $t04122441403._1
923+ let badDebt = $t04122441403._2
915924 if ((badDebt != 0))
916925 then throw("Invalid removed margin amount")
917926 else {
918927 let marginRatio = calcMarginRatio(remainMargin, badDebt, oldPositionOpenNotional)
919928 if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
920929 then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
921930 else {
922931 let unstake = invoke(quoteAssetStaking(), "unlockNeutrino", [_amount, toBase58String(quoteAsset())], nil)
923932 if ((unstake == unstake))
924933 then ((updatePosition(toString(i.caller), oldPositionSize, remainMargin, oldPositionOpenNotional, latestCumulativePremiumFraction(oldPositionSize)) ++ withdraw(i.caller, _amount)) ++ updateBalance((cbalance() - _amount)))
925934 else throw("Strict value is not equal to itself.")
926935 }
927936 }
928937 }
929938
930939
931940
932941 @Callable(i)
933942 func closePosition () = if (if (if (!(requireOpenPosition(toString(i.caller))))
934943 then true
935944 else !(initialized()))
936945 then true
937946 else paused())
938947 then throw("Invalid closePosition parameters")
939948 else {
940- let $t04209642486 = internalClosePosition(toString(i.caller), true)
941- let x1 = $t04209642486._1
942- let positionBadDebt = $t04209642486._2
943- let realizedPnl = $t04209642486._3
944- let marginToVault = $t04209642486._4
945- let quoteAssetReserveAfter = $t04209642486._5
946- let baseAssetReserveAfter = $t04209642486._6
947- let totalPositionSizeAfter = $t04209642486._7
948- let cumulativeNotionalAfter = $t04209642486._8
949- let openInterestNotionalAfter = $t04209642486._9
950- let x2 = $t04209642486._10
951- let totalLongAfter = $t04209642486._11
952- let totalShortAfter = $t04209642486._12
949+ let $t04249942889 = internalClosePosition(toString(i.caller), true)
950+ let x1 = $t04249942889._1
951+ let positionBadDebt = $t04249942889._2
952+ let realizedPnl = $t04249942889._3
953+ let marginToVault = $t04249942889._4
954+ let quoteAssetReserveAfter = $t04249942889._5
955+ let baseAssetReserveAfter = $t04249942889._6
956+ let totalPositionSizeAfter = $t04249942889._7
957+ let cumulativeNotionalAfter = $t04249942889._8
958+ let openInterestNotionalAfter = $t04249942889._9
959+ let x2 = $t04249942889._10
960+ let totalLongAfter = $t04249942889._11
961+ let totalShortAfter = $t04249942889._12
953962 if ((positionBadDebt > 0))
954963 then throw("Unable to close position with bad debt")
955964 else {
956965 let withdrawAmount = abs(marginToVault)
957966 let ammBalance = (cbalance() - withdrawAmount)
958- let $t04269542837 = if ((0 > ammBalance))
967+ let $t04309843240 = if ((0 > ammBalance))
959968 then $Tuple2(0, abs(ammBalance))
960969 else $Tuple2(ammBalance, 0)
961- let ammNewBalance = $t04269542837._1
962- let getFromInsurance = $t04269542837._2
970+ let ammNewBalance = $t04309843240._1
971+ let getFromInsurance = $t04309843240._2
963972 let x = if ((getFromInsurance > 0))
964973 then {
965974 let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [getFromInsurance], nil)
966975 if ((withdrawInsurance == withdrawInsurance))
967976 then nil
968977 else throw("Strict value is not equal to itself.")
969978 }
970979 else nil
971980 if ((x == x))
972981 then {
973982 let unstake = invoke(quoteAssetStaking(), "unlockNeutrino", [(withdrawAmount - getFromInsurance), toBase58String(quoteAsset())], nil)
974983 if ((unstake == unstake))
975984 then {
976985 let notifyNotional = invoke(minerAddress(), "notifyNotional", [toString(i.caller), 0], nil)
977986 if ((notifyNotional == notifyNotional))
978987 then (((deletePosition(toString(i.caller)) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(i.caller, withdrawAmount)) ++ updateBalance(ammNewBalance))
979988 else throw("Strict value is not equal to itself.")
980989 }
981990 else throw("Strict value is not equal to itself.")
982991 }
983992 else throw("Strict value is not equal to itself.")
984993 }
985994 }
986995
987996
988997
989998 @Callable(i)
990999 func liquidate (_trader) = {
9911000 let spotMarginRatio = getMarginRatioByOption(_trader, PNL_OPTION_SPOT)
9921001 let marginRatio = if (isOverFluctuationLimit())
9931002 then {
9941003 let oracleMarginRatio = getMarginRatioByOption(_trader, PNL_OPTION_ORACLE)
9951004 vmax(spotMarginRatio, oracleMarginRatio)
9961005 }
9971006 else spotMarginRatio
9981007 if (if (if (if (!(requireMoreMarginRatio(marginRatio, maintenanceMarginRatio(), false)))
9991008 then true
10001009 else !(requireOpenPosition(_trader)))
10011010 then true
10021011 else !(initialized()))
10031012 then true
10041013 else paused())
10051014 then throw("Unable to liquidate")
10061015 else {
1007- let $t04452744950 = internalClosePosition(_trader, false)
1008- let x1 = $t04452744950._1
1009- let badDebt = $t04452744950._2
1010- let x2 = $t04452744950._3
1011- let x3 = $t04452744950._4
1012- let quoteAssetReserveAfter = $t04452744950._5
1013- let baseAssetReserveAfter = $t04452744950._6
1014- let totalPositionSizeAfter = $t04452744950._7
1015- let cumulativeNotionalAfter = $t04452744950._8
1016- let openInterestNotionalAfter = $t04452744950._9
1017- let exchangedQuoteAssetAmount = $t04452744950._10
1018- let totalLongAfter = $t04452744950._11
1019- let totalShortAfter = $t04452744950._12
1016+ let $t04493045353 = internalClosePosition(_trader, false)
1017+ let x1 = $t04493045353._1
1018+ let badDebt = $t04493045353._2
1019+ let x2 = $t04493045353._3
1020+ let x3 = $t04493045353._4
1021+ let quoteAssetReserveAfter = $t04493045353._5
1022+ let baseAssetReserveAfter = $t04493045353._6
1023+ let totalPositionSizeAfter = $t04493045353._7
1024+ let cumulativeNotionalAfter = $t04493045353._8
1025+ let openInterestNotionalAfter = $t04493045353._9
1026+ let exchangedQuoteAssetAmount = $t04493045353._10
1027+ let totalLongAfter = $t04493045353._11
1028+ let totalShortAfter = $t04493045353._12
10201029 let liquidationPenalty = muld(exchangedQuoteAssetAmount, liquidationFeeRatio())
10211030 let feeToLiquidator = (liquidationPenalty / 2)
10221031 let feeToInsurance = (liquidationPenalty - feeToLiquidator)
10231032 let ammBalance = (cbalance() - liquidationPenalty)
1024- let $t04534845483 = if ((0 > ammBalance))
1033+ let $t04575145886 = if ((0 > ammBalance))
10251034 then $Tuple2(0, abs(ammBalance))
10261035 else $Tuple2(ammBalance, 0)
1027- let newAmmBalance = $t04534845483._1
1028- let takeFromInsurance = $t04534845483._2
1036+ let newAmmBalance = $t04575145886._1
1037+ let takeFromInsurance = $t04575145886._2
10291038 let x = if ((takeFromInsurance > 0))
10301039 then {
10311040 let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
10321041 if ((withdrawInsurance == withdrawInsurance))
10331042 then nil
10341043 else throw("Strict value is not equal to itself.")
10351044 }
10361045 else nil
10371046 if ((x == x))
10381047 then {
10391048 let unstake = invoke(quoteAssetStaking(), "unlockNeutrino", [(liquidationPenalty - takeFromInsurance), toBase58String(quoteAsset())], nil)
10401049 if ((unstake == unstake))
10411050 then {
10421051 let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
10431052 if ((depositInsurance == depositInsurance))
10441053 then {
10451054 let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
10461055 if ((notifyNotional == notifyNotional))
10471056 then (((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
10481057 else throw("Strict value is not equal to itself.")
10491058 }
10501059 else throw("Strict value is not equal to itself.")
10511060 }
10521061 else throw("Strict value is not equal to itself.")
10531062 }
10541063 else throw("Strict value is not equal to itself.")
10551064 }
10561065 }
10571066
10581067
10591068
10601069 @Callable(i)
10611070 func payFunding () = {
10621071 let fundingBlockTimestamp = nextFundingBlockTimestamp()
10631072 if (if (if ((fundingBlockTimestamp > lastBlock.timestamp))
10641073 then true
10651074 else !(initialized()))
10661075 then true
10671076 else paused())
10681077 then throw(((("Invalid funding block timestamp: " + toString(lastBlock.timestamp)) + " < ") + toString(fundingBlockTimestamp)))
10691078 else {
10701079 let underlyingPrice = getOracleTwapPrice()
10711080 let spotTwapPrice = getTwapSpotPrice()
10721081 let premium = (spotTwapPrice - underlyingPrice)
1073- let $t04697748312 = if (if ((totalShortPositionSize() == 0))
1082+ let $t04738048715 = if (if ((totalShortPositionSize() == 0))
10741083 then true
10751084 else (totalLongPositionSize() == 0))
10761085 then $Tuple2(0, 0)
10771086 else if ((0 > premium))
10781087 then {
10791088 let shortPremiumFraction = divd(muld(premium, fundingPeriodDecimal()), ONE_DAY)
10801089 let longPremiumFraction = divd(muld(shortPremiumFraction, totalShortPositionSize()), totalLongPositionSize())
10811090 $Tuple2(shortPremiumFraction, longPremiumFraction)
10821091 }
10831092 else {
10841093 let longPremiumFraction = divd(muld(premium, fundingPeriodDecimal()), ONE_DAY)
10851094 let shortPremiumFraction = divd(muld(longPremiumFraction, totalLongPositionSize()), totalShortPositionSize())
10861095 $Tuple2(shortPremiumFraction, longPremiumFraction)
10871096 }
1088- let shortPremiumFraction = $t04697748312._1
1089- let longPremiumFraction = $t04697748312._2
1097+ let shortPremiumFraction = $t04738048715._1
1098+ let longPremiumFraction = $t04738048715._2
10901099 updateFunding((fundingBlockTimestamp + fundingPeriodSeconds()), (latestLongCumulativePremiumFraction() + longPremiumFraction), (latestShortCumulativePremiumFraction() + shortPremiumFraction), divd(longPremiumFraction, underlyingPrice), divd(shortPremiumFraction, underlyingPrice))
10911100 }
10921101 }
10931102
10941103
10951104
10961105 @Callable(i)
10971106 func v_get (_trader) = {
1098- let $t04868748747 = internalClosePosition(_trader, false)
1099- let x1 = $t04868748747._1
1100- let x2 = $t04868748747._2
1101- let x3 = $t04868748747._3
1102- let x4 = $t04868748747._4
1107+ let $t04909049150 = internalClosePosition(_trader, false)
1108+ let x1 = $t04909049150._1
1109+ let x2 = $t04909049150._2
1110+ let x3 = $t04909049150._3
1111+ let x4 = $t04909049150._4
11031112 throw((((s(x2) + s(x3)) + s(x4)) + s(getMarginRatio(_trader))))
11041113 }
11051114
11061115
11071116
11081117 @Callable(i)
11091118 func view_calcRemainMarginWithFundingPayment (_trader) = {
1110- let $t04889449005 = getPosition(_trader)
1111- let positionSize = $t04889449005._1
1112- let positionMargin = $t04889449005._2
1113- let pon = $t04889449005._3
1114- let positionLstUpdCPF = $t04889449005._4
1115- let $t04901049111 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1116- let positionNotional = $t04901049111._1
1117- let unrealizedPnl = $t04901049111._2
1118- let $t04911649298 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1119- let remainMargin = $t04911649298._1
1120- let badDebt = $t04911649298._2
1121- let fundingPayment = $t04911649298._3
1119+ let $t04929749408 = getPosition(_trader)
1120+ let positionSize = $t04929749408._1
1121+ let positionMargin = $t04929749408._2
1122+ let pon = $t04929749408._3
1123+ let positionLstUpdCPF = $t04929749408._4
1124+ let $t04941349514 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1125+ let positionNotional = $t04941349514._1
1126+ let unrealizedPnl = $t04941349514._2
1127+ let $t04951949701 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1128+ let remainMargin = $t04951949701._1
1129+ let badDebt = $t04951949701._2
1130+ let fundingPayment = $t04951949701._3
11221131 throw(((((s(remainMargin) + s(fundingPayment)) + s(getMarginRatio(_trader))) + s(unrealizedPnl)) + s(badDebt)))
11231132 }
11241133
11251134
11261135
11271136 @Callable(i)
11281137 func forceMoveAsset (_trader,_amount) = if (if ((addressFromPublicKey(adminPublicKey()) != i.caller))
11291138 then true
11301139 else (0 > _amount))
11311140 then throw("Invalid forceMoveAsset parameters")
11321141 else {
11331142 let unstake = invoke(quoteAssetStaking(), "unlockNeutrino", [_amount, toBase58String(quoteAsset())], nil)
11341143 if ((unstake == unstake))
11351144 then (withdraw(addressFromStringValue(_trader), _amount) ++ updateBalance((cbalance() - _amount)))
11361145 else throw("Strict value is not equal to itself.")
11371146 }
11381147
11391148
11401149 @Verifier(tx)
11411150 func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], adminPublicKey())
11421151

github/deemru/w8io/6500d08 
138.62 ms