tx · 12jwsvNBMHpiNmbzbvMVEzyfmn7cxFCr2QvdXxmqvY2v

3MyKAZyGakn6DvvgePYccmeEFR5XufZHPmD:  -0.06500000 Waves

2022.11.10 12:46 [2310890] smart account 3MyKAZyGakn6DvvgePYccmeEFR5XufZHPmD > SELF 0.00000000 Waves

{ "type": 13, "id": "12jwsvNBMHpiNmbzbvMVEzyfmn7cxFCr2QvdXxmqvY2v", "fee": 6500000, "feeAssetId": null, "timestamp": 1668073613027, "version": 2, "chainId": 84, "sender": "3MyKAZyGakn6DvvgePYccmeEFR5XufZHPmD", "senderPublicKey": "6gi3c71QVs4Magr73Vi1QYfh6uvwCWeRju3mGGSVT2R4", "proofs": [ "2m3FvU6j3LzZeCsGMa3Awy3LoK7AS5iaRBuuV23Fa2g2h2FKF3nzmoMkx5T5zCyeL4GwAnbfnw9mX5ZSBXxggYfh" ], "script": "base64:BgJ2CAISABIAEgMKAQESAwoBARILCgkBAQEBAQEBAQESEAoOAQEBAQEBAQgICAEBAQESAwoBARIFCgMBAQESBgoEAQEBCBIAEgMKAQESABIDCgEIEgASBAoCCAESAwoBARIAEgMKAQgSAwoBCBIDCgEBEgASAwoBCKEBAAlrX29yYV9rZXkCCWtfb3JhX2tleQAPa19vcmFfYmxvY2tfa2V5Ag9rX29yYV9ibG9ja19rZXkABWtfb3JhAgVrX29yYQAJa19iYWxhbmNlAglrX2JhbGFuY2UADmtfcG9zaXRpb25TaXplAg5rX3Bvc2l0aW9uU2l6ZQAQa19wb3NpdGlvbk1hcmdpbgIQa19wb3NpdGlvbk1hcmdpbgAWa19wb3NpdGlvbk9wZW5Ob3Rpb25hbAIWa19wb3NpdGlvbk9wZW5Ob3Rpb25hbAAua19wb3NpdGlvbkxhc3RVcGRhdGVkQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgISa19wb3NpdGlvbkZyYWN0aW9uABRrX3Bvc2l0aW9uQ2xvc2VkRGF0ZQIUa19wb3NpdGlvbkNsb3NlZERhdGUAD2tfcG9zaXRpb25Bc3NldAIPa19wb3NpdGlvbkFzc2V0AA1rX2luaXRpYWxpemVkAg1rX2luaXRpYWxpemVkAAhrX3BhdXNlZAIIa19wYXVzZWQABWtfZmVlAgVrX2ZlZQAPa19mdW5kaW5nUGVyaW9kAg9rX2Z1bmRpbmdQZXJpb2QAEWtfaW5pdE1hcmdpblJhdGlvAhFrX2luaXRNYXJnaW5SYXRpbwAYa19tYWludGVuYW5jZU1hcmdpblJhdGlvAgVrX21tcgAVa19saXF1aWRhdGlvbkZlZVJhdGlvAhVrX2xpcXVpZGF0aW9uRmVlUmF0aW8AGWtfcGFydGlhbExpcXVpZGF0aW9uUmF0aW8CFmtfcGFydExpcXVpZGF0aW9uUmF0aW8ADWtfc3ByZWFkTGltaXQCDWtfc3ByZWFkTGltaXQAEGtfbWF4UHJpY2VJbXBhY3QCEGtfbWF4UHJpY2VJbXBhY3QAEGtfbWF4UHJpY2VTcHJlYWQCEGtfbWF4UHJpY2VTcHJlYWQADWtfbGFzdERhdGFTdHICDWtfbGFzdERhdGFTdHIADmtfbGFzdE1pbnV0ZUlkAg5rX2xhc3RNaW51dGVJZAAda190d2FwRGF0YUxhc3RDdW11bGF0aXZlUHJpY2UCHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlABNrX3R3YXBEYXRhTGFzdFByaWNlAhNrX3R3YXBEYXRhTGFzdFByaWNlABprX3R3YXBEYXRhUHJldmlvdXNNaW51dGVJZAIaa190d2FwRGF0YVByZXZpb3VzTWludXRlSWQAJWtfbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24CG2tfbGF0ZXN0TG9uZ1ByZW1pdW1GcmFjdGlvbgAma19sYXRlc3RTaG9ydEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24CHGtfbGF0ZXN0U2hvcnRQcmVtaXVtRnJhY3Rpb24AEmtfbmV4dEZ1bmRpbmdCbG9jawIea19uZXh0RnVuZGluZ0Jsb2NrTWluVGltZXN0YW1wABFrX2xvbmdGdW5kaW5nUmF0ZQIRa19sb25nRnVuZGluZ1JhdGUAEmtfc2hvcnRGdW5kaW5nUmF0ZQISa19zaG9ydEZ1bmRpbmdSYXRlABNrX3F1b3RlQXNzZXRSZXNlcnZlAghrX3F0QXN0UgASa19iYXNlQXNzZXRSZXNlcnZlAghrX2JzQXN0UgATa190b3RhbFBvc2l0aW9uU2l6ZQITa190b3RhbFBvc2l0aW9uU2l6ZQAXa190b3RhbExvbmdQb3NpdGlvblNpemUCF2tfdG90YWxMb25nUG9zaXRpb25TaXplABhrX3RvdGFsU2hvcnRQb3NpdGlvblNpemUCGGtfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQAUa19jdW11bGF0aXZlTm90aW9uYWwCFGtfY3VtdWxhdGl2ZU5vdGlvbmFsABZrX29wZW5JbnRlcmVzdE5vdGlvbmFsAhZrX29wZW5JbnRlcmVzdE5vdGlvbmFsABRrX2Nvb3JkaW5hdG9yQWRkcmVzcwIUa19jb29yZGluYXRvckFkZHJlc3MAE2tfaW5zdXJhbmNlX2FkZHJlc3MCE2tfaW5zdXJhbmNlX2FkZHJlc3MAD2tfYWRtaW5fYWRkcmVzcwIPa19hZG1pbl9hZGRyZXNzABJrX2FkbWluX3B1YmxpY19rZXkCEmtfYWRtaW5fcHVibGljX2tleQANa19xdW90ZV9hc3NldAINa19xdW90ZV9hc3NldAAPa19xdW90ZV9zdGFraW5nAg9rX3F1b3RlX3N0YWtpbmcAEWtfc3Rha2luZ19hZGRyZXNzAhFrX3N0YWtpbmdfYWRkcmVzcwAPa19taW5lcl9hZGRyZXNzAg9rX21pbmVyX2FkZHJlc3MAEGtfb3JkZXJzX2FkZHJlc3MCEGtfb3JkZXJzX2FkZHJlc3MAEmtfcmVmZXJyYWxfYWRkcmVzcwISa19yZWZlcnJhbF9hZGRyZXNzABFrX21hbmFnZXJfYWRkcmVzcwIRa19tYW5hZ2VyX2FkZHJlc3MAFGtfY29sbGF0ZXJhbF9hZGRyZXNzAhRrX2NvbGxhdGVyYWxfYWRkcmVzcwASa19leGNoYW5nZV9hZGRyZXNzAhJrX2V4Y2hhbmdlX2FkZHJlc3MAFWtfbmZ0X21hbmFnZXJfYWRkcmVzcwIVa19uZnRfbWFuYWdlcl9hZGRyZXNzACBrX3RyYWRlcl9tYXJrZXRfYXNzZXRfY29sbGF0ZXJhbAIga190cmFkZXJfbWFya2V0X2Fzc2V0X2NvbGxhdGVyYWwBDnRvQ29tcG9zaXRlS2V5AgRfa2V5CF9hZGRyZXNzCQCsAgIJAKwCAgUEX2tleQIBXwUIX2FkZHJlc3MBC2Nvb3JkaW5hdG9yAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFBHRoaXMFFGtfY29vcmRpbmF0b3JBZGRyZXNzAhNDb29yZGluYXRvciBub3Qgc2V0AQxhZG1pbkFkZHJlc3MACQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFD2tfYWRtaW5fYWRkcmVzcwEOYWRtaW5QdWJsaWNLZXkACQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFEmtfYWRtaW5fcHVibGljX2tleQEKcXVvdGVBc3NldAAJANkEAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUNa19xdW90ZV9hc3NldAERcXVvdGVBc3NldFN0YWtpbmcACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUPa19xdW90ZV9zdGFraW5nAhtRdW90ZSBhc3NldCBzdGFraW5nIG5vdCBzZXQBDnN0YWtpbmdBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFEWtfc3Rha2luZ19hZGRyZXNzAg9TdGFraW5nIG5vdCBzZXQBEGluc3VyYW5jZUFkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUTa19pbnN1cmFuY2VfYWRkcmVzcwIRSW5zdXJhbmNlIG5vdCBzZXQBDG1pbmVyQWRkcmVzcwAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABQ9rX21pbmVyX2FkZHJlc3MCDU1pbmVyIG5vdCBzZXQBDW9yZGVyc0FkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUQa19vcmRlcnNfYWRkcmVzcwIOT3JkZXJzIG5vdCBzZXQBD3JlZmVycmFsQWRkcmVzcwAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABRJrX3JlZmVycmFsX2FkZHJlc3MCEFJlZmVycmFsIG5vdCBzZXQBDm1hbmFnZXJBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFEWtfbWFuYWdlcl9hZGRyZXNzAg9NYW5hZ2VyIG5vdCBzZXQBEW5mdE1hbmFnZXJBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFFWtfbmZ0X21hbmFnZXJfYWRkcmVzcwITTkZUIE1hbmFnZXIgbm90IHNldAERY29sbGF0ZXJhbEFkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBEUBleHRyTmF0aXZlKDEwNTMpAgkBC2Nvb3JkaW5hdG9yAAUUa19jb2xsYXRlcmFsX2FkZHJlc3MCGkNvbGxhdGVyYWwgTWFuYWdlciBub3Qgc2V0AQtzd2FwQWRkcmVzcwAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ0IAgkBC2Nvb3JkaW5hdG9yAAUSa19leGNoYW5nZV9hZGRyZXNzAg9ObyBzd2FwIGFkZHJlc3MCFEludmFsaWQgc3dhcCBhZGRyZXNzABFrX3doaXRlbGlzdF9hc3NldAIRa193aGl0ZWxpc3RfYXNzZXQBEGlzV2hpdGVsaXN0QXNzZXQBCF9hc3NldElkCQELdmFsdWVPckVsc2UCCQCbCAIJARFjb2xsYXRlcmFsQWRkcmVzcwAJAQ50b0NvbXBvc2l0ZUtleQIFEWtfd2hpdGVsaXN0X2Fzc2V0BQhfYXNzZXRJZAcADWtfdG9rZW5fcGFyYW0CDWtfdG9rZW5fcGFyYW0ADGtfdG9rZW5fdHlwZQIMa190b2tlbl90eXBlABhGRUVfUkVEVUNUSU9OX1RPS0VOX1RZUEUCDWZlZV9yZWR1Y3Rpb24ACERJUl9MT05HAAEACURJUl9TSE9SVAACAA1UV0FQX0lOVEVSVkFMAA8AD09SQUNMRV9JTlRFUlZBTAAPAAdTRUNPTkRTAOgHAAxERUNJTUFMX1VOSVQJAGgCAAEJAGgCCQBoAgkAaAIJAGgCCQBoAgAKAAoACgAKAAoACgAHT05FX0RBWQkAaAIAgKMFBQxERUNJTUFMX1VOSVQACEFMTF9GRUVTAGQAD1BOTF9PUFRJT05fU1BPVAABABFQTkxfT1BUSU9OX09SQUNMRQACAQFzAQJfeAkArAICCQCkAwEFAl94AgEsAQRkaXZkAgJfeAJfeQkAbgQFAl94BQxERUNJTUFMX1VOSVQFAl95BQhIQUxGRVZFTgEEbXVsZAICX3gCX3kJAG4EBQJfeAUCX3kFDERFQ0lNQUxfVU5JVAUISEFMRkVWRU4BA2FicwECX3gDCQBmAgUCX3gAAAUCX3gJAQEtAQUCX3gBBHZtYXgCAl94Al95AwkAZwIFAl94BQJfeQUCX3gFAl95AQlsaXN0VG9TdHIBBV9saXN0CgEFX2pvaW4CC2FjY3VtdWxhdG9yA3ZhbAkArAICCQCsAgIFC2FjY3VtdWxhdG9yBQN2YWwCASwECm5ld0xpc3RTdHIKAAIkbAUFX2xpc3QKAAIkcwkAkAMBBQIkbAoABSRhY2MwAgAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBBV9qb2luAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAyMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQEC25ld0xpc3RTdHJVCQCzAgIFCm5ld0xpc3RTdHIAAQQLbmV3TGlzdFN0clIDCQAAAgkArwICBQtuZXdMaXN0U3RyVQABAgEsCQCwAgIFC25ld0xpc3RTdHJVAAEFC25ld0xpc3RTdHJVBQtuZXdMaXN0U3RyUgEJc3RyVG9MaXN0AQRfc3RyCQC1CQIFBF9zdHICASwBC3B1c2hUb1F1ZXVlAwVfbGlzdAhfbWF4U2l6ZQZfdmFsdWUDCQBmAgkAkAMBBQVfbGlzdAUIX21heFNpemUJAM0IAgkA0QgCBQVfbGlzdAAABQZfdmFsdWUJAM0IAgUFX2xpc3QFBl92YWx1ZQEDaW50AQFrCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUEdGhpcwUBawkArAICAg1ubyB2YWx1ZSBmb3IgBQFrAQRzdHJBAghfYWRkcmVzcwRfa2V5BAN2YWwJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQhfYWRkcmVzcwUEX2tleQkArAICAhFObyB2YWx1ZSBmb3Iga2V5IAUEX2tleQUDdmFsAQRpbnRBAghfYWRkcmVzcwRfa2V5BAN2YWwJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQhfYWRkcmVzcwUEX2tleQkArAICAhFObyB2YWx1ZSBmb3Iga2V5IAUEX2tleQUDdmFsAQhjYmFsYW5jZQAJAQNpbnQBBQlrX2JhbGFuY2UBA2ZlZQAJAQNpbnQBBQVrX2ZlZQEPaW5pdE1hcmdpblJhdGlvAAkBA2ludAEFEWtfaW5pdE1hcmdpblJhdGlvAQZxdEFzdFIACQEDaW50AQUTa19xdW90ZUFzc2V0UmVzZXJ2ZQEGYnNBc3RSAAkBA2ludAEFEmtfYmFzZUFzc2V0UmVzZXJ2ZQERdG90YWxQb3NpdGlvblNpemUACQEDaW50AQUTa190b3RhbFBvc2l0aW9uU2l6ZQESY3VtdWxhdGl2ZU5vdGlvbmFsAAkBA2ludAEFFGtfY3VtdWxhdGl2ZU5vdGlvbmFsARRvcGVuSW50ZXJlc3ROb3Rpb25hbAAJAQNpbnQBBRZrX29wZW5JbnRlcmVzdE5vdGlvbmFsARluZXh0RnVuZGluZ0Jsb2NrVGltZXN0YW1wAAkBA2ludAEFEmtfbmV4dEZ1bmRpbmdCbG9jawEQZnVuZGluZ1BlcmlvZFJhdwAJAQNpbnQBBQ9rX2Z1bmRpbmdQZXJpb2QBFGZ1bmRpbmdQZXJpb2REZWNpbWFsAAkAaAIJARBmdW5kaW5nUGVyaW9kUmF3AAUMREVDSU1BTF9VTklUARRmdW5kaW5nUGVyaW9kU2Vjb25kcwAJAGgCCQEQZnVuZGluZ1BlcmlvZFJhdwAFB1NFQ09ORFMBFm1haW50ZW5hbmNlTWFyZ2luUmF0aW8ACQEDaW50AQUYa19tYWludGVuYW5jZU1hcmdpblJhdGlvARNsaXF1aWRhdGlvbkZlZVJhdGlvAAkBA2ludAEFFWtfbGlxdWlkYXRpb25GZWVSYXRpbwEXcGFydGlhbExpcXVpZGF0aW9uUmF0aW8ACQEDaW50AQUZa19wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwELc3ByZWFkTGltaXQACQEDaW50AQUNa19zcHJlYWRMaW1pdAEObWF4UHJpY2VJbXBhY3QACQEDaW50AQUQa19tYXhQcmljZUltcGFjdAEObWF4UHJpY2VTcHJlYWQACQEDaW50AQUQa19tYXhQcmljZVNwcmVhZAEjbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24ACQEDaW50AQUla19sYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgEkbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAAkBA2ludAEFJmtfbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAkBA2ludAEFGGtfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQEVdG90YWxMb25nUG9zaXRpb25TaXplAAkBA2ludAEFF2tfdG90YWxMb25nUG9zaXRpb25TaXplAQ9nZXRBY3R1YWxDYWxsZXIBAWkJAQt2YWx1ZU9yRWxzZQIJAJ0IAgkBDW9yZGVyc0FkZHJlc3MAAghrX3NlbmRlcgkApQgBCAUBaQZjYWxsZXIBFnJlcXVpcmVNb3JlTWFyZ2luUmF0aW8DDF9tYXJnaW5SYXRpbxBfYmFzZU1hcmdpblJhdGlvFF9sYXJnZXJUaGFuT3JFcXVhbFRvBBRyZW1haW5pbmdNYXJnaW5SYXRpbwkAZQIFDF9tYXJnaW5SYXRpbwUQX2Jhc2VNYXJnaW5SYXRpbwMDBRRfbGFyZ2VyVGhhbk9yRXF1YWxUbwkAZgIAAAUUcmVtYWluaW5nTWFyZ2luUmF0aW8HCQACAQIOSW52YWxpZCBtYXJnaW4DAwkBASEBBRRfbGFyZ2VyVGhhbk9yRXF1YWxUbwkAZwIFFHJlbWFpbmluZ01hcmdpblJhdGlvAAAHCQACAQIOSW52YWxpZCBtYXJnaW4GAR9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAQ1fcG9zaXRpb25TaXplAwkAAAIFDV9wb3NpdGlvblNpemUAAAkAAgECLFNob3VsZCBub3QgYmUgY2FsbGVkIHdpdGggX3Bvc2l0aW9uU2l6ZSA9PSAwAwkAZgIFDV9wb3NpdGlvblNpemUAAAkBI2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAAkBJGxhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgABC2dldFBvc2l0aW9uAQdfdHJhZGVyBA9wb3NpdGlvblNpemVPcHQJAJoIAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUOa19wb3NpdGlvblNpemUFB190cmFkZXIEByRtYXRjaDAFD3Bvc2l0aW9uU2l6ZU9wdAMJAAECBQckbWF0Y2gwAgNJbnQEDHBvc2l0aW9uU2l6ZQUHJG1hdGNoMAkAlgoEBQxwb3NpdGlvblNpemUJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFEGtfcG9zaXRpb25NYXJnaW4FB190cmFkZXIJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFFmtfcG9zaXRpb25PcGVuTm90aW9uYWwFB190cmFkZXIJARFAZXh0ck5hdGl2ZSgxMDUwKQIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFLmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FB190cmFkZXIJAJYKBAAAAAAAAAAAARBnZXRQb3NpdGlvbkFzc2V0AQdfdHJhZGVyBBBwb3NpdGlvbkFzc2V0T3B0CQCdCAIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFD2tfcG9zaXRpb25Bc3NldAUHX3RyYWRlcgQHJG1hdGNoMAUQcG9zaXRpb25Bc3NldE9wdAMJAAECBQckbWF0Y2gwAgZTdHJpbmcEDXBvc2l0aW9uQXNzZXQFByRtYXRjaDAFDXBvc2l0aW9uQXNzZXQJANgEAQkBCnF1b3RlQXNzZXQAARNyZXF1aXJlT3BlblBvc2l0aW9uAQdfdHJhZGVyAwkAAAIICQELZ2V0UG9zaXRpb24BBQdfdHJhZGVyAl8xAAAJAAIBAhBObyBvcGVuIHBvc2l0aW9uBgELaW5pdGlhbGl6ZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFDWtfaW5pdGlhbGl6ZWQHAQZwYXVzZWQACQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFCGtfcGF1c2VkBwENdXBkYXRlUmVzZXJ2ZQMGX2lzQWRkEV9xdW90ZUFzc2V0QW1vdW50EF9iYXNlQXNzZXRBbW91bnQDBQZfaXNBZGQEB25ld0Jhc2UJAGUCCQEGYnNBc3RSAAUQX2Jhc2VBc3NldEFtb3VudAMJAGcCAAAFB25ld0Jhc2UJAAIBAipUeCBsZWFkIHRvIGJhc2UgYXNzZXQgcmVzZXJ2ZSA8PSAwLCByZXZlcnQJAJYKBAkAZAIJAQZxdEFzdFIABRFfcXVvdGVBc3NldEFtb3VudAUHbmV3QmFzZQkAZAIJARF0b3RhbFBvc2l0aW9uU2l6ZQAFEF9iYXNlQXNzZXRBbW91bnQJAGQCCQESY3VtdWxhdGl2ZU5vdGlvbmFsAAURX3F1b3RlQXNzZXRBbW91bnQECG5ld1F1b3RlCQBlAgkBBnF0QXN0UgAFEV9xdW90ZUFzc2V0QW1vdW50AwkAZwIAAAUIbmV3UXVvdGUJAAIBAipUeCBsZWFkIHRvIGJhc2UgcXVvdGUgcmVzZXJ2ZSA8PSAwLCByZXZlcnQJAJYKBAUIbmV3UXVvdGUJAGQCCQEGYnNBc3RSAAUQX2Jhc2VBc3NldEFtb3VudAkAZQIJARF0b3RhbFBvc2l0aW9uU2l6ZQAFEF9iYXNlQXNzZXRBbW91bnQJAGUCCQESY3VtdWxhdGl2ZU5vdGlvbmFsAAURX3F1b3RlQXNzZXRBbW91bnQBCXN3YXBJbnB1dAIGX2lzQWRkEV9xdW90ZUFzc2V0QW1vdW50BAdfcXRBc3RSCQEGcXRBc3RSAAQHX2JzQXN0UgkBBmJzQXN0UgAEAWsJAQRtdWxkAgUHX3F0QXN0UgUHX2JzQXN0UgQWcXVvdGVBc3NldFJlc2VydmVBZnRlcgMFBl9pc0FkZAkAZAIFB19xdEFzdFIFEV9xdW90ZUFzc2V0QW1vdW50CQBlAgUHX3F0QXN0UgURX3F1b3RlQXNzZXRBbW91bnQEFWJhc2VBc3NldFJlc2VydmVBZnRlcgkBBGRpdmQCBQFrBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBBhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMJAQNhYnMBCQBlAgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBQdfYnNBc3RSBBVhbW91bnRCYXNlQXNzZXRCb3VnaHQDBQZfaXNBZGQFGGFtb3VudEJhc2VBc3NldEJvdWdodEFicwkBAS0BBRhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMEDSR0MDE0OTc0MTUxNjcJAQ11cGRhdGVSZXNlcnZlAwUGX2lzQWRkBRFfcXVvdGVBc3NldEFtb3VudAUYYW1vdW50QmFzZUFzc2V0Qm91Z2h0QWJzBBdxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyMQgFDSR0MDE0OTc0MTUxNjcCXzEEFmJhc2VBc3NldFJlc2VydmVBZnRlcjEIBQ0kdDAxNDk3NDE1MTY3Al8yBBd0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyMQgFDSR0MDE0OTc0MTUxNjcCXzMEGGN1bXVsYXRpdmVOb3Rpb25hbEFmdGVyMQgFDSR0MDE0OTc0MTUxNjcCXzQEC3ByaWNlQmVmb3JlCQEEZGl2ZAIFB19xdEFzdFIFB19ic0FzdFIEC21hcmtldFByaWNlCQEEZGl2ZAIFEV9xdW90ZUFzc2V0QW1vdW50BRhhbW91bnRCYXNlQXNzZXRCb3VnaHRBYnMECXByaWNlRGlmZgkBA2FicwEJAGUCBQtwcmljZUJlZm9yZQULbWFya2V0UHJpY2UEC3ByaWNlSW1wYWN0CQBlAgUMREVDSU1BTF9VTklUCQEEZGl2ZAIFC3ByaWNlQmVmb3JlCQBkAgULcHJpY2VCZWZvcmUFCXByaWNlRGlmZgQTbWF4UHJpY2VJbXBhY3RWYWx1ZQkBDm1heFByaWNlSW1wYWN0AAMJAGYCBQtwcmljZUltcGFjdAUTbWF4UHJpY2VJbXBhY3RWYWx1ZQkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgINUHJpY2UgaW1wYWN0IAkApAMBBQtwcmljZUltcGFjdAIUID4gbWF4IHByaWNlIGltcGFjdCAJAKQDAQUTbWF4UHJpY2VJbXBhY3RWYWx1ZQIVIGJlZm9yZSBxdW90ZSBhc3NldDogCQCkAwEFB19xdEFzdFICFCBiZWZvcmUgYmFzZSBhc3NldDogCQCkAwEFB19ic0FzdFICISBxdW90ZSBhc3NldCBhbW91bnQgdG8gZXhjaGFuZ2U6IAkApAMBBRFfcXVvdGVBc3NldEFtb3VudAIPIHByaWNlIGJlZm9yZTogCQCkAwEFC3ByaWNlQmVmb3JlAg4gbWFya2V0UHJpY2U6IAkApAMBBQttYXJrZXRQcmljZQkAlwoFBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQFF3F1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIxBRZiYXNlQXNzZXRSZXNlcnZlQWZ0ZXIxBRd0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyMQUYY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIxASJjYWxjUmVtYWluTWFyZ2luV2l0aEZ1bmRpbmdQYXltZW50BBBfb2xkUG9zaXRpb25TaXplEl9vbGRQb3NpdGlvbk1hcmdpbiVfb2xkUG9zaXRpb25DdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uDF9tYXJnaW5EZWx0YQQOZnVuZGluZ1BheW1lbnQDCQECIT0CBRBfb2xkUG9zaXRpb25TaXplAAAEIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uCQEfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgEFEF9vbGRQb3NpdGlvblNpemUJAQRtdWxkAgkAZQIFIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBSVfb2xkUG9zaXRpb25DdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBRBfb2xkUG9zaXRpb25TaXplAAAEDHNpZ25lZE1hcmdpbgkAZAIJAGUCBQxfbWFyZ2luRGVsdGEFDmZ1bmRpbmdQYXltZW50BRJfb2xkUG9zaXRpb25NYXJnaW4EDSR0MDE2NjU4MTY3ODUDCQBmAgAABQxzaWduZWRNYXJnaW4JAJQKAgAACQEDYWJzAQUMc2lnbmVkTWFyZ2luCQCUCgIJAQNhYnMBBQxzaWduZWRNYXJnaW4AAAQMcmVtYWluTWFyZ2luCAUNJHQwMTY2NTgxNjc4NQJfMQQHYmFkRGVidAgFDSR0MDE2NjU4MTY3ODUCXzIJAJUKAwUMcmVtYWluTWFyZ2luBQdiYWREZWJ0BQ5mdW5kaW5nUGF5bWVudAEWc3dhcE91dHB1dFdpdGhSZXNlcnZlcwUGX2lzQWRkEF9iYXNlQXNzZXRBbW91bnQUX2NoZWNrTWF4UHJpY2VJbXBhY3QSX3F1b3RlQXNzZXRSZXNlcnZlEV9iYXNlQXNzZXRSZXNlcnZlBAtwcmljZUJlZm9yZQkBBGRpdmQCBRJfcXVvdGVBc3NldFJlc2VydmUFEV9iYXNlQXNzZXRSZXNlcnZlAwkAAAIFEF9iYXNlQXNzZXRBbW91bnQAAAkAAgECGUludmFsaWQgYmFzZSBhc3NldCBhbW91bnQEAWsJAQRtdWxkAgUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfYmFzZUFzc2V0UmVzZXJ2ZQQYYmFzZUFzc2V0UG9vbEFtb3VudEFmdGVyAwUGX2lzQWRkCQBkAgURX2Jhc2VBc3NldFJlc2VydmUFEF9iYXNlQXNzZXRBbW91bnQJAGUCBRFfYmFzZUFzc2V0UmVzZXJ2ZQUQX2Jhc2VBc3NldEFtb3VudAQPcXVvdGVBc3NldEFmdGVyCQEEZGl2ZAIFAWsFGGJhc2VBc3NldFBvb2xBbW91bnRBZnRlcgQOcXVvdGVBc3NldFNvbGQJAQNhYnMBCQBlAgUPcXVvdGVBc3NldEFmdGVyBRJfcXVvdGVBc3NldFJlc2VydmUEE21heFByaWNlSW1wYWN0VmFsdWUJAQ5tYXhQcmljZUltcGFjdAAEDSR0MDE3NzYyMTc5NTUJAQ11cGRhdGVSZXNlcnZlAwkBASEBBQZfaXNBZGQFDnF1b3RlQXNzZXRTb2xkBRBfYmFzZUFzc2V0QW1vdW50BBdxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyMQgFDSR0MDE3NzYyMTc5NTUCXzEEFmJhc2VBc3NldFJlc2VydmVBZnRlcjEIBQ0kdDAxNzc2MjE3OTU1Al8yBBd0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyMQgFDSR0MDE3NzYyMTc5NTUCXzMEGGN1bXVsYXRpdmVOb3Rpb25hbEFmdGVyMQgFDSR0MDE3NzYyMTc5NTUCXzQEC21hcmtldFByaWNlCQEEZGl2ZAIFDnF1b3RlQXNzZXRTb2xkBRBfYmFzZUFzc2V0QW1vdW50BAlwcmljZURpZmYJAQNhYnMBCQBlAgULcHJpY2VCZWZvcmUFC21hcmtldFByaWNlBAtwcmljZUltcGFjdAkAZQIFDERFQ0lNQUxfVU5JVAkBBGRpdmQCBQtwcmljZUJlZm9yZQkAZAIFC3ByaWNlQmVmb3JlBQlwcmljZURpZmYDAwkAZgIFC3ByaWNlSW1wYWN0BRNtYXhQcmljZUltcGFjdFZhbHVlBRRfY2hlY2tNYXhQcmljZUltcGFjdAcJAAIBCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICDVByaWNlIGltcGFjdCAJAKQDAQULcHJpY2VJbXBhY3QCFCA+IG1heCBwcmljZSBpbXBhY3QgCQCkAwEFE21heFByaWNlSW1wYWN0VmFsdWUCFSBiZWZvcmUgcXVvdGUgYXNzZXQ6IAkApAMBBRJfcXVvdGVBc3NldFJlc2VydmUCFCBiZWZvcmUgYmFzZSBhc3NldDogCQCkAwEFEV9iYXNlQXNzZXRSZXNlcnZlAiAgYmFzZSBhc3NldCBhbW91bnQgdG8gZXhjaGFuZ2U6IAkApAMBBRBfYmFzZUFzc2V0QW1vdW50Ag8gcHJpY2UgYmVmb3JlOiAJAKQDAQULcHJpY2VCZWZvcmUCDyBtYXJrZXQgcHJpY2U6IAkApAMBBQttYXJrZXRQcmljZQkAmgoIBQ5xdW90ZUFzc2V0U29sZAUXcXVvdGVBc3NldFJlc2VydmVBZnRlcjEFFmJhc2VBc3NldFJlc2VydmVBZnRlcjEFF3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIxBRhjdW11bGF0aXZlTm90aW9uYWxBZnRlcjEJAGUCCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAMFBl9pc0FkZAkBA2FicwEFEF9iYXNlQXNzZXRBbW91bnQAAAkAZQIJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAMJAQEhAQUGX2lzQWRkCQEDYWJzAQUQX2Jhc2VBc3NldEFtb3VudAAABQtwcmljZUltcGFjdAEKc3dhcE91dHB1dAMGX2lzQWRkEF9iYXNlQXNzZXRBbW91bnQUX2NoZWNrTWF4UHJpY2VJbXBhY3QJARZzd2FwT3V0cHV0V2l0aFJlc2VydmVzBQUGX2lzQWRkBRBfYmFzZUFzc2V0QW1vdW50BRRfY2hlY2tNYXhQcmljZUltcGFjdAkBBnF0QXN0UgAJAQZic0FzdFIAARJnZXRPcmFjbGVUd2FwUHJpY2UABAZvcmFjbGUJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCBQR0aGlzBQVrX29yYQIABAhwcmljZUtleQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhpcwUJa19vcmFfa2V5BAhibG9ja0tleQkBEUBleHRyTmF0aXZlKDEwNTMpAgUEdGhpcwUPa19vcmFfYmxvY2tfa2V5BAlsYXN0VmFsdWUJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQZvcmFjbGUFCHByaWNlS2V5CQCsAgIJAKwCAgkArAICAiJDYW4gbm90IGdldCBvcmFjbGUgcHJpY2UuIE9yYWNsZTogCQClCAEFBm9yYWNsZQIGIGtleTogBQhwcmljZUtleQUJbGFzdFZhbHVlARlyZXF1aXJlTm90T3ZlclNwcmVhZExpbWl0AhJfcXVvdGVBc3NldFJlc2VydmURX2Jhc2VBc3NldFJlc2VydmUEC29yYWNsZVByaWNlCQESZ2V0T3JhY2xlVHdhcFByaWNlAAQKcHJpY2VBZnRlcgkBBGRpdmQCBRJfcXVvdGVBc3NldFJlc2VydmUFEV9iYXNlQXNzZXRSZXNlcnZlBAxhdmVyYWdlUHJpY2UJAQRkaXZkAgkAZAIFC29yYWNsZVByaWNlBQpwcmljZUFmdGVyCQBoAgACBQxERUNJTUFMX1VOSVQEDGFic1ByaWNlRGlmZgkBBGRpdmQCCQEDYWJzAQkAZQIFC29yYWNsZVByaWNlBQpwcmljZUFmdGVyBQxhdmVyYWdlUHJpY2UDCQBmAgUMYWJzUHJpY2VEaWZmCQEObWF4UHJpY2VTcHJlYWQACQACAQkArAICCQCsAgIJAKwCAgINUHJpY2Ugc3ByZWFkIAkApAMBBQxhYnNQcmljZURpZmYCFCA+IG1heCBwcmljZSBzcHJlYWQgCQCkAwEJAQ5tYXhQcmljZVNwcmVhZAAGAQxnZXRTcG90UHJpY2UABBJfcXVvdGVBc3NldFJlc2VydmUJAQZxdEFzdFIABBFfYmFzZUFzc2V0UmVzZXJ2ZQkBBmJzQXN0UgAJAQRkaXZkAgUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfYmFzZUFzc2V0UmVzZXJ2ZQEWaXNPdmVyRmx1Y3R1YXRpb25MaW1pdAAEC29yYWNsZVByaWNlCQESZ2V0T3JhY2xlVHdhcFByaWNlAAQMY3VycmVudFByaWNlCQEMZ2V0U3BvdFByaWNlAAkAZgIJAQRkaXZkAgkBA2FicwEJAGUCBQtvcmFjbGVQcmljZQUMY3VycmVudFByaWNlBQtvcmFjbGVQcmljZQkBC3NwcmVhZExpbWl0AAEfZ2V0UG9zaXRpb25BZGp1c3RlZE9wZW5Ob3Rpb25hbAQNX3Bvc2l0aW9uU2l6ZQdfb3B0aW9uEl9xdW90ZUFzc2V0UmVzZXJ2ZRFfYmFzZUFzc2V0UmVzZXJ2ZQQPcG9zaXRpb25TaXplQWJzCQEDYWJzAQUNX3Bvc2l0aW9uU2l6ZQQHaXNTaG9ydAkAZgIAAAUNX3Bvc2l0aW9uU2l6ZQQQcG9zaXRpb25Ob3Rpb25hbAMJAAACBQdfb3B0aW9uBQ9QTkxfT1BUSU9OX1NQT1QEDSR0MDIxMTczMjEzNDQJARZzd2FwT3V0cHV0V2l0aFJlc2VydmVzBQkBASEBBQdpc1Nob3J0BQ9wb3NpdGlvblNpemVBYnMHBRJfcXVvdGVBc3NldFJlc2VydmUFEV9iYXNlQXNzZXRSZXNlcnZlBBNvdXRQb3NpdGlvbk5vdGlvbmFsCAUNJHQwMjExNzMyMTM0NAJfMQQCeDEIBQ0kdDAyMTE3MzIxMzQ0Al8yBAJ4MggFDSR0MDIxMTczMjEzNDQCXzMEAngzCAUNJHQwMjExNzMyMTM0NAJfNAUTb3V0UG9zaXRpb25Ob3Rpb25hbAkBBG11bGQCBQ9wb3NpdGlvblNpemVBYnMJARJnZXRPcmFjbGVUd2FwUHJpY2UABRBwb3NpdGlvbk5vdGlvbmFsAStnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubEJ5VmFsdWVzBQ1fcG9zaXRpb25TaXplFV9wb3NpdGlvbk9wZW5Ob3Rpb25hbBJfcXVvdGVBc3NldFJlc2VydmURX2Jhc2VBc3NldFJlc2VydmUHX29wdGlvbgMJAAACBQ1fcG9zaXRpb25TaXplAAAJAAIBAhVJbnZhbGlkIHBvc2l0aW9uIHNpemUEB2lzU2hvcnQJAGYCAAAFDV9wb3NpdGlvblNpemUEEHBvc2l0aW9uTm90aW9uYWwJAR9nZXRQb3NpdGlvbkFkanVzdGVkT3Blbk5vdGlvbmFsBAUNX3Bvc2l0aW9uU2l6ZQUHX29wdGlvbgUSX3F1b3RlQXNzZXRSZXNlcnZlBRFfYmFzZUFzc2V0UmVzZXJ2ZQQNdW5yZWFsaXplZFBubAMFB2lzU2hvcnQJAGUCBRVfcG9zaXRpb25PcGVuTm90aW9uYWwFEHBvc2l0aW9uTm90aW9uYWwJAGUCBRBwb3NpdGlvbk5vdGlvbmFsBRVfcG9zaXRpb25PcGVuTm90aW9uYWwJAJQKAgUQcG9zaXRpb25Ob3Rpb25hbAUNdW5yZWFsaXplZFBubAEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwCB190cmFkZXIHX29wdGlvbgQNJHQwMjI2NjEyMjc4OQkBC2dldFBvc2l0aW9uAQUHX3RyYWRlcgQMcG9zaXRpb25TaXplCAUNJHQwMjI2NjEyMjc4OQJfMQQOcG9zaXRpb25NYXJnaW4IBQ0kdDAyMjY2MTIyNzg5Al8yBBRwb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDIyNjYxMjI3ODkCXzMEEXBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwMjI2NjEyMjc4OQJfNAkBK2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sQnlWYWx1ZXMFBQxwb3NpdGlvblNpemUFFHBvc2l0aW9uT3Blbk5vdGlvbmFsCQEGcXRBc3RSAAkBBmJzQXN0UgAFB19vcHRpb24BD2NhbGNNYXJnaW5SYXRpbwMNX3JlbWFpbk1hcmdpbghfYmFkRGVidBFfcG9zaXRpb25Ob3Rpb25hbAkBBGRpdmQCCQBlAgUNX3JlbWFpbk1hcmdpbgUIX2JhZERlYnQFEV9wb3NpdGlvbk5vdGlvbmFsARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAgdfdHJhZGVyB19vcHRpb24EDSR0MDIzMjcwMjMzODEJAQtnZXRQb3NpdGlvbgEFB190cmFkZXIEDHBvc2l0aW9uU2l6ZQgFDSR0MDIzMjcwMjMzODECXzEEDnBvc2l0aW9uTWFyZ2luCAUNJHQwMjMyNzAyMzM4MQJfMgQDcG9uCAUNJHQwMjMyNzAyMzM4MQJfMwQRcG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDAyMzI3MDIzMzgxAl80BA0kdDAyMzM4NzIzNDgwCQEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwCBQdfdHJhZGVyBQdfb3B0aW9uBBBwb3NpdGlvbk5vdGlvbmFsCAUNJHQwMjMzODcyMzQ4MAJfMQQNdW5yZWFsaXplZFBubAgFDSR0MDIzMzg3MjM0ODACXzIEDSR0MDIzNDg1MjM2NTEJASJjYWxjUmVtYWluTWFyZ2luV2l0aEZ1bmRpbmdQYXltZW50BAUMcG9zaXRpb25TaXplBQ5wb3NpdGlvbk1hcmdpbgURcG9zaXRpb25Mc3RVcGRDUEYFDXVucmVhbGl6ZWRQbmwEDHJlbWFpbk1hcmdpbggFDSR0MDIzNDg1MjM2NTECXzEEB2JhZERlYnQIBQ0kdDAyMzQ4NTIzNjUxAl8yCQEPY2FsY01hcmdpblJhdGlvAwUMcmVtYWluTWFyZ2luBQdiYWREZWJ0BRBwb3NpdGlvbk5vdGlvbmFsAQ5nZXRNYXJnaW5SYXRpbwEHX3RyYWRlcgkBFmdldE1hcmdpblJhdGlvQnlPcHRpb24CBQdfdHJhZGVyBQ9QTkxfT1BUSU9OX1NQT1QBG2dldFBhcnRpYWxMaXF1aWRhdGlvbkFtb3VudAIHX3RyYWRlcg1fcG9zaXRpb25TaXplBAxtYXhpbXVtUmF0aW8JAQR2bWF4AgkBF3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvAAkAZQIFDERFQ0lNQUxfVU5JVAkBBGRpdmQCCQEOZ2V0TWFyZ2luUmF0aW8BBQdfdHJhZGVyCQEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAEGG1heEV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQkBBG11bGQCCQEDYWJzAQUNX3Bvc2l0aW9uU2l6ZQUMbWF4aW11bVJhdGlvBApzd2FwUmVzdWx0CQEKc3dhcE91dHB1dAMJAGYCBQ1fcG9zaXRpb25TaXplAAAFGG1heEV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQcEHG1heEV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQIBQpzd2FwUmVzdWx0Al8xBAtwcmljZUltcGFjdAgFCnN3YXBSZXN1bHQCXzgDCQBmAgkBDm1heFByaWNlSW1wYWN0AAULcHJpY2VJbXBhY3QFHG1heEV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQEFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQkBBG11bGQCCQEDYWJzAQUNX3Bvc2l0aW9uU2l6ZQkBF3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvAAQZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAgJAQpzd2FwT3V0cHV0AwkAZgIFDV9wb3NpdGlvblNpemUAAAUVZXhjaGFuZ2VkUG9zaXRpb25TaXplBwJfMQUZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAEVaW50ZXJuYWxDbG9zZVBvc2l0aW9uAgdfdHJhZGVyFF9jaGVja01heFByaWNlSW1wYWN0BA0kdDAyNDg4NDI1MDEyCQELZ2V0UG9zaXRpb24BBQdfdHJhZGVyBAxwb3NpdGlvblNpemUIBQ0kdDAyNDg4NDI1MDEyAl8xBA5wb3NpdGlvbk1hcmdpbggFDSR0MDI0ODg0MjUwMTICXzIEFHBvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwMjQ4ODQyNTAxMgJfMwQRcG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDAyNDg4NDI1MDEyAl80BA11bnJlYWxpemVkUG5sCAkBI2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sAgUHX3RyYWRlcgUPUE5MX09QVElPTl9TUE9UAl8yBA0kdDAyNTEwNzI1Mjc1CQEiY2FsY1JlbWFpbk1hcmdpbldpdGhGdW5kaW5nUGF5bWVudAQFDHBvc2l0aW9uU2l6ZQUOcG9zaXRpb25NYXJnaW4FEXBvc2l0aW9uTHN0VXBkQ1BGBQ11bnJlYWxpemVkUG5sBAxyZW1haW5NYXJnaW4IBQ0kdDAyNTEwNzI1Mjc1Al8xBAdiYWREZWJ0CAUNJHQwMjUxMDcyNTI3NQJfMgQVZXhjaGFuZ2VkUG9zaXRpb25TaXplCQEBLQEFDHBvc2l0aW9uU2l6ZQQLcmVhbGl6ZWRQbmwFDXVucmVhbGl6ZWRQbmwEDW1hcmdpblRvVmF1bHQJAQEtAQUMcmVtYWluTWFyZ2luBA0kdDAyNTQwMjI1NzEzCQEKc3dhcE91dHB1dAMJAGYCBQxwb3NpdGlvblNpemUAAAkBA2FicwEFDHBvc2l0aW9uU2l6ZQUUX2NoZWNrTWF4UHJpY2VJbXBhY3QEGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQIBQ0kdDAyNTQwMjI1NzEzAl8xBBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMjU0MDIyNTcxMwJfMgQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwMjU0MDIyNTcxMwJfMwQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDI1NDAyMjU3MTMCXzQEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwMjU0MDIyNTcxMwJfNQQOdG90YWxMb25nQWZ0ZXIIBQ0kdDAyNTQwMjI1NzEzAl82BA90b3RhbFNob3J0QWZ0ZXIIBQ0kdDAyNTQwMjI1NzEzAl83BBlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCQBlAgkBFG9wZW5JbnRlcmVzdE5vdGlvbmFsAAUUcG9zaXRpb25PcGVuTm90aW9uYWwJAJ4KDAUVZXhjaGFuZ2VkUG9zaXRpb25TaXplBQdiYWREZWJ0BQtyZWFsaXplZFBubAUNbWFyZ2luVG9WYXVsdAUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyBRZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyBRdjdW11bGF0aXZlTm90aW9uYWxBZnRlcgUZb3BlbkludGVyZXN0Tm90aW9uYWxBZnRlcgUZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAUOdG90YWxMb25nQWZ0ZXIFD3RvdGFsU2hvcnRBZnRlcgEQZ2V0VHdhcFNwb3RQcmljZQAECG1pbnV0ZUlkCQBpAgkAaQIIBQlsYXN0QmxvY2sJdGltZXN0YW1wAOgHADwEDXN0YXJ0TWludXRlSWQJAGUCBQhtaW51dGVJZAUNVFdBUF9JTlRFUlZBTAQHbGlzdFN0cgkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzBQ1rX2xhc3REYXRhU3RyAgAEBGxpc3QJALUJAgUHbGlzdFN0cgIBLAoBCGZpbHRlckZuAgthY2N1bXVsYXRvcgRuZXh0AwkAZwIFDXN0YXJ0TWludXRlSWQJAQ1wYXJzZUludFZhbHVlAQUEbmV4dAkAzQgCBQthY2N1bXVsYXRvcgkBDXBhcnNlSW50VmFsdWUBBQRuZXh0BQthY2N1bXVsYXRvcgQFbGlzdEYKAAIkbAUEbGlzdAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEIZmlsdGVyRm4CBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDIwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAQIbWF4SW5kZXgDCQBmAgkAkAMBBQVsaXN0RgAACQCWAwEFBWxpc3RGCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUEbGlzdAAABAxsYXN0TWludXRlSWQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUOa19sYXN0TWludXRlSWQAAAQWZW5kTGFzdEN1bXVsYXRpdmVQcmljZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIJAKwCAgUda190d2FwRGF0YUxhc3RDdW11bGF0aXZlUHJpY2UCAV8JAKQDAQUMbGFzdE1pbnV0ZUlkAAAEDGVuZExhc3RQcmljZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIJAKwCAgUTa190d2FwRGF0YUxhc3RQcmljZQIBXwkApAMBBQxsYXN0TWludXRlSWQAAAQSbm93Q3VtdWxhdGl2ZVByaWNlCQBkAgUWZW5kTGFzdEN1bXVsYXRpdmVQcmljZQkAaAIJAGUCBQhtaW51dGVJZAUMbGFzdE1pbnV0ZUlkBQxlbmRMYXN0UHJpY2UEGHN0YXJ0TGFzdEN1bXVsYXRpdmVQcmljZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIJAKwCAgUda190d2FwRGF0YUxhc3RDdW11bGF0aXZlUHJpY2UCAV8JAKQDAQUIbWF4SW5kZXgAAAQOc3RhcnRMYXN0UHJpY2UJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkArAICCQCsAgIFE2tfdHdhcERhdGFMYXN0UHJpY2UCAV8JAKQDAQUIbWF4SW5kZXgAAAQUc3RhcnRDdW11bGF0aXZlUHJpY2UJAGQCBRhzdGFydExhc3RDdW11bGF0aXZlUHJpY2UJAGgCCQBlAgUNc3RhcnRNaW51dGVJZAUIbWF4SW5kZXgFDnN0YXJ0TGFzdFByaWNlCQBpAgkAZQIFEm5vd0N1bXVsYXRpdmVQcmljZQUUc3RhcnRDdW11bGF0aXZlUHJpY2UFDVRXQVBfSU5URVJWQUwBEGdldFBlZ0FkanVzdENvc3QBBl9wcmljZQQNX3Bvc2l0aW9uU2l6ZQkBEXRvdGFsUG9zaXRpb25TaXplAAQJZGlyZWN0aW9uCQBmAgUNX3Bvc2l0aW9uU2l6ZQAABBVjdXJyZW50TmV0TWFya2V0VmFsdWUICQEKc3dhcE91dHB1dAMFCWRpcmVjdGlvbgkBA2FicwEFDV9wb3NpdGlvblNpemUHAl8xBBBiYXNlQXNzZXRSZXNlcnZlCQEGYnNBc3RSAAQUbmV3UXVvdGVBc3NldFJlc2VydmUJAQRtdWxkAgUQYmFzZUFzc2V0UmVzZXJ2ZQUGX3ByaWNlBARjb3N0CAkBK2dldFBvc2l0aW9uTm90aW9uYWxBbmRVbnJlYWxpemVkUG5sQnlWYWx1ZXMFBQ1fcG9zaXRpb25TaXplBRVjdXJyZW50TmV0TWFya2V0VmFsdWUFFG5ld1F1b3RlQXNzZXRSZXNlcnZlBRBiYXNlQXNzZXRSZXNlcnZlBQ9QTkxfT1BUSU9OX1NQT1QCXzIJAJQKAgUUbmV3UXVvdGVBc3NldFJlc2VydmUFBGNvc3QBCmdldEZ1bmRpbmcABA91bmRlcmx5aW5nUHJpY2UJARJnZXRPcmFjbGVUd2FwUHJpY2UABA1zcG90VHdhcFByaWNlCQEQZ2V0VHdhcFNwb3RQcmljZQAEB3ByZW1pdW0JAGUCBQ1zcG90VHdhcFByaWNlBQ91bmRlcmx5aW5nUHJpY2UDAwkAAAIJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAAABgkAAAIJARV0b3RhbExvbmdQb3NpdGlvblNpemUAAAAJAJQKAgAAAAADCQBmAgAABQdwcmVtaXVtBBRzaG9ydFByZW1pdW1GcmFjdGlvbgkBBGRpdmQCCQEEbXVsZAIFB3ByZW1pdW0JARRmdW5kaW5nUGVyaW9kRGVjaW1hbAAFB09ORV9EQVkEE2xvbmdQcmVtaXVtRnJhY3Rpb24JAQRkaXZkAgkBBG11bGQCBRRzaG9ydFByZW1pdW1GcmFjdGlvbgkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUACQEVdG90YWxMb25nUG9zaXRpb25TaXplAAkAlAoCBRRzaG9ydFByZW1pdW1GcmFjdGlvbgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgQTbG9uZ1ByZW1pdW1GcmFjdGlvbgkBBGRpdmQCCQEEbXVsZAIFB3ByZW1pdW0JARRmdW5kaW5nUGVyaW9kRGVjaW1hbAAFB09ORV9EQVkEFHNob3J0UHJlbWl1bUZyYWN0aW9uCQEEZGl2ZAIJAQRtdWxkAgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQAJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAkAlAoCBRRzaG9ydFByZW1pdW1GcmFjdGlvbgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgEOZ2V0QWRqdXN0ZWRGZWUBAWkEB2Jhc2VGZWUJAQNmZWUABA0kdDAyOTM3NjI5OTg0AwkAZgIJAJADAQgFAWkIcGF5bWVudHMAAQQKYXJ0aWZhY3RJZAkA2AQBCQETdmFsdWVPckVycm9yTWVzc2FnZQIICQCRAwIIBQFpCHBheW1lbnRzAAEHYXNzZXRJZAIZSW52YWxpZCBhdHRhY2hlZCBhcnRpZmFjdAQMYXJ0aWZhY3RLaW5kCQEEc3RyQQIJARFuZnRNYW5hZ2VyQWRkcmVzcwAJAQ50b0NvbXBvc2l0ZUtleQIFDGtfdG9rZW5fdHlwZQUKYXJ0aWZhY3RJZAMJAAACBQxhcnRpZmFjdEtpbmQFGEZFRV9SRURVQ1RJT05fVE9LRU5fVFlQRQQJcmVkdWN0aW9uCQEEaW50QQIJARFuZnRNYW5hZ2VyQWRkcmVzcwAJAQ50b0NvbXBvc2l0ZUtleQIFDWtfdG9rZW5fcGFyYW0FCmFydGlmYWN0SWQEC2FkanVzdGVkRmVlCQEEbXVsZAIFB2Jhc2VGZWUFCXJlZHVjdGlvbgkAlAoCBQthZGp1c3RlZEZlZQYJAAIBAhlJbnZhbGlkIGF0dGFjaGVkIGFydGlmYWN0CQCUCgIFB2Jhc2VGZWUHBAthZGp1c3RlZEZlZQgFDSR0MDI5Mzc2Mjk5ODQCXzEEDGJ1cm5BcnRpZmFjdAgFDSR0MDI5Mzc2Mjk5ODQCXzIJAJQKAgULYWRqdXN0ZWRGZWUFDGJ1cm5BcnRpZmFjdAEOdXBkYXRlU2V0dGluZ3MJEF9pbml0TWFyZ2luUmF0aW8EX21tchRfbGlxdWlkYXRpb25GZWVSYXRpbw5fZnVuZGluZ1BlcmlvZARfZmVlDF9zcHJlYWRMaW1pdA9fbWF4UHJpY2VJbXBhY3QYX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvD19tYXhQcmljZVNwcmVhZAkAzAgCCQEMSW50ZWdlckVudHJ5AgURa19pbml0TWFyZ2luUmF0aW8FEF9pbml0TWFyZ2luUmF0aW8JAMwIAgkBDEludGVnZXJFbnRyeQIFGGtfbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwUEX21tcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUVa19saXF1aWRhdGlvbkZlZVJhdGlvBRRfbGlxdWlkYXRpb25GZWVSYXRpbwkAzAgCCQEMSW50ZWdlckVudHJ5AgUPa19mdW5kaW5nUGVyaW9kBQ5fZnVuZGluZ1BlcmlvZAkAzAgCCQEMSW50ZWdlckVudHJ5AgUFa19mZWUFBF9mZWUJAMwIAgkBDEludGVnZXJFbnRyeQIFDWtfc3ByZWFkTGltaXQFDF9zcHJlYWRMaW1pdAkAzAgCCQEMSW50ZWdlckVudHJ5AgUQa19tYXhQcmljZUltcGFjdAUPX21heFByaWNlSW1wYWN0CQDMCAIJAQxJbnRlZ2VyRW50cnkCBRlrX3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvBRhfcGFydGlhbExpcXVpZGF0aW9uUmF0aW8JAMwIAgkBDEludGVnZXJFbnRyeQIFEGtfbWF4UHJpY2VTcHJlYWQFD19tYXhQcmljZVNwcmVhZAUDbmlsAQ11cGRhdGVGdW5kaW5nBRFfbmV4dEZ1bmRpbmdCbG9jayRfbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24lX2xhdGVzdFNob3J0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbhBfbG9uZ0Z1bmRpbmdSYXRlEV9zaG9ydEZ1bmRpbmdSYXRlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRJrX25leHRGdW5kaW5nQmxvY2sFEV9uZXh0RnVuZGluZ0Jsb2NrCQDMCAIJAQxJbnRlZ2VyRW50cnkCBSVrX2xhdGVzdExvbmdDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBSRfbGF0ZXN0TG9uZ0N1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24JAMwIAgkBDEludGVnZXJFbnRyeQIFJmtfbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBSVfbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRFrX2xvbmdGdW5kaW5nUmF0ZQUQX2xvbmdGdW5kaW5nUmF0ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUSa19zaG9ydEZ1bmRpbmdSYXRlBRFfc2hvcnRGdW5kaW5nUmF0ZQUDbmlsARN1cGRhdGVQb3NpdGlvbkFzc2V0AghfYWRkcmVzcwhfYXNzZXRJZAkAzAgCCQELU3RyaW5nRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBQ9rX3Bvc2l0aW9uQXNzZXQFCF9hZGRyZXNzBQhfYXNzZXRJZAUDbmlsAQ51cGRhdGVQb3NpdGlvbgUIX2FkZHJlc3MFX3NpemUHX21hcmdpbg1fb3Blbk5vdGlvbmFsIF9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBQ5rX3Bvc2l0aW9uU2l6ZQUIX2FkZHJlc3MFBV9zaXplCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRBrX3Bvc2l0aW9uTWFyZ2luBQhfYWRkcmVzcwUHX21hcmdpbgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUWa19wb3NpdGlvbk9wZW5Ob3Rpb25hbAUIX2FkZHJlc3MFDV9vcGVuTm90aW9uYWwJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFLmtfcG9zaXRpb25MYXN0VXBkYXRlZEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24FCF9hZGRyZXNzBSBfbGF0ZXN0Q3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgUDbmlsAQphcHBlbmRUd2FwAQVwcmljZQQIbWludXRlSWQJAGkCCQBpAggFCWxhc3RCbG9jawl0aW1lc3RhbXAA6AcAPAQQcHJldmlvdXNNaW51dGVJZAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBQ5rX2xhc3RNaW51dGVJZAAAAwkAZgIFEHByZXZpb3VzTWludXRlSWQFCG1pbnV0ZUlkCQACAQIRVFdBUCBvdXQtb2Ytb3JkZXIEDGxhc3RNaW51dGVJZAMJAAACBRBwcmV2aW91c01pbnV0ZUlkAAAFCG1pbnV0ZUlkBRBwcmV2aW91c01pbnV0ZUlkAwkAZgIFCG1pbnV0ZUlkBRBwcmV2aW91c01pbnV0ZUlkBBNwcmV2Q3VtdWxhdGl2ZVByaWNlCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAKwCAgkArAICBR1rX3R3YXBEYXRhTGFzdEN1bXVsYXRpdmVQcmljZQIBXwkApAMBBRBwcmV2aW91c01pbnV0ZUlkAAAECXByZXZQcmljZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQCsAgIJAKwCAgUTa190d2FwRGF0YUxhc3RQcmljZQIBXwkApAMBBRBwcmV2aW91c01pbnV0ZUlkBQVwcmljZQQTbGFzdEN1bXVsYXRpdmVQcmljZQkAZAIFE3ByZXZDdW11bGF0aXZlUHJpY2UJAGgCCQBlAgUIbWludXRlSWQFDGxhc3RNaW51dGVJZAUJcHJldlByaWNlBARsaXN0CQELcHVzaFRvUXVldWUDCQEJc3RyVG9MaXN0AQkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzBQ1rX2xhc3REYXRhU3RyAgAFDVRXQVBfSU5URVJWQUwJAKQDAQUIbWludXRlSWQJAMwIAgkBDEludGVnZXJFbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFHWtfdHdhcERhdGFMYXN0Q3VtdWxhdGl2ZVByaWNlCQCkAwEFCG1pbnV0ZUlkBRNsYXN0Q3VtdWxhdGl2ZVByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRNrX3R3YXBEYXRhTGFzdFByaWNlCQCkAwEFCG1pbnV0ZUlkBQVwcmljZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUaa190d2FwRGF0YVByZXZpb3VzTWludXRlSWQJAKQDAQUIbWludXRlSWQFEHByZXZpb3VzTWludXRlSWQJAMwIAgkBDEludGVnZXJFbnRyeQIFDmtfbGFzdE1pbnV0ZUlkBQhtaW51dGVJZAkAzAgCCQELU3RyaW5nRW50cnkCBQ1rX2xhc3REYXRhU3RyCQEJbGlzdFRvU3RyAQUEbGlzdAUDbmlsBBh0d2FwRGF0YVByZXZpb3VzTWludXRlSWQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBDnRvQ29tcG9zaXRlS2V5AgUaa190d2FwRGF0YVByZXZpb3VzTWludXRlSWQJAKQDAQUIbWludXRlSWQAAAQTcHJldkN1bXVsYXRpdmVQcmljZQkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEOdG9Db21wb3NpdGVLZXkCBR1rX3R3YXBEYXRhTGFzdEN1bXVsYXRpdmVQcmljZQkApAMBBRh0d2FwRGF0YVByZXZpb3VzTWludXRlSWQAAAQJcHJldlByaWNlCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQ50b0NvbXBvc2l0ZUtleQIFE2tfdHdhcERhdGFMYXN0UHJpY2UJAKQDAQUYdHdhcERhdGFQcmV2aW91c01pbnV0ZUlkBQVwcmljZQQTbGFzdEN1bXVsYXRpdmVQcmljZQkAZAIFE3ByZXZDdW11bGF0aXZlUHJpY2UJAGgCCQBlAgUIbWludXRlSWQFGHR3YXBEYXRhUHJldmlvdXNNaW51dGVJZAUJcHJldlByaWNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBR1rX3R3YXBEYXRhTGFzdEN1bXVsYXRpdmVQcmljZQkApAMBBQhtaW51dGVJZAUTbGFzdEN1bXVsYXRpdmVQcmljZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBDnRvQ29tcG9zaXRlS2V5AgUTa190d2FwRGF0YUxhc3RQcmljZQkApAMBBQhtaW51dGVJZAUFcHJpY2UFA25pbAERdXBkYXRlQW1tUmVzZXJ2ZXMCB19xdEFzdFIHX2JzQXN0UgkAzAgCCQEMSW50ZWdlckVudHJ5AgUTa19xdW90ZUFzc2V0UmVzZXJ2ZQUHX3F0QXN0UgkAzAgCCQEMSW50ZWdlckVudHJ5AgUSa19iYXNlQXNzZXRSZXNlcnZlBQdfYnNBc3RSBQNuaWwBCXVwZGF0ZUFtbQcHX3F0QXN0UgdfYnNBc3RSF190b3RhbFBvc2l0aW9uU2l6ZUFmdGVyGF9jdW11bGF0aXZlTm90aW9uYWxBZnRlchVfb3BlbkludGVyZXN0Tm90aW9uYWwWX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZRdfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQMJAQIhPQIJAGUCBRZfdG90YWxMb25nUG9zaXRpb25TaXplBRdfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQUXX3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIJAAIBCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgIYSW52YWxpZCBBTU0gc3RhdGUgZGF0YTogCQCkAwEFFl90b3RhbExvbmdQb3NpdGlvblNpemUCBCArICAJAKQDAQUXX3RvdGFsU2hvcnRQb3NpdGlvblNpemUCBCAhPSAJAKQDAQUXX3RvdGFsUG9zaXRpb25TaXplQWZ0ZXIJAM4IAgkAzggCCQERdXBkYXRlQW1tUmVzZXJ2ZXMCBQdfcXRBc3RSBQdfYnNBc3RSCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRNrX3RvdGFsUG9zaXRpb25TaXplBRdfdG90YWxQb3NpdGlvblNpemVBZnRlcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUUa19jdW11bGF0aXZlTm90aW9uYWwFGF9jdW11bGF0aXZlTm90aW9uYWxBZnRlcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUWa19vcGVuSW50ZXJlc3ROb3Rpb25hbAUVX29wZW5JbnRlcmVzdE5vdGlvbmFsCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRdrX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQUWX3RvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQkAzAgCCQEMSW50ZWdlckVudHJ5AgUYa190b3RhbFNob3J0UG9zaXRpb25TaXplBRdfdG90YWxTaG9ydFBvc2l0aW9uU2l6ZQUDbmlsCQEKYXBwZW5kVHdhcAEJAQRkaXZkAgUHX3F0QXN0UgUHX2JzQXN0UgEOZGVsZXRlUG9zaXRpb24BCF9hZGRyZXNzCQDMCAIJAQtEZWxldGVFbnRyeQEJAQ50b0NvbXBvc2l0ZUtleQIFDmtfcG9zaXRpb25TaXplBQhfYWRkcmVzcwkAzAgCCQELRGVsZXRlRW50cnkBCQEOdG9Db21wb3NpdGVLZXkCBRBrX3Bvc2l0aW9uTWFyZ2luBQhfYWRkcmVzcwkAzAgCCQELRGVsZXRlRW50cnkBCQEOdG9Db21wb3NpdGVLZXkCBRZrX3Bvc2l0aW9uT3Blbk5vdGlvbmFsBQhfYWRkcmVzcwkAzAgCCQELRGVsZXRlRW50cnkBCQEOdG9Db21wb3NpdGVLZXkCBS5rX3Bvc2l0aW9uTGFzdFVwZGF0ZWRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uBQhfYWRkcmVzcwkAzAgCCQELRGVsZXRlRW50cnkBCQEOdG9Db21wb3NpdGVLZXkCBQ9rX3Bvc2l0aW9uQXNzZXQFCF9hZGRyZXNzCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEOdG9Db21wb3NpdGVLZXkCBRRrX3Bvc2l0aW9uQ2xvc2VkRGF0ZQUIX2FkZHJlc3MIBQlsYXN0QmxvY2sJdGltZXN0YW1wBQNuaWwBCHdpdGhkcmF3AghfYWRkcmVzcwdfYW1vdW50BAdiYWxhbmNlCQDwBwIFBHRoaXMJAQpxdW90ZUFzc2V0AAMJAGYCBQdfYW1vdW50BQdiYWxhbmNlCQACAQkArAICCQCsAgIJAKwCAgITVW5hYmxlIHRvIHdpdGhkcmF3IAkApAMBBQdfYW1vdW50AhcgZnJvbSBjb250cmFjdCBiYWxhbmNlIAkApAMBBQdiYWxhbmNlCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMFCF9hZGRyZXNzBQdfYW1vdW50CQEKcXVvdGVBc3NldAAFA25pbAENdXBkYXRlQmFsYW5jZQEBaQMJAGYCAAAFAWkJAAIBAgdCYWxhbmNlCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQlrX2JhbGFuY2UFAWkFA25pbAELdHJhbnNmZXJGZWUBAWkJAMwIAgkBDlNjcmlwdFRyYW5zZmVyAwkBDnN0YWtpbmdBZGRyZXNzAAUBaQkBCnF1b3RlQXNzZXQABQNuaWwBDmRvQnVybkFydGlmYWN0Ag1fYnVybkFydGlmYWN0AWkDBQ1fYnVybkFydGlmYWN0CQDMCAIJAQRCdXJuAgkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCAkAkQMCCAUBaQhwYXltZW50cwABB2Fzc2V0SWQCEEludmFsaWQgYXJ0aWZhY3QAAQUDbmlsBQNuaWwBF2lzU2FtZUFzc2V0T3JOb1Bvc2l0aW9uAgdfdHJhZGVyCF9hc3NldElkBA9vbGRQb3NpdGlvblNpemUICQELZ2V0UG9zaXRpb24BBQdfdHJhZGVyAl8xAwkAAAIFD29sZFBvc2l0aW9uU2l6ZQAABgkAAAIJARBnZXRQb3NpdGlvbkFzc2V0AQUHX3RyYWRlcgUIX2Fzc2V0SWQBC2lzU2FtZUFzc2V0AgdfdHJhZGVyCF9hc3NldElkCQAAAgkBEGdldFBvc2l0aW9uQXNzZXQBBQdfdHJhZGVyBQhfYXNzZXRJZAEeZ2V0Qm9ycm93ZWRCeVRyYWRlckluTWFya2V0S2V5AwRfYW1tCF9hc3NldElkB190cmFkZXIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgIFIGtfdHJhZGVyX21hcmtldF9hc3NldF9jb2xsYXRlcmFsAgFfBQRfYW1tAgFfBQhfYXNzZXRJZAIBXwUHX3RyYWRlcgETZ2V0Qm9ycm93ZWRCeVRyYWRlcgEHX3RyYWRlcgQNcG9zaXRpb25Bc3NldAkBEGdldFBvc2l0aW9uQXNzZXQBBQdfdHJhZGVyAwkAAAIFDXBvc2l0aW9uQXNzZXQJANgEAQkBCnF1b3RlQXNzZXQACQCUCgIAAAUNcG9zaXRpb25Bc3NldAQDa2V5CQEeZ2V0Qm9ycm93ZWRCeVRyYWRlckluTWFya2V0S2V5AwkApQgBBQR0aGlzBQ1wb3NpdGlvbkFzc2V0BQdfdHJhZGVyBAZib3Jyb3cJAQt2YWx1ZU9yRWxzZQIJAJoIAgkBEWNvbGxhdGVyYWxBZGRyZXNzAAUDa2V5AAAJAJQKAgUGYm9ycm93BQ1wb3NpdGlvbkFzc2V0FgFpAQVwYXVzZQADCQECIT0CCAUBaQZjYWxsZXIJAQxhZG1pbkFkZHJlc3MACQACAQIaSW52YWxpZCB0b2dnbGVQYXVzZSBwYXJhbXMJAMwIAgkBDEJvb2xlYW5FbnRyeQIFCGtfcGF1c2VkBgUDbmlsAWkBB3VucGF1c2UAAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAkAAgECGkludmFsaWQgdG9nZ2xlUGF1c2UgcGFyYW1zCQDMCAIJAQxCb29sZWFuRW50cnkCBQhrX3BhdXNlZAcFA25pbAFpAQxhZGRMaXF1aWRpdHkBEV9xdW90ZUFzc2V0QW1vdW50AwMJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAGCQBnAgAABRFfcXVvdGVBc3NldEFtb3VudAkAAgECG0ludmFsaWQgYWRkTGlxdWlkaXR5IHBhcmFtcwQHX3F0QXN0UgkBBnF0QXN0UgAEB19ic0FzdFIJAQZic0FzdFIABAVwcmljZQkBBGRpdmQCBQdfcXRBc3RSBQdfYnNBc3RSBBRiYXNlQXNzZXRBbW91bnRUb0FkZAkBBGRpdmQCBRFfcXVvdGVBc3NldEFtb3VudAUFcHJpY2UEC3F0QXN0UkFmdGVyCQBkAgUHX3F0QXN0UgURX3F1b3RlQXNzZXRBbW91bnQEC2JzQXN0UkFmdGVyCQBkAgUHX2JzQXN0UgUUYmFzZUFzc2V0QW1vdW50VG9BZGQJARF1cGRhdGVBbW1SZXNlcnZlcwIFC3F0QXN0UkFmdGVyBQtic0FzdFJBZnRlcgFpAQ9yZW1vdmVMaXF1aWRpdHkBEV9xdW90ZUFzc2V0QW1vdW50AwMJAQIhPQIIBQFpBmNhbGxlcgkBDGFkbWluQWRkcmVzcwAGCQBnAgAABRFfcXVvdGVBc3NldEFtb3VudAkAAgECHkludmFsaWQgcmVtb3ZlTGlxdWlkaXR5IHBhcmFtcwQHX3F0QXN0UgkBBnF0QXN0UgAEB19ic0FzdFIJAQZic0FzdFIABAVwcmljZQkBBGRpdmQCBQdfcXRBc3RSBQdfYnNBc3RSBBdiYXNlQXNzZXRBbW91bnRUb1JlbW92ZQkBBGRpdmQCBRFfcXVvdGVBc3NldEFtb3VudAUFcHJpY2UEC3F0QXN0UkFmdGVyCQBlAgUHX3F0QXN0UgURX3F1b3RlQXNzZXRBbW91bnQEC2JzQXN0UkFmdGVyCQBlAgUHX2JzQXN0UgUXYmFzZUFzc2V0QW1vdW50VG9SZW1vdmUJARF1cGRhdGVBbW1SZXNlcnZlcwIFC3F0QXN0UkFmdGVyBQtic0FzdFJBZnRlcgFpAQ5jaGFuZ2VTZXR0aW5ncwkQX2luaXRNYXJnaW5SYXRpbwRfbW1yFF9saXF1aWRhdGlvbkZlZVJhdGlvDl9mdW5kaW5nUGVyaW9kBF9mZWUMX3NwcmVhZExpbWl0D19tYXhQcmljZUltcGFjdBhfcGFydGlhbExpcXVpZGF0aW9uUmF0aW8PX21heFByaWNlU3ByZWFkAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAkAAgECHUludmFsaWQgY2hhbmdlU2V0dGluZ3MgcGFyYW1zCQEOdXBkYXRlU2V0dGluZ3MJBRBfaW5pdE1hcmdpblJhdGlvBQRfbW1yBRRfbGlxdWlkYXRpb25GZWVSYXRpbwUOX2Z1bmRpbmdQZXJpb2QFBF9mZWUFDF9zcHJlYWRMaW1pdAUPX21heFByaWNlSW1wYWN0BRhfcGFydGlhbExpcXVpZGF0aW9uUmF0aW8FD19tYXhQcmljZVNwcmVhZAFpAQppbml0aWFsaXplDgdfcXRBc3RSB19ic0FzdFIOX2Z1bmRpbmdQZXJpb2QQX2luaXRNYXJnaW5SYXRpbwRfbW1yFF9saXF1aWRhdGlvbkZlZVJhdGlvBF9mZWUHX29yYWNsZQpfb3JhY2xlS2V5DF9jb29yZGluYXRvcgxfc3ByZWFkTGltaXQPX21heFByaWNlSW1wYWN0GF9wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbw9fbWF4UHJpY2VTcHJlYWQDAwMDAwMDAwMDAwMJAGcCAAAFB19xdEFzdFIGCQBnAgAABQdfYnNBc3RSBgkAZwIAAAUOX2Z1bmRpbmdQZXJpb2QGCQBnAgAABRBfaW5pdE1hcmdpblJhdGlvBgkAZwIAAAUEX21tcgYJAGcCAAAFFF9saXF1aWRhdGlvbkZlZVJhdGlvBgkAZwIAAAUEX2ZlZQYJAGcCAAAFDF9zcHJlYWRMaW1pdAYJAGcCAAAFD19tYXhQcmljZUltcGFjdAYJAGcCAAAFGF9wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwYJAGcCAAAFD19tYXhQcmljZVNwcmVhZAYJAQtpbml0aWFsaXplZAAJAAIBAh1JbnZhbGlkIGluaXRpYWxpemUgcGFyYW1ldGVycwkAzggCCQDOCAIJAM4IAgkAzggCCQEJdXBkYXRlQW1tBwUHX3F0QXN0UgUHX2JzQXN0UgAAAAAAAAAAAAAJAQ51cGRhdGVTZXR0aW5ncwkFEF9pbml0TWFyZ2luUmF0aW8FBF9tbXIFFF9saXF1aWRhdGlvbkZlZVJhdGlvBQ5fZnVuZGluZ1BlcmlvZAUEX2ZlZQUMX3NwcmVhZExpbWl0BQ9fbWF4UHJpY2VJbXBhY3QFGF9wYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwUPX21heFByaWNlU3ByZWFkCQENdXBkYXRlRnVuZGluZwUJAGQCCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAUOX2Z1bmRpbmdQZXJpb2QAAAAAAAAAAAkBDXVwZGF0ZUJhbGFuY2UBAAAJAMwIAgkBDEJvb2xlYW5FbnRyeQIFDWtfaW5pdGlhbGl6ZWQGCQDMCAIJAQtTdHJpbmdFbnRyeQIFBWtfb3JhBQdfb3JhY2xlCQDMCAIJAQtTdHJpbmdFbnRyeQIFCWtfb3JhX2tleQUKX29yYWNsZUtleQkAzAgCCQELU3RyaW5nRW50cnkCBRRrX2Nvb3JkaW5hdG9yQWRkcmVzcwUMX2Nvb3JkaW5hdG9yBQNuaWwBaQESc2V0SW5pdE1hcmdpblJhdGlvARBfaW5pdE1hcmdpblJhdGlvAwMJAGcCAAAFEF9pbml0TWFyZ2luUmF0aW8GCQEBIQEJAQtpbml0aWFsaXplZAAJAAIBAiVJbnZhbGlkIHNldEluaXRNYXJnaW5SYXRpbyBwYXJhbWV0ZXJzCQEOdXBkYXRlU2V0dGluZ3MJBRBfaW5pdE1hcmdpblJhdGlvCQEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAJARNsaXF1aWRhdGlvbkZlZVJhdGlvAAkBEGZ1bmRpbmdQZXJpb2RSYXcACQEDZmVlAAkBC3NwcmVhZExpbWl0AAkBDm1heFByaWNlSW1wYWN0AAkBF3BhcnRpYWxMaXF1aWRhdGlvblJhdGlvAAkBDm1heFByaWNlU3ByZWFkAAFpARBkZWNyZWFzZVBvc2l0aW9uAwdfYW1vdW50CV9sZXZlcmFnZRNfbWluQmFzZUFzc2V0QW1vdW50AwMDAwMJAGcCAAAFB19hbW91bnQGCQEBIQEJAQtpbml0aWFsaXplZAAGCQEBIQEJARZyZXF1aXJlTW9yZU1hcmdpblJhdGlvAwkBBGRpdmQCBQxERUNJTUFMX1VOSVQFCV9sZXZlcmFnZQkBD2luaXRNYXJnaW5SYXRpbwAGBgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBgkBBnBhdXNlZAAJAAIBAiNJbnZhbGlkIGRlY3JlYXNlUG9zaXRpb24gcGFyYW1ldGVycwQNJHQwNDE0OTI0MTY0NAkBC2dldFBvc2l0aW9uAQkApQgBCAUBaQZjYWxsZXIED29sZFBvc2l0aW9uU2l6ZQgFDSR0MDQxNDkyNDE2NDQCXzEEEW9sZFBvc2l0aW9uTWFyZ2luCAUNJHQwNDE0OTI0MTY0NAJfMgQXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDA0MTQ5MjQxNjQ0Al8zBBRvbGRQb3NpdGlvbkxzdFVwZENQRggFDSR0MDQxNDkyNDE2NDQCXzQECl9kaXJlY3Rpb24DCQBmAgUPb2xkUG9zaXRpb25TaXplAAAFCURJUl9TSE9SVAUIRElSX0xPTkcEBWlzQWRkCQAAAgUKX2RpcmVjdGlvbgUIRElSX0xPTkcEDG9wZW5Ob3Rpb25hbAkBBG11bGQCBQdfYW1vdW50BQlfbGV2ZXJhZ2UEDSR0MDQxODE3NDE5MzMJASNnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubAIJAKUIAQgFAWkGY2FsbGVyBQ9QTkxfT1BUSU9OX1NQT1QEE29sZFBvc2l0aW9uTm90aW9uYWwIBQ0kdDA0MTgxNzQxOTMzAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwNDE4MTc0MTkzMwJfMgQNJHQwNDE5Mzk0NDQ4OAMJAGYCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBQxvcGVuTm90aW9uYWwEDSR0MDQyMzE2NDI1MzUJAQlzd2FwSW5wdXQCBQVpc0FkZAUMb3Blbk5vdGlvbmFsBBVleGNoYW5nZWRQb3NpdGlvblNpemUIBQ0kdDA0MjMxNjQyNTM1Al8xBBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNDIzMTY0MjUzNQJfMgQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNDIzMTY0MjUzNQJfMwQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDQyMzE2NDI1MzUCXzQEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwNDIzMTY0MjUzNQJfNQQYZXhjaGFuZ2VkUG9zaXRpb25TaXplQWJzCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplAwMJAQIhPQIFE19taW5CYXNlQXNzZXRBbW91bnQAAAkAZgIFE19taW5CYXNlQXNzZXRBbW91bnQFGGV4Y2hhbmdlZFBvc2l0aW9uU2l6ZUFicwcJAAIBCQCsAgIJAKwCAgkArAICAiVUb28gbGl0dGxlIGJhc2UgYXNzZXQgZXhjaGFuZ2VkLCBnb3QgCQCkAwEFGGV4Y2hhbmdlZFBvc2l0aW9uU2l6ZUFicwIKIGV4cGVjdGVkIAkApAMBBRNfbWluQmFzZUFzc2V0QW1vdW50BAtyZWFsaXplZFBubAkBBGRpdmQCCQEEbXVsZAIFDXVucmVhbGl6ZWRQbmwFGGV4Y2hhbmdlZFBvc2l0aW9uU2l6ZUFicwkBA2FicwEFD29sZFBvc2l0aW9uU2l6ZQQNJHQwNDI5NzI0MzIxNwkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQ9vbGRQb3NpdGlvblNpemUFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvbkxzdFVwZENQRgULcmVhbGl6ZWRQbmwEDHJlbWFpbk1hcmdpbggFDSR0MDQyOTcyNDMyMTcCXzEEB2JhZERlYnQIBQ0kdDA0Mjk3MjQzMjE3Al8yBA5mdW5kaW5nUGF5bWVudAgFDSR0MDQyOTcyNDMyMTcCXzMEGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQFDG9wZW5Ob3Rpb25hbAQSdW5yZWFsaXplZFBubEFmdGVyCQBlAgUNdW5yZWFsaXplZFBubAULcmVhbGl6ZWRQbmwEEnJlbWFpbk9wZW5Ob3Rpb25hbAMJAGYCBQ9vbGRQb3NpdGlvblNpemUAAAkAZQIJAGUCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BRJ1bnJlYWxpemVkUG5sQWZ0ZXIJAGUCCQBkAgUSdW5yZWFsaXplZFBubEFmdGVyBRNvbGRQb3NpdGlvbk5vdGlvbmFsBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BA9uZXdQb3NpdGlvblNpemUJAGQCBQ9vbGRQb3NpdGlvblNpemUFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQkAnQoLBQ9uZXdQb3NpdGlvblNpemUFDHJlbWFpbk1hcmdpbgkBA2FicwEFEnJlbWFpbk9wZW5Ob3Rpb25hbAkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBQ9uZXdQb3NpdGlvblNpemUFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIJAGUCCQEUb3BlbkludGVyZXN0Tm90aW9uYWwABQxvcGVuTm90aW9uYWwJAGUCCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAMJAGYCBQ9uZXdQb3NpdGlvblNpemUAAAkBA2FicwEFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQAACQBlAgkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUAAwkAZgIAAAUPbmV3UG9zaXRpb25TaXplCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplAAAJAAIBAhRDbG9zZSBwb3NpdGlvbiBmaXJzdAQPbmV3UG9zaXRpb25TaXplCAUNJHQwNDE5Mzk0NDQ4OAJfMQQXbmV3UG9zaXRpb25SZW1haW5NYXJnaW4IBQ0kdDA0MTkzOTQ0NDg4Al8yBBduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDQxOTM5NDQ0ODgCXzMEFG5ld1Bvc2l0aW9uTGF0ZXN0Q1BGCAUNJHQwNDE5Mzk0NDQ4OAJfNAQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNDE5Mzk0NDQ4OAJfNQQWcXVvdGVBc3NldFJlc2VydmVBZnRlcggFDSR0MDQxOTM5NDQ0ODgCXzYEFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIIBQ0kdDA0MTkzOTQ0NDg4Al83BBdjdW11bGF0aXZlTm90aW9uYWxBZnRlcggFDSR0MDQxOTM5NDQ0ODgCXzgEGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIIBQ0kdDA0MTkzOTQ0NDg4Al85BA50b3RhbExvbmdBZnRlcggFDSR0MDQxOTM5NDQ0ODgDXzEwBA90b3RhbFNob3J0QWZ0ZXIIBQ0kdDA0MTkzOTQ0NDg4A18xMQQObm90aWZ5Tm90aW9uYWwJAPwHBAkBDG1pbmVyQWRkcmVzcwACDm5vdGlmeU5vdGlvbmFsCQDMCAIJAKUIAQgFAWkGY2FsbGVyCQDMCAIFF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBQNuaWwFA25pbAMJAAACBQ5ub3RpZnlOb3Rpb25hbAUObm90aWZ5Tm90aW9uYWwJAM4IAgkBDnVwZGF0ZVBvc2l0aW9uBQkApQgBCAUBaQZjYWxsZXIFD25ld1Bvc2l0aW9uU2l6ZQUXbmV3UG9zaXRpb25SZW1haW5NYXJnaW4FF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBRRuZXdQb3NpdGlvbkxhdGVzdENQRgkBCXVwZGF0ZUFtbQcFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIFGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIFDnRvdGFsTG9uZ0FmdGVyBQ90b3RhbFNob3J0QWZ0ZXIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEQaW5jcmVhc2VQb3NpdGlvbgQKX2RpcmVjdGlvbglfbGV2ZXJhZ2UTX21pbkJhc2VBc3NldEFtb3VudAhfcmVmTGluawQHX3RyYWRlcgkApQgBCAUBaQZjYWxsZXIECl9yYXdBbW91bnQICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BAhfYXNzZXRJZAgJAJEDAggFAWkIcGF5bWVudHMAAAdhc3NldElkBAtfYXNzZXRJZFN0cgkA2AQBCQEFdmFsdWUBBQhfYXNzZXRJZAQMaXNRdW90ZUFzc2V0CQAAAgUIX2Fzc2V0SWQJAQpxdW90ZUFzc2V0AAQRaXNDb2xsYXRlcmFsQXNzZXQJARBpc1doaXRlbGlzdEFzc2V0AQULX2Fzc2V0SWRTdHIDAwMDAwMDAwkBAiE9AgUKX2RpcmVjdGlvbgUIRElSX0xPTkcJAQIhPQIFCl9kaXJlY3Rpb24FCURJUl9TSE9SVAcGCQBnAgAABQpfcmF3QW1vdW50BgkBASEBCQELaW5pdGlhbGl6ZWQABgMJAQEhAQUMaXNRdW90ZUFzc2V0CQEBIQEFEWlzQ29sbGF0ZXJhbEFzc2V0BwYJAQEhAQkBF2lzU2FtZUFzc2V0T3JOb1Bvc2l0aW9uAgUHX3RyYWRlcgULX2Fzc2V0SWRTdHIGCQEBIQEJARZyZXF1aXJlTW9yZU1hcmdpblJhdGlvAwkBBGRpdmQCBQxERUNJTUFMX1VOSVQFCV9sZXZlcmFnZQkBD2luaXRNYXJnaW5SYXRpbwAGBgkBBnBhdXNlZAAJAAIBAiNJbnZhbGlkIGluY3JlYXNlUG9zaXRpb24gcGFyYW1ldGVycwQHYmFzZUZlZQkBA2ZlZQAEDSR0MDQ1ODUwNDU5MDEJAQ5nZXRBZGp1c3RlZEZlZQEFAWkEC2FkanVzdGVkRmVlCAUNJHQwNDU4NTA0NTkwMQJfMQQMYnVybkFydGlmYWN0CAUNJHQwNDU4NTA0NTkwMQJfMgQMcmF3RmVlQW1vdW50CQEEbXVsZAIFCl9yYXdBbW91bnQFC2FkanVzdGVkRmVlBAdfYW1vdW50CQBlAgUKX3Jhd0Ftb3VudAUMcmF3RmVlQW1vdW50BBNkaXN0cmlidXRlRmVlQW1vdW50AwURaXNDb2xsYXRlcmFsQXNzZXQECGRvQm9ycm93CQD8BwQJARFjb2xsYXRlcmFsQWRkcmVzcwACBmJvcnJvdwkAzAgCBQdfdHJhZGVyBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIFCF9hc3NldElkBQdfYW1vdW50BQNuaWwDCQAAAgUIZG9Cb3Jyb3cFCGRvQm9ycm93BA1iYWxhbmNlQmVmb3JlCQDwBwIFBHRoaXMJAQpxdW90ZUFzc2V0AAMJAAACBQ1iYWxhbmNlQmVmb3JlBQ1iYWxhbmNlQmVmb3JlBAZkb1N3YXAJAPwHBAkBC3N3YXBBZGRyZXNzAAIEc3dhcAkAzAgCCQDYBAEJAQpxdW90ZUFzc2V0AAkAzAgCAAAFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgUIX2Fzc2V0SWQFDHJhd0ZlZUFtb3VudAUDbmlsAwkAAAIFBmRvU3dhcAUGZG9Td2FwBAxiYWxhbmNlQWZ0ZXIJAPAHAgUEdGhpcwkBCnF1b3RlQXNzZXQAAwkAAAIFDGJhbGFuY2VBZnRlcgUMYmFsYW5jZUFmdGVyBA9leGNoYW5nZWRBbW91bnQJAGUCBQxiYWxhbmNlQWZ0ZXIFDWJhbGFuY2VCZWZvcmUDCQAAAgUPZXhjaGFuZ2VkQW1vdW50BQ9leGNoYW5nZWRBbW91bnQFD2V4Y2hhbmdlZEFtb3VudAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgUMcmF3RmVlQW1vdW50AwkAAAIFE2Rpc3RyaWJ1dGVGZWVBbW91bnQFE2Rpc3RyaWJ1dGVGZWVBbW91bnQEDnJlZmVycmVyRmVlQW55CQD8BwQJAQ9yZWZlcnJhbEFkZHJlc3MAAhVhY2NlcHRQYXltZW50V2l0aExpbmsJAMwIAgUHX3RyYWRlcgkAzAgCBQhfcmVmTGluawUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQEKcXVvdGVBc3NldAAFE2Rpc3RyaWJ1dGVGZWVBbW91bnQFA25pbAMJAAACBQ5yZWZlcnJlckZlZUFueQUOcmVmZXJyZXJGZWVBbnkEC3JlZmVycmVyRmVlBAckbWF0Y2gwBQ5yZWZlcnJlckZlZUFueQMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAhNJbnZhbGlkIHJlZmVycmVyRmVlBAlmZWVBbW91bnQJAGUCBRNkaXN0cmlidXRlRmVlQW1vdW50BQtyZWZlcnJlckZlZQQNJHQwNDcyMTc0NzM1NwkBC2dldFBvc2l0aW9uAQUHX3RyYWRlcgQPb2xkUG9zaXRpb25TaXplCAUNJHQwNDcyMTc0NzM1NwJfMQQRb2xkUG9zaXRpb25NYXJnaW4IBQ0kdDA0NzIxNzQ3MzU3Al8yBBdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDQ3MjE3NDczNTcCXzMEFG9sZFBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwNDcyMTc0NzM1NwJfNAQNaXNOZXdQb3NpdGlvbgkAAAIFD29sZFBvc2l0aW9uU2l6ZQAABA9pc1NhbWVEaXJlY3Rpb24DCQBmAgUPb2xkUG9zaXRpb25TaXplAAAJAAACBQpfZGlyZWN0aW9uBQhESVJfTE9ORwkAAAIFCl9kaXJlY3Rpb24FCURJUl9TSE9SVAQOZXhwYW5kRXhpc3RpbmcDCQEBIQEFDWlzTmV3UG9zaXRpb24FD2lzU2FtZURpcmVjdGlvbgcEBWlzQWRkCQAAAgUKX2RpcmVjdGlvbgUIRElSX0xPTkcEDSR0MDQ3NjQ2NTAyMzEDAwUNaXNOZXdQb3NpdGlvbgYFDmV4cGFuZEV4aXN0aW5nBAxvcGVuTm90aW9uYWwJAQRtdWxkAgUHX2Ftb3VudAUJX2xldmVyYWdlBA0kdDA0ODA3MDQ4Mjc2CQEJc3dhcElucHV0AgUFaXNBZGQFDG9wZW5Ob3Rpb25hbAQVYW1vdW50QmFzZUFzc2V0Qm91Z2h0CAUNJHQwNDgwNzA0ODI3NgJfMQQWcXVvdGVBc3NldFJlc2VydmVBZnRlcggFDSR0MDQ4MDcwNDgyNzYCXzIEFWJhc2VBc3NldFJlc2VydmVBZnRlcggFDSR0MDQ4MDcwNDgyNzYCXzMEFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIIBQ0kdDA0ODA3MDQ4Mjc2Al80BBdjdW11bGF0aXZlTm90aW9uYWxBZnRlcggFDSR0MDQ4MDcwNDgyNzYCXzUDAwkBAiE9AgUTX21pbkJhc2VBc3NldEFtb3VudAAACQBmAgUTX21pbkJhc2VBc3NldEFtb3VudAkBA2FicwEFFWFtb3VudEJhc2VBc3NldEJvdWdodAcJAAIBCQCsAgIJAKwCAgkArAICAg1MaW1pdCBlcnJvcjogCQCkAwEJAQNhYnMBBRVhbW91bnRCYXNlQXNzZXRCb3VnaHQCAyA8IAkApAMBBRNfbWluQmFzZUFzc2V0QW1vdW50BA9uZXdQb3NpdGlvblNpemUJAGQCBQ9vbGRQb3NpdGlvblNpemUFFWFtb3VudEJhc2VBc3NldEJvdWdodAQZaW5jcmVhc2VNYXJnaW5SZXF1aXJlbWVudAkBBGRpdmQCBQxvcGVuTm90aW9uYWwFCV9sZXZlcmFnZQQNJHQwNDg2NTc0ODg5NgkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQ9vbGRQb3NpdGlvblNpemUFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvbkxzdFVwZENQRgUZaW5jcmVhc2VNYXJnaW5SZXF1aXJlbWVudAQMcmVtYWluTWFyZ2luCAUNJHQwNDg2NTc0ODg5NgJfMQQCeDEIBQ0kdDA0ODY1NzQ4ODk2Al8yBAJ4MggFDSR0MDQ4NjU3NDg4OTYCXzMDCQEBIQEJARlyZXF1aXJlTm90T3ZlclNwcmVhZExpbWl0AgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCQACAQIVT3ZlciBtYXggc3ByZWFkIGxpbWl0CQCdCgsFD25ld1Bvc2l0aW9uU2l6ZQUMcmVtYWluTWFyZ2luCQBkAgUXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwFDG9wZW5Ob3Rpb25hbAkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBQ9uZXdQb3NpdGlvblNpemUFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWcXVvdGVBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIJAGQCCQEUb3BlbkludGVyZXN0Tm90aW9uYWwABQxvcGVuTm90aW9uYWwJAGQCCQEVdG90YWxMb25nUG9zaXRpb25TaXplAAMJAGYCBQ9uZXdQb3NpdGlvblNpemUAAAkBA2FicwEFFWFtb3VudEJhc2VBc3NldEJvdWdodAAACQBkAgkBFnRvdGFsU2hvcnRQb3NpdGlvblNpemUAAwkAZgIAAAUPbmV3UG9zaXRpb25TaXplCQEDYWJzAQUVYW1vdW50QmFzZUFzc2V0Qm91Z2h0AAAEDG9wZW5Ob3Rpb25hbAkBBG11bGQCBQdfYW1vdW50BQlfbGV2ZXJhZ2UEDSR0MDQ5OTI0NTAwNDAJASNnZXRQb3NpdGlvbk5vdGlvbmFsQW5kVW5yZWFsaXplZFBubAIJAKUIAQgFAWkGY2FsbGVyBQ9QTkxfT1BUSU9OX1NQT1QEE29sZFBvc2l0aW9uTm90aW9uYWwIBQ0kdDA0OTkyNDUwMDQwAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwNDk5MjQ1MDA0MAJfMgMJAGYCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBQxvcGVuTm90aW9uYWwJAAIBAi5Vc2UgZGVjcmVhc2VQb3NpdGlvbiB0byBkZWNyZWFzZSBwb3NpdGlvbiBzaXplCQACAQIUQ2xvc2UgcG9zaXRpb24gZmlyc3QED25ld1Bvc2l0aW9uU2l6ZQgFDSR0MDQ3NjQ2NTAyMzECXzEEF25ld1Bvc2l0aW9uUmVtYWluTWFyZ2luCAUNJHQwNDc2NDY1MDIzMQJfMgQXbmV3UG9zaXRpb25PcGVuTm90aW9uYWwIBQ0kdDA0NzY0NjUwMjMxAl8zBBRuZXdQb3NpdGlvbkxhdGVzdENQRggFDSR0MDQ3NjQ2NTAyMzECXzQEFWJhc2VBc3NldFJlc2VydmVBZnRlcggFDSR0MDQ3NjQ2NTAyMzECXzUEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA0NzY0NjUwMjMxAl82BBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwNDc2NDY1MDIzMQJfNwQXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIIBQ0kdDA0NzY0NjUwMjMxAl84BBlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCAUNJHQwNDc2NDY1MDIzMQJfOQQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA0NzY0NjUwMjMxA18xMAQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNDc2NDY1MDIzMQNfMTEEDGZlZVRvU3Rha2VycwkAaQIFCWZlZUFtb3VudAACBA5mZWVUb0luc3VyYW5jZQkAZQIFCWZlZUFtb3VudAUMZmVlVG9TdGFrZXJzBAVzdGFrZQkA/AcECQEObWFuYWdlckFkZHJlc3MAAgdkZXBvc2l0BQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUHX2Ftb3VudAUDbmlsAwkAAAIFBXN0YWtlBQVzdGFrZQQQZGVwb3NpdEluc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQ5mZWVUb0luc3VyYW5jZQUDbmlsAwkAAAIFEGRlcG9zaXRJbnN1cmFuY2UFEGRlcG9zaXRJbnN1cmFuY2UECW5vdGlmeUZlZQkA/AcECQEMbWluZXJBZGRyZXNzAAIKbm90aWZ5RmVlcwkAzAgCBQdfdHJhZGVyCQDMCAIFCWZlZUFtb3VudAUDbmlsBQNuaWwDCQAAAgUJbm90aWZ5RmVlBQlub3RpZnlGZWUEDm5vdGlmeU5vdGlvbmFsCQD8BwQJAQxtaW5lckFkZHJlc3MAAg5ub3RpZnlOb3Rpb25hbAkAzAgCBQdfdHJhZGVyCQDMCAIFF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsBQNuaWwFA25pbAMJAAACBQ5ub3RpZnlOb3Rpb25hbAUObm90aWZ5Tm90aW9uYWwJAM4IAgkAzggCCQDOCAIJAM4IAgkAzggCCQEOdXBkYXRlUG9zaXRpb24FBQdfdHJhZGVyBQ9uZXdQb3NpdGlvblNpemUFF25ld1Bvc2l0aW9uUmVtYWluTWFyZ2luBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUUbmV3UG9zaXRpb25MYXRlc3RDUEYJARN1cGRhdGVQb3NpdGlvbkFzc2V0AgUHX3RyYWRlcgULX2Fzc2V0SWRTdHIJAQl1cGRhdGVBbW0HBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBRViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIFFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIFF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyBRlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyBQ50b3RhbExvbmdBZnRlcgUPdG90YWxTaG9ydEFmdGVyCQELdHJhbnNmZXJGZWUBBQxmZWVUb1N0YWtlcnMJAQ11cGRhdGVCYWxhbmNlAQkAZAIJAQhjYmFsYW5jZQAFB19hbW91bnQJAQ5kb0J1cm5BcnRpZmFjdAIFDGJ1cm5BcnRpZmFjdAUBaQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQlhZGRNYXJnaW4ABAdfdHJhZGVyCQClCAEIBQFpBmNhbGxlcgQKX3Jhd0Ftb3VudAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQECF9hc3NldElkCAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQEC19hc3NldElkU3RyCQDYBAEJAQV2YWx1ZQEFCF9hc3NldElkBAxpc1F1b3RlQXNzZXQJAAACBQhfYXNzZXRJZAkBCnF1b3RlQXNzZXQABBFpc0NvbGxhdGVyYWxBc3NldAkBEGlzV2hpdGVsaXN0QXNzZXQBBQtfYXNzZXRJZFN0cgMDAwMDAwkBASEBBQxpc1F1b3RlQXNzZXQJAQEhAQURaXNDb2xsYXRlcmFsQXNzZXQHBgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEJAKUIAQgFAWkGY2FsbGVyBgkBASEBCQELaXNTYW1lQXNzZXQCBQdfdHJhZGVyBQtfYXNzZXRJZFN0cgYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQZwYXVzZWQACQACAQIcSW52YWxpZCBhZGRNYXJnaW4gcGFyYW1ldGVycwQNJHQwNTE5ODc1MjAzOAkBDmdldEFkanVzdGVkRmVlAQUBaQQLYWRqdXN0ZWRGZWUIBQ0kdDA1MTk4NzUyMDM4Al8xBAxidXJuQXJ0aWZhY3QIBQ0kdDA1MTk4NzUyMDM4Al8yBAxyYXdGZWVBbW91bnQJAQRtdWxkAgUKX3Jhd0Ftb3VudAULYWRqdXN0ZWRGZWUEB19hbW91bnQJAGUCBQpfcmF3QW1vdW50BQxyYXdGZWVBbW91bnQEE2Rpc3RyaWJ1dGVGZWVBbW91bnQDBRFpc0NvbGxhdGVyYWxBc3NldAQIZG9Cb3Jyb3cJAPwHBAkBEWNvbGxhdGVyYWxBZGRyZXNzAAIGYm9ycm93CQDMCAIFB190cmFkZXIFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgUIX2Fzc2V0SWQFB19hbW91bnQFA25pbAMJAAACBQhkb0JvcnJvdwUIZG9Cb3Jyb3cEDWJhbGFuY2VCZWZvcmUJAPAHAgUEdGhpcwkBCnF1b3RlQXNzZXQAAwkAAAIFDWJhbGFuY2VCZWZvcmUFDWJhbGFuY2VCZWZvcmUEBmRvU3dhcAkA/AcECQELc3dhcEFkZHJlc3MAAgRzd2FwCQDMCAIJANgEAQkBCnF1b3RlQXNzZXQACQDMCAIAAAUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCBQhfYXNzZXRJZAUMcmF3RmVlQW1vdW50BQNuaWwDCQAAAgUGZG9Td2FwBQZkb1N3YXAEDGJhbGFuY2VBZnRlcgkA8AcCBQR0aGlzCQEKcXVvdGVBc3NldAADCQAAAgUMYmFsYW5jZUFmdGVyBQxiYWxhbmNlQWZ0ZXIED2V4Y2hhbmdlZEFtb3VudAkAZQIFDGJhbGFuY2VBZnRlcgUNYmFsYW5jZUJlZm9yZQMJAAACBQ9leGNoYW5nZWRBbW91bnQFD2V4Y2hhbmdlZEFtb3VudAUPZXhjaGFuZ2VkQW1vdW50CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQxyYXdGZWVBbW91bnQDCQAAAgUTZGlzdHJpYnV0ZUZlZUFtb3VudAUTZGlzdHJpYnV0ZUZlZUFtb3VudAQOcmVmZXJyZXJGZWVBbnkJAPwHBAkBD3JlZmVycmFsQWRkcmVzcwACDWFjY2VwdFBheW1lbnQJAMwIAgUHX3RyYWRlcgUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQEKcXVvdGVBc3NldAAFE2Rpc3RyaWJ1dGVGZWVBbW91bnQFA25pbAMJAAACBQ5yZWZlcnJlckZlZUFueQUOcmVmZXJyZXJGZWVBbnkEC3JlZmVycmVyRmVlBAckbWF0Y2gwBQ5yZWZlcnJlckZlZUFueQMJAAECBQckbWF0Y2gwAgNJbnQEAXgFByRtYXRjaDAFAXgJAAIBAhNJbnZhbGlkIHJlZmVycmVyRmVlBAlmZWVBbW91bnQJAGUCBRNkaXN0cmlidXRlRmVlQW1vdW50BQtyZWZlcnJlckZlZQQNJHQwNTMzMzc1MzQ3NwkBC2dldFBvc2l0aW9uAQUHX3RyYWRlcgQPb2xkUG9zaXRpb25TaXplCAUNJHQwNTMzMzc1MzQ3NwJfMQQRb2xkUG9zaXRpb25NYXJnaW4IBQ0kdDA1MzMzNzUzNDc3Al8yBBdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDUzMzM3NTM0NzcCXzMEFG9sZFBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwNTMzMzc1MzQ3NwJfNAQMZmVlVG9TdGFrZXJzCQBpAgUJZmVlQW1vdW50AAIEDmZlZVRvSW5zdXJhbmNlCQBlAgUJZmVlQW1vdW50BQxmZWVUb1N0YWtlcnMEBXN0YWtlCQD8BwQJAQ5tYW5hZ2VyQWRkcmVzcwACB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQdfYW1vdW50BQNuaWwDCQAAAgUFc3Rha2UFBXN0YWtlBBBkZXBvc2l0SW5zdXJhbmNlCQD8BwQJARBpbnN1cmFuY2VBZGRyZXNzAAIHZGVwb3NpdAUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQEKcXVvdGVBc3NldAAFDmZlZVRvSW5zdXJhbmNlBQNuaWwDCQAAAgUQZGVwb3NpdEluc3VyYW5jZQUQZGVwb3NpdEluc3VyYW5jZQQJbm90aWZ5RmVlCQD8BwQJAQxtaW5lckFkZHJlc3MAAgpub3RpZnlGZWVzCQDMCAIFB190cmFkZXIJAMwIAgUJZmVlQW1vdW50BQNuaWwFA25pbAMJAAACBQlub3RpZnlGZWUFCW5vdGlmeUZlZQkAzggCCQDOCAIJAM4IAgkBDnVwZGF0ZVBvc2l0aW9uBQUHX3RyYWRlcgUPb2xkUG9zaXRpb25TaXplCQBkAgURb2xkUG9zaXRpb25NYXJnaW4FB19hbW91bnQFF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsBRRvbGRQb3NpdGlvbkxzdFVwZENQRgkBC3RyYW5zZmVyRmVlAQUMZmVlVG9TdGFrZXJzCQENdXBkYXRlQmFsYW5jZQEJAGQCCQEIY2JhbGFuY2UABQdfYW1vdW50CQEOZG9CdXJuQXJ0aWZhY3QCBQxidXJuQXJ0aWZhY3QFAWkJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEMcmVtb3ZlTWFyZ2luAQdfYW1vdW50BAdfdHJhZGVyCQClCAEIBQFpBmNhbGxlcgMDAwMJAGcCAAAFB19hbW91bnQGCQEBIQEJARNyZXF1aXJlT3BlblBvc2l0aW9uAQUHX3RyYWRlcgYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQZwYXVzZWQACQACAQIfSW52YWxpZCByZW1vdmVNYXJnaW4gcGFyYW1ldGVycwQNJHQwNTQ1MjA1NDY2MAkBC2dldFBvc2l0aW9uAQUHX3RyYWRlcgQPb2xkUG9zaXRpb25TaXplCAUNJHQwNTQ1MjA1NDY2MAJfMQQRb2xkUG9zaXRpb25NYXJnaW4IBQ0kdDA1NDUyMDU0NjYwAl8yBBdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAgFDSR0MDU0NTIwNTQ2NjACXzMEFG9sZFBvc2l0aW9uTHN0VXBkQ1BGCAUNJHQwNTQ1MjA1NDY2MAJfNAQLbWFyZ2luRGVsdGEJAQEtAQUHX2Ftb3VudAQNJHQwNTQ2OTc1NDg3NgkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQ9vbGRQb3NpdGlvblNpemUFEW9sZFBvc2l0aW9uTWFyZ2luBRRvbGRQb3NpdGlvbkxzdFVwZENQRgULbWFyZ2luRGVsdGEEDHJlbWFpbk1hcmdpbggFDSR0MDU0Njk3NTQ4NzYCXzEEB2JhZERlYnQIBQ0kdDA1NDY5NzU0ODc2Al8yAwkBAiE9AgUHYmFkRGVidAAACQACAQIdSW52YWxpZCByZW1vdmVkIG1hcmdpbiBhbW91bnQEC21hcmdpblJhdGlvCQEPY2FsY01hcmdpblJhdGlvAwUMcmVtYWluTWFyZ2luBQdiYWREZWJ0BRdvbGRQb3NpdGlvbk9wZW5Ob3Rpb25hbAMJAQEhAQkBFnJlcXVpcmVNb3JlTWFyZ2luUmF0aW8DBQttYXJnaW5SYXRpbwkBD2luaXRNYXJnaW5SYXRpbwAGCQACAQkArAICCQCsAgIJAKwCAgIZVG9vIG11Y2ggbWFyZ2luIHJlbW92ZWQ6IAkApAMBBQttYXJnaW5SYXRpbwIDIDwgCQCkAwEJAQ9pbml0TWFyZ2luUmF0aW8ABA1xdW90ZUFzc2V0U3RyCQDYBAEJAQpxdW90ZUFzc2V0AAQNJHQwNTUzMjA1NTM3NAkBE2dldEJvcnJvd2VkQnlUcmFkZXIBBQdfdHJhZGVyBAhib3Jyb3dlZAgFDSR0MDU1MzIwNTUzNzQCXzEEB2Fzc2V0SWQIBQ0kdDA1NTMyMDU1Mzc0Al8yBAd0b1JlcGF5AwkAZgIFB19hbW91bnQFCGJvcnJvd2VkBQhib3Jyb3dlZAUHX2Ftb3VudAQKdG9XaXRoZHJhdwMJAGYCBQhib3Jyb3dlZAUHX2Ftb3VudAAACQBlAgUHX2Ftb3VudAUIYm9ycm93ZWQEC2ZpbmFsQm9ycm93CQBlAgUIYm9ycm93ZWQFB3RvUmVwYXkEFXN3aXRjaFBvc2l0aW9uVG9RdW90ZQMJAGYCBQtmaW5hbEJvcnJvdwAABQNuaWwJARN1cGRhdGVQb3NpdGlvbkFzc2V0AgUHX3RyYWRlcgUNcXVvdGVBc3NldFN0cgQNZG9TYW5pdHlDaGVjawMJAQIhPQIJAGQCBQd0b1JlcGF5BQp0b1dpdGhkcmF3BQdfYW1vdW50CQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgICCHRvUmVwYXk9CQCkAwEFB3RvUmVwYXkCDiArIHRvV2l0aGRyYXc9CQCkAwEFCnRvV2l0aGRyYXcCBCAhPSAJAKQDAQUHX2Ftb3VudAUDbmlsAwkAAAIFDWRvU2FuaXR5Q2hlY2sFDWRvU2FuaXR5Q2hlY2sECWRvVW5zdGFrZQkA/AcECQEObWFuYWdlckFkZHJlc3MAAgh3aXRoZHJhdwkAzAgCBQ1xdW90ZUFzc2V0U3RyCQDMCAIFB19hbW91bnQFA25pbAUDbmlsAwkAAAIFCWRvVW5zdGFrZQUJZG9VbnN0YWtlBBZyZXR1cm5Db2xsYXRlcmFsQWN0aW9uAwkAZgIFB3RvUmVwYXkAAAQHZG9SZXBheQkA/AcECQERY29sbGF0ZXJhbEFkZHJlc3MAAgVyZXBheQkAzAgCBQdfdHJhZGVyCQDMCAIFB2Fzc2V0SWQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQd0b1JlcGF5BQNuaWwDCQAAAgUHZG9SZXBheQUHZG9SZXBheQkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDCAUBaQZjYWxsZXIFB3RvUmVwYXkJANkEAQUHYXNzZXRJZAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUWcmV0dXJuQ29sbGF0ZXJhbEFjdGlvbgUWcmV0dXJuQ29sbGF0ZXJhbEFjdGlvbgkAzggCCQEOdXBkYXRlUG9zaXRpb24FBQdfdHJhZGVyBQ9vbGRQb3NpdGlvblNpemUFDHJlbWFpbk1hcmdpbgUXb2xkUG9zaXRpb25PcGVuTm90aW9uYWwJAR9sYXRlc3RDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAQUPb2xkUG9zaXRpb25TaXplAwkAZgIFCnRvV2l0aGRyYXcAAAkBCHdpdGhkcmF3AggFAWkGY2FsbGVyBQp0b1dpdGhkcmF3CQDOCAIJAM4IAgkAzggCBQNuaWwJAQ11cGRhdGVCYWxhbmNlAQkAZQIJAQhjYmFsYW5jZQAFB19hbW91bnQFFXN3aXRjaFBvc2l0aW9uVG9RdW90ZQUWcmV0dXJuQ29sbGF0ZXJhbEFjdGlvbgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQ1jbG9zZVBvc2l0aW9uAAQHX3RyYWRlcgkBD2dldEFjdHVhbENhbGxlcgEFAWkEDl90cmFkZXJBZGRyZXNzCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQUHX3RyYWRlcgIOSW52YWxpZCBjYWxsZXIDAwMJAQEhAQkBE3JlcXVpcmVPcGVuUG9zaXRpb24BBQdfdHJhZGVyBgkBASEBCQELaW5pdGlhbGl6ZWQABgkBBnBhdXNlZAAJAAIBAiBJbnZhbGlkIGNsb3NlUG9zaXRpb24gcGFyYW1ldGVycwQNJHQwNTc1NjY1Nzk0NAkBFWludGVybmFsQ2xvc2VQb3NpdGlvbgIFB190cmFkZXIGBAJ4MQgFDSR0MDU3NTY2NTc5NDQCXzEED3Bvc2l0aW9uQmFkRGVidAgFDSR0MDU3NTY2NTc5NDQCXzIEC3JlYWxpemVkUG5sCAUNJHQwNTc1NjY1Nzk0NAJfMwQNbWFyZ2luVG9WYXVsdAgFDSR0MDU3NTY2NTc5NDQCXzQEFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA1NzU2NjU3OTQ0Al81BBViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIIBQ0kdDA1NzU2NjU3OTQ0Al82BBZ0b3RhbFBvc2l0aW9uU2l6ZUFmdGVyCAUNJHQwNTc1NjY1Nzk0NAJfNwQXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIIBQ0kdDA1NzU2NjU3OTQ0Al84BBlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCAUNJHQwNTc1NjY1Nzk0NAJfOQQCeDIIBQ0kdDA1NzU2NjU3OTQ0A18xMAQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA1NzU2NjU3OTQ0A18xMQQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNTc1NjY1Nzk0NANfMTIDCQBmAgUPcG9zaXRpb25CYWREZWJ0AAAJAAIBAiZVbmFibGUgdG8gY2xvc2UgcG9zaXRpb24gd2l0aCBiYWQgZGVidAQOd2l0aGRyYXdBbW91bnQJAQNhYnMBBQ1tYXJnaW5Ub1ZhdWx0BAphbW1CYWxhbmNlCQBlAgkBCGNiYWxhbmNlAAUOd2l0aGRyYXdBbW91bnQEDSR0MDU4MTUzNTgyOTUDCQBmAgAABQphbW1CYWxhbmNlCQCUCgIAAAkBA2FicwEFCmFtbUJhbGFuY2UJAJQKAgUKYW1tQmFsYW5jZQAABA1hbW1OZXdCYWxhbmNlCAUNJHQwNTgxNTM1ODI5NQJfMQQQZ2V0RnJvbUluc3VyYW5jZQgFDSR0MDU4MTUzNTgyOTUCXzIEAXgDCQBmAgUQZ2V0RnJvbUluc3VyYW5jZQAABBF3aXRoZHJhd0luc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACCHdpdGhkcmF3CQDMCAIFEGdldEZyb21JbnN1cmFuY2UFA25pbAUDbmlsAwkAAAIFEXdpdGhkcmF3SW5zdXJhbmNlBRF3aXRoZHJhd0luc3VyYW5jZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUBeAUBeAQHdW5zdGFrZQkA/AcECQEObWFuYWdlckFkZHJlc3MAAgh3aXRoZHJhdwkAzAgCCQDYBAEJAQpxdW90ZUFzc2V0AAkAzAgCCQBlAgUOd2l0aGRyYXdBbW91bnQFEGdldEZyb21JbnN1cmFuY2UFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UEDSR0MDU4Nzk3NTg4NTEJARNnZXRCb3Jyb3dlZEJ5VHJhZGVyAQUHX3RyYWRlcgQIYm9ycm93ZWQIBQ0kdDA1ODc5NzU4ODUxAl8xBAdhc3NldElkCAUNJHQwNTg3OTc1ODg1MQJfMgQNJHQwNTg4NjY1OTc4MwMJAGYCBQhib3Jyb3dlZAAAAwkAZwIFDndpdGhkcmF3QW1vdW50BQhib3Jyb3dlZAQHZG9SZXBheQkA/AcECQERY29sbGF0ZXJhbEFkZHJlc3MAAgVyZXBheQkAzAgCBQdfdHJhZGVyCQDMCAIFB2Fzc2V0SWQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQhib3Jyb3dlZAUDbmlsAwkAAAIFB2RvUmVwYXkFB2RvUmVwYXkJAJQKAgkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQ5fdHJhZGVyQWRkcmVzcwUIYm9ycm93ZWQJANkEAQUHYXNzZXRJZAUDbmlsCQBlAgUOd2l0aGRyYXdBbW91bnQFCGJvcnJvd2VkCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBA9yZWFsaXplQW5kQ2xvc2UJAPwHBAkBEWNvbGxhdGVyYWxBZGRyZXNzAAIYcmVhbGl6ZVBhcnRpYWxseUFuZENsb3NlCQDMCAIFB190cmFkZXIJAMwIAgUHYXNzZXRJZAUDbmlsCQDMCAIJAQ9BdHRhY2hlZFBheW1lbnQCCQEKcXVvdGVBc3NldAAFDndpdGhkcmF3QW1vdW50BQNuaWwDCQAAAgUPcmVhbGl6ZUFuZENsb3NlBQ9yZWFsaXplQW5kQ2xvc2UJAJQKAgkAzAgCCQEOU2NyaXB0VHJhbnNmZXIDBQ5fdHJhZGVyQWRkcmVzcwUOd2l0aGRyYXdBbW91bnQJANkEAQUHYXNzZXRJZAUDbmlsAAAJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAJQKAgUDbmlsBQ53aXRoZHJhd0Ftb3VudAMJAAACBQ0kdDA1ODg2NjU5NzgzBQ0kdDA1ODg2NjU5NzgzBBNxdW90ZVdpdGhkcmF3QW1vdW50CAUNJHQwNTg4NjY1OTc4MwJfMgQUc2VuZENvbGxhdGVyYWxBY3Rpb24IBQ0kdDA1ODg2NjU5NzgzAl8xBA5ub3RpZnlOb3Rpb25hbAkA/AcECQEMbWluZXJBZGRyZXNzAAIObm90aWZ5Tm90aW9uYWwJAMwIAgUHX3RyYWRlcgkAzAgCAAAFA25pbAUDbmlsAwkAAAIFDm5vdGlmeU5vdGlvbmFsBQ5ub3RpZnlOb3Rpb25hbAkAzggCCQDOCAIJAQ5kZWxldGVQb3NpdGlvbgEFB190cmFkZXIJAQl1cGRhdGVBbW0HBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBRViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIFFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIFF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyBRlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyBQ50b3RhbExvbmdBZnRlcgUPdG90YWxTaG9ydEFmdGVyAwkAZgIFE3F1b3RlV2l0aGRyYXdBbW91bnQAAAkBCHdpdGhkcmF3AgUOX3RyYWRlckFkZHJlc3MFE3F1b3RlV2l0aGRyYXdBbW91bnQJAM4IAgkAzggCBQNuaWwJAQ11cGRhdGVCYWxhbmNlAQUNYW1tTmV3QmFsYW5jZQUUc2VuZENvbGxhdGVyYWxBY3Rpb24JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEJbGlxdWlkYXRlAQdfdHJhZGVyBA9zcG90TWFyZ2luUmF0aW8JARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAgUHX3RyYWRlcgUPUE5MX09QVElPTl9TUE9UBAttYXJnaW5SYXRpbwMJARZpc092ZXJGbHVjdHVhdGlvbkxpbWl0AAQRb3JhY2xlTWFyZ2luUmF0aW8JARZnZXRNYXJnaW5SYXRpb0J5T3B0aW9uAgUHX3RyYWRlcgURUE5MX09QVElPTl9PUkFDTEUJAQR2bWF4AgUPc3BvdE1hcmdpblJhdGlvBRFvcmFjbGVNYXJnaW5SYXRpbwUPc3BvdE1hcmdpblJhdGlvAwMDAwkBASEBCQEWcmVxdWlyZU1vcmVNYXJnaW5SYXRpbwMFC21hcmdpblJhdGlvCQEWbWFpbnRlbmFuY2VNYXJnaW5SYXRpbwAHBgkBASEBCQETcmVxdWlyZU9wZW5Qb3NpdGlvbgEFB190cmFkZXIGCQEBIQEJAQtpbml0aWFsaXplZAAGCQEGcGF1c2VkAAkAAgECE1VuYWJsZSB0byBsaXF1aWRhdGUDAwMJAGYCBQ9zcG90TWFyZ2luUmF0aW8JARNsaXF1aWRhdGlvbkZlZVJhdGlvAAkAZgIJARdwYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwAAAAcJAGYCBQxERUNJTUFMX1VOSVQJARdwYXJ0aWFsTGlxdWlkYXRpb25SYXRpbwAHBA0kdDA2MTQ4MzYxNjMzCQELZ2V0UG9zaXRpb24BBQdfdHJhZGVyBA9vbGRQb3NpdGlvblNpemUIBQ0kdDA2MTQ4MzYxNjMzAl8xBBFvbGRQb3NpdGlvbk1hcmdpbggFDSR0MDYxNDgzNjE2MzMCXzIEF29sZFBvc2l0aW9uT3Blbk5vdGlvbmFsCAUNJHQwNjE0ODM2MTYzMwJfMwQUb2xkUG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDA2MTQ4MzYxNjMzAl80BApfZGlyZWN0aW9uAwkAZgIFD29sZFBvc2l0aW9uU2l6ZQAABQlESVJfU0hPUlQFCERJUl9MT05HBAVpc0FkZAkAAAIFCl9kaXJlY3Rpb24FCERJUl9MT05HBBlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50CQEbZ2V0UGFydGlhbExpcXVpZGF0aW9uQW1vdW50AgUHX3RyYWRlcgUPb2xkUG9zaXRpb25TaXplBA0kdDA2MTg1ODYxOTYyCQEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwCBQdfdHJhZGVyBQ9QTkxfT1BUSU9OX1NQT1QEE29sZFBvc2l0aW9uTm90aW9uYWwIBQ0kdDA2MTg1ODYxOTYyAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwNjE4NTg2MTk2MgJfMgQNJHQwNjE5NzA2MjIwMgkBCXN3YXBJbnB1dAIFBWlzQWRkBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BBVleGNoYW5nZWRQb3NpdGlvblNpemUIBQ0kdDA2MTk3MDYyMjAyAl8xBBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNjE5NzA2MjIwMgJfMgQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNjE5NzA2MjIwMgJfMwQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDYxOTcwNjIyMDICXzQEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwNjE5NzA2MjIwMgJfNQQQbGlxdWlkYXRpb25SYXRpbwkBBGRpdmQCCQEDYWJzAQUVZXhjaGFuZ2VkUG9zaXRpb25TaXplCQEDYWJzAQUPb2xkUG9zaXRpb25TaXplBAtyZWFsaXplZFBubAkBBG11bGQCBQ11bnJlYWxpemVkUG5sBRBsaXF1aWRhdGlvblJhdGlvBA0kdDA2MjQxMzYyNjQ2CQEiY2FsY1JlbWFpbk1hcmdpbldpdGhGdW5kaW5nUGF5bWVudAQFD29sZFBvc2l0aW9uU2l6ZQURb2xkUG9zaXRpb25NYXJnaW4FFG9sZFBvc2l0aW9uTHN0VXBkQ1BGBQtyZWFsaXplZFBubAQMcmVtYWluTWFyZ2luCAUNJHQwNjI0MTM2MjY0NgJfMQQHYmFkRGVidAgFDSR0MDYyNDEzNjI2NDYCXzIEDmZ1bmRpbmdQYXltZW50CAUNJHQwNjI0MTM2MjY0NgJfMwQSdW5yZWFsaXplZFBubEFmdGVyCQBlAgUNdW5yZWFsaXplZFBubAULcmVhbGl6ZWRQbmwEEnJlbWFpbk9wZW5Ob3Rpb25hbAMJAGYCBQ9vbGRQb3NpdGlvblNpemUAAAkAZQIJAGUCBRNvbGRQb3NpdGlvbk5vdGlvbmFsBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BRJ1bnJlYWxpemVkUG5sQWZ0ZXIJAGUCCQBkAgUSdW5yZWFsaXplZFBubEFmdGVyBRNvbGRQb3NpdGlvbk5vdGlvbmFsBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BBJsaXF1aWRhdGlvblBlbmFsdHkJAQRtdWxkAgUZZXhjaGFuZ2VkUXVvdGVBc3NldEFtb3VudAkBE2xpcXVpZGF0aW9uRmVlUmF0aW8ABA9mZWVUb0xpcXVpZGF0b3IJAGkCBRJsaXF1aWRhdGlvblBlbmFsdHkAAgQOZmVlVG9JbnN1cmFuY2UJAGUCBRJsaXF1aWRhdGlvblBlbmFsdHkFD2ZlZVRvTGlxdWlkYXRvcgQRbmV3UG9zaXRpb25NYXJnaW4JAGUCBQxyZW1haW5NYXJnaW4FEmxpcXVpZGF0aW9uUGVuYWx0eQQPbmV3UG9zaXRpb25TaXplCQBkAgUPb2xkUG9zaXRpb25TaXplBRVleGNoYW5nZWRQb3NpdGlvblNpemUEF25ld1Bvc2l0aW9uT3Blbk5vdGlvbmFsCQEDYWJzAQUScmVtYWluT3Blbk5vdGlvbmFsBBRuZXdQb3NpdGlvbkxzdFVwZENQRgkBH2xhdGVzdEN1bXVsYXRpdmVQcmVtaXVtRnJhY3Rpb24BBQ9uZXdQb3NpdGlvblNpemUEGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIJAGUCCQEUb3BlbkludGVyZXN0Tm90aW9uYWwABRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50BAphbW1CYWxhbmNlCQBlAgkBCGNiYWxhbmNlAAUSbGlxdWlkYXRpb25QZW5hbHR5BA0kdDA2MzgxOTYzOTYyAwkAZgIAAAUKYW1tQmFsYW5jZQkAlAoCAAAJAQNhYnMBBQphbW1CYWxhbmNlCQCUCgIFCmFtbUJhbGFuY2UAAAQNbmV3QW1tQmFsYW5jZQgFDSR0MDYzODE5NjM5NjICXzEEEXRha2VGcm9tSW5zdXJhbmNlCAUNJHQwNjM4MTk2Mzk2MgJfMgQNJHQwNjM5NzA2NDAyNAkBE2dldEJvcnJvd2VkQnlUcmFkZXIBBQdfdHJhZGVyBAhib3Jyb3dlZAgFDSR0MDYzOTcwNjQwMjQCXzEEB2Fzc2V0SWQIBQ0kdDA2Mzk3MDY0MDI0Al8yBBVkb0xpcXVpZGF0ZUNvbGxhdGVyYWwDCQBmAgUIYm9ycm93ZWQAAAQQY29sbGF0ZXJhbFRvU2VsbAkBBG11bGQCBQhib3Jyb3dlZAUQbGlxdWlkYXRpb25SYXRpbwQPcmVhbGl6ZUFuZENsb3NlCQD8BwQJARFjb2xsYXRlcmFsQWRkcmVzcwACEHJlYWxpemVQYXJ0aWFsbHkJAMwIAgUHX3RyYWRlcgkAzAgCBQdhc3NldElkCQDMCAIFEGNvbGxhdGVyYWxUb1NlbGwFA25pbAUDbmlsAwkAAAIFD3JlYWxpemVBbmRDbG9zZQUPcmVhbGl6ZUFuZENsb3NlBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBRVkb0xpcXVpZGF0ZUNvbGxhdGVyYWwFFWRvTGlxdWlkYXRlQ29sbGF0ZXJhbAQBeAMJAGYCBRF0YWtlRnJvbUluc3VyYW5jZQAABBF3aXRoZHJhd0luc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACCHdpdGhkcmF3CQDMCAIFEXRha2VGcm9tSW5zdXJhbmNlBQNuaWwFA25pbAMJAAACBRF3aXRoZHJhd0luc3VyYW5jZQURd2l0aGRyYXdJbnN1cmFuY2UFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgUDbmlsAwkAAAIFAXgFAXgEB3Vuc3Rha2UJAPwHBAkBDm1hbmFnZXJBZGRyZXNzAAIId2l0aGRyYXcJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAJAMwIAgkAZQIFEmxpcXVpZGF0aW9uUGVuYWx0eQURdGFrZUZyb21JbnN1cmFuY2UFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UEEGRlcG9zaXRJbnN1cmFuY2UJAPwHBAkBEGluc3VyYW5jZUFkZHJlc3MAAgdkZXBvc2l0BQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUOZmVlVG9JbnN1cmFuY2UFA25pbAMJAAACBRBkZXBvc2l0SW5zdXJhbmNlBRBkZXBvc2l0SW5zdXJhbmNlBA5ub3RpZnlOb3Rpb25hbAkA/AcECQEMbWluZXJBZGRyZXNzAAIObm90aWZ5Tm90aW9uYWwJAMwIAgUHX3RyYWRlcgkAzAgCBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUDbmlsBQNuaWwDCQAAAgUObm90aWZ5Tm90aW9uYWwFDm5vdGlmeU5vdGlvbmFsCQDOCAIJAM4IAgkAzggCCQEOdXBkYXRlUG9zaXRpb24FBQdfdHJhZGVyBQ9uZXdQb3NpdGlvblNpemUFEW5ld1Bvc2l0aW9uTWFyZ2luBRduZXdQb3NpdGlvbk9wZW5Ob3Rpb25hbAUUbmV3UG9zaXRpb25Mc3RVcGRDUEYJAQl1cGRhdGVBbW0HBRZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyBRViYXNlQXNzZXRSZXNlcnZlQWZ0ZXIFFnRvdGFsUG9zaXRpb25TaXplQWZ0ZXIFF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyBRlvcGVuSW50ZXJlc3ROb3Rpb25hbEFmdGVyCQBlAgkBFXRvdGFsTG9uZ1Bvc2l0aW9uU2l6ZQADCQBmAgUPbmV3UG9zaXRpb25TaXplAAAJAQNhYnMBBRVleGNoYW5nZWRQb3NpdGlvblNpemUAAAkAZQIJARZ0b3RhbFNob3J0UG9zaXRpb25TaXplAAMJAGYCAAAFD25ld1Bvc2l0aW9uU2l6ZQkBA2FicwEFFWV4Y2hhbmdlZFBvc2l0aW9uU2l6ZQAACQEId2l0aGRyYXcCCAUBaQZjYWxsZXIFD2ZlZVRvTGlxdWlkYXRvcgkBDXVwZGF0ZUJhbGFuY2UBBQ1uZXdBbW1CYWxhbmNlCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBA0kdDA2NTg2NjY2MzIxCQEVaW50ZXJuYWxDbG9zZVBvc2l0aW9uAgUHX3RyYWRlcgcEAngxCAUNJHQwNjU4NjY2NjMyMQJfMQQHYmFkRGVidAgFDSR0MDY1ODY2NjYzMjECXzIEAngyCAUNJHQwNjU4NjY2NjMyMQJfMwQCeDMIBQ0kdDA2NTg2NjY2MzIxAl80BBZxdW90ZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNjU4NjY2NjMyMQJfNQQVYmFzZUFzc2V0UmVzZXJ2ZUFmdGVyCAUNJHQwNjU4NjY2NjMyMQJfNgQWdG90YWxQb3NpdGlvblNpemVBZnRlcggFDSR0MDY1ODY2NjYzMjECXzcEF2N1bXVsYXRpdmVOb3Rpb25hbEFmdGVyCAUNJHQwNjU4NjY2NjMyMQJfOAQZb3BlbkludGVyZXN0Tm90aW9uYWxBZnRlcggFDSR0MDY1ODY2NjYzMjECXzkEGWV4Y2hhbmdlZFF1b3RlQXNzZXRBbW91bnQIBQ0kdDA2NTg2NjY2MzIxA18xMAQOdG90YWxMb25nQWZ0ZXIIBQ0kdDA2NTg2NjY2MzIxA18xMQQPdG90YWxTaG9ydEFmdGVyCAUNJHQwNjU4NjY2NjMyMQNfMTIEEmxpcXVpZGF0aW9uUGVuYWx0eQkBBG11bGQCBRlleGNoYW5nZWRRdW90ZUFzc2V0QW1vdW50CQETbGlxdWlkYXRpb25GZWVSYXRpbwAED2ZlZVRvTGlxdWlkYXRvcgkAaQIFEmxpcXVpZGF0aW9uUGVuYWx0eQACBA5mZWVUb0luc3VyYW5jZQkAZQIFEmxpcXVpZGF0aW9uUGVuYWx0eQUPZmVlVG9MaXF1aWRhdG9yBAphbW1CYWxhbmNlCQBlAgkBCGNiYWxhbmNlAAUSbGlxdWlkYXRpb25QZW5hbHR5BA0kdDA2NjczMzY2ODc2AwkAZgIAAAUKYW1tQmFsYW5jZQkAlAoCAAAJAQNhYnMBBQphbW1CYWxhbmNlCQCUCgIFCmFtbUJhbGFuY2UAAAQNbmV3QW1tQmFsYW5jZQgFDSR0MDY2NzMzNjY4NzYCXzEEEXRha2VGcm9tSW5zdXJhbmNlCAUNJHQwNjY3MzM2Njg3NgJfMgQNJHQwNjY4ODQ2NjkzOAkBE2dldEJvcnJvd2VkQnlUcmFkZXIBBQdfdHJhZGVyBAhib3Jyb3dlZAgFDSR0MDY2ODg0NjY5MzgCXzEEB2Fzc2V0SWQIBQ0kdDA2Njg4NDY2OTM4Al8yBBVkb0xpcXVpZGF0ZUNvbGxhdGVyYWwDCQBmAgUIYm9ycm93ZWQAAAQPcmVhbGl6ZUFuZENsb3NlCQD8BwQJARFjb2xsYXRlcmFsQWRkcmVzcwACGHJlYWxpemVQYXJ0aWFsbHlBbmRDbG9zZQkAzAgCBQdfdHJhZGVyCQDMCAIFB2Fzc2V0SWQFA25pbAUDbmlsAwkAAAIFD3JlYWxpemVBbmRDbG9zZQUPcmVhbGl6ZUFuZENsb3NlBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBRVkb0xpcXVpZGF0ZUNvbGxhdGVyYWwFFWRvTGlxdWlkYXRlQ29sbGF0ZXJhbAQBeAMJAGYCBRF0YWtlRnJvbUluc3VyYW5jZQAABBF3aXRoZHJhd0luc3VyYW5jZQkA/AcECQEQaW5zdXJhbmNlQWRkcmVzcwACCHdpdGhkcmF3CQDMCAIFEXRha2VGcm9tSW5zdXJhbmNlBQNuaWwFA25pbAMJAAACBRF3aXRoZHJhd0luc3VyYW5jZQURd2l0aGRyYXdJbnN1cmFuY2UFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgUDbmlsAwkAAAIFAXgFAXgEB3Vuc3Rha2UJAPwHBAkBDm1hbmFnZXJBZGRyZXNzAAIId2l0aGRyYXcJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAJAMwIAgkAZQIFEmxpcXVpZGF0aW9uUGVuYWx0eQURdGFrZUZyb21JbnN1cmFuY2UFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UEEGRlcG9zaXRJbnN1cmFuY2UJAPwHBAkBEGluc3VyYW5jZUFkZHJlc3MAAgdkZXBvc2l0BQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUOZmVlVG9JbnN1cmFuY2UFA25pbAMJAAACBRBkZXBvc2l0SW5zdXJhbmNlBRBkZXBvc2l0SW5zdXJhbmNlBA5ub3RpZnlOb3Rpb25hbAkA/AcECQEMbWluZXJBZGRyZXNzAAIObm90aWZ5Tm90aW9uYWwJAMwIAgUHX3RyYWRlcgkAzAgCAAAFA25pbAUDbmlsAwkAAAIFDm5vdGlmeU5vdGlvbmFsBQ5ub3RpZnlOb3Rpb25hbAkAzggCCQDOCAIJAM4IAgkBDmRlbGV0ZVBvc2l0aW9uAQUHX3RyYWRlcgkBCXVwZGF0ZUFtbQcFFnF1b3RlQXNzZXRSZXNlcnZlQWZ0ZXIFFWJhc2VBc3NldFJlc2VydmVBZnRlcgUWdG90YWxQb3NpdGlvblNpemVBZnRlcgUXY3VtdWxhdGl2ZU5vdGlvbmFsQWZ0ZXIFGW9wZW5JbnRlcmVzdE5vdGlvbmFsQWZ0ZXIFDnRvdGFsTG9uZ0FmdGVyBQ90b3RhbFNob3J0QWZ0ZXIJAQh3aXRoZHJhdwIIBQFpBmNhbGxlcgUPZmVlVG9MaXF1aWRhdG9yCQENdXBkYXRlQmFsYW5jZQEFDW5ld0FtbUJhbGFuY2UJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEKcGF5RnVuZGluZwAEFWZ1bmRpbmdCbG9ja1RpbWVzdGFtcAkBGW5leHRGdW5kaW5nQmxvY2tUaW1lc3RhbXAAAwMDCQBmAgUVZnVuZGluZ0Jsb2NrVGltZXN0YW1wCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAYJAQEhAQkBC2luaXRpYWxpemVkAAYJAQZwYXVzZWQACQACAQkArAICCQCsAgIJAKwCAgIhSW52YWxpZCBmdW5kaW5nIGJsb2NrIHRpbWVzdGFtcDogCQCkAwEIBQlsYXN0QmxvY2sJdGltZXN0YW1wAgMgPCAJAKQDAQUVZnVuZGluZ0Jsb2NrVGltZXN0YW1wBA91bmRlcmx5aW5nUHJpY2UJARJnZXRPcmFjbGVUd2FwUHJpY2UABA0kdDA2ODczNjY4Nzk4CQEKZ2V0RnVuZGluZwAEFHNob3J0UHJlbWl1bUZyYWN0aW9uCAUNJHQwNjg3MzY2ODc5OAJfMQQTbG9uZ1ByZW1pdW1GcmFjdGlvbggFDSR0MDY4NzM2Njg3OTgCXzIJAQ11cGRhdGVGdW5kaW5nBQkAZAIFFWZ1bmRpbmdCbG9ja1RpbWVzdGFtcAkBFGZ1bmRpbmdQZXJpb2RTZWNvbmRzAAkAZAIJASNsYXRlc3RMb25nQ3VtdWxhdGl2ZVByZW1pdW1GcmFjdGlvbgAFE2xvbmdQcmVtaXVtRnJhY3Rpb24JAGQCCQEkbGF0ZXN0U2hvcnRDdW11bGF0aXZlUHJlbWl1bUZyYWN0aW9uAAUUc2hvcnRQcmVtaXVtRnJhY3Rpb24JAQRkaXZkAgUTbG9uZ1ByZW1pdW1GcmFjdGlvbgUPdW5kZXJseWluZ1ByaWNlCQEEZGl2ZAIFFHNob3J0UHJlbWl1bUZyYWN0aW9uBQ91bmRlcmx5aW5nUHJpY2UBaQEOZm9yY2VNb3ZlQXNzZXQCB190cmFkZXIHX2Ftb3VudAMDCQECIT0CCQCnCAEJAQ5hZG1pblB1YmxpY0tleQAIBQFpBmNhbGxlcgYJAGYCAAAFB19hbW91bnQJAAIBAiFJbnZhbGlkIGZvcmNlTW92ZUFzc2V0IHBhcmFtZXRlcnMEB3Vuc3Rha2UJAPwHBAkBDm1hbmFnZXJBZGRyZXNzAAIId2l0aGRyYXcJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAJAMwIAgUHX2Ftb3VudAUDbmlsBQNuaWwDCQAAAgUHdW5zdGFrZQUHdW5zdGFrZQkAzggCCQEId2l0aGRyYXcCCQERQGV4dHJOYXRpdmUoMTA2MikBBQdfdHJhZGVyBQdfYW1vdW50CQENdXBkYXRlQmFsYW5jZQEJAGUCCQEIY2JhbGFuY2UABQdfYW1vdW50CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBCWFkanVzdFBlZwEGX3ByaWNlAwMJAQIhPQIJAKcIAQkBDmFkbWluUHVibGljS2V5AAgFAWkGY2FsbGVyBgkAZgIAAAUGX3ByaWNlCQACAQIcSW52YWxpZCBhZGp1c3RQZWcgcGFyYW1ldGVycwQNJHQwNjk3ODk2OTg3NQkBEGdldFBlZ0FkanVzdENvc3QBBQZfcHJpY2UEFG5ld1F1b3RlQXNzZXRSZXNlcnZlCAUNJHQwNjk3ODk2OTg3NQJfMQQNcGVnQ2hhbmdlQ29zdAgFDSR0MDY5Nzg5Njk4NzUCXzIDCQAAAgUNcGVnQ2hhbmdlQ29zdAAACQACAQIRTm90aGluZyB0byBhZGp1c3QDCQBmAgUNcGVnQ2hhbmdlQ29zdAAACQERdXBkYXRlQW1tUmVzZXJ2ZXMCBRRuZXdRdW90ZUFzc2V0UmVzZXJ2ZQkBBmJzQXN0UgAEB3Vuc3Rha2UJAPwHBAkBDm1hbmFnZXJBZGRyZXNzAAIId2l0aGRyYXcJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAJAMwIAgkBA2FicwEFDXBlZ0NoYW5nZUNvc3QFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UEEGRlcG9zaXRJbnN1cmFuY2UJAPwHBAkBEGluc3VyYW5jZUFkZHJlc3MAAgdkZXBvc2l0BQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAkBA2FicwEFDXBlZ0NoYW5nZUNvc3QFA25pbAMJAAACBRBkZXBvc2l0SW5zdXJhbmNlBRBkZXBvc2l0SW5zdXJhbmNlCQDOCAIJAQ11cGRhdGVCYWxhbmNlAQkAZQIJAQhjYmFsYW5jZQAJAQNhYnMBBQ1wZWdDaGFuZ2VDb3N0CQERdXBkYXRlQW1tUmVzZXJ2ZXMCBRRuZXdRdW90ZUFzc2V0UmVzZXJ2ZQkBBmJzQXN0UgAJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEQbWlncmF0ZUxpcXVpZGl0eQAEBmFtb3VudAkBCGNiYWxhbmNlAAMJAGYCBQZhbW91bnQAAAQHdW5zdGFrZQkA/AcECQERcXVvdGVBc3NldFN0YWtpbmcAAg51bmxvY2tOZXV0cmlubwkAzAgCBQZhbW91bnQJAMwIAgkA2AQBCQEKcXVvdGVBc3NldAAFA25pbAUDbmlsAwkAAAIFB3Vuc3Rha2UFB3Vuc3Rha2UEBXN0YWtlCQD8BwQJAQ5tYW5hZ2VyQWRkcmVzcwACB2RlcG9zaXQFA25pbAkAzAgCCQEPQXR0YWNoZWRQYXltZW50AgkBCnF1b3RlQXNzZXQABQZhbW91bnQFA25pbAMJAAACBQVzdGFrZQUFc3Rha2UFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgUDbmlsAWkBBXZfZ2V0AQdfdHJhZGVyBA0kdDA3MTUxNjcxNTc2CQEVaW50ZXJuYWxDbG9zZVBvc2l0aW9uAgUHX3RyYWRlcgcEAngxCAUNJHQwNzE1MTY3MTU3NgJfMQQCeDIIBQ0kdDA3MTUxNjcxNTc2Al8yBAJ4MwgFDSR0MDcxNTE2NzE1NzYCXzMEAng0CAUNJHQwNzE1MTY3MTU3NgJfNAkAAgEJAKwCAgkArAICCQCsAgIJAQFzAQUCeDIJAQFzAQUCeDMJAQFzAQUCeDQJAQFzAQkBDmdldE1hcmdpblJhdGlvAQUHX3RyYWRlcgFpASd2aWV3X2NhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQBB190cmFkZXIEDSR0MDcxNzIzNzE4MzQJAQtnZXRQb3NpdGlvbgEFB190cmFkZXIEDHBvc2l0aW9uU2l6ZQgFDSR0MDcxNzIzNzE4MzQCXzEEDnBvc2l0aW9uTWFyZ2luCAUNJHQwNzE3MjM3MTgzNAJfMgQDcG9uCAUNJHQwNzE3MjM3MTgzNAJfMwQRcG9zaXRpb25Mc3RVcGRDUEYIBQ0kdDA3MTcyMzcxODM0Al80BA0kdDA3MTgzOTcxOTQwCQEjZ2V0UG9zaXRpb25Ob3Rpb25hbEFuZFVucmVhbGl6ZWRQbmwCBQdfdHJhZGVyBQ9QTkxfT1BUSU9OX1NQT1QEEHBvc2l0aW9uTm90aW9uYWwIBQ0kdDA3MTgzOTcxOTQwAl8xBA11bnJlYWxpemVkUG5sCAUNJHQwNzE4Mzk3MTk0MAJfMgQNJHQwNzE5NDU3MjEyNwkBImNhbGNSZW1haW5NYXJnaW5XaXRoRnVuZGluZ1BheW1lbnQEBQxwb3NpdGlvblNpemUFDnBvc2l0aW9uTWFyZ2luBRFwb3NpdGlvbkxzdFVwZENQRgUNdW5yZWFsaXplZFBubAQMcmVtYWluTWFyZ2luCAUNJHQwNzE5NDU3MjEyNwJfMQQHYmFkRGVidAgFDSR0MDcxOTQ1NzIxMjcCXzIEDmZ1bmRpbmdQYXltZW50CAUNJHQwNzE5NDU3MjEyNwJfMwkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQEBcwEFDHJlbWFpbk1hcmdpbgkBAXMBBQ5mdW5kaW5nUGF5bWVudAkBAXMBCQEOZ2V0TWFyZ2luUmF0aW8BBQdfdHJhZGVyCQEBcwEFDXVucmVhbGl6ZWRQbmwJAQFzAQUHYmFkRGVidAkBAXMBBRBwb3NpdGlvbk5vdGlvbmFsAWkBFXZpZXdfZ2V0UGVnQWRqdXN0Q29zdAEGX3ByaWNlBARjb3N0CQEQZ2V0UGVnQWRqdXN0Q29zdAEFBl9wcmljZQkAAgEJAKQDAQgFBGNvc3QCXzIBaQEPdmlld19nZXRGdW5kaW5nAAQPdW5kZXJseWluZ1ByaWNlCQESZ2V0T3JhY2xlVHdhcFByaWNlAAQNJHQwNzI1MTg3MjU4MAkBCmdldEZ1bmRpbmcABBRzaG9ydFByZW1pdW1GcmFjdGlvbggFDSR0MDcyNTE4NzI1ODACXzEEE2xvbmdQcmVtaXVtRnJhY3Rpb24IBQ0kdDA3MjUxODcyNTgwAl8yBAtsb25nRnVuZGluZwkBBGRpdmQCBRNsb25nUHJlbWl1bUZyYWN0aW9uBQ91bmRlcmx5aW5nUHJpY2UEDHNob3J0RnVuZGluZwkBBGRpdmQCBRRzaG9ydFByZW1pdW1GcmFjdGlvbgUPdW5kZXJseWluZ1ByaWNlCQACAQkArAICCQCsAgIJAKwCAgkBAXMBBQtsb25nRnVuZGluZwkBAXMBBQxzaG9ydEZ1bmRpbmcJAQFzAQkBEGdldFR3YXBTcG90UHJpY2UACQEBcwEJARJnZXRPcmFjbGVUd2FwUHJpY2UAAWkBGHZpZXdfZ2V0Qm9ycm93ZWRCeVRyYWRlcgEHX3RyYWRlcgQNJHQwNzI4NzA3MjkyNAkBE2dldEJvcnJvd2VkQnlUcmFkZXIBBQdfdHJhZGVyBAhib3Jyb3dlZAgFDSR0MDcyODcwNzI5MjQCXzEEB2Fzc2V0SWQIBQ0kdDA3Mjg3MDcyOTI0Al8yCQACAQkArAICCQEBcwEFCGJvcnJvd2VkBQdhc3NldElkAQJ0eAEGdmVyaWZ5AAkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAkBDmFkbWluUHVibGljS2V5AMmgpy8=", "height": 2310890, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: BSwAua8E9TJprwEtm7rx2UdLFgcSskLpGXvFA6XwnJHL Next: none Diff:
OldNewDifferences
1818 let k_positionLastUpdatedCumulativePremiumFraction = "k_positionFraction"
1919
2020 let k_positionClosedDate = "k_positionClosedDate"
21+
22+let k_positionAsset = "k_positionAsset"
2123
2224 let k_initialized = "k_initialized"
2325
9799
98100 let k_manager_address = "k_manager_address"
99101
102+let k_collateral_address = "k_collateral_address"
103+
104+let k_exchange_address = "k_exchange_address"
105+
100106 let k_nft_manager_address = "k_nft_manager_address"
107+
108+let k_trader_market_asset_collateral = "k_trader_market_asset_collateral"
109+
110+func toCompositeKey (_key,_address) = ((_key + "_") + _address)
111+
101112
102113 func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
103114
133144
134145
135146 func nftManagerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_nft_manager_address)), "NFT Manager not set")
147+
148+
149+func collateralAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_collateral_address)), "Collateral Manager not set")
150+
151+
152+func swapAddress () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(coordinator(), k_exchange_address), "No swap address")), "Invalid swap address")
153+
154+
155+let k_whitelist_asset = "k_whitelist_asset"
156+
157+func isWhitelistAsset (_assetId) = valueOrElse(getBoolean(collateralAddress(), toCompositeKey(k_whitelist_asset, _assetId)), false)
136158
137159
138160 let k_token_param = "k_token_param"
178200 func vmax (_x,_y) = if ((_x >= _y))
179201 then _x
180202 else _y
181-
182-
183-func toCompositeKey (_key,_address) = ((_key + "_") + _address)
184203
185204
186205 func listToStr (_list) = {
332351 }
333352
334353
354+func getPositionAsset (_trader) = {
355+ let positionAssetOpt = getString(this, toCompositeKey(k_positionAsset, _trader))
356+ match positionAssetOpt {
357+ case positionAsset: String =>
358+ positionAsset
359+ case _ =>
360+ toBase58String(quoteAsset())
361+ }
362+ }
363+
364+
335365 func requireOpenPosition (_trader) = if ((getPosition(_trader)._1 == 0))
336366 then throw("No open position")
337367 else true
370400 let amountBaseAssetBought = if (_isAdd)
371401 then amountBaseAssetBoughtAbs
372402 else -(amountBaseAssetBoughtAbs)
373- let $t01366013853 = updateReserve(_isAdd, _quoteAssetAmount, amountBaseAssetBoughtAbs)
374- let quoteAssetReserveAfter1 = $t01366013853._1
375- let baseAssetReserveAfter1 = $t01366013853._2
376- let totalPositionSizeAfter1 = $t01366013853._3
377- let cumulativeNotionalAfter1 = $t01366013853._4
403+ let $t01497415167 = updateReserve(_isAdd, _quoteAssetAmount, amountBaseAssetBoughtAbs)
404+ let quoteAssetReserveAfter1 = $t01497415167._1
405+ let baseAssetReserveAfter1 = $t01497415167._2
406+ let totalPositionSizeAfter1 = $t01497415167._3
407+ let cumulativeNotionalAfter1 = $t01497415167._4
378408 let priceBefore = divd(_qtAstR, _bsAstR)
379409 let marketPrice = divd(_quoteAssetAmount, amountBaseAssetBoughtAbs)
380410 let priceDiff = abs((priceBefore - marketPrice))
394424 }
395425 else 0
396426 let signedMargin = ((_marginDelta - fundingPayment) + _oldPositionMargin)
397- let $t01534415471 = if ((0 > signedMargin))
427+ let $t01665816785 = if ((0 > signedMargin))
398428 then $Tuple2(0, abs(signedMargin))
399429 else $Tuple2(abs(signedMargin), 0)
400- let remainMargin = $t01534415471._1
401- let badDebt = $t01534415471._2
430+ let remainMargin = $t01665816785._1
431+ let badDebt = $t01665816785._2
402432 $Tuple3(remainMargin, badDebt, fundingPayment)
403433 }
404434
415445 let quoteAssetAfter = divd(k, baseAssetPoolAmountAfter)
416446 let quoteAssetSold = abs((quoteAssetAfter - _quoteAssetReserve))
417447 let maxPriceImpactValue = maxPriceImpact()
418- let $t01644816641 = updateReserve(!(_isAdd), quoteAssetSold, _baseAssetAmount)
419- let quoteAssetReserveAfter1 = $t01644816641._1
420- let baseAssetReserveAfter1 = $t01644816641._2
421- let totalPositionSizeAfter1 = $t01644816641._3
422- let cumulativeNotionalAfter1 = $t01644816641._4
448+ let $t01776217955 = updateReserve(!(_isAdd), quoteAssetSold, _baseAssetAmount)
449+ let quoteAssetReserveAfter1 = $t01776217955._1
450+ let baseAssetReserveAfter1 = $t01776217955._2
451+ let totalPositionSizeAfter1 = $t01776217955._3
452+ let cumulativeNotionalAfter1 = $t01776217955._4
423453 let marketPrice = divd(quoteAssetSold, _baseAssetAmount)
424454 let priceDiff = abs((priceBefore - marketPrice))
425455 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
478508 let isShort = (0 > _positionSize)
479509 let positionNotional = if ((_option == PNL_OPTION_SPOT))
480510 then {
481- let $t01985920030 = swapOutputWithReserves(!(isShort), positionSizeAbs, false, _quoteAssetReserve, _baseAssetReserve)
482- let outPositionNotional = $t01985920030._1
483- let x1 = $t01985920030._2
484- let x2 = $t01985920030._3
485- let x3 = $t01985920030._4
511+ let $t02117321344 = swapOutputWithReserves(!(isShort), positionSizeAbs, false, _quoteAssetReserve, _baseAssetReserve)
512+ let outPositionNotional = $t02117321344._1
513+ let x1 = $t02117321344._2
514+ let x2 = $t02117321344._3
515+ let x3 = $t02117321344._4
486516 outPositionNotional
487517 }
488518 else muld(positionSizeAbs, getOracleTwapPrice())
503533
504534
505535 func getPositionNotionalAndUnrealizedPnl (_trader,_option) = {
506- let $t02134721475 = getPosition(_trader)
507- let positionSize = $t02134721475._1
508- let positionMargin = $t02134721475._2
509- let positionOpenNotional = $t02134721475._3
510- let positionLstUpdCPF = $t02134721475._4
536+ let $t02266122789 = getPosition(_trader)
537+ let positionSize = $t02266122789._1
538+ let positionMargin = $t02266122789._2
539+ let positionOpenNotional = $t02266122789._3
540+ let positionLstUpdCPF = $t02266122789._4
511541 getPositionNotionalAndUnrealizedPnlByValues(positionSize, positionOpenNotional, qtAstR(), bsAstR(), _option)
512542 }
513543
516546
517547
518548 func getMarginRatioByOption (_trader,_option) = {
519- let $t02195622067 = getPosition(_trader)
520- let positionSize = $t02195622067._1
521- let positionMargin = $t02195622067._2
522- let pon = $t02195622067._3
523- let positionLstUpdCPF = $t02195622067._4
524- let $t02207322166 = getPositionNotionalAndUnrealizedPnl(_trader, _option)
525- let positionNotional = $t02207322166._1
526- let unrealizedPnl = $t02207322166._2
527- let $t02217122337 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
528- let remainMargin = $t02217122337._1
529- let badDebt = $t02217122337._2
549+ let $t02327023381 = getPosition(_trader)
550+ let positionSize = $t02327023381._1
551+ let positionMargin = $t02327023381._2
552+ let pon = $t02327023381._3
553+ let positionLstUpdCPF = $t02327023381._4
554+ let $t02338723480 = getPositionNotionalAndUnrealizedPnl(_trader, _option)
555+ let positionNotional = $t02338723480._1
556+ let unrealizedPnl = $t02338723480._2
557+ let $t02348523651 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
558+ let remainMargin = $t02348523651._1
559+ let badDebt = $t02348523651._2
530560 calcMarginRatio(remainMargin, badDebt, positionNotional)
531561 }
532562
551581
552582
553583 func internalClosePosition (_trader,_checkMaxPriceImpact) = {
554- let $t02357023698 = getPosition(_trader)
555- let positionSize = $t02357023698._1
556- let positionMargin = $t02357023698._2
557- let positionOpenNotional = $t02357023698._3
558- let positionLstUpdCPF = $t02357023698._4
584+ let $t02488425012 = getPosition(_trader)
585+ let positionSize = $t02488425012._1
586+ let positionMargin = $t02488425012._2
587+ let positionOpenNotional = $t02488425012._3
588+ let positionLstUpdCPF = $t02488425012._4
559589 let unrealizedPnl = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)._2
560- let $t02379323961 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
561- let remainMargin = $t02379323961._1
562- let badDebt = $t02379323961._2
590+ let $t02510725275 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
591+ let remainMargin = $t02510725275._1
592+ let badDebt = $t02510725275._2
563593 let exchangedPositionSize = -(positionSize)
564594 let realizedPnl = unrealizedPnl
565595 let marginToVault = -(remainMargin)
566- let $t02408824399 = swapOutput((positionSize > 0), abs(positionSize), _checkMaxPriceImpact)
567- let exchangedQuoteAssetAmount = $t02408824399._1
568- let quoteAssetReserveAfter = $t02408824399._2
569- let baseAssetReserveAfter = $t02408824399._3
570- let totalPositionSizeAfter = $t02408824399._4
571- let cumulativeNotionalAfter = $t02408824399._5
572- let totalLongAfter = $t02408824399._6
573- let totalShortAfter = $t02408824399._7
596+ let $t02540225713 = swapOutput((positionSize > 0), abs(positionSize), _checkMaxPriceImpact)
597+ let exchangedQuoteAssetAmount = $t02540225713._1
598+ let quoteAssetReserveAfter = $t02540225713._2
599+ let baseAssetReserveAfter = $t02540225713._3
600+ let totalPositionSizeAfter = $t02540225713._4
601+ let cumulativeNotionalAfter = $t02540225713._5
602+ let totalLongAfter = $t02540225713._6
603+ let totalShortAfter = $t02540225713._7
574604 let openInterestNotionalAfter = (openInterestNotional() - positionOpenNotional)
575605 $Tuple12(exchangedPositionSize, badDebt, realizedPnl, marginToVault, quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, exchangedQuoteAssetAmount, totalLongAfter, totalShortAfter)
576606 }
648678
649679 func getAdjustedFee (i) = {
650680 let baseFee = fee()
651- let $t02806228670 = if ((size(i.payments) > 1))
681+ let $t02937629984 = if ((size(i.payments) > 1))
652682 then {
653683 let artifactId = toBase58String(valueOrErrorMessage(i.payments[1].assetId, "Invalid attached artifact"))
654684 let artifactKind = strA(nftManagerAddress(), toCompositeKey(k_token_type, artifactId))
661691 else throw("Invalid attached artifact")
662692 }
663693 else $Tuple2(baseFee, false)
664- let adjustedFee = $t02806228670._1
665- let burnArtifact = $t02806228670._2
694+ let adjustedFee = $t02937629984._1
695+ let burnArtifact = $t02937629984._2
666696 $Tuple2(adjustedFee, burnArtifact)
667697 }
668698
671701
672702
673703 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)]
704+
705+
706+func updatePositionAsset (_address,_assetId) = [StringEntry(toCompositeKey(k_positionAsset, _address), _assetId)]
674707
675708
676709 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)]
712745 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)))
713746
714747
715-func deletePosition (_address) = [DeleteEntry(toCompositeKey(k_positionSize, _address)), DeleteEntry(toCompositeKey(k_positionMargin, _address)), DeleteEntry(toCompositeKey(k_positionOpenNotional, _address)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _address)), IntegerEntry(toCompositeKey(k_positionClosedDate, _address), lastBlock.timestamp)]
748+func deletePosition (_address) = [DeleteEntry(toCompositeKey(k_positionSize, _address)), DeleteEntry(toCompositeKey(k_positionMargin, _address)), DeleteEntry(toCompositeKey(k_positionOpenNotional, _address)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _address)), DeleteEntry(toCompositeKey(k_positionAsset, _address)), IntegerEntry(toCompositeKey(k_positionClosedDate, _address), lastBlock.timestamp)]
716749
717750
718751 func withdraw (_address,_amount) = {
734767 func doBurnArtifact (_burnArtifact,i) = if (_burnArtifact)
735768 then [Burn(valueOrErrorMessage(i.payments[1].assetId, "Invalid artifact"), 1)]
736769 else nil
770+
771+
772+func isSameAssetOrNoPosition (_trader,_assetId) = {
773+ let oldPositionSize = getPosition(_trader)._1
774+ if ((oldPositionSize == 0))
775+ then true
776+ else (getPositionAsset(_trader) == _assetId)
777+ }
778+
779+
780+func isSameAsset (_trader,_assetId) = (getPositionAsset(_trader) == _assetId)
781+
782+
783+func getBorrowedByTraderInMarketKey (_amm,_assetId,_trader) = ((((((k_trader_market_asset_collateral + "_") + _amm) + "_") + _assetId) + "_") + _trader)
784+
785+
786+func getBorrowedByTrader (_trader) = {
787+ let positionAsset = getPositionAsset(_trader)
788+ if ((positionAsset == toBase58String(quoteAsset())))
789+ then $Tuple2(0, positionAsset)
790+ else {
791+ let key = getBorrowedByTraderInMarketKey(toString(this), positionAsset, _trader)
792+ let borrow = valueOrElse(getInteger(collateralAddress(), key), 0)
793+ $Tuple2(borrow, positionAsset)
794+ }
795+ }
737796
738797
739798 @Callable(i)
841900 else paused())
842901 then throw("Invalid decreasePosition parameters")
843902 else {
844- let $t03908539237 = getPosition(toString(i.caller))
845- let oldPositionSize = $t03908539237._1
846- let oldPositionMargin = $t03908539237._2
847- let oldPositionOpenNotional = $t03908539237._3
848- let oldPositionLstUpdCPF = $t03908539237._4
903+ let $t04149241644 = getPosition(toString(i.caller))
904+ let oldPositionSize = $t04149241644._1
905+ let oldPositionMargin = $t04149241644._2
906+ let oldPositionOpenNotional = $t04149241644._3
907+ let oldPositionLstUpdCPF = $t04149241644._4
849908 let _direction = if ((oldPositionSize > 0))
850909 then DIR_SHORT
851910 else DIR_LONG
852911 let isAdd = (_direction == DIR_LONG)
853912 let openNotional = muld(_amount, _leverage)
854- let $t03941039526 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
855- let oldPositionNotional = $t03941039526._1
856- let unrealizedPnl = $t03941039526._2
857- let $t03953242081 = if ((oldPositionNotional > openNotional))
913+ let $t04181741933 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
914+ let oldPositionNotional = $t04181741933._1
915+ let unrealizedPnl = $t04181741933._2
916+ let $t04193944488 = if ((oldPositionNotional > openNotional))
858917 then {
859- let $t03990940128 = swapInput(isAdd, openNotional)
860- let exchangedPositionSize = $t03990940128._1
861- let quoteAssetReserveAfter = $t03990940128._2
862- let baseAssetReserveAfter = $t03990940128._3
863- let totalPositionSizeAfter = $t03990940128._4
864- let cumulativeNotionalAfter = $t03990940128._5
918+ let $t04231642535 = swapInput(isAdd, openNotional)
919+ let exchangedPositionSize = $t04231642535._1
920+ let quoteAssetReserveAfter = $t04231642535._2
921+ let baseAssetReserveAfter = $t04231642535._3
922+ let totalPositionSizeAfter = $t04231642535._4
923+ let cumulativeNotionalAfter = $t04231642535._5
865924 let exchangedPositionSizeAbs = abs(exchangedPositionSize)
866925 if (if ((_minBaseAssetAmount != 0))
867926 then (_minBaseAssetAmount > exchangedPositionSizeAbs)
869928 then throw(((("Too little base asset exchanged, got " + toString(exchangedPositionSizeAbs)) + " expected ") + toString(_minBaseAssetAmount)))
870929 else {
871930 let realizedPnl = divd(muld(unrealizedPnl, exchangedPositionSizeAbs), abs(oldPositionSize))
872- let $t04056540810 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
873- let remainMargin = $t04056540810._1
874- let badDebt = $t04056540810._2
875- let fundingPayment = $t04056540810._3
931+ let $t04297243217 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
932+ let remainMargin = $t04297243217._1
933+ let badDebt = $t04297243217._2
934+ let fundingPayment = $t04297243217._3
876935 let exchangedQuoteAssetAmount = openNotional
877936 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
878937 let remainOpenNotional = if ((oldPositionSize > 0))
887946 }
888947 }
889948 else throw("Close position first")
890- let newPositionSize = $t03953242081._1
891- let newPositionRemainMargin = $t03953242081._2
892- let newPositionOpenNotional = $t03953242081._3
893- let newPositionLatestCPF = $t03953242081._4
894- let baseAssetReserveAfter = $t03953242081._5
895- let quoteAssetReserveAfter = $t03953242081._6
896- let totalPositionSizeAfter = $t03953242081._7
897- let cumulativeNotionalAfter = $t03953242081._8
898- let openInterestNotionalAfter = $t03953242081._9
899- let totalLongAfter = $t03953242081._10
900- let totalShortAfter = $t03953242081._11
949+ let newPositionSize = $t04193944488._1
950+ let newPositionRemainMargin = $t04193944488._2
951+ let newPositionOpenNotional = $t04193944488._3
952+ let newPositionLatestCPF = $t04193944488._4
953+ let baseAssetReserveAfter = $t04193944488._5
954+ let quoteAssetReserveAfter = $t04193944488._6
955+ let totalPositionSizeAfter = $t04193944488._7
956+ let cumulativeNotionalAfter = $t04193944488._8
957+ let openInterestNotionalAfter = $t04193944488._9
958+ let totalLongAfter = $t04193944488._10
959+ let totalShortAfter = $t04193944488._11
901960 let notifyNotional = invoke(minerAddress(), "notifyNotional", [toString(i.caller), newPositionOpenNotional], nil)
902961 if ((notifyNotional == notifyNotional))
903962 then (updatePosition(toString(i.caller), newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter))
908967
909968 @Callable(i)
910969 func increasePosition (_direction,_leverage,_minBaseAssetAmount,_refLink) = {
970+ let _trader = toString(i.caller)
911971 let _rawAmount = i.payments[0].amount
912- if (if (if (if (if (if (if ((_direction != DIR_LONG))
972+ let _assetId = i.payments[0].assetId
973+ let _assetIdStr = toBase58String(value(_assetId))
974+ let isQuoteAsset = (_assetId == quoteAsset())
975+ let isCollateralAsset = isWhitelistAsset(_assetIdStr)
976+ if (if (if (if (if (if (if (if ((_direction != DIR_LONG))
913977 then (_direction != DIR_SHORT)
914978 else false)
915979 then true
917981 then true
918982 else !(initialized()))
919983 then true
920- else (i.payments[0].assetId != quoteAsset()))
984+ else if (!(isQuoteAsset))
985+ then !(isCollateralAsset)
986+ else false)
987+ then true
988+ else !(isSameAssetOrNoPosition(_trader, _assetIdStr)))
921989 then true
922990 else !(requireMoreMarginRatio(divd(DECIMAL_UNIT, _leverage), initMarginRatio(), true)))
923991 then true
924992 else paused())
925993 then throw("Invalid increasePosition parameters")
926994 else {
927- let _trader = toString(i.caller)
928995 let baseFee = fee()
929- let $t04319643247 = getAdjustedFee(i)
930- let adjustedFee = $t04319643247._1
931- let burnArtifact = $t04319643247._2
996+ let $t04585045901 = getAdjustedFee(i)
997+ let adjustedFee = $t04585045901._1
998+ let burnArtifact = $t04585045901._2
932999 let rawFeeAmount = muld(_rawAmount, adjustedFee)
933- let referrerFeeAny = invoke(referralAddress(), "acceptPaymentWithLink", [_trader, _refLink], [AttachedPayment(quoteAsset(), rawFeeAmount)])
934- if ((referrerFeeAny == referrerFeeAny))
1000+ let _amount = (_rawAmount - rawFeeAmount)
1001+ let distributeFeeAmount = if (isCollateralAsset)
9351002 then {
936- let referrerFee = match referrerFeeAny {
937- case x: Int =>
938- x
939- case _ =>
940- throw("Invalid referrerFee")
1003+ let doBorrow = invoke(collateralAddress(), "borrow", [_trader], [AttachedPayment(_assetId, _amount)])
1004+ if ((doBorrow == doBorrow))
1005+ then {
1006+ let balanceBefore = assetBalance(this, quoteAsset())
1007+ if ((balanceBefore == balanceBefore))
1008+ then {
1009+ let doSwap = invoke(swapAddress(), "swap", [toBase58String(quoteAsset()), 0], [AttachedPayment(_assetId, rawFeeAmount)])
1010+ if ((doSwap == doSwap))
1011+ then {
1012+ let balanceAfter = assetBalance(this, quoteAsset())
1013+ if ((balanceAfter == balanceAfter))
1014+ then {
1015+ let exchangedAmount = (balanceAfter - balanceBefore)
1016+ if ((exchangedAmount == exchangedAmount))
1017+ then exchangedAmount
1018+ else throw("Strict value is not equal to itself.")
1019+ }
1020+ else throw("Strict value is not equal to itself.")
1021+ }
1022+ else throw("Strict value is not equal to itself.")
1023+ }
1024+ else throw("Strict value is not equal to itself.")
1025+ }
1026+ else throw("Strict value is not equal to itself.")
9411027 }
942- let _amount = (_rawAmount - rawFeeAmount)
943- let feeAmount = (rawFeeAmount - referrerFee)
944- let $t04369643836 = getPosition(_trader)
945- let oldPositionSize = $t04369643836._1
946- let oldPositionMargin = $t04369643836._2
947- let oldPositionOpenNotional = $t04369643836._3
948- let oldPositionLstUpdCPF = $t04369643836._4
949- let isNewPosition = (oldPositionSize == 0)
950- let isSameDirection = if ((oldPositionSize > 0))
951- then (_direction == DIR_LONG)
952- else (_direction == DIR_SHORT)
953- let expandExisting = if (!(isNewPosition))
954- then isSameDirection
955- else false
956- let isAdd = (_direction == DIR_LONG)
957- let $t04412546710 = if (if (isNewPosition)
958- then true
959- else expandExisting)
1028+ else rawFeeAmount
1029+ if ((distributeFeeAmount == distributeFeeAmount))
1030+ then {
1031+ let referrerFeeAny = invoke(referralAddress(), "acceptPaymentWithLink", [_trader, _refLink], [AttachedPayment(quoteAsset(), distributeFeeAmount)])
1032+ if ((referrerFeeAny == referrerFeeAny))
9601033 then {
961- let openNotional = muld(_amount, _leverage)
962- let $t04454944755 = swapInput(isAdd, openNotional)
963- let amountBaseAssetBought = $t04454944755._1
964- let quoteAssetReserveAfter = $t04454944755._2
965- let baseAssetReserveAfter = $t04454944755._3
966- let totalPositionSizeAfter = $t04454944755._4
967- let cumulativeNotionalAfter = $t04454944755._5
968- if (if ((_minBaseAssetAmount != 0))
969- then (_minBaseAssetAmount > abs(amountBaseAssetBought))
970- else false)
971- then throw(((("Limit error: " + toString(abs(amountBaseAssetBought))) + " < ") + toString(_minBaseAssetAmount)))
1034+ let referrerFee = match referrerFeeAny {
1035+ case x: Int =>
1036+ x
1037+ case _ =>
1038+ throw("Invalid referrerFee")
1039+ }
1040+ let feeAmount = (distributeFeeAmount - referrerFee)
1041+ let $t04721747357 = getPosition(_trader)
1042+ let oldPositionSize = $t04721747357._1
1043+ let oldPositionMargin = $t04721747357._2
1044+ let oldPositionOpenNotional = $t04721747357._3
1045+ let oldPositionLstUpdCPF = $t04721747357._4
1046+ let isNewPosition = (oldPositionSize == 0)
1047+ let isSameDirection = if ((oldPositionSize > 0))
1048+ then (_direction == DIR_LONG)
1049+ else (_direction == DIR_SHORT)
1050+ let expandExisting = if (!(isNewPosition))
1051+ then isSameDirection
1052+ else false
1053+ let isAdd = (_direction == DIR_LONG)
1054+ let $t04764650231 = if (if (isNewPosition)
1055+ then true
1056+ else expandExisting)
1057+ then {
1058+ let openNotional = muld(_amount, _leverage)
1059+ let $t04807048276 = swapInput(isAdd, openNotional)
1060+ let amountBaseAssetBought = $t04807048276._1
1061+ let quoteAssetReserveAfter = $t04807048276._2
1062+ let baseAssetReserveAfter = $t04807048276._3
1063+ let totalPositionSizeAfter = $t04807048276._4
1064+ let cumulativeNotionalAfter = $t04807048276._5
1065+ if (if ((_minBaseAssetAmount != 0))
1066+ then (_minBaseAssetAmount > abs(amountBaseAssetBought))
1067+ else false)
1068+ then throw(((("Limit error: " + toString(abs(amountBaseAssetBought))) + " < ") + toString(_minBaseAssetAmount)))
1069+ else {
1070+ let newPositionSize = (oldPositionSize + amountBaseAssetBought)
1071+ let increaseMarginRequirement = divd(openNotional, _leverage)
1072+ let $t04865748896 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
1073+ let remainMargin = $t04865748896._1
1074+ let x1 = $t04865748896._2
1075+ let x2 = $t04865748896._3
1076+ if (!(requireNotOverSpreadLimit(quoteAssetReserveAfter, baseAssetReserveAfter)))
1077+ then throw("Over max spread limit")
1078+ else $Tuple11(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
1079+ then abs(amountBaseAssetBought)
1080+ else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
1081+ then abs(amountBaseAssetBought)
1082+ else 0)))
1083+ }
1084+ }
9721085 else {
973- let newPositionSize = (oldPositionSize + amountBaseAssetBought)
974- let increaseMarginRequirement = divd(openNotional, _leverage)
975- let $t04513645375 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
976- let remainMargin = $t04513645375._1
977- let x1 = $t04513645375._2
978- let x2 = $t04513645375._3
979- if (!(requireNotOverSpreadLimit(quoteAssetReserveAfter, baseAssetReserveAfter)))
980- then throw("Over max spread limit")
981- else $Tuple11(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
982- then abs(amountBaseAssetBought)
983- else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
984- then abs(amountBaseAssetBought)
985- else 0)))
1086+ let openNotional = muld(_amount, _leverage)
1087+ let $t04992450040 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
1088+ let oldPositionNotional = $t04992450040._1
1089+ let unrealizedPnl = $t04992450040._2
1090+ if ((oldPositionNotional > openNotional))
1091+ then throw("Use decreasePosition to decrease position size")
1092+ else throw("Close position first")
9861093 }
987- }
988- else {
989- let openNotional = muld(_amount, _leverage)
990- let $t04640346519 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
991- let oldPositionNotional = $t04640346519._1
992- let unrealizedPnl = $t04640346519._2
993- if ((oldPositionNotional > openNotional))
994- then throw("Use decreasePosition to decrease position size")
995- else throw("Close position first")
996- }
997- let newPositionSize = $t04412546710._1
998- let newPositionRemainMargin = $t04412546710._2
999- let newPositionOpenNotional = $t04412546710._3
1000- let newPositionLatestCPF = $t04412546710._4
1001- let baseAssetReserveAfter = $t04412546710._5
1002- let quoteAssetReserveAfter = $t04412546710._6
1003- let totalPositionSizeAfter = $t04412546710._7
1004- let cumulativeNotionalAfter = $t04412546710._8
1005- let openInterestNotionalAfter = $t04412546710._9
1006- let totalLongAfter = $t04412546710._10
1007- let totalShortAfter = $t04412546710._11
1008- let feeToStakers = (feeAmount / 2)
1009- let feeToInsurance = (feeAmount - feeToStakers)
1010- let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1011- if ((stake == stake))
1012- then {
1013- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1014- if ((depositInsurance == depositInsurance))
1094+ let newPositionSize = $t04764650231._1
1095+ let newPositionRemainMargin = $t04764650231._2
1096+ let newPositionOpenNotional = $t04764650231._3
1097+ let newPositionLatestCPF = $t04764650231._4
1098+ let baseAssetReserveAfter = $t04764650231._5
1099+ let quoteAssetReserveAfter = $t04764650231._6
1100+ let totalPositionSizeAfter = $t04764650231._7
1101+ let cumulativeNotionalAfter = $t04764650231._8
1102+ let openInterestNotionalAfter = $t04764650231._9
1103+ let totalLongAfter = $t04764650231._10
1104+ let totalShortAfter = $t04764650231._11
1105+ let feeToStakers = (feeAmount / 2)
1106+ let feeToInsurance = (feeAmount - feeToStakers)
1107+ let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1108+ if ((stake == stake))
10151109 then {
1016- let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1017- if ((notifyFee == notifyFee))
1110+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1111+ if ((depositInsurance == depositInsurance))
10181112 then {
1019- let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1020- if ((notifyNotional == notifyNotional))
1021- then ((((updatePosition(_trader, newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1113+ let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1114+ if ((notifyFee == notifyFee))
1115+ then {
1116+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1117+ if ((notifyNotional == notifyNotional))
1118+ then (((((updatePosition(_trader, newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updatePositionAsset(_trader, _assetIdStr)) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1119+ else throw("Strict value is not equal to itself.")
1120+ }
10221121 else throw("Strict value is not equal to itself.")
10231122 }
10241123 else throw("Strict value is not equal to itself.")
10351134
10361135 @Callable(i)
10371136 func addMargin () = {
1137+ let _trader = toString(i.caller)
10381138 let _rawAmount = i.payments[0].amount
1039- if (if (if (if ((i.payments[0].assetId != quoteAsset()))
1139+ let _assetId = i.payments[0].assetId
1140+ let _assetIdStr = toBase58String(value(_assetId))
1141+ let isQuoteAsset = (_assetId == quoteAsset())
1142+ let isCollateralAsset = isWhitelistAsset(_assetIdStr)
1143+ if (if (if (if (if (if (!(isQuoteAsset))
1144+ then !(isCollateralAsset)
1145+ else false)
10401146 then true
10411147 else !(requireOpenPosition(toString(i.caller))))
1148+ then true
1149+ else !(isSameAsset(_trader, _assetIdStr)))
10421150 then true
10431151 else !(initialized()))
10441152 then true
10451153 else paused())
10461154 then throw("Invalid addMargin parameters")
10471155 else {
1048- let _trader = toString(i.caller)
1049- let $t04817848229 = getAdjustedFee(i)
1050- let adjustedFee = $t04817848229._1
1051- let burnArtifact = $t04817848229._2
1156+ let $t05198752038 = getAdjustedFee(i)
1157+ let adjustedFee = $t05198752038._1
1158+ let burnArtifact = $t05198752038._2
10521159 let rawFeeAmount = muld(_rawAmount, adjustedFee)
1053- let referrerFeeAny = invoke(referralAddress(), "acceptPayment", [_trader], [AttachedPayment(quoteAsset(), rawFeeAmount)])
1054- if ((referrerFeeAny == referrerFeeAny))
1160+ let _amount = (_rawAmount - rawFeeAmount)
1161+ let distributeFeeAmount = if (isCollateralAsset)
10551162 then {
1056- let referrerFee = match referrerFeeAny {
1057- case x: Int =>
1058- x
1059- case _ =>
1060- throw("Invalid referrerFee")
1163+ let doBorrow = invoke(collateralAddress(), "borrow", [_trader], [AttachedPayment(_assetId, _amount)])
1164+ if ((doBorrow == doBorrow))
1165+ then {
1166+ let balanceBefore = assetBalance(this, quoteAsset())
1167+ if ((balanceBefore == balanceBefore))
1168+ then {
1169+ let doSwap = invoke(swapAddress(), "swap", [toBase58String(quoteAsset()), 0], [AttachedPayment(_assetId, rawFeeAmount)])
1170+ if ((doSwap == doSwap))
1171+ then {
1172+ let balanceAfter = assetBalance(this, quoteAsset())
1173+ if ((balanceAfter == balanceAfter))
1174+ then {
1175+ let exchangedAmount = (balanceAfter - balanceBefore)
1176+ if ((exchangedAmount == exchangedAmount))
1177+ then exchangedAmount
1178+ else throw("Strict value is not equal to itself.")
1179+ }
1180+ else throw("Strict value is not equal to itself.")
1181+ }
1182+ else throw("Strict value is not equal to itself.")
1183+ }
1184+ else throw("Strict value is not equal to itself.")
1185+ }
1186+ else throw("Strict value is not equal to itself.")
10611187 }
1062- let feeAmount = (rawFeeAmount - referrerFee)
1063- let _amount = (_rawAmount - rawFeeAmount)
1064- let $t04866148801 = getPosition(_trader)
1065- let oldPositionSize = $t04866148801._1
1066- let oldPositionMargin = $t04866148801._2
1067- let oldPositionOpenNotional = $t04866148801._3
1068- let oldPositionLstUpdCPF = $t04866148801._4
1069- let feeToStakers = (feeAmount / 2)
1070- let feeToInsurance = (feeAmount - feeToStakers)
1071- let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1072- if ((stake == stake))
1188+ else rawFeeAmount
1189+ if ((distributeFeeAmount == distributeFeeAmount))
1190+ then {
1191+ let referrerFeeAny = invoke(referralAddress(), "acceptPayment", [_trader], [AttachedPayment(quoteAsset(), distributeFeeAmount)])
1192+ if ((referrerFeeAny == referrerFeeAny))
10731193 then {
1074- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1075- if ((depositInsurance == depositInsurance))
1194+ let referrerFee = match referrerFeeAny {
1195+ case x: Int =>
1196+ x
1197+ case _ =>
1198+ throw("Invalid referrerFee")
1199+ }
1200+ let feeAmount = (distributeFeeAmount - referrerFee)
1201+ let $t05333753477 = getPosition(_trader)
1202+ let oldPositionSize = $t05333753477._1
1203+ let oldPositionMargin = $t05333753477._2
1204+ let oldPositionOpenNotional = $t05333753477._3
1205+ let oldPositionLstUpdCPF = $t05333753477._4
1206+ let feeToStakers = (feeAmount / 2)
1207+ let feeToInsurance = (feeAmount - feeToStakers)
1208+ let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1209+ if ((stake == stake))
10761210 then {
1077- let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1078- if ((notifyFee == notifyFee))
1079- then (((updatePosition(_trader, oldPositionSize, (oldPositionMargin + _amount), oldPositionOpenNotional, oldPositionLstUpdCPF) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1211+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1212+ if ((depositInsurance == depositInsurance))
1213+ then {
1214+ let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1215+ if ((notifyFee == notifyFee))
1216+ then (((updatePosition(_trader, oldPositionSize, (oldPositionMargin + _amount), oldPositionOpenNotional, oldPositionLstUpdCPF) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1217+ else throw("Strict value is not equal to itself.")
1218+ }
10801219 else throw("Strict value is not equal to itself.")
10811220 }
10821221 else throw("Strict value is not equal to itself.")
10901229
10911230
10921231 @Callable(i)
1093-func removeMargin (_amount) = if (if (if (if ((0 >= _amount))
1094- then true
1095- else !(requireOpenPosition(toString(i.caller))))
1096- then true
1097- else !(initialized()))
1098- then true
1099- else paused())
1100- then throw("Invalid removeMargin parameters")
1101- else {
1102- let $t04982049972 = getPosition(toString(i.caller))
1103- let oldPositionSize = $t04982049972._1
1104- let oldPositionMargin = $t04982049972._2
1105- let oldPositionOpenNotional = $t04982049972._3
1106- let oldPositionLstUpdCPF = $t04982049972._4
1107- let marginDelta = -(_amount)
1108- let $t05000950188 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
1109- let remainMargin = $t05000950188._1
1110- let badDebt = $t05000950188._2
1111- if ((badDebt != 0))
1112- then throw("Invalid removed margin amount")
1113- else {
1114- let marginRatio = calcMarginRatio(remainMargin, badDebt, oldPositionOpenNotional)
1115- if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
1116- then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
1117- else {
1118- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), _amount], nil)
1119- if ((unstake == unstake))
1120- then ((updatePosition(toString(i.caller), oldPositionSize, remainMargin, oldPositionOpenNotional, latestCumulativePremiumFraction(oldPositionSize)) ++ withdraw(i.caller, _amount)) ++ updateBalance((cbalance() - _amount)))
1121- else throw("Strict value is not equal to itself.")
1122- }
1123- }
1124- }
1232+func removeMargin (_amount) = {
1233+ let _trader = toString(i.caller)
1234+ if (if (if (if ((0 >= _amount))
1235+ then true
1236+ else !(requireOpenPosition(_trader)))
1237+ then true
1238+ else !(initialized()))
1239+ then true
1240+ else paused())
1241+ then throw("Invalid removeMargin parameters")
1242+ else {
1243+ let $t05452054660 = getPosition(_trader)
1244+ let oldPositionSize = $t05452054660._1
1245+ let oldPositionMargin = $t05452054660._2
1246+ let oldPositionOpenNotional = $t05452054660._3
1247+ let oldPositionLstUpdCPF = $t05452054660._4
1248+ let marginDelta = -(_amount)
1249+ let $t05469754876 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
1250+ let remainMargin = $t05469754876._1
1251+ let badDebt = $t05469754876._2
1252+ if ((badDebt != 0))
1253+ then throw("Invalid removed margin amount")
1254+ else {
1255+ let marginRatio = calcMarginRatio(remainMargin, badDebt, oldPositionOpenNotional)
1256+ if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
1257+ then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
1258+ else {
1259+ let quoteAssetStr = toBase58String(quoteAsset())
1260+ let $t05532055374 = getBorrowedByTrader(_trader)
1261+ let borrowed = $t05532055374._1
1262+ let assetId = $t05532055374._2
1263+ let toRepay = if ((_amount > borrowed))
1264+ then borrowed
1265+ else _amount
1266+ let toWithdraw = if ((borrowed > _amount))
1267+ then 0
1268+ else (_amount - borrowed)
1269+ let finalBorrow = (borrowed - toRepay)
1270+ let switchPositionToQuote = if ((finalBorrow > 0))
1271+ then nil
1272+ else updatePositionAsset(_trader, quoteAssetStr)
1273+ let doSanityCheck = if (((toRepay + toWithdraw) != _amount))
1274+ then throw(((((("toRepay=" + toString(toRepay)) + " + toWithdraw=") + toString(toWithdraw)) + " != ") + toString(_amount)))
1275+ else nil
1276+ if ((doSanityCheck == doSanityCheck))
1277+ then {
1278+ let doUnstake = invoke(managerAddress(), "withdraw", [quoteAssetStr, _amount], nil)
1279+ if ((doUnstake == doUnstake))
1280+ then {
1281+ let returnCollateralAction = if ((toRepay > 0))
1282+ then {
1283+ let doRepay = invoke(collateralAddress(), "repay", [_trader, assetId], [AttachedPayment(quoteAsset(), toRepay)])
1284+ if ((doRepay == doRepay))
1285+ then [ScriptTransfer(i.caller, toRepay, fromBase58String(assetId))]
1286+ else throw("Strict value is not equal to itself.")
1287+ }
1288+ else nil
1289+ if ((returnCollateralAction == returnCollateralAction))
1290+ then (updatePosition(_trader, oldPositionSize, remainMargin, oldPositionOpenNotional, latestCumulativePremiumFraction(oldPositionSize)) ++ (if ((toWithdraw > 0))
1291+ then withdraw(i.caller, toWithdraw)
1292+ else (((nil ++ updateBalance((cbalance() - _amount))) ++ switchPositionToQuote) ++ returnCollateralAction)))
1293+ else throw("Strict value is not equal to itself.")
1294+ }
1295+ else throw("Strict value is not equal to itself.")
1296+ }
1297+ else throw("Strict value is not equal to itself.")
1298+ }
1299+ }
1300+ }
1301+ }
11251302
11261303
11271304
11281305 @Callable(i)
11291306 func closePosition () = {
1130- let caller = getActualCaller(i)
1131- let callerAddress = valueOrErrorMessage(addressFromString(caller), "Invalid caller")
1132- if (if (if (!(requireOpenPosition(caller)))
1307+ let _trader = getActualCaller(i)
1308+ let _traderAddress = valueOrErrorMessage(addressFromString(_trader), "Invalid caller")
1309+ if (if (if (!(requireOpenPosition(_trader)))
11331310 then true
11341311 else !(initialized()))
11351312 then true
11361313 else paused())
11371314 then throw("Invalid closePosition parameters")
11381315 else {
1139- let $t05138251759 = internalClosePosition(caller, true)
1140- let x1 = $t05138251759._1
1141- let positionBadDebt = $t05138251759._2
1142- let realizedPnl = $t05138251759._3
1143- let marginToVault = $t05138251759._4
1144- let quoteAssetReserveAfter = $t05138251759._5
1145- let baseAssetReserveAfter = $t05138251759._6
1146- let totalPositionSizeAfter = $t05138251759._7
1147- let cumulativeNotionalAfter = $t05138251759._8
1148- let openInterestNotionalAfter = $t05138251759._9
1149- let x2 = $t05138251759._10
1150- let totalLongAfter = $t05138251759._11
1151- let totalShortAfter = $t05138251759._12
1316+ let $t05756657944 = internalClosePosition(_trader, true)
1317+ let x1 = $t05756657944._1
1318+ let positionBadDebt = $t05756657944._2
1319+ let realizedPnl = $t05756657944._3
1320+ let marginToVault = $t05756657944._4
1321+ let quoteAssetReserveAfter = $t05756657944._5
1322+ let baseAssetReserveAfter = $t05756657944._6
1323+ let totalPositionSizeAfter = $t05756657944._7
1324+ let cumulativeNotionalAfter = $t05756657944._8
1325+ let openInterestNotionalAfter = $t05756657944._9
1326+ let x2 = $t05756657944._10
1327+ let totalLongAfter = $t05756657944._11
1328+ let totalShortAfter = $t05756657944._12
11521329 if ((positionBadDebt > 0))
11531330 then throw("Unable to close position with bad debt")
11541331 else {
11551332 let withdrawAmount = abs(marginToVault)
11561333 let ammBalance = (cbalance() - withdrawAmount)
1157- let $t05196852110 = if ((0 > ammBalance))
1334+ let $t05815358295 = if ((0 > ammBalance))
11581335 then $Tuple2(0, abs(ammBalance))
11591336 else $Tuple2(ammBalance, 0)
1160- let ammNewBalance = $t05196852110._1
1161- let getFromInsurance = $t05196852110._2
1337+ let ammNewBalance = $t05815358295._1
1338+ let getFromInsurance = $t05815358295._2
11621339 let x = if ((getFromInsurance > 0))
11631340 then {
11641341 let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [getFromInsurance], nil)
11721349 let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (withdrawAmount - getFromInsurance)], nil)
11731350 if ((unstake == unstake))
11741351 then {
1175- let notifyNotional = invoke(minerAddress(), "notifyNotional", [caller, 0], nil)
1176- if ((notifyNotional == notifyNotional))
1177- then (((deletePosition(caller) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(callerAddress, withdrawAmount)) ++ updateBalance(ammNewBalance))
1352+ let $t05879758851 = getBorrowedByTrader(_trader)
1353+ let borrowed = $t05879758851._1
1354+ let assetId = $t05879758851._2
1355+ let $t05886659783 = if ((borrowed > 0))
1356+ then if ((withdrawAmount >= borrowed))
1357+ then {
1358+ let doRepay = invoke(collateralAddress(), "repay", [_trader, assetId], [AttachedPayment(quoteAsset(), borrowed)])
1359+ if ((doRepay == doRepay))
1360+ then $Tuple2([ScriptTransfer(_traderAddress, borrowed, fromBase58String(assetId))], (withdrawAmount - borrowed))
1361+ else throw("Strict value is not equal to itself.")
1362+ }
1363+ else {
1364+ let realizeAndClose = invoke(collateralAddress(), "realizePartiallyAndClose", [_trader, assetId], [AttachedPayment(quoteAsset(), withdrawAmount)])
1365+ if ((realizeAndClose == realizeAndClose))
1366+ then $Tuple2([ScriptTransfer(_traderAddress, withdrawAmount, fromBase58String(assetId))], 0)
1367+ else throw("Strict value is not equal to itself.")
1368+ }
1369+ else $Tuple2(nil, withdrawAmount)
1370+ if (($t05886659783 == $t05886659783))
1371+ then {
1372+ let quoteWithdrawAmount = $t05886659783._2
1373+ let sendCollateralAction = $t05886659783._1
1374+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
1375+ if ((notifyNotional == notifyNotional))
1376+ then ((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ (if ((quoteWithdrawAmount > 0))
1377+ then withdraw(_traderAddress, quoteWithdrawAmount)
1378+ else ((nil ++ updateBalance(ammNewBalance)) ++ sendCollateralAction)))
1379+ else throw("Strict value is not equal to itself.")
1380+ }
11781381 else throw("Strict value is not equal to itself.")
11791382 }
11801383 else throw("Strict value is not equal to itself.")
12091412 then (DECIMAL_UNIT > partialLiquidationRatio())
12101413 else false)
12111414 then {
1212- let $t05418154331 = getPosition(_trader)
1213- let oldPositionSize = $t05418154331._1
1214- let oldPositionMargin = $t05418154331._2
1215- let oldPositionOpenNotional = $t05418154331._3
1216- let oldPositionLstUpdCPF = $t05418154331._4
1415+ let $t06148361633 = getPosition(_trader)
1416+ let oldPositionSize = $t06148361633._1
1417+ let oldPositionMargin = $t06148361633._2
1418+ let oldPositionOpenNotional = $t06148361633._3
1419+ let oldPositionLstUpdCPF = $t06148361633._4
12171420 let _direction = if ((oldPositionSize > 0))
12181421 then DIR_SHORT
12191422 else DIR_LONG
12201423 let isAdd = (_direction == DIR_LONG)
12211424 let exchangedQuoteAssetAmount = getPartialLiquidationAmount(_trader, oldPositionSize)
1222- let $t05455654660 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1223- let oldPositionNotional = $t05455654660._1
1224- let unrealizedPnl = $t05455654660._2
1225- let $t05466854958 = swapInput(isAdd, exchangedQuoteAssetAmount)
1226- let exchangedPositionSize = $t05466854958._1
1227- let quoteAssetReserveAfter = $t05466854958._2
1228- let baseAssetReserveAfter = $t05466854958._3
1229- let totalPositionSizeAfter = $t05466854958._4
1230- let cumulativeNotionalAfter = $t05466854958._5
1231- let realizedPnl = divd(muld(unrealizedPnl, abs(exchangedPositionSize)), abs(oldPositionSize))
1232- let $t05506555298 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
1233- let remainMargin = $t05506555298._1
1234- let badDebt = $t05506555298._2
1235- let fundingPayment = $t05506555298._3
1425+ let $t06185861962 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1426+ let oldPositionNotional = $t06185861962._1
1427+ let unrealizedPnl = $t06185861962._2
1428+ let $t06197062202 = swapInput(isAdd, exchangedQuoteAssetAmount)
1429+ let exchangedPositionSize = $t06197062202._1
1430+ let quoteAssetReserveAfter = $t06197062202._2
1431+ let baseAssetReserveAfter = $t06197062202._3
1432+ let totalPositionSizeAfter = $t06197062202._4
1433+ let cumulativeNotionalAfter = $t06197062202._5
1434+ let liquidationRatio = divd(abs(exchangedPositionSize), abs(oldPositionSize))
1435+ let realizedPnl = muld(unrealizedPnl, liquidationRatio)
1436+ let $t06241362646 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
1437+ let remainMargin = $t06241362646._1
1438+ let badDebt = $t06241362646._2
1439+ let fundingPayment = $t06241362646._3
12361440 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
12371441 let remainOpenNotional = if ((oldPositionSize > 0))
12381442 then ((oldPositionNotional - exchangedQuoteAssetAmount) - unrealizedPnlAfter)
12461450 let newPositionLstUpdCPF = latestCumulativePremiumFraction(newPositionSize)
12471451 let openInterestNotionalAfter = (openInterestNotional() - exchangedQuoteAssetAmount)
12481452 let ammBalance = (cbalance() - liquidationPenalty)
1249- let $t05647156614 = if ((0 > ammBalance))
1453+ let $t06381963962 = if ((0 > ammBalance))
12501454 then $Tuple2(0, abs(ammBalance))
12511455 else $Tuple2(ammBalance, 0)
1252- let newAmmBalance = $t05647156614._1
1253- let takeFromInsurance = $t05647156614._2
1254- let x = if ((takeFromInsurance > 0))
1456+ let newAmmBalance = $t06381963962._1
1457+ let takeFromInsurance = $t06381963962._2
1458+ let $t06397064024 = getBorrowedByTrader(_trader)
1459+ let borrowed = $t06397064024._1
1460+ let assetId = $t06397064024._2
1461+ let doLiquidateCollateral = if ((borrowed > 0))
12551462 then {
1256- let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1257- if ((withdrawInsurance == withdrawInsurance))
1463+ let collateralToSell = muld(borrowed, liquidationRatio)
1464+ let realizeAndClose = invoke(collateralAddress(), "realizePartially", [_trader, assetId, collateralToSell], nil)
1465+ if ((realizeAndClose == realizeAndClose))
12581466 then nil
12591467 else throw("Strict value is not equal to itself.")
12601468 }
12611469 else nil
1262- if ((x == x))
1470+ if ((doLiquidateCollateral == doLiquidateCollateral))
12631471 then {
1264- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1265- if ((unstake == unstake))
1472+ let x = if ((takeFromInsurance > 0))
12661473 then {
1267- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1268- if ((depositInsurance == depositInsurance))
1474+ let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1475+ if ((withdrawInsurance == withdrawInsurance))
1476+ then nil
1477+ else throw("Strict value is not equal to itself.")
1478+ }
1479+ else nil
1480+ if ((x == x))
1481+ then {
1482+ let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1483+ if ((unstake == unstake))
12691484 then {
1270- let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1271- if ((notifyNotional == notifyNotional))
1272- then (((updatePosition(_trader, newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, (totalLongPositionSize() - (if ((newPositionSize > 0))
1273- then abs(exchangedPositionSize)
1274- else 0)), (totalShortPositionSize() - (if ((0 > newPositionSize))
1275- then abs(exchangedPositionSize)
1276- else 0)))) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1485+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1486+ if ((depositInsurance == depositInsurance))
1487+ then {
1488+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1489+ if ((notifyNotional == notifyNotional))
1490+ then (((updatePosition(_trader, newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, (totalLongPositionSize() - (if ((newPositionSize > 0))
1491+ then abs(exchangedPositionSize)
1492+ else 0)), (totalShortPositionSize() - (if ((0 > newPositionSize))
1493+ then abs(exchangedPositionSize)
1494+ else 0)))) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1495+ else throw("Strict value is not equal to itself.")
1496+ }
12771497 else throw("Strict value is not equal to itself.")
12781498 }
12791499 else throw("Strict value is not equal to itself.")
12831503 else throw("Strict value is not equal to itself.")
12841504 }
12851505 else {
1286- let $t05806158516 = internalClosePosition(_trader, false)
1287- let x1 = $t05806158516._1
1288- let badDebt = $t05806158516._2
1289- let x2 = $t05806158516._3
1290- let x3 = $t05806158516._4
1291- let quoteAssetReserveAfter = $t05806158516._5
1292- let baseAssetReserveAfter = $t05806158516._6
1293- let totalPositionSizeAfter = $t05806158516._7
1294- let cumulativeNotionalAfter = $t05806158516._8
1295- let openInterestNotionalAfter = $t05806158516._9
1296- let exchangedQuoteAssetAmount = $t05806158516._10
1297- let totalLongAfter = $t05806158516._11
1298- let totalShortAfter = $t05806158516._12
1506+ let $t06586666321 = internalClosePosition(_trader, false)
1507+ let x1 = $t06586666321._1
1508+ let badDebt = $t06586666321._2
1509+ let x2 = $t06586666321._3
1510+ let x3 = $t06586666321._4
1511+ let quoteAssetReserveAfter = $t06586666321._5
1512+ let baseAssetReserveAfter = $t06586666321._6
1513+ let totalPositionSizeAfter = $t06586666321._7
1514+ let cumulativeNotionalAfter = $t06586666321._8
1515+ let openInterestNotionalAfter = $t06586666321._9
1516+ let exchangedQuoteAssetAmount = $t06586666321._10
1517+ let totalLongAfter = $t06586666321._11
1518+ let totalShortAfter = $t06586666321._12
12991519 let liquidationPenalty = muld(exchangedQuoteAssetAmount, liquidationFeeRatio())
13001520 let feeToLiquidator = (liquidationPenalty / 2)
13011521 let feeToInsurance = (liquidationPenalty - feeToLiquidator)
13021522 let ammBalance = (cbalance() - liquidationPenalty)
1303- let $t05892859071 = if ((0 > ammBalance))
1523+ let $t06673366876 = if ((0 > ammBalance))
13041524 then $Tuple2(0, abs(ammBalance))
13051525 else $Tuple2(ammBalance, 0)
1306- let newAmmBalance = $t05892859071._1
1307- let takeFromInsurance = $t05892859071._2
1308- let x = if ((takeFromInsurance > 0))
1526+ let newAmmBalance = $t06673366876._1
1527+ let takeFromInsurance = $t06673366876._2
1528+ let $t06688466938 = getBorrowedByTrader(_trader)
1529+ let borrowed = $t06688466938._1
1530+ let assetId = $t06688466938._2
1531+ let doLiquidateCollateral = if ((borrowed > 0))
13091532 then {
1310- let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1311- if ((withdrawInsurance == withdrawInsurance))
1533+ let realizeAndClose = invoke(collateralAddress(), "realizePartiallyAndClose", [_trader, assetId], nil)
1534+ if ((realizeAndClose == realizeAndClose))
13121535 then nil
13131536 else throw("Strict value is not equal to itself.")
13141537 }
13151538 else nil
1316- if ((x == x))
1539+ if ((doLiquidateCollateral == doLiquidateCollateral))
13171540 then {
1318- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1319- if ((unstake == unstake))
1541+ let x = if ((takeFromInsurance > 0))
13201542 then {
1321- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1322- if ((depositInsurance == depositInsurance))
1543+ let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1544+ if ((withdrawInsurance == withdrawInsurance))
1545+ then nil
1546+ else throw("Strict value is not equal to itself.")
1547+ }
1548+ else nil
1549+ if ((x == x))
1550+ then {
1551+ let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1552+ if ((unstake == unstake))
13231553 then {
1324- let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
1325- if ((notifyNotional == notifyNotional))
1326- then (((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1554+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1555+ if ((depositInsurance == depositInsurance))
1556+ then {
1557+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
1558+ if ((notifyNotional == notifyNotional))
1559+ then (((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1560+ else throw("Strict value is not equal to itself.")
1561+ }
13271562 else throw("Strict value is not equal to itself.")
13281563 }
13291564 else throw("Strict value is not equal to itself.")
13471582 then throw(((("Invalid funding block timestamp: " + toString(lastBlock.timestamp)) + " < ") + toString(fundingBlockTimestamp)))
13481583 else {
13491584 let underlyingPrice = getOracleTwapPrice()
1350- let $t06055760619 = getFunding()
1351- let shortPremiumFraction = $t06055760619._1
1352- let longPremiumFraction = $t06055760619._2
1585+ let $t06873668798 = getFunding()
1586+ let shortPremiumFraction = $t06873668798._1
1587+ let longPremiumFraction = $t06873668798._2
13531588 updateFunding((fundingBlockTimestamp + fundingPeriodSeconds()), (latestLongCumulativePremiumFraction() + longPremiumFraction), (latestShortCumulativePremiumFraction() + shortPremiumFraction), divd(longPremiumFraction, underlyingPrice), divd(shortPremiumFraction, underlyingPrice))
13541589 }
13551590 }
13761611 else (0 > _price))
13771612 then throw("Invalid adjustPeg parameters")
13781613 else {
1379- let $t06161061696 = getPegAdjustCost(_price)
1380- let newQuoteAssetReserve = $t06161061696._1
1381- let pegChangeCost = $t06161061696._2
1614+ let $t06978969875 = getPegAdjustCost(_price)
1615+ let newQuoteAssetReserve = $t06978969875._1
1616+ let pegChangeCost = $t06978969875._2
13821617 if ((pegChangeCost == 0))
13831618 then throw("Nothing to adjust")
13841619 else if ((pegChangeCost > 0))
14201655
14211656 @Callable(i)
14221657 func v_get (_trader) = {
1423- let $t06333763397 = internalClosePosition(_trader, false)
1424- let x1 = $t06333763397._1
1425- let x2 = $t06333763397._2
1426- let x3 = $t06333763397._3
1427- let x4 = $t06333763397._4
1658+ let $t07151671576 = internalClosePosition(_trader, false)
1659+ let x1 = $t07151671576._1
1660+ let x2 = $t07151671576._2
1661+ let x3 = $t07151671576._3
1662+ let x4 = $t07151671576._4
14281663 throw((((s(x2) + s(x3)) + s(x4)) + s(getMarginRatio(_trader))))
14291664 }
14301665
14321667
14331668 @Callable(i)
14341669 func view_calcRemainMarginWithFundingPayment (_trader) = {
1435- let $t06354463655 = getPosition(_trader)
1436- let positionSize = $t06354463655._1
1437- let positionMargin = $t06354463655._2
1438- let pon = $t06354463655._3
1439- let positionLstUpdCPF = $t06354463655._4
1440- let $t06366063761 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1441- let positionNotional = $t06366063761._1
1442- let unrealizedPnl = $t06366063761._2
1443- let $t06376663948 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1444- let remainMargin = $t06376663948._1
1445- let badDebt = $t06376663948._2
1446- let fundingPayment = $t06376663948._3
1670+ let $t07172371834 = getPosition(_trader)
1671+ let positionSize = $t07172371834._1
1672+ let positionMargin = $t07172371834._2
1673+ let pon = $t07172371834._3
1674+ let positionLstUpdCPF = $t07172371834._4
1675+ let $t07183971940 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1676+ let positionNotional = $t07183971940._1
1677+ let unrealizedPnl = $t07183971940._2
1678+ let $t07194572127 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1679+ let remainMargin = $t07194572127._1
1680+ let badDebt = $t07194572127._2
1681+ let fundingPayment = $t07194572127._3
14471682 throw((((((s(remainMargin) + s(fundingPayment)) + s(getMarginRatio(_trader))) + s(unrealizedPnl)) + s(badDebt)) + s(positionNotional)))
14481683 }
14491684
14601695 @Callable(i)
14611696 func view_getFunding () = {
14621697 let underlyingPrice = getOracleTwapPrice()
1463- let $t06433964401 = getFunding()
1464- let shortPremiumFraction = $t06433964401._1
1465- let longPremiumFraction = $t06433964401._2
1698+ let $t07251872580 = getFunding()
1699+ let shortPremiumFraction = $t07251872580._1
1700+ let longPremiumFraction = $t07251872580._2
14661701 let longFunding = divd(longPremiumFraction, underlyingPrice)
14671702 let shortFunding = divd(shortPremiumFraction, underlyingPrice)
14681703 throw((((s(longFunding) + s(shortFunding)) + s(getTwapSpotPrice())) + s(getOracleTwapPrice())))
1704+ }
1705+
1706+
1707+
1708+@Callable(i)
1709+func view_getBorrowedByTrader (_trader) = {
1710+ let $t07287072924 = getBorrowedByTrader(_trader)
1711+ let borrowed = $t07287072924._1
1712+ let assetId = $t07287072924._2
1713+ throw((s(borrowed) + assetId))
14691714 }
14701715
14711716
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_positionClosedDate = "k_positionClosedDate"
21+
22+let k_positionAsset = "k_positionAsset"
2123
2224 let k_initialized = "k_initialized"
2325
2426 let k_paused = "k_paused"
2527
2628 let k_fee = "k_fee"
2729
2830 let k_fundingPeriod = "k_fundingPeriod"
2931
3032 let k_initMarginRatio = "k_initMarginRatio"
3133
3234 let k_maintenanceMarginRatio = "k_mmr"
3335
3436 let k_liquidationFeeRatio = "k_liquidationFeeRatio"
3537
3638 let k_partialLiquidationRatio = "k_partLiquidationRatio"
3739
3840 let k_spreadLimit = "k_spreadLimit"
3941
4042 let k_maxPriceImpact = "k_maxPriceImpact"
4143
4244 let k_maxPriceSpread = "k_maxPriceSpread"
4345
4446 let k_lastDataStr = "k_lastDataStr"
4547
4648 let k_lastMinuteId = "k_lastMinuteId"
4749
4850 let k_twapDataLastCumulativePrice = "k_twapDataLastCumulativePrice"
4951
5052 let k_twapDataLastPrice = "k_twapDataLastPrice"
5153
5254 let k_twapDataPreviousMinuteId = "k_twapDataPreviousMinuteId"
5355
5456 let k_latestLongCumulativePremiumFraction = "k_latestLongPremiumFraction"
5557
5658 let k_latestShortCumulativePremiumFraction = "k_latestShortPremiumFraction"
5759
5860 let k_nextFundingBlock = "k_nextFundingBlockMinTimestamp"
5961
6062 let k_longFundingRate = "k_longFundingRate"
6163
6264 let k_shortFundingRate = "k_shortFundingRate"
6365
6466 let k_quoteAssetReserve = "k_qtAstR"
6567
6668 let k_baseAssetReserve = "k_bsAstR"
6769
6870 let k_totalPositionSize = "k_totalPositionSize"
6971
7072 let k_totalLongPositionSize = "k_totalLongPositionSize"
7173
7274 let k_totalShortPositionSize = "k_totalShortPositionSize"
7375
7476 let k_cumulativeNotional = "k_cumulativeNotional"
7577
7678 let k_openInterestNotional = "k_openInterestNotional"
7779
7880 let k_coordinatorAddress = "k_coordinatorAddress"
7981
8082 let k_insurance_address = "k_insurance_address"
8183
8284 let k_admin_address = "k_admin_address"
8385
8486 let k_admin_public_key = "k_admin_public_key"
8587
8688 let k_quote_asset = "k_quote_asset"
8789
8890 let k_quote_staking = "k_quote_staking"
8991
9092 let k_staking_address = "k_staking_address"
9193
9294 let k_miner_address = "k_miner_address"
9395
9496 let k_orders_address = "k_orders_address"
9597
9698 let k_referral_address = "k_referral_address"
9799
98100 let k_manager_address = "k_manager_address"
99101
102+let k_collateral_address = "k_collateral_address"
103+
104+let k_exchange_address = "k_exchange_address"
105+
100106 let k_nft_manager_address = "k_nft_manager_address"
107+
108+let k_trader_market_asset_collateral = "k_trader_market_asset_collateral"
109+
110+func toCompositeKey (_key,_address) = ((_key + "_") + _address)
111+
101112
102113 func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
103114
104115
105116 func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address))
106117
107118
108119 func adminPublicKey () = fromBase58String(getStringValue(coordinator(), k_admin_public_key))
109120
110121
111122 func quoteAsset () = fromBase58String(getStringValue(coordinator(), k_quote_asset))
112123
113124
114125 func quoteAssetStaking () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_quote_staking)), "Quote asset staking not set")
115126
116127
117128 func stakingAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_staking_address)), "Staking not set")
118129
119130
120131 func insuranceAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_insurance_address)), "Insurance not set")
121132
122133
123134 func minerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_miner_address)), "Miner not set")
124135
125136
126137 func ordersAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_orders_address)), "Orders not set")
127138
128139
129140 func referralAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_referral_address)), "Referral not set")
130141
131142
132143 func managerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_manager_address)), "Manager not set")
133144
134145
135146 func nftManagerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_nft_manager_address)), "NFT Manager not set")
147+
148+
149+func collateralAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_collateral_address)), "Collateral Manager not set")
150+
151+
152+func swapAddress () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(coordinator(), k_exchange_address), "No swap address")), "Invalid swap address")
153+
154+
155+let k_whitelist_asset = "k_whitelist_asset"
156+
157+func isWhitelistAsset (_assetId) = valueOrElse(getBoolean(collateralAddress(), toCompositeKey(k_whitelist_asset, _assetId)), false)
136158
137159
138160 let k_token_param = "k_token_param"
139161
140162 let k_token_type = "k_token_type"
141163
142164 let FEE_REDUCTION_TOKEN_TYPE = "fee_reduction"
143165
144166 let DIR_LONG = 1
145167
146168 let DIR_SHORT = 2
147169
148170 let TWAP_INTERVAL = 15
149171
150172 let ORACLE_INTERVAL = 15
151173
152174 let SECONDS = 1000
153175
154176 let DECIMAL_UNIT = (1 * (((((10 * 10) * 10) * 10) * 10) * 10))
155177
156178 let ONE_DAY = (86400 * DECIMAL_UNIT)
157179
158180 let ALL_FEES = 100
159181
160182 let PNL_OPTION_SPOT = 1
161183
162184 let PNL_OPTION_ORACLE = 2
163185
164186 func s (_x) = (toString(_x) + ",")
165187
166188
167189 func divd (_x,_y) = fraction(_x, DECIMAL_UNIT, _y, HALFEVEN)
168190
169191
170192 func muld (_x,_y) = fraction(_x, _y, DECIMAL_UNIT, HALFEVEN)
171193
172194
173195 func abs (_x) = if ((_x > 0))
174196 then _x
175197 else -(_x)
176198
177199
178200 func vmax (_x,_y) = if ((_x >= _y))
179201 then _x
180202 else _y
181-
182-
183-func toCompositeKey (_key,_address) = ((_key + "_") + _address)
184203
185204
186205 func listToStr (_list) = {
187206 func _join (accumulator,val) = ((accumulator + val) + ",")
188207
189208 let newListStr = {
190209 let $l = _list
191210 let $s = size($l)
192211 let $acc0 = ""
193212 func $f0_1 ($a,$i) = if (($i >= $s))
194213 then $a
195214 else _join($a, $l[$i])
196215
197216 func $f0_2 ($a,$i) = if (($i >= $s))
198217 then $a
199218 else throw("List size exceeds 20")
200219
201220 $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)
202221 }
203222 let newListStrU = dropRight(newListStr, 1)
204223 let newListStrR = if ((take(newListStrU, 1) == ","))
205224 then drop(newListStrU, 1)
206225 else newListStrU
207226 newListStrR
208227 }
209228
210229
211230 func strToList (_str) = split(_str, ",")
212231
213232
214233 func pushToQueue (_list,_maxSize,_value) = if ((size(_list) > _maxSize))
215234 then (removeByIndex(_list, 0) :+ _value)
216235 else (_list :+ _value)
217236
218237
219238 func int (k) = valueOrErrorMessage(getInteger(this, k), ("no value for " + k))
220239
221240
222241 func strA (_address,_key) = {
223242 let val = valueOrErrorMessage(getString(_address, _key), ("No value for key " + _key))
224243 val
225244 }
226245
227246
228247 func intA (_address,_key) = {
229248 let val = valueOrErrorMessage(getInteger(_address, _key), ("No value for key " + _key))
230249 val
231250 }
232251
233252
234253 func cbalance () = int(k_balance)
235254
236255
237256 func fee () = int(k_fee)
238257
239258
240259 func initMarginRatio () = int(k_initMarginRatio)
241260
242261
243262 func qtAstR () = int(k_quoteAssetReserve)
244263
245264
246265 func bsAstR () = int(k_baseAssetReserve)
247266
248267
249268 func totalPositionSize () = int(k_totalPositionSize)
250269
251270
252271 func cumulativeNotional () = int(k_cumulativeNotional)
253272
254273
255274 func openInterestNotional () = int(k_openInterestNotional)
256275
257276
258277 func nextFundingBlockTimestamp () = int(k_nextFundingBlock)
259278
260279
261280 func fundingPeriodRaw () = int(k_fundingPeriod)
262281
263282
264283 func fundingPeriodDecimal () = (fundingPeriodRaw() * DECIMAL_UNIT)
265284
266285
267286 func fundingPeriodSeconds () = (fundingPeriodRaw() * SECONDS)
268287
269288
270289 func maintenanceMarginRatio () = int(k_maintenanceMarginRatio)
271290
272291
273292 func liquidationFeeRatio () = int(k_liquidationFeeRatio)
274293
275294
276295 func partialLiquidationRatio () = int(k_partialLiquidationRatio)
277296
278297
279298 func spreadLimit () = int(k_spreadLimit)
280299
281300
282301 func maxPriceImpact () = int(k_maxPriceImpact)
283302
284303
285304 func maxPriceSpread () = int(k_maxPriceSpread)
286305
287306
288307 func latestLongCumulativePremiumFraction () = int(k_latestLongCumulativePremiumFraction)
289308
290309
291310 func latestShortCumulativePremiumFraction () = int(k_latestShortCumulativePremiumFraction)
292311
293312
294313 func totalShortPositionSize () = int(k_totalShortPositionSize)
295314
296315
297316 func totalLongPositionSize () = int(k_totalLongPositionSize)
298317
299318
300319 func getActualCaller (i) = valueOrElse(getString(ordersAddress(), "k_sender"), toString(i.caller))
301320
302321
303322 func requireMoreMarginRatio (_marginRatio,_baseMarginRatio,_largerThanOrEqualTo) = {
304323 let remainingMarginRatio = (_marginRatio - _baseMarginRatio)
305324 if (if (_largerThanOrEqualTo)
306325 then (0 > remainingMarginRatio)
307326 else false)
308327 then throw("Invalid margin")
309328 else if (if (!(_largerThanOrEqualTo))
310329 then (remainingMarginRatio >= 0)
311330 else false)
312331 then throw("Invalid margin")
313332 else true
314333 }
315334
316335
317336 func latestCumulativePremiumFraction (_positionSize) = if ((_positionSize == 0))
318337 then throw("Should not be called with _positionSize == 0")
319338 else if ((_positionSize > 0))
320339 then latestLongCumulativePremiumFraction()
321340 else latestShortCumulativePremiumFraction()
322341
323342
324343 func getPosition (_trader) = {
325344 let positionSizeOpt = getInteger(this, toCompositeKey(k_positionSize, _trader))
326345 match positionSizeOpt {
327346 case positionSize: Int =>
328347 $Tuple4(positionSize, getIntegerValue(this, toCompositeKey(k_positionMargin, _trader)), getIntegerValue(this, toCompositeKey(k_positionOpenNotional, _trader)), getIntegerValue(this, toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _trader)))
329348 case _ =>
330349 $Tuple4(0, 0, 0, 0)
331350 }
332351 }
333352
334353
354+func getPositionAsset (_trader) = {
355+ let positionAssetOpt = getString(this, toCompositeKey(k_positionAsset, _trader))
356+ match positionAssetOpt {
357+ case positionAsset: String =>
358+ positionAsset
359+ case _ =>
360+ toBase58String(quoteAsset())
361+ }
362+ }
363+
364+
335365 func requireOpenPosition (_trader) = if ((getPosition(_trader)._1 == 0))
336366 then throw("No open position")
337367 else true
338368
339369
340370 func initialized () = valueOrElse(getBoolean(this, k_initialized), false)
341371
342372
343373 func paused () = valueOrElse(getBoolean(this, k_paused), false)
344374
345375
346376 func updateReserve (_isAdd,_quoteAssetAmount,_baseAssetAmount) = if (_isAdd)
347377 then {
348378 let newBase = (bsAstR() - _baseAssetAmount)
349379 if ((0 >= newBase))
350380 then throw("Tx lead to base asset reserve <= 0, revert")
351381 else $Tuple4((qtAstR() + _quoteAssetAmount), newBase, (totalPositionSize() + _baseAssetAmount), (cumulativeNotional() + _quoteAssetAmount))
352382 }
353383 else {
354384 let newQuote = (qtAstR() - _quoteAssetAmount)
355385 if ((0 >= newQuote))
356386 then throw("Tx lead to base quote reserve <= 0, revert")
357387 else $Tuple4(newQuote, (bsAstR() + _baseAssetAmount), (totalPositionSize() - _baseAssetAmount), (cumulativeNotional() - _quoteAssetAmount))
358388 }
359389
360390
361391 func swapInput (_isAdd,_quoteAssetAmount) = {
362392 let _qtAstR = qtAstR()
363393 let _bsAstR = bsAstR()
364394 let k = muld(_qtAstR, _bsAstR)
365395 let quoteAssetReserveAfter = if (_isAdd)
366396 then (_qtAstR + _quoteAssetAmount)
367397 else (_qtAstR - _quoteAssetAmount)
368398 let baseAssetReserveAfter = divd(k, quoteAssetReserveAfter)
369399 let amountBaseAssetBoughtAbs = abs((baseAssetReserveAfter - _bsAstR))
370400 let amountBaseAssetBought = if (_isAdd)
371401 then amountBaseAssetBoughtAbs
372402 else -(amountBaseAssetBoughtAbs)
373- let $t01366013853 = updateReserve(_isAdd, _quoteAssetAmount, amountBaseAssetBoughtAbs)
374- let quoteAssetReserveAfter1 = $t01366013853._1
375- let baseAssetReserveAfter1 = $t01366013853._2
376- let totalPositionSizeAfter1 = $t01366013853._3
377- let cumulativeNotionalAfter1 = $t01366013853._4
403+ let $t01497415167 = updateReserve(_isAdd, _quoteAssetAmount, amountBaseAssetBoughtAbs)
404+ let quoteAssetReserveAfter1 = $t01497415167._1
405+ let baseAssetReserveAfter1 = $t01497415167._2
406+ let totalPositionSizeAfter1 = $t01497415167._3
407+ let cumulativeNotionalAfter1 = $t01497415167._4
378408 let priceBefore = divd(_qtAstR, _bsAstR)
379409 let marketPrice = divd(_quoteAssetAmount, amountBaseAssetBoughtAbs)
380410 let priceDiff = abs((priceBefore - marketPrice))
381411 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
382412 let maxPriceImpactValue = maxPriceImpact()
383413 if ((priceImpact > maxPriceImpactValue))
384414 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)))
385415 else $Tuple5(amountBaseAssetBought, quoteAssetReserveAfter1, baseAssetReserveAfter1, totalPositionSizeAfter1, cumulativeNotionalAfter1)
386416 }
387417
388418
389419 func calcRemainMarginWithFundingPayment (_oldPositionSize,_oldPositionMargin,_oldPositionCumulativePremiumFraction,_marginDelta) = {
390420 let fundingPayment = if ((_oldPositionSize != 0))
391421 then {
392422 let _latestCumulativePremiumFraction = latestCumulativePremiumFraction(_oldPositionSize)
393423 muld((_latestCumulativePremiumFraction - _oldPositionCumulativePremiumFraction), _oldPositionSize)
394424 }
395425 else 0
396426 let signedMargin = ((_marginDelta - fundingPayment) + _oldPositionMargin)
397- let $t01534415471 = if ((0 > signedMargin))
427+ let $t01665816785 = if ((0 > signedMargin))
398428 then $Tuple2(0, abs(signedMargin))
399429 else $Tuple2(abs(signedMargin), 0)
400- let remainMargin = $t01534415471._1
401- let badDebt = $t01534415471._2
430+ let remainMargin = $t01665816785._1
431+ let badDebt = $t01665816785._2
402432 $Tuple3(remainMargin, badDebt, fundingPayment)
403433 }
404434
405435
406436 func swapOutputWithReserves (_isAdd,_baseAssetAmount,_checkMaxPriceImpact,_quoteAssetReserve,_baseAssetReserve) = {
407437 let priceBefore = divd(_quoteAssetReserve, _baseAssetReserve)
408438 if ((_baseAssetAmount == 0))
409439 then throw("Invalid base asset amount")
410440 else {
411441 let k = muld(_quoteAssetReserve, _baseAssetReserve)
412442 let baseAssetPoolAmountAfter = if (_isAdd)
413443 then (_baseAssetReserve + _baseAssetAmount)
414444 else (_baseAssetReserve - _baseAssetAmount)
415445 let quoteAssetAfter = divd(k, baseAssetPoolAmountAfter)
416446 let quoteAssetSold = abs((quoteAssetAfter - _quoteAssetReserve))
417447 let maxPriceImpactValue = maxPriceImpact()
418- let $t01644816641 = updateReserve(!(_isAdd), quoteAssetSold, _baseAssetAmount)
419- let quoteAssetReserveAfter1 = $t01644816641._1
420- let baseAssetReserveAfter1 = $t01644816641._2
421- let totalPositionSizeAfter1 = $t01644816641._3
422- let cumulativeNotionalAfter1 = $t01644816641._4
448+ let $t01776217955 = updateReserve(!(_isAdd), quoteAssetSold, _baseAssetAmount)
449+ let quoteAssetReserveAfter1 = $t01776217955._1
450+ let baseAssetReserveAfter1 = $t01776217955._2
451+ let totalPositionSizeAfter1 = $t01776217955._3
452+ let cumulativeNotionalAfter1 = $t01776217955._4
423453 let marketPrice = divd(quoteAssetSold, _baseAssetAmount)
424454 let priceDiff = abs((priceBefore - marketPrice))
425455 let priceImpact = (DECIMAL_UNIT - divd(priceBefore, (priceBefore + priceDiff)))
426456 if (if ((priceImpact > maxPriceImpactValue))
427457 then _checkMaxPriceImpact
428458 else false)
429459 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)))
430460 else $Tuple8(quoteAssetSold, quoteAssetReserveAfter1, baseAssetReserveAfter1, totalPositionSizeAfter1, cumulativeNotionalAfter1, (totalLongPositionSize() - (if (_isAdd)
431461 then abs(_baseAssetAmount)
432462 else 0)), (totalShortPositionSize() - (if (!(_isAdd))
433463 then abs(_baseAssetAmount)
434464 else 0)), priceImpact)
435465 }
436466 }
437467
438468
439469 func swapOutput (_isAdd,_baseAssetAmount,_checkMaxPriceImpact) = swapOutputWithReserves(_isAdd, _baseAssetAmount, _checkMaxPriceImpact, qtAstR(), bsAstR())
440470
441471
442472 func getOracleTwapPrice () = {
443473 let oracle = valueOrErrorMessage(addressFromString(getStringValue(this, k_ora)), "")
444474 let priceKey = getStringValue(this, k_ora_key)
445475 let blockKey = getStringValue(this, k_ora_block_key)
446476 let lastValue = valueOrErrorMessage(getInteger(oracle, priceKey), ((("Can not get oracle price. Oracle: " + toString(oracle)) + " key: ") + priceKey))
447477 lastValue
448478 }
449479
450480
451481 func requireNotOverSpreadLimit (_quoteAssetReserve,_baseAssetReserve) = {
452482 let oraclePrice = getOracleTwapPrice()
453483 let priceAfter = divd(_quoteAssetReserve, _baseAssetReserve)
454484 let averagePrice = divd((oraclePrice + priceAfter), (2 * DECIMAL_UNIT))
455485 let absPriceDiff = divd(abs((oraclePrice - priceAfter)), averagePrice)
456486 if ((absPriceDiff > maxPriceSpread()))
457487 then throw(((("Price spread " + toString(absPriceDiff)) + " > max price spread ") + toString(maxPriceSpread())))
458488 else true
459489 }
460490
461491
462492 func getSpotPrice () = {
463493 let _quoteAssetReserve = qtAstR()
464494 let _baseAssetReserve = bsAstR()
465495 divd(_quoteAssetReserve, _baseAssetReserve)
466496 }
467497
468498
469499 func isOverFluctuationLimit () = {
470500 let oraclePrice = getOracleTwapPrice()
471501 let currentPrice = getSpotPrice()
472502 (divd(abs((oraclePrice - currentPrice)), oraclePrice) > spreadLimit())
473503 }
474504
475505
476506 func getPositionAdjustedOpenNotional (_positionSize,_option,_quoteAssetReserve,_baseAssetReserve) = {
477507 let positionSizeAbs = abs(_positionSize)
478508 let isShort = (0 > _positionSize)
479509 let positionNotional = if ((_option == PNL_OPTION_SPOT))
480510 then {
481- let $t01985920030 = swapOutputWithReserves(!(isShort), positionSizeAbs, false, _quoteAssetReserve, _baseAssetReserve)
482- let outPositionNotional = $t01985920030._1
483- let x1 = $t01985920030._2
484- let x2 = $t01985920030._3
485- let x3 = $t01985920030._4
511+ let $t02117321344 = swapOutputWithReserves(!(isShort), positionSizeAbs, false, _quoteAssetReserve, _baseAssetReserve)
512+ let outPositionNotional = $t02117321344._1
513+ let x1 = $t02117321344._2
514+ let x2 = $t02117321344._3
515+ let x3 = $t02117321344._4
486516 outPositionNotional
487517 }
488518 else muld(positionSizeAbs, getOracleTwapPrice())
489519 positionNotional
490520 }
491521
492522
493523 func getPositionNotionalAndUnrealizedPnlByValues (_positionSize,_positionOpenNotional,_quoteAssetReserve,_baseAssetReserve,_option) = if ((_positionSize == 0))
494524 then throw("Invalid position size")
495525 else {
496526 let isShort = (0 > _positionSize)
497527 let positionNotional = getPositionAdjustedOpenNotional(_positionSize, _option, _quoteAssetReserve, _baseAssetReserve)
498528 let unrealizedPnl = if (isShort)
499529 then (_positionOpenNotional - positionNotional)
500530 else (positionNotional - _positionOpenNotional)
501531 $Tuple2(positionNotional, unrealizedPnl)
502532 }
503533
504534
505535 func getPositionNotionalAndUnrealizedPnl (_trader,_option) = {
506- let $t02134721475 = getPosition(_trader)
507- let positionSize = $t02134721475._1
508- let positionMargin = $t02134721475._2
509- let positionOpenNotional = $t02134721475._3
510- let positionLstUpdCPF = $t02134721475._4
536+ let $t02266122789 = getPosition(_trader)
537+ let positionSize = $t02266122789._1
538+ let positionMargin = $t02266122789._2
539+ let positionOpenNotional = $t02266122789._3
540+ let positionLstUpdCPF = $t02266122789._4
511541 getPositionNotionalAndUnrealizedPnlByValues(positionSize, positionOpenNotional, qtAstR(), bsAstR(), _option)
512542 }
513543
514544
515545 func calcMarginRatio (_remainMargin,_badDebt,_positionNotional) = divd((_remainMargin - _badDebt), _positionNotional)
516546
517547
518548 func getMarginRatioByOption (_trader,_option) = {
519- let $t02195622067 = getPosition(_trader)
520- let positionSize = $t02195622067._1
521- let positionMargin = $t02195622067._2
522- let pon = $t02195622067._3
523- let positionLstUpdCPF = $t02195622067._4
524- let $t02207322166 = getPositionNotionalAndUnrealizedPnl(_trader, _option)
525- let positionNotional = $t02207322166._1
526- let unrealizedPnl = $t02207322166._2
527- let $t02217122337 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
528- let remainMargin = $t02217122337._1
529- let badDebt = $t02217122337._2
549+ let $t02327023381 = getPosition(_trader)
550+ let positionSize = $t02327023381._1
551+ let positionMargin = $t02327023381._2
552+ let pon = $t02327023381._3
553+ let positionLstUpdCPF = $t02327023381._4
554+ let $t02338723480 = getPositionNotionalAndUnrealizedPnl(_trader, _option)
555+ let positionNotional = $t02338723480._1
556+ let unrealizedPnl = $t02338723480._2
557+ let $t02348523651 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
558+ let remainMargin = $t02348523651._1
559+ let badDebt = $t02348523651._2
530560 calcMarginRatio(remainMargin, badDebt, positionNotional)
531561 }
532562
533563
534564 func getMarginRatio (_trader) = getMarginRatioByOption(_trader, PNL_OPTION_SPOT)
535565
536566
537567 func getPartialLiquidationAmount (_trader,_positionSize) = {
538568 let maximumRatio = vmax(partialLiquidationRatio(), (DECIMAL_UNIT - divd(getMarginRatio(_trader), maintenanceMarginRatio())))
539569 let maxExchangedPositionSize = muld(abs(_positionSize), maximumRatio)
540570 let swapResult = swapOutput((_positionSize > 0), maxExchangedPositionSize, false)
541571 let maxExchangedQuoteAssetAmount = swapResult._1
542572 let priceImpact = swapResult._8
543573 if ((maxPriceImpact() > priceImpact))
544574 then maxExchangedQuoteAssetAmount
545575 else {
546576 let exchangedPositionSize = muld(abs(_positionSize), partialLiquidationRatio())
547577 let exchangedQuoteAssetAmount = swapOutput((_positionSize > 0), exchangedPositionSize, false)._1
548578 exchangedQuoteAssetAmount
549579 }
550580 }
551581
552582
553583 func internalClosePosition (_trader,_checkMaxPriceImpact) = {
554- let $t02357023698 = getPosition(_trader)
555- let positionSize = $t02357023698._1
556- let positionMargin = $t02357023698._2
557- let positionOpenNotional = $t02357023698._3
558- let positionLstUpdCPF = $t02357023698._4
584+ let $t02488425012 = getPosition(_trader)
585+ let positionSize = $t02488425012._1
586+ let positionMargin = $t02488425012._2
587+ let positionOpenNotional = $t02488425012._3
588+ let positionLstUpdCPF = $t02488425012._4
559589 let unrealizedPnl = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)._2
560- let $t02379323961 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
561- let remainMargin = $t02379323961._1
562- let badDebt = $t02379323961._2
590+ let $t02510725275 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
591+ let remainMargin = $t02510725275._1
592+ let badDebt = $t02510725275._2
563593 let exchangedPositionSize = -(positionSize)
564594 let realizedPnl = unrealizedPnl
565595 let marginToVault = -(remainMargin)
566- let $t02408824399 = swapOutput((positionSize > 0), abs(positionSize), _checkMaxPriceImpact)
567- let exchangedQuoteAssetAmount = $t02408824399._1
568- let quoteAssetReserveAfter = $t02408824399._2
569- let baseAssetReserveAfter = $t02408824399._3
570- let totalPositionSizeAfter = $t02408824399._4
571- let cumulativeNotionalAfter = $t02408824399._5
572- let totalLongAfter = $t02408824399._6
573- let totalShortAfter = $t02408824399._7
596+ let $t02540225713 = swapOutput((positionSize > 0), abs(positionSize), _checkMaxPriceImpact)
597+ let exchangedQuoteAssetAmount = $t02540225713._1
598+ let quoteAssetReserveAfter = $t02540225713._2
599+ let baseAssetReserveAfter = $t02540225713._3
600+ let totalPositionSizeAfter = $t02540225713._4
601+ let cumulativeNotionalAfter = $t02540225713._5
602+ let totalLongAfter = $t02540225713._6
603+ let totalShortAfter = $t02540225713._7
574604 let openInterestNotionalAfter = (openInterestNotional() - positionOpenNotional)
575605 $Tuple12(exchangedPositionSize, badDebt, realizedPnl, marginToVault, quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, exchangedQuoteAssetAmount, totalLongAfter, totalShortAfter)
576606 }
577607
578608
579609 func getTwapSpotPrice () = {
580610 let minuteId = ((lastBlock.timestamp / 1000) / 60)
581611 let startMinuteId = (minuteId - TWAP_INTERVAL)
582612 let listStr = valueOrElse(getString(this, k_lastDataStr), "")
583613 let list = split(listStr, ",")
584614 func filterFn (accumulator,next) = if ((startMinuteId >= parseIntValue(next)))
585615 then (accumulator :+ parseIntValue(next))
586616 else accumulator
587617
588618 let listF = {
589619 let $l = list
590620 let $s = size($l)
591621 let $acc0 = nil
592622 func $f0_1 ($a,$i) = if (($i >= $s))
593623 then $a
594624 else filterFn($a, $l[$i])
595625
596626 func $f0_2 ($a,$i) = if (($i >= $s))
597627 then $a
598628 else throw("List size exceeds 20")
599629
600630 $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)
601631 }
602632 let maxIndex = if ((size(listF) > 0))
603633 then max(listF)
604634 else parseIntValue(list[0])
605635 let lastMinuteId = valueOrElse(getInteger(this, k_lastMinuteId), 0)
606636 let endLastCumulativePrice = valueOrElse(getInteger(this, ((k_twapDataLastCumulativePrice + "_") + toString(lastMinuteId))), 0)
607637 let endLastPrice = valueOrElse(getInteger(this, ((k_twapDataLastPrice + "_") + toString(lastMinuteId))), 0)
608638 let nowCumulativePrice = (endLastCumulativePrice + ((minuteId - lastMinuteId) * endLastPrice))
609639 let startLastCumulativePrice = valueOrElse(getInteger(this, ((k_twapDataLastCumulativePrice + "_") + toString(maxIndex))), 0)
610640 let startLastPrice = valueOrElse(getInteger(this, ((k_twapDataLastPrice + "_") + toString(maxIndex))), 0)
611641 let startCumulativePrice = (startLastCumulativePrice + ((startMinuteId - maxIndex) * startLastPrice))
612642 ((nowCumulativePrice - startCumulativePrice) / TWAP_INTERVAL)
613643 }
614644
615645
616646 func getPegAdjustCost (_price) = {
617647 let _positionSize = totalPositionSize()
618648 let direction = (_positionSize > 0)
619649 let currentNetMarketValue = swapOutput(direction, abs(_positionSize), false)._1
620650 let baseAssetReserve = bsAstR()
621651 let newQuoteAssetReserve = muld(baseAssetReserve, _price)
622652 let cost = getPositionNotionalAndUnrealizedPnlByValues(_positionSize, currentNetMarketValue, newQuoteAssetReserve, baseAssetReserve, PNL_OPTION_SPOT)._2
623653 $Tuple2(newQuoteAssetReserve, cost)
624654 }
625655
626656
627657 func getFunding () = {
628658 let underlyingPrice = getOracleTwapPrice()
629659 let spotTwapPrice = getTwapSpotPrice()
630660 let premium = (spotTwapPrice - underlyingPrice)
631661 if (if ((totalShortPositionSize() == 0))
632662 then true
633663 else (totalLongPositionSize() == 0))
634664 then $Tuple2(0, 0)
635665 else if ((0 > premium))
636666 then {
637667 let shortPremiumFraction = divd(muld(premium, fundingPeriodDecimal()), ONE_DAY)
638668 let longPremiumFraction = divd(muld(shortPremiumFraction, totalShortPositionSize()), totalLongPositionSize())
639669 $Tuple2(shortPremiumFraction, longPremiumFraction)
640670 }
641671 else {
642672 let longPremiumFraction = divd(muld(premium, fundingPeriodDecimal()), ONE_DAY)
643673 let shortPremiumFraction = divd(muld(longPremiumFraction, totalLongPositionSize()), totalShortPositionSize())
644674 $Tuple2(shortPremiumFraction, longPremiumFraction)
645675 }
646676 }
647677
648678
649679 func getAdjustedFee (i) = {
650680 let baseFee = fee()
651- let $t02806228670 = if ((size(i.payments) > 1))
681+ let $t02937629984 = if ((size(i.payments) > 1))
652682 then {
653683 let artifactId = toBase58String(valueOrErrorMessage(i.payments[1].assetId, "Invalid attached artifact"))
654684 let artifactKind = strA(nftManagerAddress(), toCompositeKey(k_token_type, artifactId))
655685 if ((artifactKind == FEE_REDUCTION_TOKEN_TYPE))
656686 then {
657687 let reduction = intA(nftManagerAddress(), toCompositeKey(k_token_param, artifactId))
658688 let adjustedFee = muld(baseFee, reduction)
659689 $Tuple2(adjustedFee, true)
660690 }
661691 else throw("Invalid attached artifact")
662692 }
663693 else $Tuple2(baseFee, false)
664- let adjustedFee = $t02806228670._1
665- let burnArtifact = $t02806228670._2
694+ let adjustedFee = $t02937629984._1
695+ let burnArtifact = $t02937629984._2
666696 $Tuple2(adjustedFee, burnArtifact)
667697 }
668698
669699
670700 func updateSettings (_initMarginRatio,_mmr,_liquidationFeeRatio,_fundingPeriod,_fee,_spreadLimit,_maxPriceImpact,_partialLiquidationRatio,_maxPriceSpread) = [IntegerEntry(k_initMarginRatio, _initMarginRatio), IntegerEntry(k_maintenanceMarginRatio, _mmr), IntegerEntry(k_liquidationFeeRatio, _liquidationFeeRatio), IntegerEntry(k_fundingPeriod, _fundingPeriod), IntegerEntry(k_fee, _fee), IntegerEntry(k_spreadLimit, _spreadLimit), IntegerEntry(k_maxPriceImpact, _maxPriceImpact), IntegerEntry(k_partialLiquidationRatio, _partialLiquidationRatio), IntegerEntry(k_maxPriceSpread, _maxPriceSpread)]
671701
672702
673703 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)]
704+
705+
706+func updatePositionAsset (_address,_assetId) = [StringEntry(toCompositeKey(k_positionAsset, _address), _assetId)]
674707
675708
676709 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)]
677710
678711
679712 func appendTwap (price) = {
680713 let minuteId = ((lastBlock.timestamp / 1000) / 60)
681714 let previousMinuteId = valueOrElse(getInteger(this, k_lastMinuteId), 0)
682715 if ((previousMinuteId > minuteId))
683716 then throw("TWAP out-of-order")
684717 else {
685718 let lastMinuteId = if ((previousMinuteId == 0))
686719 then minuteId
687720 else previousMinuteId
688721 if ((minuteId > previousMinuteId))
689722 then {
690723 let prevCumulativePrice = valueOrElse(getInteger(this, ((k_twapDataLastCumulativePrice + "_") + toString(previousMinuteId))), 0)
691724 let prevPrice = valueOrElse(getInteger(this, ((k_twapDataLastPrice + "_") + toString(previousMinuteId))), price)
692725 let lastCumulativePrice = (prevCumulativePrice + ((minuteId - lastMinuteId) * prevPrice))
693726 let list = pushToQueue(strToList(valueOrElse(getString(this, k_lastDataStr), "")), TWAP_INTERVAL, toString(minuteId))
694727 [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))]
695728 }
696729 else {
697730 let twapDataPreviousMinuteId = valueOrElse(getInteger(this, toCompositeKey(k_twapDataPreviousMinuteId, toString(minuteId))), 0)
698731 let prevCumulativePrice = valueOrElse(getInteger(this, toCompositeKey(k_twapDataLastCumulativePrice, toString(twapDataPreviousMinuteId))), 0)
699732 let prevPrice = valueOrElse(getInteger(this, toCompositeKey(k_twapDataLastPrice, toString(twapDataPreviousMinuteId))), price)
700733 let lastCumulativePrice = (prevCumulativePrice + ((minuteId - twapDataPreviousMinuteId) * prevPrice))
701734 [IntegerEntry(toCompositeKey(k_twapDataLastCumulativePrice, toString(minuteId)), lastCumulativePrice), IntegerEntry(toCompositeKey(k_twapDataLastPrice, toString(minuteId)), price)]
702735 }
703736 }
704737 }
705738
706739
707740 func updateAmmReserves (_qtAstR,_bsAstR) = [IntegerEntry(k_quoteAssetReserve, _qtAstR), IntegerEntry(k_baseAssetReserve, _bsAstR)]
708741
709742
710743 func updateAmm (_qtAstR,_bsAstR,_totalPositionSizeAfter,_cumulativeNotionalAfter,_openInterestNotional,_totalLongPositionSize,_totalShortPositionSize) = if (((_totalLongPositionSize - _totalShortPositionSize) != _totalPositionSizeAfter))
711744 then throw(((((("Invalid AMM state data: " + toString(_totalLongPositionSize)) + " + ") + toString(_totalShortPositionSize)) + " != ") + toString(_totalPositionSizeAfter)))
712745 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)))
713746
714747
715-func deletePosition (_address) = [DeleteEntry(toCompositeKey(k_positionSize, _address)), DeleteEntry(toCompositeKey(k_positionMargin, _address)), DeleteEntry(toCompositeKey(k_positionOpenNotional, _address)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _address)), IntegerEntry(toCompositeKey(k_positionClosedDate, _address), lastBlock.timestamp)]
748+func deletePosition (_address) = [DeleteEntry(toCompositeKey(k_positionSize, _address)), DeleteEntry(toCompositeKey(k_positionMargin, _address)), DeleteEntry(toCompositeKey(k_positionOpenNotional, _address)), DeleteEntry(toCompositeKey(k_positionLastUpdatedCumulativePremiumFraction, _address)), DeleteEntry(toCompositeKey(k_positionAsset, _address)), IntegerEntry(toCompositeKey(k_positionClosedDate, _address), lastBlock.timestamp)]
716749
717750
718751 func withdraw (_address,_amount) = {
719752 let balance = assetBalance(this, quoteAsset())
720753 if ((_amount > balance))
721754 then throw(((("Unable to withdraw " + toString(_amount)) + " from contract balance ") + toString(balance)))
722755 else [ScriptTransfer(_address, _amount, quoteAsset())]
723756 }
724757
725758
726759 func updateBalance (i) = if ((0 > i))
727760 then throw("Balance")
728761 else [IntegerEntry(k_balance, i)]
729762
730763
731764 func transferFee (i) = [ScriptTransfer(stakingAddress(), i, quoteAsset())]
732765
733766
734767 func doBurnArtifact (_burnArtifact,i) = if (_burnArtifact)
735768 then [Burn(valueOrErrorMessage(i.payments[1].assetId, "Invalid artifact"), 1)]
736769 else nil
770+
771+
772+func isSameAssetOrNoPosition (_trader,_assetId) = {
773+ let oldPositionSize = getPosition(_trader)._1
774+ if ((oldPositionSize == 0))
775+ then true
776+ else (getPositionAsset(_trader) == _assetId)
777+ }
778+
779+
780+func isSameAsset (_trader,_assetId) = (getPositionAsset(_trader) == _assetId)
781+
782+
783+func getBorrowedByTraderInMarketKey (_amm,_assetId,_trader) = ((((((k_trader_market_asset_collateral + "_") + _amm) + "_") + _assetId) + "_") + _trader)
784+
785+
786+func getBorrowedByTrader (_trader) = {
787+ let positionAsset = getPositionAsset(_trader)
788+ if ((positionAsset == toBase58String(quoteAsset())))
789+ then $Tuple2(0, positionAsset)
790+ else {
791+ let key = getBorrowedByTraderInMarketKey(toString(this), positionAsset, _trader)
792+ let borrow = valueOrElse(getInteger(collateralAddress(), key), 0)
793+ $Tuple2(borrow, positionAsset)
794+ }
795+ }
737796
738797
739798 @Callable(i)
740799 func pause () = if ((i.caller != adminAddress()))
741800 then throw("Invalid togglePause params")
742801 else [BooleanEntry(k_paused, true)]
743802
744803
745804
746805 @Callable(i)
747806 func unpause () = if ((i.caller != adminAddress()))
748807 then throw("Invalid togglePause params")
749808 else [BooleanEntry(k_paused, false)]
750809
751810
752811
753812 @Callable(i)
754813 func addLiquidity (_quoteAssetAmount) = if (if ((i.caller != adminAddress()))
755814 then true
756815 else (0 >= _quoteAssetAmount))
757816 then throw("Invalid addLiquidity params")
758817 else {
759818 let _qtAstR = qtAstR()
760819 let _bsAstR = bsAstR()
761820 let price = divd(_qtAstR, _bsAstR)
762821 let baseAssetAmountToAdd = divd(_quoteAssetAmount, price)
763822 let qtAstRAfter = (_qtAstR + _quoteAssetAmount)
764823 let bsAstRAfter = (_bsAstR + baseAssetAmountToAdd)
765824 updateAmmReserves(qtAstRAfter, bsAstRAfter)
766825 }
767826
768827
769828
770829 @Callable(i)
771830 func removeLiquidity (_quoteAssetAmount) = if (if ((i.caller != adminAddress()))
772831 then true
773832 else (0 >= _quoteAssetAmount))
774833 then throw("Invalid removeLiquidity params")
775834 else {
776835 let _qtAstR = qtAstR()
777836 let _bsAstR = bsAstR()
778837 let price = divd(_qtAstR, _bsAstR)
779838 let baseAssetAmountToRemove = divd(_quoteAssetAmount, price)
780839 let qtAstRAfter = (_qtAstR - _quoteAssetAmount)
781840 let bsAstRAfter = (_bsAstR - baseAssetAmountToRemove)
782841 updateAmmReserves(qtAstRAfter, bsAstRAfter)
783842 }
784843
785844
786845
787846 @Callable(i)
788847 func changeSettings (_initMarginRatio,_mmr,_liquidationFeeRatio,_fundingPeriod,_fee,_spreadLimit,_maxPriceImpact,_partialLiquidationRatio,_maxPriceSpread) = if ((i.caller != adminAddress()))
789848 then throw("Invalid changeSettings params")
790849 else updateSettings(_initMarginRatio, _mmr, _liquidationFeeRatio, _fundingPeriod, _fee, _spreadLimit, _maxPriceImpact, _partialLiquidationRatio, _maxPriceSpread)
791850
792851
793852
794853 @Callable(i)
795854 func initialize (_qtAstR,_bsAstR,_fundingPeriod,_initMarginRatio,_mmr,_liquidationFeeRatio,_fee,_oracle,_oracleKey,_coordinator,_spreadLimit,_maxPriceImpact,_partialLiquidationRatio,_maxPriceSpread) = if (if (if (if (if (if (if (if (if (if (if (if ((0 >= _qtAstR))
796855 then true
797856 else (0 >= _bsAstR))
798857 then true
799858 else (0 >= _fundingPeriod))
800859 then true
801860 else (0 >= _initMarginRatio))
802861 then true
803862 else (0 >= _mmr))
804863 then true
805864 else (0 >= _liquidationFeeRatio))
806865 then true
807866 else (0 >= _fee))
808867 then true
809868 else (0 >= _spreadLimit))
810869 then true
811870 else (0 >= _maxPriceImpact))
812871 then true
813872 else (0 >= _partialLiquidationRatio))
814873 then true
815874 else (0 >= _maxPriceSpread))
816875 then true
817876 else initialized())
818877 then throw("Invalid initialize parameters")
819878 else ((((updateAmm(_qtAstR, _bsAstR, 0, 0, 0, 0, 0) ++ updateSettings(_initMarginRatio, _mmr, _liquidationFeeRatio, _fundingPeriod, _fee, _spreadLimit, _maxPriceImpact, _partialLiquidationRatio, _maxPriceSpread)) ++ 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)])
820879
821880
822881
823882 @Callable(i)
824883 func setInitMarginRatio (_initMarginRatio) = if (if ((0 >= _initMarginRatio))
825884 then true
826885 else !(initialized()))
827886 then throw("Invalid setInitMarginRatio parameters")
828887 else updateSettings(_initMarginRatio, maintenanceMarginRatio(), liquidationFeeRatio(), fundingPeriodRaw(), fee(), spreadLimit(), maxPriceImpact(), partialLiquidationRatio(), maxPriceSpread())
829888
830889
831890
832891 @Callable(i)
833892 func decreasePosition (_amount,_leverage,_minBaseAssetAmount) = if (if (if (if (if ((0 >= _amount))
834893 then true
835894 else !(initialized()))
836895 then true
837896 else !(requireMoreMarginRatio(divd(DECIMAL_UNIT, _leverage), initMarginRatio(), true)))
838897 then true
839898 else !(requireOpenPosition(toString(i.caller))))
840899 then true
841900 else paused())
842901 then throw("Invalid decreasePosition parameters")
843902 else {
844- let $t03908539237 = getPosition(toString(i.caller))
845- let oldPositionSize = $t03908539237._1
846- let oldPositionMargin = $t03908539237._2
847- let oldPositionOpenNotional = $t03908539237._3
848- let oldPositionLstUpdCPF = $t03908539237._4
903+ let $t04149241644 = getPosition(toString(i.caller))
904+ let oldPositionSize = $t04149241644._1
905+ let oldPositionMargin = $t04149241644._2
906+ let oldPositionOpenNotional = $t04149241644._3
907+ let oldPositionLstUpdCPF = $t04149241644._4
849908 let _direction = if ((oldPositionSize > 0))
850909 then DIR_SHORT
851910 else DIR_LONG
852911 let isAdd = (_direction == DIR_LONG)
853912 let openNotional = muld(_amount, _leverage)
854- let $t03941039526 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
855- let oldPositionNotional = $t03941039526._1
856- let unrealizedPnl = $t03941039526._2
857- let $t03953242081 = if ((oldPositionNotional > openNotional))
913+ let $t04181741933 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
914+ let oldPositionNotional = $t04181741933._1
915+ let unrealizedPnl = $t04181741933._2
916+ let $t04193944488 = if ((oldPositionNotional > openNotional))
858917 then {
859- let $t03990940128 = swapInput(isAdd, openNotional)
860- let exchangedPositionSize = $t03990940128._1
861- let quoteAssetReserveAfter = $t03990940128._2
862- let baseAssetReserveAfter = $t03990940128._3
863- let totalPositionSizeAfter = $t03990940128._4
864- let cumulativeNotionalAfter = $t03990940128._5
918+ let $t04231642535 = swapInput(isAdd, openNotional)
919+ let exchangedPositionSize = $t04231642535._1
920+ let quoteAssetReserveAfter = $t04231642535._2
921+ let baseAssetReserveAfter = $t04231642535._3
922+ let totalPositionSizeAfter = $t04231642535._4
923+ let cumulativeNotionalAfter = $t04231642535._5
865924 let exchangedPositionSizeAbs = abs(exchangedPositionSize)
866925 if (if ((_minBaseAssetAmount != 0))
867926 then (_minBaseAssetAmount > exchangedPositionSizeAbs)
868927 else false)
869928 then throw(((("Too little base asset exchanged, got " + toString(exchangedPositionSizeAbs)) + " expected ") + toString(_minBaseAssetAmount)))
870929 else {
871930 let realizedPnl = divd(muld(unrealizedPnl, exchangedPositionSizeAbs), abs(oldPositionSize))
872- let $t04056540810 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
873- let remainMargin = $t04056540810._1
874- let badDebt = $t04056540810._2
875- let fundingPayment = $t04056540810._3
931+ let $t04297243217 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
932+ let remainMargin = $t04297243217._1
933+ let badDebt = $t04297243217._2
934+ let fundingPayment = $t04297243217._3
876935 let exchangedQuoteAssetAmount = openNotional
877936 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
878937 let remainOpenNotional = if ((oldPositionSize > 0))
879938 then ((oldPositionNotional - exchangedQuoteAssetAmount) - unrealizedPnlAfter)
880939 else ((unrealizedPnlAfter + oldPositionNotional) - exchangedQuoteAssetAmount)
881940 let newPositionSize = (oldPositionSize + exchangedPositionSize)
882941 $Tuple11(newPositionSize, remainMargin, abs(remainOpenNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() - openNotional), (totalLongPositionSize() - (if ((newPositionSize > 0))
883942 then abs(exchangedPositionSize)
884943 else 0)), (totalShortPositionSize() - (if ((0 > newPositionSize))
885944 then abs(exchangedPositionSize)
886945 else 0)))
887946 }
888947 }
889948 else throw("Close position first")
890- let newPositionSize = $t03953242081._1
891- let newPositionRemainMargin = $t03953242081._2
892- let newPositionOpenNotional = $t03953242081._3
893- let newPositionLatestCPF = $t03953242081._4
894- let baseAssetReserveAfter = $t03953242081._5
895- let quoteAssetReserveAfter = $t03953242081._6
896- let totalPositionSizeAfter = $t03953242081._7
897- let cumulativeNotionalAfter = $t03953242081._8
898- let openInterestNotionalAfter = $t03953242081._9
899- let totalLongAfter = $t03953242081._10
900- let totalShortAfter = $t03953242081._11
949+ let newPositionSize = $t04193944488._1
950+ let newPositionRemainMargin = $t04193944488._2
951+ let newPositionOpenNotional = $t04193944488._3
952+ let newPositionLatestCPF = $t04193944488._4
953+ let baseAssetReserveAfter = $t04193944488._5
954+ let quoteAssetReserveAfter = $t04193944488._6
955+ let totalPositionSizeAfter = $t04193944488._7
956+ let cumulativeNotionalAfter = $t04193944488._8
957+ let openInterestNotionalAfter = $t04193944488._9
958+ let totalLongAfter = $t04193944488._10
959+ let totalShortAfter = $t04193944488._11
901960 let notifyNotional = invoke(minerAddress(), "notifyNotional", [toString(i.caller), newPositionOpenNotional], nil)
902961 if ((notifyNotional == notifyNotional))
903962 then (updatePosition(toString(i.caller), newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter))
904963 else throw("Strict value is not equal to itself.")
905964 }
906965
907966
908967
909968 @Callable(i)
910969 func increasePosition (_direction,_leverage,_minBaseAssetAmount,_refLink) = {
970+ let _trader = toString(i.caller)
911971 let _rawAmount = i.payments[0].amount
912- if (if (if (if (if (if (if ((_direction != DIR_LONG))
972+ let _assetId = i.payments[0].assetId
973+ let _assetIdStr = toBase58String(value(_assetId))
974+ let isQuoteAsset = (_assetId == quoteAsset())
975+ let isCollateralAsset = isWhitelistAsset(_assetIdStr)
976+ if (if (if (if (if (if (if (if ((_direction != DIR_LONG))
913977 then (_direction != DIR_SHORT)
914978 else false)
915979 then true
916980 else (0 >= _rawAmount))
917981 then true
918982 else !(initialized()))
919983 then true
920- else (i.payments[0].assetId != quoteAsset()))
984+ else if (!(isQuoteAsset))
985+ then !(isCollateralAsset)
986+ else false)
987+ then true
988+ else !(isSameAssetOrNoPosition(_trader, _assetIdStr)))
921989 then true
922990 else !(requireMoreMarginRatio(divd(DECIMAL_UNIT, _leverage), initMarginRatio(), true)))
923991 then true
924992 else paused())
925993 then throw("Invalid increasePosition parameters")
926994 else {
927- let _trader = toString(i.caller)
928995 let baseFee = fee()
929- let $t04319643247 = getAdjustedFee(i)
930- let adjustedFee = $t04319643247._1
931- let burnArtifact = $t04319643247._2
996+ let $t04585045901 = getAdjustedFee(i)
997+ let adjustedFee = $t04585045901._1
998+ let burnArtifact = $t04585045901._2
932999 let rawFeeAmount = muld(_rawAmount, adjustedFee)
933- let referrerFeeAny = invoke(referralAddress(), "acceptPaymentWithLink", [_trader, _refLink], [AttachedPayment(quoteAsset(), rawFeeAmount)])
934- if ((referrerFeeAny == referrerFeeAny))
1000+ let _amount = (_rawAmount - rawFeeAmount)
1001+ let distributeFeeAmount = if (isCollateralAsset)
9351002 then {
936- let referrerFee = match referrerFeeAny {
937- case x: Int =>
938- x
939- case _ =>
940- throw("Invalid referrerFee")
1003+ let doBorrow = invoke(collateralAddress(), "borrow", [_trader], [AttachedPayment(_assetId, _amount)])
1004+ if ((doBorrow == doBorrow))
1005+ then {
1006+ let balanceBefore = assetBalance(this, quoteAsset())
1007+ if ((balanceBefore == balanceBefore))
1008+ then {
1009+ let doSwap = invoke(swapAddress(), "swap", [toBase58String(quoteAsset()), 0], [AttachedPayment(_assetId, rawFeeAmount)])
1010+ if ((doSwap == doSwap))
1011+ then {
1012+ let balanceAfter = assetBalance(this, quoteAsset())
1013+ if ((balanceAfter == balanceAfter))
1014+ then {
1015+ let exchangedAmount = (balanceAfter - balanceBefore)
1016+ if ((exchangedAmount == exchangedAmount))
1017+ then exchangedAmount
1018+ else throw("Strict value is not equal to itself.")
1019+ }
1020+ else throw("Strict value is not equal to itself.")
1021+ }
1022+ else throw("Strict value is not equal to itself.")
1023+ }
1024+ else throw("Strict value is not equal to itself.")
1025+ }
1026+ else throw("Strict value is not equal to itself.")
9411027 }
942- let _amount = (_rawAmount - rawFeeAmount)
943- let feeAmount = (rawFeeAmount - referrerFee)
944- let $t04369643836 = getPosition(_trader)
945- let oldPositionSize = $t04369643836._1
946- let oldPositionMargin = $t04369643836._2
947- let oldPositionOpenNotional = $t04369643836._3
948- let oldPositionLstUpdCPF = $t04369643836._4
949- let isNewPosition = (oldPositionSize == 0)
950- let isSameDirection = if ((oldPositionSize > 0))
951- then (_direction == DIR_LONG)
952- else (_direction == DIR_SHORT)
953- let expandExisting = if (!(isNewPosition))
954- then isSameDirection
955- else false
956- let isAdd = (_direction == DIR_LONG)
957- let $t04412546710 = if (if (isNewPosition)
958- then true
959- else expandExisting)
1028+ else rawFeeAmount
1029+ if ((distributeFeeAmount == distributeFeeAmount))
1030+ then {
1031+ let referrerFeeAny = invoke(referralAddress(), "acceptPaymentWithLink", [_trader, _refLink], [AttachedPayment(quoteAsset(), distributeFeeAmount)])
1032+ if ((referrerFeeAny == referrerFeeAny))
9601033 then {
961- let openNotional = muld(_amount, _leverage)
962- let $t04454944755 = swapInput(isAdd, openNotional)
963- let amountBaseAssetBought = $t04454944755._1
964- let quoteAssetReserveAfter = $t04454944755._2
965- let baseAssetReserveAfter = $t04454944755._3
966- let totalPositionSizeAfter = $t04454944755._4
967- let cumulativeNotionalAfter = $t04454944755._5
968- if (if ((_minBaseAssetAmount != 0))
969- then (_minBaseAssetAmount > abs(amountBaseAssetBought))
970- else false)
971- then throw(((("Limit error: " + toString(abs(amountBaseAssetBought))) + " < ") + toString(_minBaseAssetAmount)))
1034+ let referrerFee = match referrerFeeAny {
1035+ case x: Int =>
1036+ x
1037+ case _ =>
1038+ throw("Invalid referrerFee")
1039+ }
1040+ let feeAmount = (distributeFeeAmount - referrerFee)
1041+ let $t04721747357 = getPosition(_trader)
1042+ let oldPositionSize = $t04721747357._1
1043+ let oldPositionMargin = $t04721747357._2
1044+ let oldPositionOpenNotional = $t04721747357._3
1045+ let oldPositionLstUpdCPF = $t04721747357._4
1046+ let isNewPosition = (oldPositionSize == 0)
1047+ let isSameDirection = if ((oldPositionSize > 0))
1048+ then (_direction == DIR_LONG)
1049+ else (_direction == DIR_SHORT)
1050+ let expandExisting = if (!(isNewPosition))
1051+ then isSameDirection
1052+ else false
1053+ let isAdd = (_direction == DIR_LONG)
1054+ let $t04764650231 = if (if (isNewPosition)
1055+ then true
1056+ else expandExisting)
1057+ then {
1058+ let openNotional = muld(_amount, _leverage)
1059+ let $t04807048276 = swapInput(isAdd, openNotional)
1060+ let amountBaseAssetBought = $t04807048276._1
1061+ let quoteAssetReserveAfter = $t04807048276._2
1062+ let baseAssetReserveAfter = $t04807048276._3
1063+ let totalPositionSizeAfter = $t04807048276._4
1064+ let cumulativeNotionalAfter = $t04807048276._5
1065+ if (if ((_minBaseAssetAmount != 0))
1066+ then (_minBaseAssetAmount > abs(amountBaseAssetBought))
1067+ else false)
1068+ then throw(((("Limit error: " + toString(abs(amountBaseAssetBought))) + " < ") + toString(_minBaseAssetAmount)))
1069+ else {
1070+ let newPositionSize = (oldPositionSize + amountBaseAssetBought)
1071+ let increaseMarginRequirement = divd(openNotional, _leverage)
1072+ let $t04865748896 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
1073+ let remainMargin = $t04865748896._1
1074+ let x1 = $t04865748896._2
1075+ let x2 = $t04865748896._3
1076+ if (!(requireNotOverSpreadLimit(quoteAssetReserveAfter, baseAssetReserveAfter)))
1077+ then throw("Over max spread limit")
1078+ else $Tuple11(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
1079+ then abs(amountBaseAssetBought)
1080+ else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
1081+ then abs(amountBaseAssetBought)
1082+ else 0)))
1083+ }
1084+ }
9721085 else {
973- let newPositionSize = (oldPositionSize + amountBaseAssetBought)
974- let increaseMarginRequirement = divd(openNotional, _leverage)
975- let $t04513645375 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, increaseMarginRequirement)
976- let remainMargin = $t04513645375._1
977- let x1 = $t04513645375._2
978- let x2 = $t04513645375._3
979- if (!(requireNotOverSpreadLimit(quoteAssetReserveAfter, baseAssetReserveAfter)))
980- then throw("Over max spread limit")
981- else $Tuple11(newPositionSize, remainMargin, (oldPositionOpenNotional + openNotional), latestCumulativePremiumFraction(newPositionSize), baseAssetReserveAfter, quoteAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, (openInterestNotional() + openNotional), (totalLongPositionSize() + (if ((newPositionSize > 0))
982- then abs(amountBaseAssetBought)
983- else 0)), (totalShortPositionSize() + (if ((0 > newPositionSize))
984- then abs(amountBaseAssetBought)
985- else 0)))
1086+ let openNotional = muld(_amount, _leverage)
1087+ let $t04992450040 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
1088+ let oldPositionNotional = $t04992450040._1
1089+ let unrealizedPnl = $t04992450040._2
1090+ if ((oldPositionNotional > openNotional))
1091+ then throw("Use decreasePosition to decrease position size")
1092+ else throw("Close position first")
9861093 }
987- }
988- else {
989- let openNotional = muld(_amount, _leverage)
990- let $t04640346519 = getPositionNotionalAndUnrealizedPnl(toString(i.caller), PNL_OPTION_SPOT)
991- let oldPositionNotional = $t04640346519._1
992- let unrealizedPnl = $t04640346519._2
993- if ((oldPositionNotional > openNotional))
994- then throw("Use decreasePosition to decrease position size")
995- else throw("Close position first")
996- }
997- let newPositionSize = $t04412546710._1
998- let newPositionRemainMargin = $t04412546710._2
999- let newPositionOpenNotional = $t04412546710._3
1000- let newPositionLatestCPF = $t04412546710._4
1001- let baseAssetReserveAfter = $t04412546710._5
1002- let quoteAssetReserveAfter = $t04412546710._6
1003- let totalPositionSizeAfter = $t04412546710._7
1004- let cumulativeNotionalAfter = $t04412546710._8
1005- let openInterestNotionalAfter = $t04412546710._9
1006- let totalLongAfter = $t04412546710._10
1007- let totalShortAfter = $t04412546710._11
1008- let feeToStakers = (feeAmount / 2)
1009- let feeToInsurance = (feeAmount - feeToStakers)
1010- let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1011- if ((stake == stake))
1012- then {
1013- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1014- if ((depositInsurance == depositInsurance))
1094+ let newPositionSize = $t04764650231._1
1095+ let newPositionRemainMargin = $t04764650231._2
1096+ let newPositionOpenNotional = $t04764650231._3
1097+ let newPositionLatestCPF = $t04764650231._4
1098+ let baseAssetReserveAfter = $t04764650231._5
1099+ let quoteAssetReserveAfter = $t04764650231._6
1100+ let totalPositionSizeAfter = $t04764650231._7
1101+ let cumulativeNotionalAfter = $t04764650231._8
1102+ let openInterestNotionalAfter = $t04764650231._9
1103+ let totalLongAfter = $t04764650231._10
1104+ let totalShortAfter = $t04764650231._11
1105+ let feeToStakers = (feeAmount / 2)
1106+ let feeToInsurance = (feeAmount - feeToStakers)
1107+ let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1108+ if ((stake == stake))
10151109 then {
1016- let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1017- if ((notifyFee == notifyFee))
1110+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1111+ if ((depositInsurance == depositInsurance))
10181112 then {
1019- let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1020- if ((notifyNotional == notifyNotional))
1021- then ((((updatePosition(_trader, newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1113+ let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1114+ if ((notifyFee == notifyFee))
1115+ then {
1116+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1117+ if ((notifyNotional == notifyNotional))
1118+ then (((((updatePosition(_trader, newPositionSize, newPositionRemainMargin, newPositionOpenNotional, newPositionLatestCPF) ++ updatePositionAsset(_trader, _assetIdStr)) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1119+ else throw("Strict value is not equal to itself.")
1120+ }
10221121 else throw("Strict value is not equal to itself.")
10231122 }
10241123 else throw("Strict value is not equal to itself.")
10251124 }
10261125 else throw("Strict value is not equal to itself.")
10271126 }
10281127 else throw("Strict value is not equal to itself.")
10291128 }
10301129 else throw("Strict value is not equal to itself.")
10311130 }
10321131 }
10331132
10341133
10351134
10361135 @Callable(i)
10371136 func addMargin () = {
1137+ let _trader = toString(i.caller)
10381138 let _rawAmount = i.payments[0].amount
1039- if (if (if (if ((i.payments[0].assetId != quoteAsset()))
1139+ let _assetId = i.payments[0].assetId
1140+ let _assetIdStr = toBase58String(value(_assetId))
1141+ let isQuoteAsset = (_assetId == quoteAsset())
1142+ let isCollateralAsset = isWhitelistAsset(_assetIdStr)
1143+ if (if (if (if (if (if (!(isQuoteAsset))
1144+ then !(isCollateralAsset)
1145+ else false)
10401146 then true
10411147 else !(requireOpenPosition(toString(i.caller))))
1148+ then true
1149+ else !(isSameAsset(_trader, _assetIdStr)))
10421150 then true
10431151 else !(initialized()))
10441152 then true
10451153 else paused())
10461154 then throw("Invalid addMargin parameters")
10471155 else {
1048- let _trader = toString(i.caller)
1049- let $t04817848229 = getAdjustedFee(i)
1050- let adjustedFee = $t04817848229._1
1051- let burnArtifact = $t04817848229._2
1156+ let $t05198752038 = getAdjustedFee(i)
1157+ let adjustedFee = $t05198752038._1
1158+ let burnArtifact = $t05198752038._2
10521159 let rawFeeAmount = muld(_rawAmount, adjustedFee)
1053- let referrerFeeAny = invoke(referralAddress(), "acceptPayment", [_trader], [AttachedPayment(quoteAsset(), rawFeeAmount)])
1054- if ((referrerFeeAny == referrerFeeAny))
1160+ let _amount = (_rawAmount - rawFeeAmount)
1161+ let distributeFeeAmount = if (isCollateralAsset)
10551162 then {
1056- let referrerFee = match referrerFeeAny {
1057- case x: Int =>
1058- x
1059- case _ =>
1060- throw("Invalid referrerFee")
1163+ let doBorrow = invoke(collateralAddress(), "borrow", [_trader], [AttachedPayment(_assetId, _amount)])
1164+ if ((doBorrow == doBorrow))
1165+ then {
1166+ let balanceBefore = assetBalance(this, quoteAsset())
1167+ if ((balanceBefore == balanceBefore))
1168+ then {
1169+ let doSwap = invoke(swapAddress(), "swap", [toBase58String(quoteAsset()), 0], [AttachedPayment(_assetId, rawFeeAmount)])
1170+ if ((doSwap == doSwap))
1171+ then {
1172+ let balanceAfter = assetBalance(this, quoteAsset())
1173+ if ((balanceAfter == balanceAfter))
1174+ then {
1175+ let exchangedAmount = (balanceAfter - balanceBefore)
1176+ if ((exchangedAmount == exchangedAmount))
1177+ then exchangedAmount
1178+ else throw("Strict value is not equal to itself.")
1179+ }
1180+ else throw("Strict value is not equal to itself.")
1181+ }
1182+ else throw("Strict value is not equal to itself.")
1183+ }
1184+ else throw("Strict value is not equal to itself.")
1185+ }
1186+ else throw("Strict value is not equal to itself.")
10611187 }
1062- let feeAmount = (rawFeeAmount - referrerFee)
1063- let _amount = (_rawAmount - rawFeeAmount)
1064- let $t04866148801 = getPosition(_trader)
1065- let oldPositionSize = $t04866148801._1
1066- let oldPositionMargin = $t04866148801._2
1067- let oldPositionOpenNotional = $t04866148801._3
1068- let oldPositionLstUpdCPF = $t04866148801._4
1069- let feeToStakers = (feeAmount / 2)
1070- let feeToInsurance = (feeAmount - feeToStakers)
1071- let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1072- if ((stake == stake))
1188+ else rawFeeAmount
1189+ if ((distributeFeeAmount == distributeFeeAmount))
1190+ then {
1191+ let referrerFeeAny = invoke(referralAddress(), "acceptPayment", [_trader], [AttachedPayment(quoteAsset(), distributeFeeAmount)])
1192+ if ((referrerFeeAny == referrerFeeAny))
10731193 then {
1074- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1075- if ((depositInsurance == depositInsurance))
1194+ let referrerFee = match referrerFeeAny {
1195+ case x: Int =>
1196+ x
1197+ case _ =>
1198+ throw("Invalid referrerFee")
1199+ }
1200+ let feeAmount = (distributeFeeAmount - referrerFee)
1201+ let $t05333753477 = getPosition(_trader)
1202+ let oldPositionSize = $t05333753477._1
1203+ let oldPositionMargin = $t05333753477._2
1204+ let oldPositionOpenNotional = $t05333753477._3
1205+ let oldPositionLstUpdCPF = $t05333753477._4
1206+ let feeToStakers = (feeAmount / 2)
1207+ let feeToInsurance = (feeAmount - feeToStakers)
1208+ let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _amount)])
1209+ if ((stake == stake))
10761210 then {
1077- let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1078- if ((notifyFee == notifyFee))
1079- then (((updatePosition(_trader, oldPositionSize, (oldPositionMargin + _amount), oldPositionOpenNotional, oldPositionLstUpdCPF) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1211+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1212+ if ((depositInsurance == depositInsurance))
1213+ then {
1214+ let notifyFee = invoke(minerAddress(), "notifyFees", [_trader, feeAmount], nil)
1215+ if ((notifyFee == notifyFee))
1216+ then (((updatePosition(_trader, oldPositionSize, (oldPositionMargin + _amount), oldPositionOpenNotional, oldPositionLstUpdCPF) ++ transferFee(feeToStakers)) ++ updateBalance((cbalance() + _amount))) ++ doBurnArtifact(burnArtifact, i))
1217+ else throw("Strict value is not equal to itself.")
1218+ }
10801219 else throw("Strict value is not equal to itself.")
10811220 }
10821221 else throw("Strict value is not equal to itself.")
10831222 }
10841223 else throw("Strict value is not equal to itself.")
10851224 }
10861225 else throw("Strict value is not equal to itself.")
10871226 }
10881227 }
10891228
10901229
10911230
10921231 @Callable(i)
1093-func removeMargin (_amount) = if (if (if (if ((0 >= _amount))
1094- then true
1095- else !(requireOpenPosition(toString(i.caller))))
1096- then true
1097- else !(initialized()))
1098- then true
1099- else paused())
1100- then throw("Invalid removeMargin parameters")
1101- else {
1102- let $t04982049972 = getPosition(toString(i.caller))
1103- let oldPositionSize = $t04982049972._1
1104- let oldPositionMargin = $t04982049972._2
1105- let oldPositionOpenNotional = $t04982049972._3
1106- let oldPositionLstUpdCPF = $t04982049972._4
1107- let marginDelta = -(_amount)
1108- let $t05000950188 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
1109- let remainMargin = $t05000950188._1
1110- let badDebt = $t05000950188._2
1111- if ((badDebt != 0))
1112- then throw("Invalid removed margin amount")
1113- else {
1114- let marginRatio = calcMarginRatio(remainMargin, badDebt, oldPositionOpenNotional)
1115- if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
1116- then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
1117- else {
1118- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), _amount], nil)
1119- if ((unstake == unstake))
1120- then ((updatePosition(toString(i.caller), oldPositionSize, remainMargin, oldPositionOpenNotional, latestCumulativePremiumFraction(oldPositionSize)) ++ withdraw(i.caller, _amount)) ++ updateBalance((cbalance() - _amount)))
1121- else throw("Strict value is not equal to itself.")
1122- }
1123- }
1124- }
1232+func removeMargin (_amount) = {
1233+ let _trader = toString(i.caller)
1234+ if (if (if (if ((0 >= _amount))
1235+ then true
1236+ else !(requireOpenPosition(_trader)))
1237+ then true
1238+ else !(initialized()))
1239+ then true
1240+ else paused())
1241+ then throw("Invalid removeMargin parameters")
1242+ else {
1243+ let $t05452054660 = getPosition(_trader)
1244+ let oldPositionSize = $t05452054660._1
1245+ let oldPositionMargin = $t05452054660._2
1246+ let oldPositionOpenNotional = $t05452054660._3
1247+ let oldPositionLstUpdCPF = $t05452054660._4
1248+ let marginDelta = -(_amount)
1249+ let $t05469754876 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, marginDelta)
1250+ let remainMargin = $t05469754876._1
1251+ let badDebt = $t05469754876._2
1252+ if ((badDebt != 0))
1253+ then throw("Invalid removed margin amount")
1254+ else {
1255+ let marginRatio = calcMarginRatio(remainMargin, badDebt, oldPositionOpenNotional)
1256+ if (!(requireMoreMarginRatio(marginRatio, initMarginRatio(), true)))
1257+ then throw(((("Too much margin removed: " + toString(marginRatio)) + " < ") + toString(initMarginRatio())))
1258+ else {
1259+ let quoteAssetStr = toBase58String(quoteAsset())
1260+ let $t05532055374 = getBorrowedByTrader(_trader)
1261+ let borrowed = $t05532055374._1
1262+ let assetId = $t05532055374._2
1263+ let toRepay = if ((_amount > borrowed))
1264+ then borrowed
1265+ else _amount
1266+ let toWithdraw = if ((borrowed > _amount))
1267+ then 0
1268+ else (_amount - borrowed)
1269+ let finalBorrow = (borrowed - toRepay)
1270+ let switchPositionToQuote = if ((finalBorrow > 0))
1271+ then nil
1272+ else updatePositionAsset(_trader, quoteAssetStr)
1273+ let doSanityCheck = if (((toRepay + toWithdraw) != _amount))
1274+ then throw(((((("toRepay=" + toString(toRepay)) + " + toWithdraw=") + toString(toWithdraw)) + " != ") + toString(_amount)))
1275+ else nil
1276+ if ((doSanityCheck == doSanityCheck))
1277+ then {
1278+ let doUnstake = invoke(managerAddress(), "withdraw", [quoteAssetStr, _amount], nil)
1279+ if ((doUnstake == doUnstake))
1280+ then {
1281+ let returnCollateralAction = if ((toRepay > 0))
1282+ then {
1283+ let doRepay = invoke(collateralAddress(), "repay", [_trader, assetId], [AttachedPayment(quoteAsset(), toRepay)])
1284+ if ((doRepay == doRepay))
1285+ then [ScriptTransfer(i.caller, toRepay, fromBase58String(assetId))]
1286+ else throw("Strict value is not equal to itself.")
1287+ }
1288+ else nil
1289+ if ((returnCollateralAction == returnCollateralAction))
1290+ then (updatePosition(_trader, oldPositionSize, remainMargin, oldPositionOpenNotional, latestCumulativePremiumFraction(oldPositionSize)) ++ (if ((toWithdraw > 0))
1291+ then withdraw(i.caller, toWithdraw)
1292+ else (((nil ++ updateBalance((cbalance() - _amount))) ++ switchPositionToQuote) ++ returnCollateralAction)))
1293+ else throw("Strict value is not equal to itself.")
1294+ }
1295+ else throw("Strict value is not equal to itself.")
1296+ }
1297+ else throw("Strict value is not equal to itself.")
1298+ }
1299+ }
1300+ }
1301+ }
11251302
11261303
11271304
11281305 @Callable(i)
11291306 func closePosition () = {
1130- let caller = getActualCaller(i)
1131- let callerAddress = valueOrErrorMessage(addressFromString(caller), "Invalid caller")
1132- if (if (if (!(requireOpenPosition(caller)))
1307+ let _trader = getActualCaller(i)
1308+ let _traderAddress = valueOrErrorMessage(addressFromString(_trader), "Invalid caller")
1309+ if (if (if (!(requireOpenPosition(_trader)))
11331310 then true
11341311 else !(initialized()))
11351312 then true
11361313 else paused())
11371314 then throw("Invalid closePosition parameters")
11381315 else {
1139- let $t05138251759 = internalClosePosition(caller, true)
1140- let x1 = $t05138251759._1
1141- let positionBadDebt = $t05138251759._2
1142- let realizedPnl = $t05138251759._3
1143- let marginToVault = $t05138251759._4
1144- let quoteAssetReserveAfter = $t05138251759._5
1145- let baseAssetReserveAfter = $t05138251759._6
1146- let totalPositionSizeAfter = $t05138251759._7
1147- let cumulativeNotionalAfter = $t05138251759._8
1148- let openInterestNotionalAfter = $t05138251759._9
1149- let x2 = $t05138251759._10
1150- let totalLongAfter = $t05138251759._11
1151- let totalShortAfter = $t05138251759._12
1316+ let $t05756657944 = internalClosePosition(_trader, true)
1317+ let x1 = $t05756657944._1
1318+ let positionBadDebt = $t05756657944._2
1319+ let realizedPnl = $t05756657944._3
1320+ let marginToVault = $t05756657944._4
1321+ let quoteAssetReserveAfter = $t05756657944._5
1322+ let baseAssetReserveAfter = $t05756657944._6
1323+ let totalPositionSizeAfter = $t05756657944._7
1324+ let cumulativeNotionalAfter = $t05756657944._8
1325+ let openInterestNotionalAfter = $t05756657944._9
1326+ let x2 = $t05756657944._10
1327+ let totalLongAfter = $t05756657944._11
1328+ let totalShortAfter = $t05756657944._12
11521329 if ((positionBadDebt > 0))
11531330 then throw("Unable to close position with bad debt")
11541331 else {
11551332 let withdrawAmount = abs(marginToVault)
11561333 let ammBalance = (cbalance() - withdrawAmount)
1157- let $t05196852110 = if ((0 > ammBalance))
1334+ let $t05815358295 = if ((0 > ammBalance))
11581335 then $Tuple2(0, abs(ammBalance))
11591336 else $Tuple2(ammBalance, 0)
1160- let ammNewBalance = $t05196852110._1
1161- let getFromInsurance = $t05196852110._2
1337+ let ammNewBalance = $t05815358295._1
1338+ let getFromInsurance = $t05815358295._2
11621339 let x = if ((getFromInsurance > 0))
11631340 then {
11641341 let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [getFromInsurance], nil)
11651342 if ((withdrawInsurance == withdrawInsurance))
11661343 then nil
11671344 else throw("Strict value is not equal to itself.")
11681345 }
11691346 else nil
11701347 if ((x == x))
11711348 then {
11721349 let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (withdrawAmount - getFromInsurance)], nil)
11731350 if ((unstake == unstake))
11741351 then {
1175- let notifyNotional = invoke(minerAddress(), "notifyNotional", [caller, 0], nil)
1176- if ((notifyNotional == notifyNotional))
1177- then (((deletePosition(caller) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(callerAddress, withdrawAmount)) ++ updateBalance(ammNewBalance))
1352+ let $t05879758851 = getBorrowedByTrader(_trader)
1353+ let borrowed = $t05879758851._1
1354+ let assetId = $t05879758851._2
1355+ let $t05886659783 = if ((borrowed > 0))
1356+ then if ((withdrawAmount >= borrowed))
1357+ then {
1358+ let doRepay = invoke(collateralAddress(), "repay", [_trader, assetId], [AttachedPayment(quoteAsset(), borrowed)])
1359+ if ((doRepay == doRepay))
1360+ then $Tuple2([ScriptTransfer(_traderAddress, borrowed, fromBase58String(assetId))], (withdrawAmount - borrowed))
1361+ else throw("Strict value is not equal to itself.")
1362+ }
1363+ else {
1364+ let realizeAndClose = invoke(collateralAddress(), "realizePartiallyAndClose", [_trader, assetId], [AttachedPayment(quoteAsset(), withdrawAmount)])
1365+ if ((realizeAndClose == realizeAndClose))
1366+ then $Tuple2([ScriptTransfer(_traderAddress, withdrawAmount, fromBase58String(assetId))], 0)
1367+ else throw("Strict value is not equal to itself.")
1368+ }
1369+ else $Tuple2(nil, withdrawAmount)
1370+ if (($t05886659783 == $t05886659783))
1371+ then {
1372+ let quoteWithdrawAmount = $t05886659783._2
1373+ let sendCollateralAction = $t05886659783._1
1374+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
1375+ if ((notifyNotional == notifyNotional))
1376+ then ((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ (if ((quoteWithdrawAmount > 0))
1377+ then withdraw(_traderAddress, quoteWithdrawAmount)
1378+ else ((nil ++ updateBalance(ammNewBalance)) ++ sendCollateralAction)))
1379+ else throw("Strict value is not equal to itself.")
1380+ }
11781381 else throw("Strict value is not equal to itself.")
11791382 }
11801383 else throw("Strict value is not equal to itself.")
11811384 }
11821385 else throw("Strict value is not equal to itself.")
11831386 }
11841387 }
11851388 }
11861389
11871390
11881391
11891392 @Callable(i)
11901393 func liquidate (_trader) = {
11911394 let spotMarginRatio = getMarginRatioByOption(_trader, PNL_OPTION_SPOT)
11921395 let marginRatio = if (isOverFluctuationLimit())
11931396 then {
11941397 let oracleMarginRatio = getMarginRatioByOption(_trader, PNL_OPTION_ORACLE)
11951398 vmax(spotMarginRatio, oracleMarginRatio)
11961399 }
11971400 else spotMarginRatio
11981401 if (if (if (if (!(requireMoreMarginRatio(marginRatio, maintenanceMarginRatio(), false)))
11991402 then true
12001403 else !(requireOpenPosition(_trader)))
12011404 then true
12021405 else !(initialized()))
12031406 then true
12041407 else paused())
12051408 then throw("Unable to liquidate")
12061409 else if (if (if ((spotMarginRatio > liquidationFeeRatio()))
12071410 then (partialLiquidationRatio() > 0)
12081411 else false)
12091412 then (DECIMAL_UNIT > partialLiquidationRatio())
12101413 else false)
12111414 then {
1212- let $t05418154331 = getPosition(_trader)
1213- let oldPositionSize = $t05418154331._1
1214- let oldPositionMargin = $t05418154331._2
1215- let oldPositionOpenNotional = $t05418154331._3
1216- let oldPositionLstUpdCPF = $t05418154331._4
1415+ let $t06148361633 = getPosition(_trader)
1416+ let oldPositionSize = $t06148361633._1
1417+ let oldPositionMargin = $t06148361633._2
1418+ let oldPositionOpenNotional = $t06148361633._3
1419+ let oldPositionLstUpdCPF = $t06148361633._4
12171420 let _direction = if ((oldPositionSize > 0))
12181421 then DIR_SHORT
12191422 else DIR_LONG
12201423 let isAdd = (_direction == DIR_LONG)
12211424 let exchangedQuoteAssetAmount = getPartialLiquidationAmount(_trader, oldPositionSize)
1222- let $t05455654660 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1223- let oldPositionNotional = $t05455654660._1
1224- let unrealizedPnl = $t05455654660._2
1225- let $t05466854958 = swapInput(isAdd, exchangedQuoteAssetAmount)
1226- let exchangedPositionSize = $t05466854958._1
1227- let quoteAssetReserveAfter = $t05466854958._2
1228- let baseAssetReserveAfter = $t05466854958._3
1229- let totalPositionSizeAfter = $t05466854958._4
1230- let cumulativeNotionalAfter = $t05466854958._5
1231- let realizedPnl = divd(muld(unrealizedPnl, abs(exchangedPositionSize)), abs(oldPositionSize))
1232- let $t05506555298 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
1233- let remainMargin = $t05506555298._1
1234- let badDebt = $t05506555298._2
1235- let fundingPayment = $t05506555298._3
1425+ let $t06185861962 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1426+ let oldPositionNotional = $t06185861962._1
1427+ let unrealizedPnl = $t06185861962._2
1428+ let $t06197062202 = swapInput(isAdd, exchangedQuoteAssetAmount)
1429+ let exchangedPositionSize = $t06197062202._1
1430+ let quoteAssetReserveAfter = $t06197062202._2
1431+ let baseAssetReserveAfter = $t06197062202._3
1432+ let totalPositionSizeAfter = $t06197062202._4
1433+ let cumulativeNotionalAfter = $t06197062202._5
1434+ let liquidationRatio = divd(abs(exchangedPositionSize), abs(oldPositionSize))
1435+ let realizedPnl = muld(unrealizedPnl, liquidationRatio)
1436+ let $t06241362646 = calcRemainMarginWithFundingPayment(oldPositionSize, oldPositionMargin, oldPositionLstUpdCPF, realizedPnl)
1437+ let remainMargin = $t06241362646._1
1438+ let badDebt = $t06241362646._2
1439+ let fundingPayment = $t06241362646._3
12361440 let unrealizedPnlAfter = (unrealizedPnl - realizedPnl)
12371441 let remainOpenNotional = if ((oldPositionSize > 0))
12381442 then ((oldPositionNotional - exchangedQuoteAssetAmount) - unrealizedPnlAfter)
12391443 else ((unrealizedPnlAfter + oldPositionNotional) - exchangedQuoteAssetAmount)
12401444 let liquidationPenalty = muld(exchangedQuoteAssetAmount, liquidationFeeRatio())
12411445 let feeToLiquidator = (liquidationPenalty / 2)
12421446 let feeToInsurance = (liquidationPenalty - feeToLiquidator)
12431447 let newPositionMargin = (remainMargin - liquidationPenalty)
12441448 let newPositionSize = (oldPositionSize + exchangedPositionSize)
12451449 let newPositionOpenNotional = abs(remainOpenNotional)
12461450 let newPositionLstUpdCPF = latestCumulativePremiumFraction(newPositionSize)
12471451 let openInterestNotionalAfter = (openInterestNotional() - exchangedQuoteAssetAmount)
12481452 let ammBalance = (cbalance() - liquidationPenalty)
1249- let $t05647156614 = if ((0 > ammBalance))
1453+ let $t06381963962 = if ((0 > ammBalance))
12501454 then $Tuple2(0, abs(ammBalance))
12511455 else $Tuple2(ammBalance, 0)
1252- let newAmmBalance = $t05647156614._1
1253- let takeFromInsurance = $t05647156614._2
1254- let x = if ((takeFromInsurance > 0))
1456+ let newAmmBalance = $t06381963962._1
1457+ let takeFromInsurance = $t06381963962._2
1458+ let $t06397064024 = getBorrowedByTrader(_trader)
1459+ let borrowed = $t06397064024._1
1460+ let assetId = $t06397064024._2
1461+ let doLiquidateCollateral = if ((borrowed > 0))
12551462 then {
1256- let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1257- if ((withdrawInsurance == withdrawInsurance))
1463+ let collateralToSell = muld(borrowed, liquidationRatio)
1464+ let realizeAndClose = invoke(collateralAddress(), "realizePartially", [_trader, assetId, collateralToSell], nil)
1465+ if ((realizeAndClose == realizeAndClose))
12581466 then nil
12591467 else throw("Strict value is not equal to itself.")
12601468 }
12611469 else nil
1262- if ((x == x))
1470+ if ((doLiquidateCollateral == doLiquidateCollateral))
12631471 then {
1264- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1265- if ((unstake == unstake))
1472+ let x = if ((takeFromInsurance > 0))
12661473 then {
1267- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1268- if ((depositInsurance == depositInsurance))
1474+ let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1475+ if ((withdrawInsurance == withdrawInsurance))
1476+ then nil
1477+ else throw("Strict value is not equal to itself.")
1478+ }
1479+ else nil
1480+ if ((x == x))
1481+ then {
1482+ let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1483+ if ((unstake == unstake))
12691484 then {
1270- let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1271- if ((notifyNotional == notifyNotional))
1272- then (((updatePosition(_trader, newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, (totalLongPositionSize() - (if ((newPositionSize > 0))
1273- then abs(exchangedPositionSize)
1274- else 0)), (totalShortPositionSize() - (if ((0 > newPositionSize))
1275- then abs(exchangedPositionSize)
1276- else 0)))) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1485+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1486+ if ((depositInsurance == depositInsurance))
1487+ then {
1488+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, newPositionOpenNotional], nil)
1489+ if ((notifyNotional == notifyNotional))
1490+ then (((updatePosition(_trader, newPositionSize, newPositionMargin, newPositionOpenNotional, newPositionLstUpdCPF) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, (totalLongPositionSize() - (if ((newPositionSize > 0))
1491+ then abs(exchangedPositionSize)
1492+ else 0)), (totalShortPositionSize() - (if ((0 > newPositionSize))
1493+ then abs(exchangedPositionSize)
1494+ else 0)))) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1495+ else throw("Strict value is not equal to itself.")
1496+ }
12771497 else throw("Strict value is not equal to itself.")
12781498 }
12791499 else throw("Strict value is not equal to itself.")
12801500 }
12811501 else throw("Strict value is not equal to itself.")
12821502 }
12831503 else throw("Strict value is not equal to itself.")
12841504 }
12851505 else {
1286- let $t05806158516 = internalClosePosition(_trader, false)
1287- let x1 = $t05806158516._1
1288- let badDebt = $t05806158516._2
1289- let x2 = $t05806158516._3
1290- let x3 = $t05806158516._4
1291- let quoteAssetReserveAfter = $t05806158516._5
1292- let baseAssetReserveAfter = $t05806158516._6
1293- let totalPositionSizeAfter = $t05806158516._7
1294- let cumulativeNotionalAfter = $t05806158516._8
1295- let openInterestNotionalAfter = $t05806158516._9
1296- let exchangedQuoteAssetAmount = $t05806158516._10
1297- let totalLongAfter = $t05806158516._11
1298- let totalShortAfter = $t05806158516._12
1506+ let $t06586666321 = internalClosePosition(_trader, false)
1507+ let x1 = $t06586666321._1
1508+ let badDebt = $t06586666321._2
1509+ let x2 = $t06586666321._3
1510+ let x3 = $t06586666321._4
1511+ let quoteAssetReserveAfter = $t06586666321._5
1512+ let baseAssetReserveAfter = $t06586666321._6
1513+ let totalPositionSizeAfter = $t06586666321._7
1514+ let cumulativeNotionalAfter = $t06586666321._8
1515+ let openInterestNotionalAfter = $t06586666321._9
1516+ let exchangedQuoteAssetAmount = $t06586666321._10
1517+ let totalLongAfter = $t06586666321._11
1518+ let totalShortAfter = $t06586666321._12
12991519 let liquidationPenalty = muld(exchangedQuoteAssetAmount, liquidationFeeRatio())
13001520 let feeToLiquidator = (liquidationPenalty / 2)
13011521 let feeToInsurance = (liquidationPenalty - feeToLiquidator)
13021522 let ammBalance = (cbalance() - liquidationPenalty)
1303- let $t05892859071 = if ((0 > ammBalance))
1523+ let $t06673366876 = if ((0 > ammBalance))
13041524 then $Tuple2(0, abs(ammBalance))
13051525 else $Tuple2(ammBalance, 0)
1306- let newAmmBalance = $t05892859071._1
1307- let takeFromInsurance = $t05892859071._2
1308- let x = if ((takeFromInsurance > 0))
1526+ let newAmmBalance = $t06673366876._1
1527+ let takeFromInsurance = $t06673366876._2
1528+ let $t06688466938 = getBorrowedByTrader(_trader)
1529+ let borrowed = $t06688466938._1
1530+ let assetId = $t06688466938._2
1531+ let doLiquidateCollateral = if ((borrowed > 0))
13091532 then {
1310- let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1311- if ((withdrawInsurance == withdrawInsurance))
1533+ let realizeAndClose = invoke(collateralAddress(), "realizePartiallyAndClose", [_trader, assetId], nil)
1534+ if ((realizeAndClose == realizeAndClose))
13121535 then nil
13131536 else throw("Strict value is not equal to itself.")
13141537 }
13151538 else nil
1316- if ((x == x))
1539+ if ((doLiquidateCollateral == doLiquidateCollateral))
13171540 then {
1318- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1319- if ((unstake == unstake))
1541+ let x = if ((takeFromInsurance > 0))
13201542 then {
1321- let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1322- if ((depositInsurance == depositInsurance))
1543+ let withdrawInsurance = invoke(insuranceAddress(), "withdraw", [takeFromInsurance], nil)
1544+ if ((withdrawInsurance == withdrawInsurance))
1545+ then nil
1546+ else throw("Strict value is not equal to itself.")
1547+ }
1548+ else nil
1549+ if ((x == x))
1550+ then {
1551+ let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), (liquidationPenalty - takeFromInsurance)], nil)
1552+ if ((unstake == unstake))
13231553 then {
1324- let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
1325- if ((notifyNotional == notifyNotional))
1326- then (((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1554+ let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), feeToInsurance)])
1555+ if ((depositInsurance == depositInsurance))
1556+ then {
1557+ let notifyNotional = invoke(minerAddress(), "notifyNotional", [_trader, 0], nil)
1558+ if ((notifyNotional == notifyNotional))
1559+ then (((deletePosition(_trader) ++ updateAmm(quoteAssetReserveAfter, baseAssetReserveAfter, totalPositionSizeAfter, cumulativeNotionalAfter, openInterestNotionalAfter, totalLongAfter, totalShortAfter)) ++ withdraw(i.caller, feeToLiquidator)) ++ updateBalance(newAmmBalance))
1560+ else throw("Strict value is not equal to itself.")
1561+ }
13271562 else throw("Strict value is not equal to itself.")
13281563 }
13291564 else throw("Strict value is not equal to itself.")
13301565 }
13311566 else throw("Strict value is not equal to itself.")
13321567 }
13331568 else throw("Strict value is not equal to itself.")
13341569 }
13351570 }
13361571
13371572
13381573
13391574 @Callable(i)
13401575 func payFunding () = {
13411576 let fundingBlockTimestamp = nextFundingBlockTimestamp()
13421577 if (if (if ((fundingBlockTimestamp > lastBlock.timestamp))
13431578 then true
13441579 else !(initialized()))
13451580 then true
13461581 else paused())
13471582 then throw(((("Invalid funding block timestamp: " + toString(lastBlock.timestamp)) + " < ") + toString(fundingBlockTimestamp)))
13481583 else {
13491584 let underlyingPrice = getOracleTwapPrice()
1350- let $t06055760619 = getFunding()
1351- let shortPremiumFraction = $t06055760619._1
1352- let longPremiumFraction = $t06055760619._2
1585+ let $t06873668798 = getFunding()
1586+ let shortPremiumFraction = $t06873668798._1
1587+ let longPremiumFraction = $t06873668798._2
13531588 updateFunding((fundingBlockTimestamp + fundingPeriodSeconds()), (latestLongCumulativePremiumFraction() + longPremiumFraction), (latestShortCumulativePremiumFraction() + shortPremiumFraction), divd(longPremiumFraction, underlyingPrice), divd(shortPremiumFraction, underlyingPrice))
13541589 }
13551590 }
13561591
13571592
13581593
13591594 @Callable(i)
13601595 func forceMoveAsset (_trader,_amount) = if (if ((addressFromPublicKey(adminPublicKey()) != i.caller))
13611596 then true
13621597 else (0 > _amount))
13631598 then throw("Invalid forceMoveAsset parameters")
13641599 else {
13651600 let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), _amount], nil)
13661601 if ((unstake == unstake))
13671602 then (withdraw(addressFromStringValue(_trader), _amount) ++ updateBalance((cbalance() - _amount)))
13681603 else throw("Strict value is not equal to itself.")
13691604 }
13701605
13711606
13721607
13731608 @Callable(i)
13741609 func adjustPeg (_price) = if (if ((addressFromPublicKey(adminPublicKey()) != i.caller))
13751610 then true
13761611 else (0 > _price))
13771612 then throw("Invalid adjustPeg parameters")
13781613 else {
1379- let $t06161061696 = getPegAdjustCost(_price)
1380- let newQuoteAssetReserve = $t06161061696._1
1381- let pegChangeCost = $t06161061696._2
1614+ let $t06978969875 = getPegAdjustCost(_price)
1615+ let newQuoteAssetReserve = $t06978969875._1
1616+ let pegChangeCost = $t06978969875._2
13821617 if ((pegChangeCost == 0))
13831618 then throw("Nothing to adjust")
13841619 else if ((pegChangeCost > 0))
13851620 then updateAmmReserves(newQuoteAssetReserve, bsAstR())
13861621 else {
13871622 let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), abs(pegChangeCost)], nil)
13881623 if ((unstake == unstake))
13891624 then {
13901625 let depositInsurance = invoke(insuranceAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), abs(pegChangeCost))])
13911626 if ((depositInsurance == depositInsurance))
13921627 then (updateBalance((cbalance() - abs(pegChangeCost))) ++ updateAmmReserves(newQuoteAssetReserve, bsAstR()))
13931628 else throw("Strict value is not equal to itself.")
13941629 }
13951630 else throw("Strict value is not equal to itself.")
13961631 }
13971632 }
13981633
13991634
14001635
14011636 @Callable(i)
14021637 func migrateLiquidity () = {
14031638 let amount = cbalance()
14041639 if ((amount > 0))
14051640 then {
14061641 let unstake = invoke(quoteAssetStaking(), "unlockNeutrino", [amount, toBase58String(quoteAsset())], nil)
14071642 if ((unstake == unstake))
14081643 then {
14091644 let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), amount)])
14101645 if ((stake == stake))
14111646 then nil
14121647 else throw("Strict value is not equal to itself.")
14131648 }
14141649 else throw("Strict value is not equal to itself.")
14151650 }
14161651 else nil
14171652 }
14181653
14191654
14201655
14211656 @Callable(i)
14221657 func v_get (_trader) = {
1423- let $t06333763397 = internalClosePosition(_trader, false)
1424- let x1 = $t06333763397._1
1425- let x2 = $t06333763397._2
1426- let x3 = $t06333763397._3
1427- let x4 = $t06333763397._4
1658+ let $t07151671576 = internalClosePosition(_trader, false)
1659+ let x1 = $t07151671576._1
1660+ let x2 = $t07151671576._2
1661+ let x3 = $t07151671576._3
1662+ let x4 = $t07151671576._4
14281663 throw((((s(x2) + s(x3)) + s(x4)) + s(getMarginRatio(_trader))))
14291664 }
14301665
14311666
14321667
14331668 @Callable(i)
14341669 func view_calcRemainMarginWithFundingPayment (_trader) = {
1435- let $t06354463655 = getPosition(_trader)
1436- let positionSize = $t06354463655._1
1437- let positionMargin = $t06354463655._2
1438- let pon = $t06354463655._3
1439- let positionLstUpdCPF = $t06354463655._4
1440- let $t06366063761 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1441- let positionNotional = $t06366063761._1
1442- let unrealizedPnl = $t06366063761._2
1443- let $t06376663948 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1444- let remainMargin = $t06376663948._1
1445- let badDebt = $t06376663948._2
1446- let fundingPayment = $t06376663948._3
1670+ let $t07172371834 = getPosition(_trader)
1671+ let positionSize = $t07172371834._1
1672+ let positionMargin = $t07172371834._2
1673+ let pon = $t07172371834._3
1674+ let positionLstUpdCPF = $t07172371834._4
1675+ let $t07183971940 = getPositionNotionalAndUnrealizedPnl(_trader, PNL_OPTION_SPOT)
1676+ let positionNotional = $t07183971940._1
1677+ let unrealizedPnl = $t07183971940._2
1678+ let $t07194572127 = calcRemainMarginWithFundingPayment(positionSize, positionMargin, positionLstUpdCPF, unrealizedPnl)
1679+ let remainMargin = $t07194572127._1
1680+ let badDebt = $t07194572127._2
1681+ let fundingPayment = $t07194572127._3
14471682 throw((((((s(remainMargin) + s(fundingPayment)) + s(getMarginRatio(_trader))) + s(unrealizedPnl)) + s(badDebt)) + s(positionNotional)))
14481683 }
14491684
14501685
14511686
14521687 @Callable(i)
14531688 func view_getPegAdjustCost (_price) = {
14541689 let cost = getPegAdjustCost(_price)
14551690 throw(toString(cost._2))
14561691 }
14571692
14581693
14591694
14601695 @Callable(i)
14611696 func view_getFunding () = {
14621697 let underlyingPrice = getOracleTwapPrice()
1463- let $t06433964401 = getFunding()
1464- let shortPremiumFraction = $t06433964401._1
1465- let longPremiumFraction = $t06433964401._2
1698+ let $t07251872580 = getFunding()
1699+ let shortPremiumFraction = $t07251872580._1
1700+ let longPremiumFraction = $t07251872580._2
14661701 let longFunding = divd(longPremiumFraction, underlyingPrice)
14671702 let shortFunding = divd(shortPremiumFraction, underlyingPrice)
14681703 throw((((s(longFunding) + s(shortFunding)) + s(getTwapSpotPrice())) + s(getOracleTwapPrice())))
1704+ }
1705+
1706+
1707+
1708+@Callable(i)
1709+func view_getBorrowedByTrader (_trader) = {
1710+ let $t07287072924 = getBorrowedByTrader(_trader)
1711+ let borrowed = $t07287072924._1
1712+ let assetId = $t07287072924._2
1713+ throw((s(borrowed) + assetId))
14691714 }
14701715
14711716
14721717 @Verifier(tx)
14731718 func verify () = sigVerify(tx.bodyBytes, tx.proofs[0], adminPublicKey())
14741719

github/deemru/w8io/873ac7e 
311.14 ms