tx · HR2Q5DwnggXfAGj8yBJQd9cMbup894NUTJy5nXqSmv1W

3N9kbR6BQEQV7pwBfDFzprJtNofgi6fSJ6Y:  -0.03700000 Waves

2023.06.23 12:17 [2635101] smart account 3N9kbR6BQEQV7pwBfDFzprJtNofgi6fSJ6Y > SELF 0.00000000 Waves

{ "type": 13, "id": "HR2Q5DwnggXfAGj8yBJQd9cMbup894NUTJy5nXqSmv1W", "fee": 3700000, "feeAssetId": null, "timestamp": 1687511941224, "version": 2, "chainId": 84, "sender": "3N9kbR6BQEQV7pwBfDFzprJtNofgi6fSJ6Y", "senderPublicKey": "EajGg1J8duApPELWPw9gVkPwAKFC9hujhvCcne4FunUa", "proofs": [ "3qyf5gakpEcQ3WNtxoNrdAYrsWjtYvs8wEJ2uicq2z3fdcRumyPNS1ViAbq8yXGXTMF6HkSD4vCwJLuH9q692YZa" ], "script": "base64:BgJsCAISBAoCCAgSAwoBCBIAEgQKAggBEgMKAQESEAoOCAEBAQEBAQgBAQEBAQgSDAoKCAEBAQgBAQEBCBIFCgMICAESCAoGCAgBAQEBEg4KDAgIAQEBAQEBCAgBARIDCgEBEgQKAgEIEgQKAgEIRAAUa19jb29yZGluYXRvckFkZHJlc3MCFGtfY29vcmRpbmF0b3JBZGRyZXNzAA9rX2FkbWluX2FkZHJlc3MCD2tfYWRtaW5fYWRkcmVzcwANa19xdW90ZV9hc3NldAINa19xdW90ZV9hc3NldAAFa19hbW0CBWtfYW1tABFrX21hbmFnZXJfYWRkcmVzcwIRa19tYW5hZ2VyX2FkZHJlc3MAEmtfcG9zaXRpb25TZXF1ZW5jZQISa19wb3NpdGlvblNlcXVlbmNlAA5rX3Bvc2l0aW9uU2l6ZQIOa19wb3NpdGlvblNpemUAEGtfZXhlY3V0ZWRPcmRlcnMCEGtfZXhlY3V0ZWRPcmRlcnMAEGtfY2FuY2VsZWRPcmRlcnMCEGtfY2FuY2VsZWRPcmRlcnMAB2tfb3JkZXICB2tfb3JkZXIADmtfb3JkZXJSZXF1ZXN0AgprX29yZGVyUmVxAA1rX2xhc3RPcmRlcklkAg1rX2xhc3RPcmRlcklkABBrX3RyYWRlck9yZGVyQ250AhBrX3RyYWRlck9yZGVyQ250ABBrX3RyYWRlck9yZGVySWRzAhBrX3RyYWRlck9yZGVySWRzAA1rX3NwcmVhZExpbWl0Ag1rX3NwcmVhZExpbWl0AAhrX3NlbmRlcgIIa19zZW5kZXIADWtfaW5pdGlhbGl6ZWQCDWtfaW5pdGlhbGl6ZWQABFNUT1AAAQAEVEFLRQACAAVMSU1JVAADAARMT05HAAEABVNIT1JUAAIACElOQ1JFQVNFAAEACERFQ1JFQVNFAAIAGU1BWF9UUkFERVJfT1JERVJTX1BFUl9BTU0ACgAEVElNRQgFCWxhc3RCbG9jawl0aW1lc3RhbXAADERFQ0lNQUxfVU5JVAkAaAIAAQkAaAIJAGgCCQBoAgkAaAIJAGgCAAoACgAKAAoACgAKAQNhYnMBAl94AwkAZgIFAl94AAAFAl94CQEBLQEFAl94AQRkaXZkAgJfeAJfeQkAbgQFAl94BQxERUNJTUFMX1VOSVQFAl95BQhIQUxGRVZFTgEEbXVsZAICX3gCX3kJAG4EBQJfeAUCX3kFDERFQ0lNQUxfVU5JVAUISEFMRkVWRU4BBG1pbnYCAl94Al95AwkAZgIFAl94BQJfeQUCX3kFAl94AQ50b0NvbXBvc2l0ZUtleQIEX2tleQhfYWRkcmVzcwkArAICCQCsAgIFBF9rZXkCAV8FCF9hZGRyZXNzARBleGVjdXRlZE9yZGVyS2V5AQhfb3JkZXJJZAkArAICCQCsAgIFEGtfZXhlY3V0ZWRPcmRlcnMCAV8JAKQDAQUIX29yZGVySWQBEGNhbmNlbGVkT3JkZXJLZXkBCF9vcmRlcklkCQCsAgIJAKwCAgUQa19jYW5jZWxlZE9yZGVycwIBXwkApAMBBQhfb3JkZXJJZAEIb3JkZXJLZXkBCF9vcmRlcklkCQEOdG9Db21wb3NpdGVLZXkCBQdrX29yZGVyCQCkAwEFCF9vcmRlcklkAQ9vcmRlclJlcXVlc3RLZXkCCF9vcmRlcklkBV90eXBlCQCsAgIJAKwCAgkArAICCQCsAgIFDmtfb3JkZXJSZXF1ZXN0AgFfCQCkAwEFCF9vcmRlcklkAgFfCQCkAwEFBV90eXBlARN0cmFkZXJPcmRlckNvdW50S2V5AgRfYW1tB190cmFkZXIJAKwCAgkArAICCQCsAgIJAKwCAgUQa190cmFkZXJPcmRlckNudAIBXwUEX2FtbQIBXwUHX3RyYWRlcgERdHJhZGVyT3JkZXJJZHNLZXkCBF9hbW0HX3RyYWRlcgkArAICCQCsAgIJAKwCAgkArAICBRBrX3RyYWRlck9yZGVySWRzAgFfBQRfYW1tAgFfBQdfdHJhZGVyAQtjb29yZGluYXRvcgAJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCBQR0aGlzBRRrX2Nvb3JkaW5hdG9yQWRkcmVzcwITQ29vcmRpbmF0b3Igbm90IHNldAEMYWRtaW5BZGRyZXNzAAkApggBCQERQGV4dHJOYXRpdmUoMTA1MykCCQELY29vcmRpbmF0b3IABQ9rX2FkbWluX2FkZHJlc3MBCnF1b3RlQXNzZXQACQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFDWtfcXVvdGVfYXNzZXQBDm1hbmFnZXJBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIJAQtjb29yZGluYXRvcgAFEWtfbWFuYWdlcl9hZGRyZXNzAg9NYW5hZ2VyIG5vdCBzZXQBC2lzV2hpdGVsaXN0AQhfYWRkcmVzcwkBC3ZhbHVlT3JFbHNlAgkAmwgCCQELY29vcmRpbmF0b3IACQEOdG9Db21wb3NpdGVLZXkCBQVrX2FtbQUIX2FkZHJlc3MHAQtpbml0aWFsaXplZAAJAQt2YWx1ZU9yRWxzZQIJAJsIAgUEdGhpcwUNa19pbml0aWFsaXplZAcBB2lzVmFsaWQBCF9vcmRlcklkAwkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzCQEQZXhlY3V0ZWRPcmRlcktleQEFCF9vcmRlcklkBwkAAgEJAKwCAgIYT3JkZXIgYWxyZWFkeSBleGVjdXRlZDogCQCkAwEFCF9vcmRlcklkAwkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzCQEQY2FuY2VsZWRPcmRlcktleQEFCF9vcmRlcklkBwkAAgEJAKwCAgIZT3JkZXIgYWxyZWFkeSBjYW5jZWxsZWQ6IAkApAMBBQhfb3JkZXJJZAYBDmdldFNwcmVhZExpbWl0AAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFBHRoaXMFDWtfc3ByZWFkTGltaXQCFFNwcmVhZCBsaW1pdCBub3Qgc2V0AQ5jdXJyZW50T3JkZXJJZAAJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUNa19sYXN0T3JkZXJJZAAAARNnZXRUcmFkZXJPcmRlckNvdW50AgRfYW1tB190cmFkZXIEA2tleQkBE3RyYWRlck9yZGVyQ291bnRLZXkCBQRfYW1tBQdfdHJhZGVyCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMFA2tleQAAARJ0cmFkZXJBbW1PcmRlcnNJZHMCBF9hbW0HX3RyYWRlcgQDa2V5CQERdHJhZGVyT3JkZXJJZHNLZXkCBQRfYW1tBQdfdHJhZGVyBAN2YWwJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwUDa2V5AgADCQAAAgUDdmFsAgAFA25pbAkAtQkCBQN2YWwCASwBEGhhdmVPcmRlclJlcXVlc3QCCF9vcmRlcklkBV90eXBlBANrZXkJAQ9vcmRlclJlcXVlc3RLZXkCBQhfb3JkZXJJZAUFX3R5cGUJAQlpc0RlZmluZWQBCQCdCAIFBHRoaXMFA2tleQEPZ2V0T3JkZXJSZXF1ZXN0Aghfb3JkZXJJZAVfdHlwZQQDa2V5CQEPb3JkZXJSZXF1ZXN0S2V5AgUIX29yZGVySWQFBV90eXBlBA9vcmRlclJlcXVlc3RTdHIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGlzBQNrZXkJAKwCAgIbTm90IG9yZGVyIHJlcXVlc3QgZm9yIGtleTogBQNrZXkEFG9yZGVyUmVxdWVzdFBhcnRMaXN0CQC1CQIFD29yZGVyUmVxdWVzdFN0cgIBLAQHb3JkZXJJZAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQC2CQEJAJEDAgUUb3JkZXJSZXF1ZXN0UGFydExpc3QAAAIPSW52YWxpZCBvcmRlcklkBANhbW0JAJEDAgUUb3JkZXJSZXF1ZXN0UGFydExpc3QAAQQGdHJhZGVyCQCRAwIFFG9yZGVyUmVxdWVzdFBhcnRMaXN0AAIEBHR5cGUJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAtgkBCQCRAwIFFG9yZGVyUmVxdWVzdFBhcnRMaXN0AAMCDEludmFsaWQgdHlwZQQMdHJpZ2dlclByaWNlCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQkAkQMCBRRvcmRlclJlcXVlc3RQYXJ0TGlzdAAEAhRJbnZhbGlkIHRyaWdnZXJQcmljZQQKbGltaXRQcmljZQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQC2CQEJAJEDAgUUb3JkZXJSZXF1ZXN0UGFydExpc3QABQISSW52YWxpZCBsaW1pdFByaWNlCQCYCgYFB29yZGVySWQFA2FtbQUGdHJhZGVyBQR0eXBlBQx0cmlnZ2VyUHJpY2UFCmxpbWl0UHJpY2UBCGdldE9yZGVyAQhfb3JkZXJJZAQIb3JkZXJTdHIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGlzCQEIb3JkZXJLZXkBBQhfb3JkZXJJZAkArAICAhJJbnZhbGlkIG9yZGVyIGlkOiAJAKQDAQUIX29yZGVySWQEDW9yZGVyUGFydExpc3QJALUJAgUIb3JkZXJTdHICASwEA2FtbQkAkQMCBQ1vcmRlclBhcnRMaXN0AAAEBnRyYWRlcgkAkQMCBQ1vcmRlclBhcnRMaXN0AAEECGFtb3VudEluCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQkAkQMCBQ1vcmRlclBhcnRMaXN0AAICEEludmFsaWQgYW1vdW50SW4ECGxldmVyYWdlCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQkAkQMCBQ1vcmRlclBhcnRMaXN0AAMCEEludmFsaWQgbGV2ZXJhZ2UEBHR5cGUJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAtgkBCQCRAwIFDW9yZGVyUGFydExpc3QABAIMSW52YWxpZCB0eXBlBAx0cmlnZ2VyUHJpY2UJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAtgkBCQCRAwIFDW9yZGVyUGFydExpc3QABQIUSW52YWxpZCB0cmlnZ2VyUHJpY2UEC3BheW1lbnRVc2RuCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQkAkQMCBQ1vcmRlclBhcnRMaXN0AAYCE0ludmFsaWQgcGF5bWVudFVzZG4EBHNpZGUJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAtgkBCQCRAwIFDW9yZGVyUGFydExpc3QABwIMSW52YWxpZCBzaWRlBAdyZWZMaW5rCQCRAwIFDW9yZGVyUGFydExpc3QACAQKcG9zaXRpb25JZAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQC2CQEJAJEDAgUNb3JkZXJQYXJ0TGlzdAAJAhJJbnZhbGlkIHBvc2l0aW9uSWQECmxpbWl0UHJpY2UJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAtgkBCQCRAwIFDW9yZGVyUGFydExpc3QACgISSW52YWxpZCBsaW1pdFByaWNlBApleHBpcmF0aW9uAwkAZgIJAJADAQUNb3JkZXJQYXJ0TGlzdAALCQETdmFsdWVPckVycm9yTWVzc2FnZQIJALYJAQkAkQMCBQ1vcmRlclBhcnRMaXN0AAsCEkludmFsaWQgZXhwaXJhdGlvbgAABAttYXJrZXRQcmljZQMJAGYCCQCQAwEFDW9yZGVyUGFydExpc3QADAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQC2CQEJAJEDAgUNb3JkZXJQYXJ0TGlzdAAMAhhJbnZhbGlkIG9yZGVyTWFya2V0UHJpY2UAAAkAnwoNBQNhbW0FBnRyYWRlcgUIYW1vdW50SW4FCGxldmVyYWdlBQR0eXBlBQx0cmlnZ2VyUHJpY2UFC3BheW1lbnRVc2RuBQRzaWRlBQdyZWZMaW5rBQpwb3NpdGlvbklkBQpsaW1pdFByaWNlBQpleHBpcmF0aW9uBQttYXJrZXRQcmljZQEOZ2V0TWFya2V0UHJpY2UBBF9hbW0EAXMJAPwHBAkBEUBleHRyTmF0aXZlKDEwNjIpAQUEX2FtbQIQY29tcHV0ZVNwb3RQcmljZQUDbmlsBQNuaWwDCQAAAgUBcwUBcwQDcmVzBAckbWF0Y2gwBQFzAwkAAQIFByRtYXRjaDACA0ludAQBdAUHJG1hdGNoMAUBdAkAAgECH0ludmFsaWQgY29tcHV0ZVNwb3RQcmljZSByZXN1bHQJAQV2YWx1ZQEFA3JlcwkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgEGZ2V0RmVlAgRfYW1tB190cmFkZXIEAXMJAPwHBAkBEUBleHRyTmF0aXZlKDEwNjIpAQUEX2FtbQIfY29tcHV0ZUZlZUZvclRyYWRlcldpdGhBcnRpZmFjdAkAzAgCBQdfdHJhZGVyCQDMCAICAAUDbmlsBQNuaWwDCQAAAgUBcwUBcwQDcmVzBAckbWF0Y2gwBQFzAwkAAQIFByRtYXRjaDACDihJbnQsIEJvb2xlYW4pBAF0BQckbWF0Y2gwCAUBdAJfMQkAAgECLkludmFsaWQgY29tcHV0ZUZlZUZvclRyYWRlcldpdGhBcnRpZmFjdCByZXN1bHQJAQV2YWx1ZQEFA3JlcwkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgEUZ2V0UG9zaXRpb25EaXJlY3Rpb24CCl9vcmRlclNpZGUKX29yZGVyVHlwZQMDCQAAAgUKX29yZGVyVHlwZQUEVEFLRQYJAAACBQpfb3JkZXJUeXBlBQRTVE9QAwkAAAIFCl9vcmRlclNpZGUFBExPTkcFBVNIT1JUBQRMT05HBQpfb3JkZXJTaWRlAQ9nZXRQb3NpdGlvblNpemUDBF9hbW0HX3RyYWRlcgpfZGlyZWN0aW9uBANhbW0JARFAZXh0ck5hdGl2ZSgxMDYyKQEFBF9hbW0EC3Bvc2l0aW9uS2V5CQCsAgIJAKwCAgUHX3RyYWRlcgIBXwkApAMBBQpfZGlyZWN0aW9uBAdzaXplS2V5CQEOdG9Db21wb3NpdGVLZXkCBQ5rX3Bvc2l0aW9uU2l6ZQULcG9zaXRpb25LZXkJAQt2YWx1ZU9yRWxzZQIJAJoIAgUDYW1tBQdzaXplS2V5AAABDWdldFBvc2l0aW9uSWQDBF9hbW0HX3RyYWRlcgpfZGlyZWN0aW9uBANhbW0JARFAZXh0ck5hdGl2ZSgxMDYyKQEFBF9hbW0EC3Bvc2l0aW9uS2V5CQCsAgIJAKwCAgUHX3RyYWRlcgIBXwkApAMBBQpfZGlyZWN0aW9uBAZzZXFLZXkJAQ50b0NvbXBvc2l0ZUtleQIFEmtfcG9zaXRpb25TZXF1ZW5jZQULcG9zaXRpb25LZXkJAQt2YWx1ZU9yRWxzZQIJAJoIAgUDYW1tBQZzZXFLZXkAAAEJZ2V0U3ByZWFkAQZfcHJpY2UJAQRtdWxkAgUGX3ByaWNlCQEOZ2V0U3ByZWFkTGltaXQAAQlzYXZlT3JkZXIOCF9vcmRlcklkBF9hbW0HX3RyYWRlcglfYW1vdW50SW4JX2xldmVyYWdlBV90eXBlDV90cmlnZ2VyUHJpY2UMX3BheW1lbnRVc2RuBV9zaWRlCF9yZWZMaW5rC19wb3NpdGlvbklkC19saW1pdFByaWNlC19leHBpcmF0aW9uEV9vcmRlck1hcmtldFByaWNlBAhvcmRlclN0cgkAuQkCCQDMCAIFBF9hbW0JAMwIAgUHX3RyYWRlcgkAzAgCCQCkAwEFCV9hbW91bnRJbgkAzAgCCQCkAwEFCV9sZXZlcmFnZQkAzAgCCQCkAwEFBV90eXBlCQDMCAIJAKQDAQUNX3RyaWdnZXJQcmljZQkAzAgCCQCkAwEFDF9wYXltZW50VXNkbgkAzAgCCQCkAwEFBV9zaWRlCQDMCAIFCF9yZWZMaW5rCQDMCAIJAKQDAQULX3Bvc2l0aW9uSWQJAMwIAgkApAMBBQtfbGltaXRQcmljZQkAzAgCCQCkAwEFC19leHBpcmF0aW9uCQDMCAIJAKQDAQURX29yZGVyTWFya2V0UHJpY2UFA25pbAIBLAkAzAgCCQELU3RyaW5nRW50cnkCCQEIb3JkZXJLZXkBBQhfb3JkZXJJZAUIb3JkZXJTdHIFA25pbAEQc2F2ZU9yZGVyUmVxdWVzdAYIX29yZGVySWQEX2FtbQdfdHJhZGVyBV90eXBlDV90cmlnZ2VyUHJpY2ULX2xpbWl0UHJpY2UEC29yZGVyUmVxU3RyCQC5CQIJAMwIAgkApAMBBQhfb3JkZXJJZAkAzAgCBQRfYW1tCQDMCAIFB190cmFkZXIJAMwIAgkApAMBBQVfdHlwZQkAzAgCCQCkAwEFDV90cmlnZ2VyUHJpY2UJAMwIAgkApAMBBQtfbGltaXRQcmljZQUDbmlsAgEsCQDMCAIJAQtTdHJpbmdFbnRyeQIJAQ9vcmRlclJlcXVlc3RLZXkCBQhfb3JkZXJJZAUFX3R5cGUFC29yZGVyUmVxU3RyBQNuaWwBEHVwZGF0ZU9yZGVySWRTdHIDD19vcmRlcklkc05ld1N0cgRfYW1tB190cmFkZXIJAMwIAgkBC1N0cmluZ0VudHJ5AgkBEXRyYWRlck9yZGVySWRzS2V5AgUEX2FtbQUHX3RyYWRlcgUPX29yZGVySWRzTmV3U3RyBQNuaWwBFGFkZFJlbW92ZU9yZGVySWRMaXN0BQlfb3JkZXJJZHMIX29yZGVySWQEX2FtbQdfdHJhZGVyBF9hZGQEC29yZGVySWRzTmV3AwUEX2FkZAkAzQgCBQlfb3JkZXJJZHMJAKQDAQUIX29yZGVySWQJANEIAgUJX29yZGVySWRzCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAM8IAgUJX29yZGVySWRzCQCkAwEFCF9vcmRlcklkCQCsAgICEk5vIG9yZGVyIHdpdGggaWQ6IAkApAMBBQhfb3JkZXJJZAULb3JkZXJJZHNOZXcBEGFkZFJlbW92ZU9yZGVySWQECF9vcmRlcklkBF9hbW0HX3RyYWRlcgRfYWRkBAhvcmRlcklkcwkBEnRyYWRlckFtbU9yZGVyc0lkcwIFBF9hbW0FB190cmFkZXIEC29yZGVySWRzTmV3CQEUYWRkUmVtb3ZlT3JkZXJJZExpc3QFBQhvcmRlcklkcwUIX29yZGVySWQFBF9hbW0FB190cmFkZXIFBF9hZGQEDm9yZGVySWRzTmV3U3RyCQC5CQIFC29yZGVySWRzTmV3AgEsCQEQdXBkYXRlT3JkZXJJZFN0cgMFDm9yZGVySWRzTmV3U3RyBQRfYW1tBQdfdHJhZGVyARZ1cGRhdGVUcmFkZXJPcmRlckNvdW50AwRfYW1tB190cmFkZXIGX2NvdW50AwkAZgIAAAUGX2NvdW50CQACAQkArAICAhVJbnZhbGlkIG9yZGVyIGNvdW50OiAJAKQDAQUGX2NvdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQETdHJhZGVyT3JkZXJDb3VudEtleQIFBF9hbW0FB190cmFkZXIFBl9jb3VudAUDbmlsARF1cGRhdGVMYXN0T3JkZXJJZAEMX2xhc3RPcmRlcklkCQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ1rX2xhc3RPcmRlcklkBQxfbGFzdE9yZGVySWQFA25pbAEQbWFya0V4ZWN1dGVPcmRlcgEIX29yZGVySWQJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFEGtfZXhlY3V0ZWRPcmRlcnMJAKQDAQUIX29yZGVySWQGBQNuaWwBD21hcmtDYW5jZWxPcmRlcgEIX29yZGVySWQJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAQ50b0NvbXBvc2l0ZUtleQIFEGtfY2FuY2VsZWRPcmRlcnMJAKQDAQUIX29yZGVySWQGBQNuaWwBDmdldFBvc2l0aW9uSWRzAgRfYW1tB190cmFkZXIEEGxvbmdQb3NpdGlvblNpemUJAQ9nZXRQb3NpdGlvblNpemUDBQRfYW1tBQdfdHJhZGVyBQRMT05HBBVjdXJyZW50TG9uZ1Bvc2l0aW9uSWQDCQECIT0CBRBsb25nUG9zaXRpb25TaXplAAAJAQ1nZXRQb3NpdGlvbklkAwUEX2FtbQUHX3RyYWRlcgUETE9ORwAABBFzaG9ydFBvc2l0aW9uU2l6ZQkBD2dldFBvc2l0aW9uU2l6ZQMFBF9hbW0FB190cmFkZXIFBVNIT1JUBBZjdXJyZW50U2hvcnRQb3NpdGlvbklkAwkBAiE9AgURc2hvcnRQb3NpdGlvblNpemUAAAkBDWdldFBvc2l0aW9uSWQDBQRfYW1tBQdfdHJhZGVyBQVTSE9SVAAACQDMCAIFFWN1cnJlbnRMb25nUG9zaXRpb25JZAkAzAgCBRZjdXJyZW50U2hvcnRQb3NpdGlvbklkBQNuaWwNAWkBEmNsZWFuVXBTdGFsZU9yZGVycwIEX2FtbQdfdHJhZGVyBAZvcmRlcnMJARJ0cmFkZXJBbW1PcmRlcnNJZHMCBQRfYW1tBQdfdHJhZGVyBAtwb3NpdGlvbklkcwkBDmdldFBvc2l0aW9uSWRzAgUEX2FtbQUHX3RyYWRlcgoBCmNsZWFuVXBPbmUCBF9hY2MIX29yZGVySWQECm9yZGVySWRJbnQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAtgkBBQhfb3JkZXJJZAIQSW52YWxpZCBvcmRlciBpZAQNJHQwMTE3NjYxMTk4MgkBCGdldE9yZGVyAQUKb3JkZXJJZEludAQDX3gxCAUNJHQwMTE3NjYxMTk4MgJfMQQDX3gyCAUNJHQwMTE3NjYxMTk4MgJfMgQDX3gzCAUNJHQwMTE3NjYxMTk4MgJfMwQDX3g0CAUNJHQwMTE3NjYxMTk4MgJfNAQFX3R5cGUIBQ0kdDAxMTc2NjExOTgyAl81BANfeDUIBQ0kdDAxMTc2NjExOTgyAl82BANfeDYIBQ0kdDAxMTc2NjExOTgyAl83BANfeDcIBQ0kdDAxMTc2NjExOTgyAl84BANfeDgIBQ0kdDAxMTc2NjExOTgyAl85BAtfcG9zaXRpb25JZAgFDSR0MDExNzY2MTE5ODIDXzEwBANfeDkIBQ0kdDAxMTc2NjExOTgyA18xMQQLX2V4cGlyYXRpb24IBQ0kdDAxMTc2NjExOTgyA18xMgQaY2FuY2VsVGFrZVN0b3BPZk5vUG9zaXRpb24DAwMJAAACBQVfdHlwZQUEU1RPUAYJAAACBQVfdHlwZQUEVEFLRQkBAiE9AgkAkQMCBQtwb3NpdGlvbklkcwAABQtfcG9zaXRpb25JZAcJAQIhPQIJAJEDAgULcG9zaXRpb25JZHMAAQULX3Bvc2l0aW9uSWQHBBRjYW5jZWxMaW1pdElmRXhwaXJlZAMJAAACBQVfdHlwZQUFTElNSVQJAGcCCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAULX2V4cGlyYXRpb24HAwMFGmNhbmNlbFRha2VTdG9wT2ZOb1Bvc2l0aW9uBgUUY2FuY2VsTGltaXRJZkV4cGlyZWQEBmNoYW5nZQkBD21hcmtDYW5jZWxPcmRlcgEFCm9yZGVySWRJbnQEDG5ld09yZGVyTGlzdAkBFGFkZFJlbW92ZU9yZGVySWRMaXN0BQgFBF9hY2MCXzEFCm9yZGVySWRJbnQFBF9hbW0FB190cmFkZXIHCQCUCgIFDG5ld09yZGVyTGlzdAkAzggCCAUEX2FjYwJfMgUGY2hhbmdlBQRfYWNjBA0kdDAxMjUxNDEyNTg5CgACJGwFBm9yZGVycwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAJAJQKAgUGb3JkZXJzBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBCmNsZWFuVXBPbmUCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoECW5ld09yZGVycwgFDSR0MDEyNTE0MTI1ODkCXzEEDWNhbmNlbEFjdGlvbnMIBQ0kdDAxMjUxNDEyNTg5Al8yCQDOCAIJAM4IAgUNY2FuY2VsQWN0aW9ucwkBFnVwZGF0ZVRyYWRlck9yZGVyQ291bnQDBQRfYW1tBQdfdHJhZGVyCQCQAwEFCW5ld09yZGVycwkBEHVwZGF0ZU9yZGVySWRTdHIDCQC5CQIFCW5ld09yZGVycwIBLAUEX2FtbQUHX3RyYWRlcgFpAQpzZXRDb250ZXh0AQdfc2VuZGVyAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIOT25seSBzZWxmLWNhbGwJAMwIAgkBC1N0cmluZ0VudHJ5AgUIa19zZW5kZXIFB19zZW5kZXIFA25pbAFpAQxyZXNldENvbnRleHQAAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIOT25seSBzZWxmLWNhbGwJAMwIAgkBC0RlbGV0ZUVudHJ5AQUIa19zZW5kZXIFA25pbAFpAQppbml0aWFsaXplAgxfY29vcmRpbmF0b3IMX3NwcmVhZExpbWl0AwMDCQELaW5pdGlhbGl6ZWQABgMJAGYCAAAFDF9zcHJlYWRMaW1pdAYJAGYCBQxfc3ByZWFkTGltaXQJAGkCBQxERUNJTUFMX1VOSVQACgYJAQIhPQIIBQFpBmNhbGxlcgUEdGhpcwkAAgECFFVuYWJsZSB0byBpbml0aWFsaXplCQDMCAIJAQtTdHJpbmdFbnRyeQIFFGtfY29vcmRpbmF0b3JBZGRyZXNzCQClCAEJARFAZXh0ck5hdGl2ZSgxMDYyKQEFDF9jb29yZGluYXRvcgkAzAgCCQEMSW50ZWdlckVudHJ5AgUNa19zcHJlYWRMaW1pdAUMX3NwcmVhZExpbWl0CQDMCAIJAQxCb29sZWFuRW50cnkCBQ1rX2luaXRpYWxpemVkBgUDbmlsAWkBDmNoYW5nZVNldHRpbmdzAQxfc3ByZWFkTGltaXQDAwkBAiE9AggFAWkGY2FsbGVyCQEMYWRtaW5BZGRyZXNzAAYDCQBmAgAABQxfc3ByZWFkTGltaXQGCQBmAgUMX3NwcmVhZExpbWl0CQBpAgUMREVDSU1BTF9VTklUAAoJAAIBAh1JbnZhbGlkIGNoYW5nZVNldHRpbmdzIHBhcmFtcwkAzAgCCQEMSW50ZWdlckVudHJ5AgUNa19zcHJlYWRMaW1pdAUMX3NwcmVhZExpbWl0BQNuaWwBaQELY3JlYXRlT3JkZXIOBF9hbW0FX3R5cGUNX3RyaWdnZXJQcmljZQtfbGltaXRQcmljZQlfYW1vdW50SW4JX2xldmVyYWdlBV9zaWRlCF9yZWZMaW5rEV9zdG9wVHJpZ2dlclByaWNlD19zdG9wTGltaXRQcmljZRFfdGFrZVRyaWdnZXJQcmljZQ9fdGFrZUxpbWl0UHJpY2ULX2V4cGlyYXRpb24KX3ByaWNlRGF0YQQHX3RyYWRlcgkApQgBCAUBaQZjYWxsZXIEC3VwZGF0ZVByaWNlCQD8BwQJARFAZXh0ck5hdGl2ZSgxMDYyKQEFBF9hbW0CDHVwZGF0ZU9yYWNsZQkAzAgCBQpfcHJpY2VEYXRhBQNuaWwFA25pbAMJAAACBQt1cGRhdGVQcmljZQULdXBkYXRlUHJpY2UEB2NsZWFuVXAJAPwHBAUEdGhpcwISY2xlYW5VcFN0YWxlT3JkZXJzCQDMCAIFBF9hbW0JAMwIAgUHX3RyYWRlcgUDbmlsBQNuaWwDCQAAAgUHY2xlYW5VcAUHY2xlYW5VcAMJAGYCCQCQAwEIBQFpCHBheW1lbnRzAAEJAAIBAjVJbnZhbGlkIGNyZWF0ZU9yZGVyIHBhcmFtZXRlcnM6IGludmFsaWQgcGF5bWVudCBjb3VudAQNJHQwMTU3MDMxNTk1NQMJAAACCQCQAwEIBQFpCHBheW1lbnRzAAEJAJQKAgkA2AQBCQETdmFsdWVPckVycm9yTWVzc2FnZQIICQCRAwIIBQFpCHBheW1lbnRzAAAHYXNzZXRJZAIQSW52YWxpZCBhc3NldCBpZAgJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQJAJQKAgIAAAAEDnBheW1lbnRBc3NldElkCAUNJHQwMTU3MDMxNTk1NQJfMQQNcGF5bWVudEFtb3VudAgFDSR0MDE1NzAzMTU5NTUCXzIEBmRvQ2FsbAkA/AcEBQR0aGlzAhNpbnRlcm5hbENyZWF0ZU9yZGVyCQDMCAIFB190cmFkZXIJAMwIAgUEX2FtbQkAzAgCBQVfdHlwZQkAzAgCBQ1fdHJpZ2dlclByaWNlCQDMCAIFC19saW1pdFByaWNlCQDMCAIFCV9hbW91bnRJbgkAzAgCBQlfbGV2ZXJhZ2UJAMwIAgUFX3NpZGUJAMwIAgUIX3JlZkxpbmsJAMwIAgUOcGF5bWVudEFzc2V0SWQJAMwIAgUNcGF5bWVudEFtb3VudAkAzAgCBQtfZXhwaXJhdGlvbgUDbmlsBQNuaWwDCQAAAgUGZG9DYWxsBQZkb0NhbGwEB29yZGVySWQEByRtYXRjaDAFBmRvQ2FsbAMJAAECBQckbWF0Y2gwAgNJbnQEAXQFByRtYXRjaDAFAXQJAAIBAhtJbnZhbGlkIElEIG9mIGNyZWF0ZWQgb3JkZXIDCQAAAgUFX3R5cGUFBUxJTUlUBAhtYWtlU3RvcAMJAGYCBRFfc3RvcFRyaWdnZXJQcmljZQAABApkb01ha2VTdG9wCQD8BwQFBHRoaXMCGmludGVybmFsQ3JlYXRlT3JkZXJSZXF1ZXN0CQDMCAIFB190cmFkZXIJAMwIAgUEX2FtbQkAzAgCBQdvcmRlcklkCQDMCAIFBFNUT1AJAMwIAgURX3N0b3BUcmlnZ2VyUHJpY2UJAMwIAgUPX3N0b3BMaW1pdFByaWNlBQNuaWwFA25pbAMJAAACBQpkb01ha2VTdG9wBQpkb01ha2VTdG9wBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAQIbWFrZVRha2UDCQBmAgURX3Rha2VUcmlnZ2VyUHJpY2UAAAQKZG9NYWtlVGFrZQkA/AcEBQR0aGlzAhppbnRlcm5hbENyZWF0ZU9yZGVyUmVxdWVzdAkAzAgCBQdfdHJhZGVyCQDMCAIFBF9hbW0JAMwIAgUHb3JkZXJJZAkAzAgCBQRUQUtFCQDMCAIFEV90YWtlVHJpZ2dlclByaWNlCQDMCAIFD190YWtlTGltaXRQcmljZQUDbmlsBQNuaWwDCQAAAgUKZG9NYWtlVGFrZQUKZG9NYWtlVGFrZQUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwJAM4IAgUIbWFrZVN0b3AFCG1ha2VUYWtlBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEcaW5jcmVhc2VQb3NpdGlvbldpdGhTdG9wTG9zcwoEX2FtbQpfZGlyZWN0aW9uCV9sZXZlcmFnZRNfbWluQmFzZUFzc2V0QW1vdW50CF9yZWZMaW5rEV9zdG9wVHJpZ2dlclByaWNlD19zdG9wTGltaXRQcmljZRFfdGFrZVRyaWdnZXJQcmljZQ9fdGFrZUxpbWl0UHJpY2UKX3ByaWNlRGF0YQQHX3RyYWRlcgkApQgBCAUBaQZjYWxsZXIEB2NsZWFuVXAJAPwHBAUEdGhpcwISY2xlYW5VcFN0YWxlT3JkZXJzCQDMCAIFBF9hbW0JAMwIAgUHX3RyYWRlcgUDbmlsBQNuaWwDCQAAAgUHY2xlYW5VcAUHY2xlYW5VcAMDCQEBIQEJAQtpbml0aWFsaXplZAAGCQEBIQEJAQtpc1doaXRlbGlzdAEFBF9hbW0JAAIBAi9JbnZhbGlkIGluY3JlYXNlUG9zaXRpb25XaXRoU3RvcExvc3MgcGFyYW1ldGVycwQMcG9zaXRpb25TaXplCQEPZ2V0UG9zaXRpb25TaXplAwUEX2FtbQUHX3RyYWRlcgUKX2RpcmVjdGlvbgMJAQIhPQIFDHBvc2l0aW9uU2l6ZQAACQACAQJCSW52YWxpZCBpbmNyZWFzZVBvc2l0aW9uV2l0aFN0b3BMb3NzIHBhcmFtZXRlcnM6IG9ubHkgbmV3IHBvc2l0aW9uBAxkb1NldENvbnRleHQJAPwHBAUEdGhpcwIKc2V0Q29udGV4dAkAzAgCBQdfdHJhZGVyBQNuaWwFA25pbAMJAAACBQxkb1NldENvbnRleHQFDGRvU2V0Q29udGV4dAQOZG9PcGVuUG9zaXRpb24JAPwHBAkBEUBleHRyTmF0aXZlKDEwNjIpAQUEX2FtbQIQaW5jcmVhc2VQb3NpdGlvbgkAzAgCBQpfZGlyZWN0aW9uCQDMCAIFCV9sZXZlcmFnZQkAzAgCBRNfbWluQmFzZUFzc2V0QW1vdW50CQDMCAIFCF9yZWZMaW5rCQDMCAIFCl9wcmljZURhdGEFA25pbAgFAWkIcGF5bWVudHMDCQAAAgUOZG9PcGVuUG9zaXRpb24FDmRvT3BlblBvc2l0aW9uBA5kb1Jlc2V0Q29udGV4dAkA/AcEBQR0aGlzAgxyZXNldENvbnRleHQFA25pbAUDbmlsAwkAAAIFDmRvUmVzZXRDb250ZXh0BQ5kb1Jlc2V0Q29udGV4dAQSb3BlbmVkUG9zaXRpb25TaXplCQEPZ2V0UG9zaXRpb25TaXplAwUEX2FtbQUHX3RyYWRlcgUKX2RpcmVjdGlvbgMJAAACBRJvcGVuZWRQb3NpdGlvblNpemUFEm9wZW5lZFBvc2l0aW9uU2l6ZQQIYW1vdW50SW4JAQNhYnMBBRJvcGVuZWRQb3NpdGlvblNpemUDCQAAAgUIYW1vdW50SW4FCGFtb3VudEluBAxzdG9wTG9zc1NpZGUDCQBmAgAABRJvcGVuZWRQb3NpdGlvblNpemUFBExPTkcFBVNIT1JUBA1jaGVja0Ftb3VudEluAwkAZwIAAAUIYW1vdW50SW4EBGRhdGEJALkJAgkAzAgCBQRfYW1tCQDMCAIFB190cmFkZXIJAMwIAgkApAMBBQpfZGlyZWN0aW9uCQDMCAIJAKQDAQUJX2xldmVyYWdlCQDMCAIJAKQDAQUTX21pbkJhc2VBc3NldEFtb3VudAkAzAgCBQhfcmVmTGluawkAzAgCBQpfcHJpY2VEYXRhBQNuaWwCASwJAAIBCQCsAgIJAKwCAgkArAICAhFJbnZhbGlkIGFtb3VudEluPQkApAMBBQhhbW91bnRJbgIhIGFmdGVyIHN1Y2Nlc3MgaW5jcmVhc2VQb3NpdGlvbjogBQRkYXRhAAADCQAAAgUNY2hlY2tBbW91bnRJbgUNY2hlY2tBbW91bnRJbgQRZG9DcmVhdGVTdG9wT3JkZXIDCQBmAgURX3N0b3BUcmlnZ2VyUHJpY2UAAAQRZG9DcmVhdGVTdG9wT3JkZXIJAPwHBAUEdGhpcwITaW50ZXJuYWxDcmVhdGVPcmRlcgkAzAgCBQdfdHJhZGVyCQDMCAIFBF9hbW0JAMwIAgUEU1RPUAkAzAgCBRFfc3RvcFRyaWdnZXJQcmljZQkAzAgCBQ9fc3RvcExpbWl0UHJpY2UJAMwIAgUIYW1vdW50SW4JAMwIAgAACQDMCAIFDHN0b3BMb3NzU2lkZQkAzAgCBQhfcmVmTGluawkAzAgCAgAJAMwIAgAACQDMCAIAAAUDbmlsBQNuaWwDCQAAAgURZG9DcmVhdGVTdG9wT3JkZXIFEWRvQ3JlYXRlU3RvcE9yZGVyBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBRFkb0NyZWF0ZVN0b3BPcmRlcgURZG9DcmVhdGVTdG9wT3JkZXIEEWRvQ3JlYXRlVGFrZU9yZGVyAwkAZgIFEV90YWtlVHJpZ2dlclByaWNlAAAEEWRvQ3JlYXRlVGFrZU9yZGVyCQD8BwQFBHRoaXMCE2ludGVybmFsQ3JlYXRlT3JkZXIJAMwIAgUHX3RyYWRlcgkAzAgCBQRfYW1tCQDMCAIFBFRBS0UJAMwIAgURX3Rha2VUcmlnZ2VyUHJpY2UJAMwIAgUPX3Rha2VMaW1pdFByaWNlCQDMCAIFCGFtb3VudEluCQDMCAIAAAkAzAgCBQxzdG9wTG9zc1NpZGUJAMwIAgUIX3JlZkxpbmsJAMwIAgIACQDMCAIAAAkAzAgCAAAFA25pbAUDbmlsAwkAAAIFEWRvQ3JlYXRlVGFrZU9yZGVyBRFkb0NyZWF0ZVRha2VPcmRlcgUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgURZG9DcmVhdGVUYWtlT3JkZXIFEWRvQ3JlYXRlVGFrZU9yZGVyBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEZaW50ZXJuYWxNYXJrT3JkZXJFeGVjdXRlZAMEX2FtbQdfdHJhZGVyCF9vcmRlcklkAwMDCQEBIQEJAQtpbml0aWFsaXplZAAGCQEBIQEJAQtpc1doaXRlbGlzdAEFBF9hbW0GCQEBIQEJAAACCAUBaQZjYWxsZXIFBHRoaXMJAAIBAixJbnZhbGlkIGludGVybmFsTWFya09yZGVyRXhlY3V0ZWQgcGFyYW1ldGVycwQTbmV3VHJhZGVyT3JkZXJDb3VudAkAZQIJARNnZXRUcmFkZXJPcmRlckNvdW50AgUEX2FtbQUHX3RyYWRlcgABCQDOCAIJAM4IAgkBFnVwZGF0ZVRyYWRlck9yZGVyQ291bnQDBQRfYW1tBQdfdHJhZGVyBRNuZXdUcmFkZXJPcmRlckNvdW50CQEQYWRkUmVtb3ZlT3JkZXJJZAQFCF9vcmRlcklkBQRfYW1tBQdfdHJhZGVyBwkBEG1hcmtFeGVjdXRlT3JkZXIBBQhfb3JkZXJJZAFpARppbnRlcm5hbENyZWF0ZU9yZGVyUmVxdWVzdAYHX3RyYWRlcgRfYW1tCF9vcmRlcklkBV90eXBlDV90cmlnZ2VyUHJpY2ULX2xpbWl0UHJpY2UDAwMDAwkBASEBCQELaW5pdGlhbGl6ZWQABgkBASEBCQELaXNXaGl0ZWxpc3QBBQRfYW1tBgkAZwIAAAUNX3RyaWdnZXJQcmljZQYJAGYCAAAFC19saW1pdFByaWNlBgkBASEBCQAAAggFAWkGY2FsbGVyBQR0aGlzCQACAQItSW52YWxpZCBpbnRlcm5hbENyZWF0ZU9yZGVyUmVxdWVzdCBwYXJhbWV0ZXJzCQEQc2F2ZU9yZGVyUmVxdWVzdAYFCF9vcmRlcklkBQRfYW1tBQdfdHJhZGVyBQVfdHlwZQUNX3RyaWdnZXJQcmljZQULX2xpbWl0UHJpY2UBaQETaW50ZXJuYWxDcmVhdGVPcmRlcgwHX3RyYWRlcgRfYW1tBV90eXBlDV90cmlnZ2VyUHJpY2ULX2xpbWl0UHJpY2UJX2Ftb3VudEluCV9sZXZlcmFnZQVfc2lkZQhfcmVmTGluaw9fcGF5bWVudEFzc2V0SWQOX3BheW1lbnRBbW91bnQLX2V4cGlyYXRpb24DAwMDAwMDAwMDCQEBIQEJAQtpbml0aWFsaXplZAAGCQEBIQEJAQtpc1doaXRlbGlzdAEFBF9hbW0GCQBnAgAABQ1fdHJpZ2dlclByaWNlBgkAZgIAAAULX2xpbWl0UHJpY2UGCQBnAgAABQlfYW1vdW50SW4GCQBmAgAABQlfbGV2ZXJhZ2UGCQEBIQEDCQAAAgUFX3NpZGUFBExPTkcGCQAAAgUFX3NpZGUFBVNIT1JUBgkBASEBAwMJAAACBQVfdHlwZQUEU1RPUAYJAAACBQVfdHlwZQUEVEFLRQYJAAACBQVfdHlwZQUFTElNSVQGCQEBIQEJAAACCAUBaQZjYWxsZXIFBHRoaXMGCQBmAgAABQtfZXhwaXJhdGlvbgkAAgECJkludmFsaWQgaW50ZXJuYWxDcmVhdGVPcmRlciBwYXJhbWV0ZXJzBAdvcmRlcklkCQBkAgkBDmN1cnJlbnRPcmRlcklkAAABBBFwb3NpdGlvbkRpcmVjdGlvbgkBFGdldFBvc2l0aW9uRGlyZWN0aW9uAgUFX3NpZGUFBV90eXBlBBNuZXdUcmFkZXJPcmRlckNvdW50CQBkAgkBE2dldFRyYWRlck9yZGVyQ291bnQCBQRfYW1tBQdfdHJhZGVyAAEEDHBvc2l0aW9uU2l6ZQkBD2dldFBvc2l0aW9uU2l6ZQMFBF9hbW0FB190cmFkZXIFEXBvc2l0aW9uRGlyZWN0aW9uBBBvcmRlck1hcmtldFByaWNlCQEOZ2V0TWFya2V0UHJpY2UBBQRfYW1tBApfZGlyZWN0aW9uAwMDCQAAAgUMcG9zaXRpb25TaXplAAAGAwkAZgIFDHBvc2l0aW9uU2l6ZQAACQAAAgUFX3NpZGUFBExPTkcHBgMJAGYCAAAFDHBvc2l0aW9uU2l6ZQkAAAIFBV9zaWRlBQVTSE9SVAcFCElOQ1JFQVNFBQhERUNSRUFTRQMDCQAAAgUMcG9zaXRpb25TaXplAAADCQAAAgUFX3R5cGUFBFNUT1AGCQAAAgUFX3R5cGUFBFRBS0UHCQACAQIrQ2FuIG5vdCBjcmVhdGUgU1RPUC9UQUtFIG9yZGVyOiBubyBwb3NpdGlvbgQLdXNkblBheW1lbnQDCQAAAgUKX2RpcmVjdGlvbgUISU5DUkVBU0UDAwkBAiE9AgUPX3BheW1lbnRBc3NldElkCQDYBAEJAQpxdW90ZUFzc2V0AAYJAQIhPQIFDl9wYXltZW50QW1vdW50BQlfYW1vdW50SW4JAAIBAjRJbnZhbGlkIGNyZWF0ZUxpbWl0T3JkZXIgcGFyYW1ldGVyczogaW52YWxpZCBwYXltZW50BAVzdGFrZQkA/AcECQEObWFuYWdlckFkZHJlc3MAAgdkZXBvc2l0BQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAUOX3BheW1lbnRBbW91bnQFA25pbAMJAAACBQVzdGFrZQUFc3Rha2UFDl9wYXltZW50QW1vdW50CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAAADCQAAAgULdXNkblBheW1lbnQFC3VzZG5QYXltZW50BApwb3NpdGlvbklkAwkBAiE9AgUMcG9zaXRpb25TaXplAAAJAQ1nZXRQb3NpdGlvbklkAwUEX2FtbQUHX3RyYWRlcgURcG9zaXRpb25EaXJlY3Rpb24AAAMDAwkAAAIFBV90eXBlBQRTVE9QBgkAAAIFBV90eXBlBQRUQUtFCQAAAgUKcG9zaXRpb25JZAAABwkAAgECP1NUT1AgYW5kIFRBS0Ugb3JkZXIgc2hvdWxkIGJlIGFzc2lnbmVkIHRvIHBvc2l0aW9uIHdpdGggaWQgIT0gMAMJAGYCBRNuZXdUcmFkZXJPcmRlckNvdW50BRlNQVhfVFJBREVSX09SREVSU19QRVJfQU1NCQACAQIwSW52YWxpZCBjcmVhdGVMaW1pdE9yZGVyIHBhcmFtZXRlcnM6IG9yZGVyIGNvdW50BAljaGFuZ2VTZXQJAM4IAgkAzggCCQDOCAIJAQlzYXZlT3JkZXIOBQdvcmRlcklkBQRfYW1tBQdfdHJhZGVyBQlfYW1vdW50SW4FCV9sZXZlcmFnZQUFX3R5cGUFDV90cmlnZ2VyUHJpY2UFC3VzZG5QYXltZW50BQVfc2lkZQUIX3JlZkxpbmsFCnBvc2l0aW9uSWQFC19saW1pdFByaWNlBQtfZXhwaXJhdGlvbgUQb3JkZXJNYXJrZXRQcmljZQkBEGFkZFJlbW92ZU9yZGVySWQEBQdvcmRlcklkBQRfYW1tBQdfdHJhZGVyBgkBFnVwZGF0ZVRyYWRlck9yZGVyQ291bnQDBQRfYW1tBQdfdHJhZGVyBRNuZXdUcmFkZXJPcmRlckNvdW50CQERdXBkYXRlTGFzdE9yZGVySWQBBQdvcmRlcklkCQCUCgIFCWNoYW5nZVNldAUHb3JkZXJJZAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQtjYW5jZWxPcmRlcgEIX29yZGVySWQEDSR0MDI1NTkxMjU3NDEJAQhnZXRPcmRlcgEFCF9vcmRlcklkBARfYW1tCAUNJHQwMjU1OTEyNTc0MQJfMQQHX3RyYWRlcggFDSR0MDI1NTkxMjU3NDECXzIECV9hbW91bnRJbggFDSR0MDI1NTkxMjU3NDECXzMECV9sZXZlcmFnZQgFDSR0MDI1NTkxMjU3NDECXzQEBV90eXBlCAUNJHQwMjU1OTEyNTc0MQJfNQQNX3RyaWdnZXJQcmljZQgFDSR0MDI1NTkxMjU3NDECXzYEC19hbW91bnRVc2RuCAUNJHQwMjU1OTEyNTc0MQJfNwMDAwkBASEBCQELaW5pdGlhbGl6ZWQABgkBASEBCQEHaXNWYWxpZAEFCF9vcmRlcklkBgkBASEBCQAAAgkApQgBCAUBaQZjYWxsZXIFB190cmFkZXIJAAIBAh5JbnZhbGlkIGNhbmNlbE9yZGVyIHBhcmFtZXRlcnMEB2NsZWFuVXAJAPwHBAUEdGhpcwISY2xlYW5VcFN0YWxlT3JkZXJzCQDMCAIFBF9hbW0JAMwIAgUHX3RyYWRlcgUDbmlsBQNuaWwDCQAAAgUHY2xlYW5VcAUHY2xlYW5VcAQTbmV3VHJhZGVyT3JkZXJDb3VudAkAZQIJARNnZXRUcmFkZXJPcmRlckNvdW50AgUEX2FtbQUHX3RyYWRlcgABBAh3aXRoZHJhdwMJAGYCBQtfYW1vdW50VXNkbgAABAd1bnN0YWtlCQD8BwQJAQ5tYW5hZ2VyQWRkcmVzcwACCHdpdGhkcmF3CQDMCAIJANgEAQkBCnF1b3RlQXNzZXQACQDMCAIFC19hbW91bnRVc2RuBQNuaWwFA25pbAMJAAACBQd1bnN0YWtlBQd1bnN0YWtlBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBQh3aXRoZHJhdwUId2l0aGRyYXcJAM4IAgkAzggCCQDOCAIJAQ9tYXJrQ2FuY2VsT3JkZXIBBQhfb3JkZXJJZAkBEGFkZFJlbW92ZU9yZGVySWQEBQhfb3JkZXJJZAUEX2FtbQUHX3RyYWRlcgcJARZ1cGRhdGVUcmFkZXJPcmRlckNvdW50AwUEX2FtbQUHX3RyYWRlcgUTbmV3VHJhZGVyT3JkZXJDb3VudAMJAGYCBQtfYW1vdW50VXNkbgAACQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMIBQFpBmNhbGxlcgULX2Ftb3VudFVzZG4JAQpxdW90ZUFzc2V0AAUDbmlsBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEMZXhlY3V0ZU9yZGVyAghfb3JkZXJJZApfcHJpY2VEYXRhBA0kdDAyNjk1OTI3MjIxCQEIZ2V0T3JkZXIBBQhfb3JkZXJJZAQEX2FtbQgFDSR0MDI2OTU5MjcyMjECXzEEB190cmFkZXIIBQ0kdDAyNjk1OTI3MjIxAl8yBAlfYW1vdW50SW4IBQ0kdDAyNjk1OTI3MjIxAl8zBAlfbGV2ZXJhZ2UIBQ0kdDAyNjk1OTI3MjIxAl80BAVfdHlwZQgFDSR0MDI2OTU5MjcyMjECXzUEDV90cmlnZ2VyUHJpY2UIBQ0kdDAyNjk1OTI3MjIxAl82BAtfYW1vdW50VXNkbggFDSR0MDI2OTU5MjcyMjECXzcEBV9zaWRlCAUNJHQwMjY5NTkyNzIyMQJfOAQIX3JlZkxpbmsIBQ0kdDAyNjk1OTI3MjIxAl85BAtfcG9zaXRpb25JZAgFDSR0MDI2OTU5MjcyMjEDXzEwBAtfbGltaXRQcmljZQgFDSR0MDI2OTU5MjcyMjEDXzExBApfdGltZXN0YW1wCAUNJHQwMjY5NTkyNzIyMQNfMTIEEV9vcmRlck1hcmtldFByaWNlCAUNJHQwMjY5NTkyNzIyMQNfMTMEC3VwZGF0ZVByaWNlCQD8BwQJARFAZXh0ck5hdGl2ZSgxMDYyKQEFBF9hbW0CDHVwZGF0ZU9yYWNsZQkAzAgCBQpfcHJpY2VEYXRhBQNuaWwFA25pbAMJAAACBQt1cGRhdGVQcmljZQULdXBkYXRlUHJpY2UEB2NsZWFuVXAJAPwHBAUEdGhpcwISY2xlYW5VcFN0YWxlT3JkZXJzCQDMCAIFBF9hbW0JAMwIAgUHX3RyYWRlcgUDbmlsBQNuaWwDCQAAAgUHY2xlYW5VcAUHY2xlYW5VcAQRcG9zaXRpb25EaXJlY3Rpb24JARRnZXRQb3NpdGlvbkRpcmVjdGlvbgIFBV9zaWRlBQVfdHlwZQMDCQEBIQEJAQtpbml0aWFsaXplZAAGCQEBIQEJAQdpc1ZhbGlkAQUIX29yZGVySWQJAAIBAh9JbnZhbGlkIGV4ZWN1dGVPcmRlciBwYXJhbWV0ZXJzBAxwb3NpdGlvblNpemUJAQ9nZXRQb3NpdGlvblNpemUDBQRfYW1tBQdfdHJhZGVyBRFwb3NpdGlvbkRpcmVjdGlvbgMJAAACBQxwb3NpdGlvblNpemUFDHBvc2l0aW9uU2l6ZQQRY3VycmVudFBvc2l0aW9uSWQDCQECIT0CBQxwb3NpdGlvblNpemUAAAkBDWdldFBvc2l0aW9uSWQDBQRfYW1tBQdfdHJhZGVyBRFwb3NpdGlvbkRpcmVjdGlvbgAABA0kdDAyNzgzMDMxNzkxAwkAAAIFBV90eXBlBQRTVE9QBBJfcG9zaXRpb25EaXJlY3Rpb24DCQBmAgUMcG9zaXRpb25TaXplAAAFBExPTkcDCQBmAgAABQxwb3NpdGlvblNpemUFBVNIT1JUCQACAQIsQ2FuIG5vdCBleGVjdXRlIFNUT1Agb3JkZXI6IG5vIG9wZW4gcG9zaXRpb24EC21hcmtldFByaWNlCQEOZ2V0TWFya2V0UHJpY2UBBQRfYW1tBAxpc0V4ZWN1dGFibGUDCQAAAgUFX3NpZGUFEl9wb3NpdGlvbkRpcmVjdGlvbgkAAgECJ0NhbiBub3QgZXhlY3V0ZSBTVE9QIG9yZGVyOiByZWR1Y2Ugb25seQMJAQIhPQIFEWN1cnJlbnRQb3NpdGlvbklkBQtfcG9zaXRpb25JZAkAAgECK0NhbiBub3QgZXhlY3V0ZSBTVE9QIG9yZGVyOiBwb3NpdGlvbiBjbG9zZWQDCQAAAgUSX3Bvc2l0aW9uRGlyZWN0aW9uBQRMT05HCQBnAgUNX3RyaWdnZXJQcmljZQULbWFya2V0UHJpY2UJAGcCBQttYXJrZXRQcmljZQUNX3RyaWdnZXJQcmljZQMFDGlzRXhlY3V0YWJsZQkAlQoDAg1jbG9zZVBvc2l0aW9uCQDMCAIJAQRtaW52AgUJX2Ftb3VudEluCQEDYWJzAQUMcG9zaXRpb25TaXplCQDMCAIFEXBvc2l0aW9uRGlyZWN0aW9uCQDMCAIJAQRtdWxkAgULX2xpbWl0UHJpY2UJAQNhYnMBBQxwb3NpdGlvblNpemUJAMwIAgcJAMwIAgUKX3ByaWNlRGF0YQUDbmlsBQNuaWwJAAIBAjFDYW4gbm90IGV4ZWN1dGUgU1RPUCBvcmRlcjogdHJpZ2dlclByaWNlIG1pc21hdGNoAwkAAAIFBV90eXBlBQRUQUtFBBJfcG9zaXRpb25EaXJlY3Rpb24DCQBmAgUMcG9zaXRpb25TaXplAAAFBExPTkcDCQBmAgAABQxwb3NpdGlvblNpemUFBVNIT1JUCQACAQIsQ2FuIG5vdCBleGVjdXRlIFNUT1Agb3JkZXI6IG5vIG9wZW4gcG9zaXRpb24EC21hcmtldFByaWNlCQEOZ2V0TWFya2V0UHJpY2UBBQRfYW1tBAxpc0V4ZWN1dGFibGUDCQAAAgUFX3NpZGUFEl9wb3NpdGlvbkRpcmVjdGlvbgkAAgECJ0NhbiBub3QgZXhlY3V0ZSBUQUtFIG9yZGVyOiByZWR1Y2Ugb25seQMJAQIhPQIFEWN1cnJlbnRQb3NpdGlvbklkBQtfcG9zaXRpb25JZAkAAgEJAKwCAgkArAICCQCsAgICLENhbiBub3QgZXhlY3V0ZSBUQUtFIG9yZGVyOiBwb3NpdGlvbiBjbG9zZWQgCQCkAwEFEWN1cnJlbnRQb3NpdGlvbklkAgIhPQkApAMBBQtfcG9zaXRpb25JZAMJAAACBRJfcG9zaXRpb25EaXJlY3Rpb24FBExPTkcJAGcCBQttYXJrZXRQcmljZQUNX3RyaWdnZXJQcmljZQkAZwIFDV90cmlnZ2VyUHJpY2UFC21hcmtldFByaWNlAwUMaXNFeGVjdXRhYmxlCQCVCgMCDWNsb3NlUG9zaXRpb24JAMwIAgkBBG1pbnYCBQlfYW1vdW50SW4JAQNhYnMBBQxwb3NpdGlvblNpemUJAMwIAgURcG9zaXRpb25EaXJlY3Rpb24JAMwIAgkBBG11bGQCBQtfbGltaXRQcmljZQkBA2FicwEFDHBvc2l0aW9uU2l6ZQkAzAgCBwkAzAgCBQpfcHJpY2VEYXRhBQNuaWwFA25pbAkAAgECMUNhbiBub3QgZXhlY3V0ZSBUQUtFIG9yZGVyOiB0cmlnZ2VyUHJpY2UgbWlzbWF0Y2gDCQAAAgUFX3R5cGUFBUxJTUlUBAttYXJrZXRQcmljZQkBDmdldE1hcmtldFByaWNlAQUEX2FtbQQSaXNFeGVjdXRhYmxlQnlTdG9wAwkAAAIFC19saW1pdFByaWNlAAAGAwkAZgIFC19saW1pdFByaWNlBRFfb3JkZXJNYXJrZXRQcmljZQkAZwIFC21hcmtldFByaWNlBQtfbGltaXRQcmljZQkAZwIFC19saW1pdFByaWNlBQttYXJrZXRQcmljZQQbaXNFeGVjdXRhYmxlRm9yVHJpZ2dlclByaWNlAwkAAAIFBV9zaWRlBQRMT05HCQBnAgUNX3RyaWdnZXJQcmljZQULbWFya2V0UHJpY2UJAGcCBQttYXJrZXRQcmljZQUNX3RyaWdnZXJQcmljZQQMaXNFeGVjdXRhYmxlAwUSaXNFeGVjdXRhYmxlQnlTdG9wBRtpc0V4ZWN1dGFibGVGb3JUcmlnZ2VyUHJpY2UHAwUMaXNFeGVjdXRhYmxlBAxtYXJnaW5BbW91bnQJAQRkaXZkAgULX2Ftb3VudFVzZG4JAGQCCQEEbXVsZAIJAQZnZXRGZWUCBQRfYW1tBQdfdHJhZGVyBQlfbGV2ZXJhZ2UFDERFQ0lNQUxfVU5JVAQMb3Blbk5vdGlvbmFsCQEEbXVsZAIFDG1hcmdpbkFtb3VudAUJX2xldmVyYWdlBBdpZGVhbE1pbkJhc2VBc3NldEFtb3VudAkBBGRpdmQCBQxvcGVuTm90aW9uYWwFDV90cmlnZ2VyUHJpY2UEEm1pbkJhc2VBc3NldEFtb3VudAkAZQIFF2lkZWFsTWluQmFzZUFzc2V0QW1vdW50CQEEbXVsZAIFF2lkZWFsTWluQmFzZUFzc2V0QW1vdW50CQEOZ2V0U3ByZWFkTGltaXQACQCVCgMCEGluY3JlYXNlUG9zaXRpb24JAMwIAgUFX3NpZGUJAMwIAgUJX2xldmVyYWdlCQDMCAIFEm1pbkJhc2VBc3NldEFtb3VudAkAzAgCBQhfcmVmTGluawkAzAgCBQpfcHJpY2VEYXRhBQNuaWwJAMwIAgkBD0F0dGFjaGVkUGF5bWVudAIJAQpxdW90ZUFzc2V0AAULX2Ftb3VudFVzZG4FA25pbAkAAgECMkNhbiBub3QgZXhlY3V0ZSBMSU1JVCBvcmRlcjogdHJpZ2dlclByaWNlIG1pc21hdGNoCQACAQkArAICAhRJbnZhbGlkIG9yZGVyIHR5cGU6IAkApAMBBQVfdHlwZQQGbWV0aG9kCAUNJHQwMjc4MzAzMTc5MQJfMQQEYXJncwgFDSR0MDI3ODMwMzE3OTECXzIECHBheW1lbnRzCAUNJHQwMjc4MzAzMTc5MQJfMwQId2l0aGRyYXcDCQAAAgkAkAMBBQhwYXltZW50cwABBAd1bnN0YWtlCQD8BwQJAQ5tYW5hZ2VyQWRkcmVzcwACCHdpdGhkcmF3CQDMCAIJANgEAQkBCnF1b3RlQXNzZXQACQDMCAIICQCRAwIFCHBheW1lbnRzAAAGYW1vdW50BQNuaWwFA25pbAMJAAACBQd1bnN0YWtlBQd1bnN0YWtlBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBQh3aXRoZHJhdwUId2l0aGRyYXcEDGRvU2V0Q29udGV4dAkA/AcEBQR0aGlzAgpzZXRDb250ZXh0CQDMCAIFB190cmFkZXIFA25pbAUDbmlsAwkAAAIFDGRvU2V0Q29udGV4dAUMZG9TZXRDb250ZXh0BA9kb0Nsb3NlUG9zaXRpb24JAPwHBAkBEUBleHRyTmF0aXZlKDEwNjIpAQUEX2FtbQUGbWV0aG9kBQRhcmdzBQhwYXltZW50cwMJAAACBQ9kb0Nsb3NlUG9zaXRpb24FD2RvQ2xvc2VQb3NpdGlvbgQOZG9SZXNldENvbnRleHQJAPwHBAUEdGhpcwIMcmVzZXRDb250ZXh0BQNuaWwFA25pbAMJAAACBQ5kb1Jlc2V0Q29udGV4dAUOZG9SZXNldENvbnRleHQEE2V4ZWN1dGVPcmRlclJlcXVlc3QDCQAAAgUFX3R5cGUFBUxJTUlUBA9uZXdQb3NpdGlvblNpemUJAQ9nZXRQb3NpdGlvblNpemUDBQRfYW1tBQdfdHJhZGVyBRFwb3NpdGlvbkRpcmVjdGlvbgMJAAACBQ9uZXdQb3NpdGlvblNpemUFD25ld1Bvc2l0aW9uU2l6ZQQNbmV3UG9zaXRpb25JZAkBDWdldFBvc2l0aW9uSWQDBQRfYW1tBQdfdHJhZGVyBRFwb3NpdGlvbkRpcmVjdGlvbgMJAAACBQ1uZXdQb3NpdGlvbklkBQ1uZXdQb3NpdGlvbklkBBFwb3NpdGlvblNpemVEZWx0YQkAZQIFD25ld1Bvc2l0aW9uU2l6ZQUMcG9zaXRpb25TaXplBA5jbG9zZU9yZGVyU2lkZQMJAGYCBQ9uZXdQb3NpdGlvblNpemUAAAUFU0hPUlQFBExPTkcEDW1ha2VUYWtlT3JkZXIDCQEQaGF2ZU9yZGVyUmVxdWVzdAIFCF9vcmRlcklkBQRUQUtFBA0kdDAzMjc1MzMyOTUyCQEPZ2V0T3JkZXJSZXF1ZXN0AgUIX29yZGVySWQFBFRBS0UECm5ld09yZGVySWQIBQ0kdDAzMjc1MzMyOTUyAl8xBAZuZXdBbW0IBQ0kdDAzMjc1MzMyOTUyAl8yBAluZXdUcmFkZXIIBQ0kdDAzMjc1MzMyOTUyAl8zBAduZXdUeXBlCAUNJHQwMzI3NTMzMjk1MgJfNAQPbmV3VHJpZ2dlclByaWNlCAUNJHQwMzI3NTMzMjk1MgJfNQQNbmV3TGltaXRQcmljZQgFDSR0MDMyNzUzMzI5NTICXzYEBmRvQ2FsbAkA/AcEBQR0aGlzAhNpbnRlcm5hbENyZWF0ZU9yZGVyCQDMCAIFCW5ld1RyYWRlcgkAzAgCBQZuZXdBbW0JAMwIAgUHbmV3VHlwZQkAzAgCBQ9uZXdUcmlnZ2VyUHJpY2UJAMwIAgUNbmV3TGltaXRQcmljZQkAzAgCBRFwb3NpdGlvblNpemVEZWx0YQkAzAgCAAAJAMwIAgUOY2xvc2VPcmRlclNpZGUJAMwIAgIACQDMCAICAAkAzAgCAAAJAMwIAgAABQNuaWwFA25pbAMJAAACBQZkb0NhbGwFBmRvQ2FsbAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBQNuaWwDCQAAAgUNbWFrZVRha2VPcmRlcgUNbWFrZVRha2VPcmRlcgQNbWFrZVN0b3BPcmRlcgMJARBoYXZlT3JkZXJSZXF1ZXN0AgUIX29yZGVySWQFBFNUT1AEDSR0MDMzNTI1MzM3MjQJAQ9nZXRPcmRlclJlcXVlc3QCBQhfb3JkZXJJZAUEU1RPUAQKbmV3T3JkZXJJZAgFDSR0MDMzNTI1MzM3MjQCXzEEBm5ld0FtbQgFDSR0MDMzNTI1MzM3MjQCXzIECW5ld1RyYWRlcggFDSR0MDMzNTI1MzM3MjQCXzMEB25ld1R5cGUIBQ0kdDAzMzUyNTMzNzI0Al80BA9uZXdUcmlnZ2VyUHJpY2UIBQ0kdDAzMzUyNTMzNzI0Al81BA1uZXdMaW1pdFByaWNlCAUNJHQwMzM1MjUzMzcyNAJfNgQGZG9DYWxsCQD8BwQFBHRoaXMCE2ludGVybmFsQ3JlYXRlT3JkZXIJAMwIAgUJbmV3VHJhZGVyCQDMCAIFBm5ld0FtbQkAzAgCBQduZXdUeXBlCQDMCAIFD25ld1RyaWdnZXJQcmljZQkAzAgCBQ1uZXdMaW1pdFByaWNlCQDMCAIFEXBvc2l0aW9uU2l6ZURlbHRhCQDMCAIAAAkAzAgCBQ5jbG9zZU9yZGVyU2lkZQkAzAgCAgAJAMwIAgIACQDMCAIAAAkAzAgCAAAFA25pbAUDbmlsAwkAAAIFBmRvQ2FsbAUGZG9DYWxsBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBQ1tYWtlU3RvcE9yZGVyBQ1tYWtlU3RvcE9yZGVyCQDOCAIFDW1ha2VUYWtlT3JkZXIFDW1ha2VTdG9wT3JkZXIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4FA25pbAMJAAACBRNleGVjdXRlT3JkZXJSZXF1ZXN0BRNleGVjdXRlT3JkZXJSZXF1ZXN0BBNkb01hcmtPcmRlckV4ZWN1dGVkCQD8BwQFBHRoaXMCGWludGVybmFsTWFya09yZGVyRXhlY3V0ZWQJAMwIAgUEX2FtbQkAzAgCBQdfdHJhZGVyCQDMCAIFCF9vcmRlcklkBQNuaWwFA25pbAMJAAACBRNkb01hcmtPcmRlckV4ZWN1dGVkBRNkb01hcmtPcmRlckV4ZWN1dGVkBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEUdmlld19jYW5FeGVjdXRlT3JkZXICCF9vcmRlcklkCl9wcmljZURhdGEEAXMJAPwHBAUEdGhpcwIMZXhlY3V0ZU9yZGVyCQDMCAIFCF9vcmRlcklkCQDMCAIFCl9wcmljZURhdGEFA25pbAUDbmlsAwkAAAIFAXMFAXMJAAIBAgdTdWNjZXNzCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQJ0eAEGdmVyaWZ5AAQOY29vcmRpbmF0b3JTdHIJAJ0IAgUEdGhpcwUUa19jb29yZGluYXRvckFkZHJlc3MDCQEJaXNEZWZpbmVkAQUOY29vcmRpbmF0b3JTdHIEBWFkbWluCQCdCAIJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQV2YWx1ZQEFDmNvb3JkaW5hdG9yU3RyBQ9rX2FkbWluX2FkZHJlc3MDCQEJaXNEZWZpbmVkAQUFYWRtaW4JAQt2YWx1ZU9yRWxzZQIJAJsIAgkBEUBleHRyTmF0aXZlKDEwNjIpAQkBBXZhbHVlAQUFYWRtaW4JAKwCAgkArAICCQCsAgICB3N0YXR1c18JAKUIAQUEdGhpcwIBXwkA2AQBCAUCdHgCaWQHCQACAQIudW5hYmxlIHRvIHZlcmlmeTogYWRtaW4gbm90IHNldCBpbiBjb29yZGluYXRvcgkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAAgFAnR4D3NlbmRlclB1YmxpY0tleaehTtI=", "height": 2635101, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 4GQDLn3cqjDQdPnmxAgefpTZtysBmz9vGVUCv99MyqAk Next: none Diff:
OldNewDifferences
2929
3030 let k_traderOrderIds = "k_traderOrderIds"
3131
32+let k_spreadLimit = "k_spreadLimit"
33+
3234 let k_sender = "k_sender"
3335
3436 let k_initialized = "k_initialized"
5254 let TIME = lastBlock.timestamp
5355
5456 let DECIMAL_UNIT = (1 * (((((10 * 10) * 10) * 10) * 10) * 10))
55-
56-let SPREAD_LIMIT = (DECIMAL_UNIT / 200)
5757
5858 func abs (_x) = if ((_x > 0))
5959 then _x
9595 func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
9696
9797
98+func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address))
99+
100+
98101 func quoteAsset () = fromBase58String(getStringValue(coordinator(), k_quote_asset))
99102
100103
112115 else if (valueOrElse(getBoolean(this, canceledOrderKey(_orderId)), false))
113116 then throw(("Order already cancelled: " + toString(_orderId)))
114117 else true
118+
119+
120+func getSpreadLimit () = valueOrErrorMessage(getInteger(this, k_spreadLimit), "Spread limit not set")
115121
116122
117123 func currentOrderId () = valueOrElse(getInteger(this, k_lastOrderId), 0)
166172 let refLink = orderPartList[8]
167173 let positionId = valueOrErrorMessage(parseInt(orderPartList[9]), "Invalid positionId")
168174 let limitPrice = valueOrErrorMessage(parseInt(orderPartList[10]), "Invalid limitPrice")
169- $Tuple11(amm, trader, amountIn, leverage, type, triggerPrice, paymentUsdn, side, refLink, positionId, limitPrice)
175+ let expiration = if ((size(orderPartList) > 11))
176+ then valueOrErrorMessage(parseInt(orderPartList[11]), "Invalid expiration")
177+ else 0
178+ let marketPrice = if ((size(orderPartList) > 12))
179+ then valueOrErrorMessage(parseInt(orderPartList[12]), "Invalid orderMarketPrice")
180+ else 0
181+ $Tuple13(amm, trader, amountIn, leverage, type, triggerPrice, paymentUsdn, side, refLink, positionId, limitPrice, expiration, marketPrice)
170182 }
171183
172184
227239 }
228240
229241
230-func getSpread (_price) = muld(_price, SPREAD_LIMIT)
242+func getSpread (_price) = muld(_price, getSpreadLimit())
231243
232244
233-func saveOrder (_orderId,_amm,_trader,_amountIn,_leverage,_type,_triggerPrice,_paymentUsdn,_side,_refLink,_positionId,_limitPrice) = {
234- let orderStr = makeString([_amm, _trader, toString(_amountIn), toString(_leverage), toString(_type), toString(_triggerPrice), toString(_paymentUsdn), toString(_side), _refLink, toString(_positionId), toString(_limitPrice)], ",")
245+func saveOrder (_orderId,_amm,_trader,_amountIn,_leverage,_type,_triggerPrice,_paymentUsdn,_side,_refLink,_positionId,_limitPrice,_expiration,_orderMarketPrice) = {
246+ let orderStr = makeString([_amm, _trader, toString(_amountIn), toString(_leverage), toString(_type), toString(_triggerPrice), toString(_paymentUsdn), toString(_side), _refLink, toString(_positionId), toString(_limitPrice), toString(_expiration), toString(_orderMarketPrice)], ",")
235247 [StringEntry(orderKey(_orderId), orderStr)]
236248 }
237249
294306 let positionIds = getPositionIds(_amm, _trader)
295307 func cleanUpOne (_acc,_orderId) = {
296308 let orderIdInt = valueOrErrorMessage(parseInt(_orderId), "Invalid order id")
297- let $t01115611351 = getOrder(orderIdInt)
298- let _x1 = $t01115611351._1
299- let _x2 = $t01115611351._2
300- let _x3 = $t01115611351._3
301- let _x4 = $t01115611351._4
302- let _type = $t01115611351._5
303- let _x5 = $t01115611351._6
304- let _x6 = $t01115611351._7
305- let _x7 = $t01115611351._8
306- let _x8 = $t01115611351._9
307- let _positionId = $t01115611351._10
308- let _x9 = $t01115611351._11
309- if (if (if (if ((_type == STOP))
309+ let $t01176611982 = getOrder(orderIdInt)
310+ let _x1 = $t01176611982._1
311+ let _x2 = $t01176611982._2
312+ let _x3 = $t01176611982._3
313+ let _x4 = $t01176611982._4
314+ let _type = $t01176611982._5
315+ let _x5 = $t01176611982._6
316+ let _x6 = $t01176611982._7
317+ let _x7 = $t01176611982._8
318+ let _x8 = $t01176611982._9
319+ let _positionId = $t01176611982._10
320+ let _x9 = $t01176611982._11
321+ let _expiration = $t01176611982._12
322+ let cancelTakeStopOfNoPosition = if (if (if ((_type == STOP))
310323 then true
311324 else (_type == TAKE))
312325 then (positionIds[0] != _positionId)
313326 else false)
314327 then (positionIds[1] != _positionId)
315- else false)
328+ else false
329+ let cancelLimitIfExpired = if ((_type == LIMIT))
330+ then (lastBlock.timestamp >= _expiration)
331+ else false
332+ if (if (cancelTakeStopOfNoPosition)
333+ then true
334+ else cancelLimitIfExpired)
316335 then {
317336 let change = markCancelOrder(orderIdInt)
318337 let newOrderList = addRemoveOrderIdList(_acc._1, orderIdInt, _amm, _trader, false)
321340 else _acc
322341 }
323342
324- let $t01170411779 = {
343+ let $t01251412589 = {
325344 let $l = orders
326345 let $s = size($l)
327346 let $acc0 = $Tuple2(orders, nil)
335354
336355 $f0_2($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)
337356 }
338- let newOrders = $t01170411779._1
339- let cancelActions = $t01170411779._2
357+ let newOrders = $t01251412589._1
358+ let cancelActions = $t01251412589._2
340359 ((cancelActions ++ updateTraderOrderCount(_amm, _trader, size(newOrders))) ++ updateOrderIdStr(makeString(newOrders, ","), _amm, _trader))
341360 }
342361
357376
358377
359378 @Callable(i)
360-func initialize (_coordinator) = if (if (initialized())
379+func initialize (_coordinator,_spreadLimit) = if (if (if (initialized())
380+ then true
381+ else if ((0 > _spreadLimit))
382+ then true
383+ else (_spreadLimit > (DECIMAL_UNIT / 10)))
361384 then true
362385 else (i.caller != this))
363386 then throw("Unable to initialize")
364- else [StringEntry(k_coordinatorAddress, toString(addressFromStringValue(_coordinator))), BooleanEntry(k_initialized, true)]
387+ else [StringEntry(k_coordinatorAddress, toString(addressFromStringValue(_coordinator))), IntegerEntry(k_spreadLimit, _spreadLimit), BooleanEntry(k_initialized, true)]
365388
366389
367390
368391 @Callable(i)
369-func createOrder (_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice) = {
392+func changeSettings (_spreadLimit) = if (if ((i.caller != adminAddress()))
393+ then true
394+ else if ((0 > _spreadLimit))
395+ then true
396+ else (_spreadLimit > (DECIMAL_UNIT / 10)))
397+ then throw("Invalid changeSettings params")
398+ else [IntegerEntry(k_spreadLimit, _spreadLimit)]
399+
400+
401+
402+@Callable(i)
403+func createOrder (_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice,_expiration,_priceData) = {
370404 let _trader = toString(i.caller)
371- let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
372- if ((cleanUp == cleanUp))
373- then if ((size(i.payments) > 1))
374- then throw("Invalid createOrder parameters: invalid payment count")
375- else {
376- let $t01427514527 = if ((size(i.payments) == 1))
377- then $Tuple2(toBase58String(valueOrErrorMessage(i.payments[0].assetId, "Invalid asset id")), i.payments[0].amount)
378- else $Tuple2("", 0)
379- let paymentAssetId = $t01427514527._1
380- let paymentAmount = $t01427514527._2
381- let doCall = invoke(this, "internalCreateOrder", [_trader, _amm, _type, _triggerPrice, _limitPrice, _amountIn, _leverage, _side, _refLink, paymentAssetId, paymentAmount], nil)
382- if ((doCall == doCall))
383- then {
384- let orderId = match doCall {
385- case t: Int =>
386- t
387- case _ =>
388- throw("Invalid ID of created order")
389- }
390- if ((_type == LIMIT))
405+ let updatePrice = invoke(addressFromStringValue(_amm), "updateOracle", [_priceData], nil)
406+ if ((updatePrice == updatePrice))
407+ then {
408+ let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
409+ if ((cleanUp == cleanUp))
410+ then if ((size(i.payments) > 1))
411+ then throw("Invalid createOrder parameters: invalid payment count")
412+ else {
413+ let $t01570315955 = if ((size(i.payments) == 1))
414+ then $Tuple2(toBase58String(valueOrErrorMessage(i.payments[0].assetId, "Invalid asset id")), i.payments[0].amount)
415+ else $Tuple2("", 0)
416+ let paymentAssetId = $t01570315955._1
417+ let paymentAmount = $t01570315955._2
418+ let doCall = invoke(this, "internalCreateOrder", [_trader, _amm, _type, _triggerPrice, _limitPrice, _amountIn, _leverage, _side, _refLink, paymentAssetId, paymentAmount, _expiration], nil)
419+ if ((doCall == doCall))
391420 then {
392- let makeStop = if ((_stopTriggerPrice > 0))
421+ let orderId = match doCall {
422+ case t: Int =>
423+ t
424+ case _ =>
425+ throw("Invalid ID of created order")
426+ }
427+ if ((_type == LIMIT))
393428 then {
394- let doMakeStop = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, STOP, _stopTriggerPrice, _stopLimitPrice], nil)
395- if ((doMakeStop == doMakeStop))
396- then nil
397- else throw("Strict value is not equal to itself.")
429+ let makeStop = if ((_stopTriggerPrice > 0))
430+ then {
431+ let doMakeStop = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, STOP, _stopTriggerPrice, _stopLimitPrice], nil)
432+ if ((doMakeStop == doMakeStop))
433+ then nil
434+ else throw("Strict value is not equal to itself.")
435+ }
436+ else nil
437+ let makeTake = if ((_takeTriggerPrice > 0))
438+ then {
439+ let doMakeTake = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, TAKE, _takeTriggerPrice, _takeLimitPrice], nil)
440+ if ((doMakeTake == doMakeTake))
441+ then nil
442+ else throw("Strict value is not equal to itself.")
443+ }
444+ else nil
445+ (makeStop ++ makeTake)
398446 }
399447 else nil
400- let makeTake = if ((_takeTriggerPrice > 0))
401- then {
402- let doMakeTake = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, TAKE, _takeTriggerPrice, _takeLimitPrice], nil)
403- if ((doMakeTake == doMakeTake))
404- then nil
405- else throw("Strict value is not equal to itself.")
406- }
407- else nil
408- (makeStop ++ makeTake)
409448 }
410- else nil
449+ else throw("Strict value is not equal to itself.")
411450 }
412- else throw("Strict value is not equal to itself.")
413- }
451+ else throw("Strict value is not equal to itself.")
452+ }
414453 else throw("Strict value is not equal to itself.")
415454 }
416455
417456
418457
419458 @Callable(i)
420-func increasePositionWithStopLoss (_amm,_direction,_leverage,_minBaseAssetAmount,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice) = {
459+func increasePositionWithStopLoss (_amm,_direction,_leverage,_minBaseAssetAmount,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice,_priceData) = {
421460 let _trader = toString(i.caller)
422461 let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
423462 if ((cleanUp == cleanUp))
433472 let doSetContext = invoke(this, "setContext", [_trader], nil)
434473 if ((doSetContext == doSetContext))
435474 then {
436- let doClosePosition = invoke(addressFromStringValue(_amm), "increasePosition", [_direction, _leverage, _minBaseAssetAmount, _refLink], i.payments)
437- if ((doClosePosition == doClosePosition))
475+ let doOpenPosition = invoke(addressFromStringValue(_amm), "increasePosition", [_direction, _leverage, _minBaseAssetAmount, _refLink, _priceData], i.payments)
476+ if ((doOpenPosition == doOpenPosition))
438477 then {
439478 let doResetContext = invoke(this, "resetContext", nil, nil)
440479 if ((doResetContext == doResetContext))
443482 if ((openedPositionSize == openedPositionSize))
444483 then {
445484 let amountIn = abs(openedPositionSize)
446- let stopLossSide = if ((0 > openedPositionSize))
447- then LONG
448- else SHORT
449- let doCreateStopOrder = if ((_stopTriggerPrice > 0))
485+ if ((amountIn == amountIn))
450486 then {
451- let doCreateStopOrder = invoke(this, "internalCreateOrder", [_trader, _amm, STOP, _stopTriggerPrice, _stopLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0], nil)
452- if ((doCreateStopOrder == doCreateStopOrder))
453- then nil
454- else throw("Strict value is not equal to itself.")
455- }
456- else nil
457- if ((doCreateStopOrder == doCreateStopOrder))
458- then {
459- let doCreateTakeOrder = if ((_takeTriggerPrice > 0))
487+ let stopLossSide = if ((0 > openedPositionSize))
488+ then LONG
489+ else SHORT
490+ let checkAmountIn = if ((0 >= amountIn))
460491 then {
461- let doCreateTakeOrder = invoke(this, "internalCreateOrder", [_trader, _amm, TAKE, _takeTriggerPrice, _takeLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0], nil)
462- if ((doCreateTakeOrder == doCreateTakeOrder))
463- then nil
492+ let data = makeString([_amm, _trader, toString(_direction), toString(_leverage), toString(_minBaseAssetAmount), _refLink, _priceData], ",")
493+ throw(((("Invalid amountIn=" + toString(amountIn)) + " after success increasePosition: ") + data))
494+ }
495+ else 0
496+ if ((checkAmountIn == checkAmountIn))
497+ then {
498+ let doCreateStopOrder = if ((_stopTriggerPrice > 0))
499+ then {
500+ let doCreateStopOrder = invoke(this, "internalCreateOrder", [_trader, _amm, STOP, _stopTriggerPrice, _stopLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0, 0], nil)
501+ if ((doCreateStopOrder == doCreateStopOrder))
502+ then nil
503+ else throw("Strict value is not equal to itself.")
504+ }
505+ else nil
506+ if ((doCreateStopOrder == doCreateStopOrder))
507+ then {
508+ let doCreateTakeOrder = if ((_takeTriggerPrice > 0))
509+ then {
510+ let doCreateTakeOrder = invoke(this, "internalCreateOrder", [_trader, _amm, TAKE, _takeTriggerPrice, _takeLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0, 0], nil)
511+ if ((doCreateTakeOrder == doCreateTakeOrder))
512+ then nil
513+ else throw("Strict value is not equal to itself.")
514+ }
515+ else nil
516+ if ((doCreateTakeOrder == doCreateTakeOrder))
517+ then nil
518+ else throw("Strict value is not equal to itself.")
519+ }
464520 else throw("Strict value is not equal to itself.")
465521 }
466- else nil
467- if ((doCreateTakeOrder == doCreateTakeOrder))
468- then nil
469522 else throw("Strict value is not equal to itself.")
470523 }
471524 else throw("Strict value is not equal to itself.")
514567
515568
516569 @Callable(i)
517-func internalCreateOrder (_trader,_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_paymentAssetId,_paymentAmount) = {
518- let x1 = toString(!(initialized()))
519- let x2 = toString(!(isWhitelist(_amm)))
520- let x3 = toString((0 >= _triggerPrice))
521- let x4 = toString((0 > _limitPrice))
522- let x5 = toString((0 >= _amountIn))
523- let x6 = toString((0 > _leverage))
524- let x7 = toString(!(if ((_side == LONG))
570+func internalCreateOrder (_trader,_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_paymentAssetId,_paymentAmount,_expiration) = if (if (if (if (if (if (if (if (if (if (!(initialized()))
571+ then true
572+ else !(isWhitelist(_amm)))
573+ then true
574+ else (0 >= _triggerPrice))
575+ then true
576+ else (0 > _limitPrice))
577+ then true
578+ else (0 >= _amountIn))
579+ then true
580+ else (0 > _leverage))
581+ then true
582+ else !(if ((_side == LONG))
525583 then true
526584 else (_side == SHORT)))
527- let x8 = toString(!(if (if ((_type == STOP))
585+ then true
586+ else !(if (if ((_type == STOP))
528587 then true
529588 else (_type == TAKE))
530589 then true
531590 else (_type == LIMIT)))
532- let x9 = toString(!((i.caller == this)))
533- let all = makeString([x1, x2, x3, x4, x5, x6, x7, x8, x9], ",")
534- if (if (if (if (if (if (if (if (if (!(initialized()))
535- then true
536- else !(isWhitelist(_amm)))
537- then true
538- else (0 >= _triggerPrice))
539- then true
540- else (0 > _limitPrice))
541- then true
542- else (0 >= _amountIn))
543- then true
544- else (0 > _leverage))
545- then true
546- else !(if ((_side == LONG))
591+ then true
592+ else !((i.caller == this)))
593+ then true
594+ else (0 > _expiration))
595+ then throw("Invalid internalCreateOrder parameters")
596+ else {
597+ let orderId = (currentOrderId() + 1)
598+ let positionDirection = getPositionDirection(_side, _type)
599+ let newTraderOrderCount = (getTraderOrderCount(_amm, _trader) + 1)
600+ let positionSize = getPositionSize(_amm, _trader, positionDirection)
601+ let orderMarketPrice = getMarketPrice(_amm)
602+ let _direction = if (if (if ((positionSize == 0))
547603 then true
548- else (_side == SHORT)))
549- then true
550- else !(if (if ((_type == STOP))
604+ else if ((positionSize > 0))
605+ then (_side == LONG)
606+ else false)
551607 then true
552- else (_type == TAKE))
553- then true
554- else (_type == LIMIT)))
555- then true
556- else !((i.caller == this)))
557- then throw(("Invalid internalCreateOrder parameters " + all))
558- else {
559- let orderId = (currentOrderId() + 1)
560- let positionDirection = getPositionDirection(_side, _type)
561- let newTraderOrderCount = (getTraderOrderCount(_amm, _trader) + 1)
562- let positionSize = getPositionSize(_amm, _trader, positionDirection)
563- let _direction = if (if (if ((positionSize == 0))
608+ else if ((0 > positionSize))
609+ then (_side == SHORT)
610+ else false)
611+ then INCREASE
612+ else DECREASE
613+ if (if ((positionSize == 0))
614+ then if ((_type == STOP))
564615 then true
565- else if ((positionSize > 0))
566- then (_side == LONG)
567- else false)
568- then true
569- else if ((0 > positionSize))
570- then (_side == SHORT)
571- else false)
572- then INCREASE
573- else DECREASE
574- if (if ((positionSize == 0))
575- then if ((_type == STOP))
576- then true
577- else (_type == TAKE)
578- else false)
579- then throw("Can not create STOP/TAKE order: no position")
580- else {
581- let usdnPayment = if ((_direction == INCREASE))
582- then if (if ((_paymentAssetId != toBase58String(quoteAsset())))
616+ else (_type == TAKE)
617+ else false)
618+ then throw("Can not create STOP/TAKE order: no position")
619+ else {
620+ let usdnPayment = if ((_direction == INCREASE))
621+ then if (if ((_paymentAssetId != toBase58String(quoteAsset())))
622+ then true
623+ else (_paymentAmount != _amountIn))
624+ then throw("Invalid createLimitOrder parameters: invalid payment")
625+ else {
626+ let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _paymentAmount)])
627+ if ((stake == stake))
628+ then _paymentAmount
629+ else throw("Strict value is not equal to itself.")
630+ }
631+ else 0
632+ if ((usdnPayment == usdnPayment))
633+ then {
634+ let positionId = if ((positionSize != 0))
635+ then getPositionId(_amm, _trader, positionDirection)
636+ else 0
637+ if (if (if ((_type == STOP))
583638 then true
584- else (_paymentAmount != _amountIn))
585- then throw("Invalid createLimitOrder parameters: invalid payment")
586- else {
587- let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _paymentAmount)])
588- if ((stake == stake))
589- then _paymentAmount
590- else throw("Strict value is not equal to itself.")
591- }
592- else 0
593- if ((usdnPayment == usdnPayment))
594- then {
595- let positionId = if ((positionSize != 0))
596- then getPositionId(_amm, _trader, positionDirection)
597- else 0
598- if (if (if ((_type == STOP))
599- then true
600- else (_type == TAKE))
601- then (positionId == 0)
602- else false)
603- then throw("STOP and TAKE order should be assigned to position with id != 0")
604- else if ((newTraderOrderCount > MAX_TRADER_ORDERS_PER_AMM))
605- then throw("Invalid createLimitOrder parameters: order count")
606- else {
607- let changeSet = (((saveOrder(orderId, _amm, _trader, _amountIn, _leverage, _type, _triggerPrice, usdnPayment, _side, _refLink, positionId, _limitPrice) ++ addRemoveOrderId(orderId, _amm, _trader, true)) ++ updateTraderOrderCount(_amm, _trader, newTraderOrderCount)) ++ updateLastOrderId(orderId))
608- $Tuple2(changeSet, orderId)
609- }
610- }
611- else throw("Strict value is not equal to itself.")
612- }
613- }
614- }
639+ else (_type == TAKE))
640+ then (positionId == 0)
641+ else false)
642+ then throw("STOP and TAKE order should be assigned to position with id != 0")
643+ else if ((newTraderOrderCount > MAX_TRADER_ORDERS_PER_AMM))
644+ then throw("Invalid createLimitOrder parameters: order count")
645+ else {
646+ let changeSet = (((saveOrder(orderId, _amm, _trader, _amountIn, _leverage, _type, _triggerPrice, usdnPayment, _side, _refLink, positionId, _limitPrice, _expiration, orderMarketPrice) ++ addRemoveOrderId(orderId, _amm, _trader, true)) ++ updateTraderOrderCount(_amm, _trader, newTraderOrderCount)) ++ updateLastOrderId(orderId))
647+ $Tuple2(changeSet, orderId)
648+ }
649+ }
650+ else throw("Strict value is not equal to itself.")
651+ }
652+ }
615653
616654
617655
618656 @Callable(i)
619657 func cancelOrder (_orderId) = {
620- let $t02378624005 = getOrder(_orderId)
621- let _amm = $t02378624005._1
622- let _trader = $t02378624005._2
623- let _amountIn = $t02378624005._3
624- let _leverage = $t02378624005._4
625- let _type = $t02378624005._5
626- let _triggerPrice = $t02378624005._6
627- let _amountUsdn = $t02378624005._7
628- let _side = $t02378624005._8
629- let _refLink = $t02378624005._9
630- let _positionId = $t02378624005._10
631- let _limitPrice = $t02378624005._11
658+ let $t02559125741 = getOrder(_orderId)
659+ let _amm = $t02559125741._1
660+ let _trader = $t02559125741._2
661+ let _amountIn = $t02559125741._3
662+ let _leverage = $t02559125741._4
663+ let _type = $t02559125741._5
664+ let _triggerPrice = $t02559125741._6
665+ let _amountUsdn = $t02559125741._7
632666 if (if (if (!(initialized()))
633667 then true
634668 else !(isValid(_orderId)))
661695
662696
663697 @Callable(i)
664-func executeOrder (_orderId) = {
665- let $t02520325422 = getOrder(_orderId)
666- let _amm = $t02520325422._1
667- let _trader = $t02520325422._2
668- let _amountIn = $t02520325422._3
669- let _leverage = $t02520325422._4
670- let _type = $t02520325422._5
671- let _triggerPrice = $t02520325422._6
672- let _amountUsdn = $t02520325422._7
673- let _side = $t02520325422._8
674- let _refLink = $t02520325422._9
675- let _positionId = $t02520325422._10
676- let _limitPrice = $t02520325422._11
677- let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
678- if ((cleanUp == cleanUp))
698+func executeOrder (_orderId,_priceData) = {
699+ let $t02695927221 = getOrder(_orderId)
700+ let _amm = $t02695927221._1
701+ let _trader = $t02695927221._2
702+ let _amountIn = $t02695927221._3
703+ let _leverage = $t02695927221._4
704+ let _type = $t02695927221._5
705+ let _triggerPrice = $t02695927221._6
706+ let _amountUsdn = $t02695927221._7
707+ let _side = $t02695927221._8
708+ let _refLink = $t02695927221._9
709+ let _positionId = $t02695927221._10
710+ let _limitPrice = $t02695927221._11
711+ let _timestamp = $t02695927221._12
712+ let _orderMarketPrice = $t02695927221._13
713+ let updatePrice = invoke(addressFromStringValue(_amm), "updateOracle", [_priceData], nil)
714+ if ((updatePrice == updatePrice))
679715 then {
680- let positionDirection = getPositionDirection(_side, _type)
681- if (if (!(initialized()))
682- then true
683- else !(isValid(_orderId)))
684- then throw("Invalid executeOrder parameters")
685- else {
686- let positionSize = getPositionSize(_amm, _trader, positionDirection)
687- if ((positionSize == positionSize))
688- then {
689- let currentPositionId = if ((positionSize != 0))
690- then getPositionId(_amm, _trader, positionDirection)
691- else 0
692- let $t02592929307 = if ((_type == STOP))
716+ let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
717+ if ((cleanUp == cleanUp))
718+ then {
719+ let positionDirection = getPositionDirection(_side, _type)
720+ if (if (!(initialized()))
721+ then true
722+ else !(isValid(_orderId)))
723+ then throw("Invalid executeOrder parameters")
724+ else {
725+ let positionSize = getPositionSize(_amm, _trader, positionDirection)
726+ if ((positionSize == positionSize))
693727 then {
694- let _positionDirection = if ((positionSize > 0))
695- then LONG
696- else if ((0 > positionSize))
697- then SHORT
698- else throw("Can not execute STOP order: no open position")
699- let marketPrice = getMarketPrice(_amm)
700- let isExecutable = if ((_side == _positionDirection))
701- then throw("Can not execute STOP order: reduce only")
702- else if ((currentPositionId != _positionId))
703- then throw("Can not execute STOP order: position closed")
704- else if ((_positionDirection == LONG))
705- then (_triggerPrice >= marketPrice)
706- else (marketPrice >= _triggerPrice)
707- if (isExecutable)
708- then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false], nil)
709- else throw("Can not execute STOP order: triggerPrice mismatch")
710- }
711- else if ((_type == TAKE))
712- then {
713- let _positionDirection = if ((positionSize > 0))
714- then LONG
715- else if ((0 > positionSize))
716- then SHORT
717- else throw("Can not execute STOP order: no open position")
718- let marketPrice = getMarketPrice(_amm)
719- let isExecutable = if ((_side == _positionDirection))
720- then throw("Can not execute TAKE order: reduce only")
721- else if ((currentPositionId != _positionId))
722- then throw(((("Can not execute TAKE order: position closed " + toString(currentPositionId)) + "!=") + toString(_positionId)))
723- else if ((_positionDirection == LONG))
724- then (marketPrice >= _triggerPrice)
725- else (_triggerPrice >= marketPrice)
726- if (isExecutable)
727- then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false], nil)
728- else throw("Can not execute TAKE order: triggerPrice mismatch")
729- }
730- else if ((_type == LIMIT))
728+ let currentPositionId = if ((positionSize != 0))
729+ then getPositionId(_amm, _trader, positionDirection)
730+ else 0
731+ let $t02783031791 = if ((_type == STOP))
731732 then {
733+ let _positionDirection = if ((positionSize > 0))
734+ then LONG
735+ else if ((0 > positionSize))
736+ then SHORT
737+ else throw("Can not execute STOP order: no open position")
732738 let marketPrice = getMarketPrice(_amm)
733- let spread = if ((_limitPrice == 0))
734- then getSpread(_triggerPrice)
735- else abs((_triggerPrice - _limitPrice))
736- let isExecutable = if ((marketPrice >= (_triggerPrice - spread)))
737- then ((_triggerPrice + spread) >= marketPrice)
738- else false
739+ let isExecutable = if ((_side == _positionDirection))
740+ then throw("Can not execute STOP order: reduce only")
741+ else if ((currentPositionId != _positionId))
742+ then throw("Can not execute STOP order: position closed")
743+ else if ((_positionDirection == LONG))
744+ then (_triggerPrice >= marketPrice)
745+ else (marketPrice >= _triggerPrice)
739746 if (isExecutable)
747+ then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false, _priceData], nil)
748+ else throw("Can not execute STOP order: triggerPrice mismatch")
749+ }
750+ else if ((_type == TAKE))
751+ then {
752+ let _positionDirection = if ((positionSize > 0))
753+ then LONG
754+ else if ((0 > positionSize))
755+ then SHORT
756+ else throw("Can not execute STOP order: no open position")
757+ let marketPrice = getMarketPrice(_amm)
758+ let isExecutable = if ((_side == _positionDirection))
759+ then throw("Can not execute TAKE order: reduce only")
760+ else if ((currentPositionId != _positionId))
761+ then throw(((("Can not execute TAKE order: position closed " + toString(currentPositionId)) + "!=") + toString(_positionId)))
762+ else if ((_positionDirection == LONG))
763+ then (marketPrice >= _triggerPrice)
764+ else (_triggerPrice >= marketPrice)
765+ if (isExecutable)
766+ then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false, _priceData], nil)
767+ else throw("Can not execute TAKE order: triggerPrice mismatch")
768+ }
769+ else if ((_type == LIMIT))
740770 then {
741- let amountInWithFee = (_amountUsdn - muld(_amountUsdn, getFee(_amm, _trader)))
742- $Tuple3("increasePosition", [_side, _leverage, if ((_limitPrice == 0))
743- then 0
744- else divd(amountInWithFee, _limitPrice), _refLink], [AttachedPayment(quoteAsset(), _amountUsdn)])
771+ let marketPrice = getMarketPrice(_amm)
772+ let isExecutableByStop = if ((_limitPrice == 0))
773+ then true
774+ else if ((_limitPrice > _orderMarketPrice))
775+ then (marketPrice >= _limitPrice)
776+ else (_limitPrice >= marketPrice)
777+ let isExecutableForTriggerPrice = if ((_side == LONG))
778+ then (_triggerPrice >= marketPrice)
779+ else (marketPrice >= _triggerPrice)
780+ let isExecutable = if (isExecutableByStop)
781+ then isExecutableForTriggerPrice
782+ else false
783+ if (isExecutable)
784+ then {
785+ let marginAmount = divd(_amountUsdn, (muld(getFee(_amm, _trader), _leverage) + DECIMAL_UNIT))
786+ let openNotional = muld(marginAmount, _leverage)
787+ let idealMinBaseAssetAmount = divd(openNotional, _triggerPrice)
788+ let minBaseAssetAmount = (idealMinBaseAssetAmount - muld(idealMinBaseAssetAmount, getSpreadLimit()))
789+ $Tuple3("increasePosition", [_side, _leverage, minBaseAssetAmount, _refLink, _priceData], [AttachedPayment(quoteAsset(), _amountUsdn)])
790+ }
791+ else throw("Can not execute LIMIT order: triggerPrice mismatch")
745792 }
746- else throw("Can not execute LIMIT order: triggerPrice mismatch")
793+ else throw(("Invalid order type: " + toString(_type)))
794+ let method = $t02783031791._1
795+ let args = $t02783031791._2
796+ let payments = $t02783031791._3
797+ let withdraw = if ((size(payments) == 1))
798+ then {
799+ let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), payments[0].amount], nil)
800+ if ((unstake == unstake))
801+ then nil
802+ else throw("Strict value is not equal to itself.")
747803 }
748- else throw(("Invalid order type: " + toString(_type)))
749- let method = $t02592929307._1
750- let args = $t02592929307._2
751- let payments = $t02592929307._3
752- let withdraw = if ((size(payments) == 1))
753- then {
754- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), payments[0].amount], nil)
755- if ((unstake == unstake))
756- then nil
757- else throw("Strict value is not equal to itself.")
758- }
759- else nil
760- if ((withdraw == withdraw))
761- then {
762- let doSetContext = invoke(this, "setContext", [_trader], nil)
763- if ((doSetContext == doSetContext))
804+ else nil
805+ if ((withdraw == withdraw))
764806 then {
765- let doClosePosition = invoke(addressFromStringValue(_amm), method, args, payments)
766- if ((doClosePosition == doClosePosition))
807+ let doSetContext = invoke(this, "setContext", [_trader], nil)
808+ if ((doSetContext == doSetContext))
767809 then {
768- let doResetContext = invoke(this, "resetContext", nil, nil)
769- if ((doResetContext == doResetContext))
810+ let doClosePosition = invoke(addressFromStringValue(_amm), method, args, payments)
811+ if ((doClosePosition == doClosePosition))
770812 then {
771- let executeOrderRequest = if ((_type == LIMIT))
813+ let doResetContext = invoke(this, "resetContext", nil, nil)
814+ if ((doResetContext == doResetContext))
772815 then {
773- let newPositionSize = getPositionSize(_amm, _trader, positionDirection)
774- if ((newPositionSize == newPositionSize))
816+ let executeOrderRequest = if ((_type == LIMIT))
775817 then {
776- let newPositionId = getPositionId(_amm, _trader, positionDirection)
777- if ((newPositionId == newPositionId))
818+ let newPositionSize = getPositionSize(_amm, _trader, positionDirection)
819+ if ((newPositionSize == newPositionSize))
778820 then {
779- let positionSizeDelta = (newPositionSize - positionSize)
780- let closeOrderSide = if ((newPositionSize > 0))
781- then SHORT
782- else LONG
783- let makeTakeOrder = if (haveOrderRequest(_orderId, TAKE))
821+ let newPositionId = getPositionId(_amm, _trader, positionDirection)
822+ if ((newPositionId == newPositionId))
784823 then {
785- let $t03026930468 = getOrderRequest(_orderId, TAKE)
786- let newOrderId = $t03026930468._1
787- let newAmm = $t03026930468._2
788- let newTrader = $t03026930468._3
789- let newType = $t03026930468._4
790- let newTriggerPrice = $t03026930468._5
791- let newLimitPrice = $t03026930468._6
792- let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0], nil)
793- if ((doCall == doCall))
794- then nil
795- else throw("Strict value is not equal to itself.")
796- }
797- else nil
798- if ((makeTakeOrder == makeTakeOrder))
799- then {
800- let makeStopOrder = if (haveOrderRequest(_orderId, STOP))
824+ let positionSizeDelta = (newPositionSize - positionSize)
825+ let closeOrderSide = if ((newPositionSize > 0))
826+ then SHORT
827+ else LONG
828+ let makeTakeOrder = if (haveOrderRequest(_orderId, TAKE))
801829 then {
802- let $t03102431223 = getOrderRequest(_orderId, STOP)
803- let newOrderId = $t03102431223._1
804- let newAmm = $t03102431223._2
805- let newTrader = $t03102431223._3
806- let newType = $t03102431223._4
807- let newTriggerPrice = $t03102431223._5
808- let newLimitPrice = $t03102431223._6
809- let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0], nil)
830+ let $t03275332952 = getOrderRequest(_orderId, TAKE)
831+ let newOrderId = $t03275332952._1
832+ let newAmm = $t03275332952._2
833+ let newTrader = $t03275332952._3
834+ let newType = $t03275332952._4
835+ let newTriggerPrice = $t03275332952._5
836+ let newLimitPrice = $t03275332952._6
837+ let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0, 0], nil)
810838 if ((doCall == doCall))
811839 then nil
812840 else throw("Strict value is not equal to itself.")
813841 }
814842 else nil
815- if ((makeStopOrder == makeStopOrder))
816- then (makeTakeOrder ++ makeStopOrder)
843+ if ((makeTakeOrder == makeTakeOrder))
844+ then {
845+ let makeStopOrder = if (haveOrderRequest(_orderId, STOP))
846+ then {
847+ let $t03352533724 = getOrderRequest(_orderId, STOP)
848+ let newOrderId = $t03352533724._1
849+ let newAmm = $t03352533724._2
850+ let newTrader = $t03352533724._3
851+ let newType = $t03352533724._4
852+ let newTriggerPrice = $t03352533724._5
853+ let newLimitPrice = $t03352533724._6
854+ let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0, 0], nil)
855+ if ((doCall == doCall))
856+ then nil
857+ else throw("Strict value is not equal to itself.")
858+ }
859+ else nil
860+ if ((makeStopOrder == makeStopOrder))
861+ then (makeTakeOrder ++ makeStopOrder)
862+ else throw("Strict value is not equal to itself.")
863+ }
817864 else throw("Strict value is not equal to itself.")
818865 }
819866 else throw("Strict value is not equal to itself.")
820867 }
821868 else throw("Strict value is not equal to itself.")
822869 }
823- else throw("Strict value is not equal to itself.")
824- }
825- else nil
826- if ((executeOrderRequest == executeOrderRequest))
827- then {
828- let doMarkOrderExecuted = invoke(this, "internalMarkOrderExecuted", [_amm, _trader, _orderId], nil)
829- if ((doMarkOrderExecuted == doMarkOrderExecuted))
830- then nil
870+ else nil
871+ if ((executeOrderRequest == executeOrderRequest))
872+ then {
873+ let doMarkOrderExecuted = invoke(this, "internalMarkOrderExecuted", [_amm, _trader, _orderId], nil)
874+ if ((doMarkOrderExecuted == doMarkOrderExecuted))
875+ then nil
876+ else throw("Strict value is not equal to itself.")
877+ }
831878 else throw("Strict value is not equal to itself.")
832879 }
833880 else throw("Strict value is not equal to itself.")
840887 }
841888 else throw("Strict value is not equal to itself.")
842889 }
843- else throw("Strict value is not equal to itself.")
844890 }
891+ else throw("Strict value is not equal to itself.")
845892 }
846893 else throw("Strict value is not equal to itself.")
847894 }
849896
850897
851898 @Callable(i)
852-func view_canExecuteOrder (_orderId) = {
853- let s = invoke(this, "executeOrder", [_orderId], nil)
899+func view_canExecuteOrder (_orderId,_priceData) = {
900+ let s = invoke(this, "executeOrder", [_orderId, _priceData], nil)
854901 if ((s == s))
855902 then throw("Success")
856903 else throw("Strict value is not equal to itself.")
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let k_coordinatorAddress = "k_coordinatorAddress"
55
66 let k_admin_address = "k_admin_address"
77
88 let k_quote_asset = "k_quote_asset"
99
1010 let k_amm = "k_amm"
1111
1212 let k_manager_address = "k_manager_address"
1313
1414 let k_positionSequence = "k_positionSequence"
1515
1616 let k_positionSize = "k_positionSize"
1717
1818 let k_executedOrders = "k_executedOrders"
1919
2020 let k_canceledOrders = "k_canceledOrders"
2121
2222 let k_order = "k_order"
2323
2424 let k_orderRequest = "k_orderReq"
2525
2626 let k_lastOrderId = "k_lastOrderId"
2727
2828 let k_traderOrderCnt = "k_traderOrderCnt"
2929
3030 let k_traderOrderIds = "k_traderOrderIds"
3131
32+let k_spreadLimit = "k_spreadLimit"
33+
3234 let k_sender = "k_sender"
3335
3436 let k_initialized = "k_initialized"
3537
3638 let STOP = 1
3739
3840 let TAKE = 2
3941
4042 let LIMIT = 3
4143
4244 let LONG = 1
4345
4446 let SHORT = 2
4547
4648 let INCREASE = 1
4749
4850 let DECREASE = 2
4951
5052 let MAX_TRADER_ORDERS_PER_AMM = 10
5153
5254 let TIME = lastBlock.timestamp
5355
5456 let DECIMAL_UNIT = (1 * (((((10 * 10) * 10) * 10) * 10) * 10))
55-
56-let SPREAD_LIMIT = (DECIMAL_UNIT / 200)
5757
5858 func abs (_x) = if ((_x > 0))
5959 then _x
6060 else -(_x)
6161
6262
6363 func divd (_x,_y) = fraction(_x, DECIMAL_UNIT, _y, HALFEVEN)
6464
6565
6666 func muld (_x,_y) = fraction(_x, _y, DECIMAL_UNIT, HALFEVEN)
6767
6868
6969 func minv (_x,_y) = if ((_x > _y))
7070 then _y
7171 else _x
7272
7373
7474 func toCompositeKey (_key,_address) = ((_key + "_") + _address)
7575
7676
7777 func executedOrderKey (_orderId) = ((k_executedOrders + "_") + toString(_orderId))
7878
7979
8080 func canceledOrderKey (_orderId) = ((k_canceledOrders + "_") + toString(_orderId))
8181
8282
8383 func orderKey (_orderId) = toCompositeKey(k_order, toString(_orderId))
8484
8585
8686 func orderRequestKey (_orderId,_type) = ((((k_orderRequest + "_") + toString(_orderId)) + "_") + toString(_type))
8787
8888
8989 func traderOrderCountKey (_amm,_trader) = ((((k_traderOrderCnt + "_") + _amm) + "_") + _trader)
9090
9191
9292 func traderOrderIdsKey (_amm,_trader) = ((((k_traderOrderIds + "_") + _amm) + "_") + _trader)
9393
9494
9595 func coordinator () = valueOrErrorMessage(addressFromString(getStringValue(this, k_coordinatorAddress)), "Coordinator not set")
9696
9797
98+func adminAddress () = addressFromString(getStringValue(coordinator(), k_admin_address))
99+
100+
98101 func quoteAsset () = fromBase58String(getStringValue(coordinator(), k_quote_asset))
99102
100103
101104 func managerAddress () = valueOrErrorMessage(addressFromString(getStringValue(coordinator(), k_manager_address)), "Manager not set")
102105
103106
104107 func isWhitelist (_address) = valueOrElse(getBoolean(coordinator(), toCompositeKey(k_amm, _address)), false)
105108
106109
107110 func initialized () = valueOrElse(getBoolean(this, k_initialized), false)
108111
109112
110113 func isValid (_orderId) = if (valueOrElse(getBoolean(this, executedOrderKey(_orderId)), false))
111114 then throw(("Order already executed: " + toString(_orderId)))
112115 else if (valueOrElse(getBoolean(this, canceledOrderKey(_orderId)), false))
113116 then throw(("Order already cancelled: " + toString(_orderId)))
114117 else true
118+
119+
120+func getSpreadLimit () = valueOrErrorMessage(getInteger(this, k_spreadLimit), "Spread limit not set")
115121
116122
117123 func currentOrderId () = valueOrElse(getInteger(this, k_lastOrderId), 0)
118124
119125
120126 func getTraderOrderCount (_amm,_trader) = {
121127 let key = traderOrderCountKey(_amm, _trader)
122128 valueOrElse(getInteger(this, key), 0)
123129 }
124130
125131
126132 func traderAmmOrdersIds (_amm,_trader) = {
127133 let key = traderOrderIdsKey(_amm, _trader)
128134 let val = valueOrElse(getString(this, key), "")
129135 if ((val == ""))
130136 then nil
131137 else split(val, ",")
132138 }
133139
134140
135141 func haveOrderRequest (_orderId,_type) = {
136142 let key = orderRequestKey(_orderId, _type)
137143 isDefined(getString(this, key))
138144 }
139145
140146
141147 func getOrderRequest (_orderId,_type) = {
142148 let key = orderRequestKey(_orderId, _type)
143149 let orderRequestStr = valueOrErrorMessage(getString(this, key), ("Not order request for key: " + key))
144150 let orderRequestPartList = split(orderRequestStr, ",")
145151 let orderId = valueOrErrorMessage(parseInt(orderRequestPartList[0]), "Invalid orderId")
146152 let amm = orderRequestPartList[1]
147153 let trader = orderRequestPartList[2]
148154 let type = valueOrErrorMessage(parseInt(orderRequestPartList[3]), "Invalid type")
149155 let triggerPrice = valueOrErrorMessage(parseInt(orderRequestPartList[4]), "Invalid triggerPrice")
150156 let limitPrice = valueOrErrorMessage(parseInt(orderRequestPartList[5]), "Invalid limitPrice")
151157 $Tuple6(orderId, amm, trader, type, triggerPrice, limitPrice)
152158 }
153159
154160
155161 func getOrder (_orderId) = {
156162 let orderStr = valueOrErrorMessage(getString(this, orderKey(_orderId)), ("Invalid order id: " + toString(_orderId)))
157163 let orderPartList = split(orderStr, ",")
158164 let amm = orderPartList[0]
159165 let trader = orderPartList[1]
160166 let amountIn = valueOrErrorMessage(parseInt(orderPartList[2]), "Invalid amountIn")
161167 let leverage = valueOrErrorMessage(parseInt(orderPartList[3]), "Invalid leverage")
162168 let type = valueOrErrorMessage(parseInt(orderPartList[4]), "Invalid type")
163169 let triggerPrice = valueOrErrorMessage(parseInt(orderPartList[5]), "Invalid triggerPrice")
164170 let paymentUsdn = valueOrErrorMessage(parseInt(orderPartList[6]), "Invalid paymentUsdn")
165171 let side = valueOrErrorMessage(parseInt(orderPartList[7]), "Invalid side")
166172 let refLink = orderPartList[8]
167173 let positionId = valueOrErrorMessage(parseInt(orderPartList[9]), "Invalid positionId")
168174 let limitPrice = valueOrErrorMessage(parseInt(orderPartList[10]), "Invalid limitPrice")
169- $Tuple11(amm, trader, amountIn, leverage, type, triggerPrice, paymentUsdn, side, refLink, positionId, limitPrice)
175+ let expiration = if ((size(orderPartList) > 11))
176+ then valueOrErrorMessage(parseInt(orderPartList[11]), "Invalid expiration")
177+ else 0
178+ let marketPrice = if ((size(orderPartList) > 12))
179+ then valueOrErrorMessage(parseInt(orderPartList[12]), "Invalid orderMarketPrice")
180+ else 0
181+ $Tuple13(amm, trader, amountIn, leverage, type, triggerPrice, paymentUsdn, side, refLink, positionId, limitPrice, expiration, marketPrice)
170182 }
171183
172184
173185 func getMarketPrice (_amm) = {
174186 let s = invoke(addressFromStringValue(_amm), "computeSpotPrice", nil, nil)
175187 if ((s == s))
176188 then {
177189 let res = match s {
178190 case t: Int =>
179191 t
180192 case _ =>
181193 throw("Invalid computeSpotPrice result")
182194 }
183195 value(res)
184196 }
185197 else throw("Strict value is not equal to itself.")
186198 }
187199
188200
189201 func getFee (_amm,_trader) = {
190202 let s = invoke(addressFromStringValue(_amm), "computeFeeForTraderWithArtifact", [_trader, ""], nil)
191203 if ((s == s))
192204 then {
193205 let res = match s {
194206 case t: (Int, Boolean) =>
195207 t._1
196208 case _ =>
197209 throw("Invalid computeFeeForTraderWithArtifact result")
198210 }
199211 value(res)
200212 }
201213 else throw("Strict value is not equal to itself.")
202214 }
203215
204216
205217 func getPositionDirection (_orderSide,_orderType) = if (if ((_orderType == TAKE))
206218 then true
207219 else (_orderType == STOP))
208220 then if ((_orderSide == LONG))
209221 then SHORT
210222 else LONG
211223 else _orderSide
212224
213225
214226 func getPositionSize (_amm,_trader,_direction) = {
215227 let amm = addressFromStringValue(_amm)
216228 let positionKey = ((_trader + "_") + toString(_direction))
217229 let sizeKey = toCompositeKey(k_positionSize, positionKey)
218230 valueOrElse(getInteger(amm, sizeKey), 0)
219231 }
220232
221233
222234 func getPositionId (_amm,_trader,_direction) = {
223235 let amm = addressFromStringValue(_amm)
224236 let positionKey = ((_trader + "_") + toString(_direction))
225237 let seqKey = toCompositeKey(k_positionSequence, positionKey)
226238 valueOrElse(getInteger(amm, seqKey), 0)
227239 }
228240
229241
230-func getSpread (_price) = muld(_price, SPREAD_LIMIT)
242+func getSpread (_price) = muld(_price, getSpreadLimit())
231243
232244
233-func saveOrder (_orderId,_amm,_trader,_amountIn,_leverage,_type,_triggerPrice,_paymentUsdn,_side,_refLink,_positionId,_limitPrice) = {
234- let orderStr = makeString([_amm, _trader, toString(_amountIn), toString(_leverage), toString(_type), toString(_triggerPrice), toString(_paymentUsdn), toString(_side), _refLink, toString(_positionId), toString(_limitPrice)], ",")
245+func saveOrder (_orderId,_amm,_trader,_amountIn,_leverage,_type,_triggerPrice,_paymentUsdn,_side,_refLink,_positionId,_limitPrice,_expiration,_orderMarketPrice) = {
246+ let orderStr = makeString([_amm, _trader, toString(_amountIn), toString(_leverage), toString(_type), toString(_triggerPrice), toString(_paymentUsdn), toString(_side), _refLink, toString(_positionId), toString(_limitPrice), toString(_expiration), toString(_orderMarketPrice)], ",")
235247 [StringEntry(orderKey(_orderId), orderStr)]
236248 }
237249
238250
239251 func saveOrderRequest (_orderId,_amm,_trader,_type,_triggerPrice,_limitPrice) = {
240252 let orderReqStr = makeString([toString(_orderId), _amm, _trader, toString(_type), toString(_triggerPrice), toString(_limitPrice)], ",")
241253 [StringEntry(orderRequestKey(_orderId, _type), orderReqStr)]
242254 }
243255
244256
245257 func updateOrderIdStr (_orderIdsNewStr,_amm,_trader) = [StringEntry(traderOrderIdsKey(_amm, _trader), _orderIdsNewStr)]
246258
247259
248260 func addRemoveOrderIdList (_orderIds,_orderId,_amm,_trader,_add) = {
249261 let orderIdsNew = if (_add)
250262 then (_orderIds :+ toString(_orderId))
251263 else removeByIndex(_orderIds, valueOrErrorMessage(indexOf(_orderIds, toString(_orderId)), ("No order with id: " + toString(_orderId))))
252264 orderIdsNew
253265 }
254266
255267
256268 func addRemoveOrderId (_orderId,_amm,_trader,_add) = {
257269 let orderIds = traderAmmOrdersIds(_amm, _trader)
258270 let orderIdsNew = addRemoveOrderIdList(orderIds, _orderId, _amm, _trader, _add)
259271 let orderIdsNewStr = makeString(orderIdsNew, ",")
260272 updateOrderIdStr(orderIdsNewStr, _amm, _trader)
261273 }
262274
263275
264276 func updateTraderOrderCount (_amm,_trader,_count) = if ((0 > _count))
265277 then throw(("Invalid order count: " + toString(_count)))
266278 else [IntegerEntry(traderOrderCountKey(_amm, _trader), _count)]
267279
268280
269281 func updateLastOrderId (_lastOrderId) = [IntegerEntry(k_lastOrderId, _lastOrderId)]
270282
271283
272284 func markExecuteOrder (_orderId) = [BooleanEntry(toCompositeKey(k_executedOrders, toString(_orderId)), true)]
273285
274286
275287 func markCancelOrder (_orderId) = [BooleanEntry(toCompositeKey(k_canceledOrders, toString(_orderId)), true)]
276288
277289
278290 func getPositionIds (_amm,_trader) = {
279291 let longPositionSize = getPositionSize(_amm, _trader, LONG)
280292 let currentLongPositionId = if ((longPositionSize != 0))
281293 then getPositionId(_amm, _trader, LONG)
282294 else 0
283295 let shortPositionSize = getPositionSize(_amm, _trader, SHORT)
284296 let currentShortPositionId = if ((shortPositionSize != 0))
285297 then getPositionId(_amm, _trader, SHORT)
286298 else 0
287299 [currentLongPositionId, currentShortPositionId]
288300 }
289301
290302
291303 @Callable(i)
292304 func cleanUpStaleOrders (_amm,_trader) = {
293305 let orders = traderAmmOrdersIds(_amm, _trader)
294306 let positionIds = getPositionIds(_amm, _trader)
295307 func cleanUpOne (_acc,_orderId) = {
296308 let orderIdInt = valueOrErrorMessage(parseInt(_orderId), "Invalid order id")
297- let $t01115611351 = getOrder(orderIdInt)
298- let _x1 = $t01115611351._1
299- let _x2 = $t01115611351._2
300- let _x3 = $t01115611351._3
301- let _x4 = $t01115611351._4
302- let _type = $t01115611351._5
303- let _x5 = $t01115611351._6
304- let _x6 = $t01115611351._7
305- let _x7 = $t01115611351._8
306- let _x8 = $t01115611351._9
307- let _positionId = $t01115611351._10
308- let _x9 = $t01115611351._11
309- if (if (if (if ((_type == STOP))
309+ let $t01176611982 = getOrder(orderIdInt)
310+ let _x1 = $t01176611982._1
311+ let _x2 = $t01176611982._2
312+ let _x3 = $t01176611982._3
313+ let _x4 = $t01176611982._4
314+ let _type = $t01176611982._5
315+ let _x5 = $t01176611982._6
316+ let _x6 = $t01176611982._7
317+ let _x7 = $t01176611982._8
318+ let _x8 = $t01176611982._9
319+ let _positionId = $t01176611982._10
320+ let _x9 = $t01176611982._11
321+ let _expiration = $t01176611982._12
322+ let cancelTakeStopOfNoPosition = if (if (if ((_type == STOP))
310323 then true
311324 else (_type == TAKE))
312325 then (positionIds[0] != _positionId)
313326 else false)
314327 then (positionIds[1] != _positionId)
315- else false)
328+ else false
329+ let cancelLimitIfExpired = if ((_type == LIMIT))
330+ then (lastBlock.timestamp >= _expiration)
331+ else false
332+ if (if (cancelTakeStopOfNoPosition)
333+ then true
334+ else cancelLimitIfExpired)
316335 then {
317336 let change = markCancelOrder(orderIdInt)
318337 let newOrderList = addRemoveOrderIdList(_acc._1, orderIdInt, _amm, _trader, false)
319338 $Tuple2(newOrderList, (_acc._2 ++ change))
320339 }
321340 else _acc
322341 }
323342
324- let $t01170411779 = {
343+ let $t01251412589 = {
325344 let $l = orders
326345 let $s = size($l)
327346 let $acc0 = $Tuple2(orders, nil)
328347 func $f0_1 ($a,$i) = if (($i >= $s))
329348 then $a
330349 else cleanUpOne($a, $l[$i])
331350
332351 func $f0_2 ($a,$i) = if (($i >= $s))
333352 then $a
334353 else throw("List size exceeds 10")
335354
336355 $f0_2($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)
337356 }
338- let newOrders = $t01170411779._1
339- let cancelActions = $t01170411779._2
357+ let newOrders = $t01251412589._1
358+ let cancelActions = $t01251412589._2
340359 ((cancelActions ++ updateTraderOrderCount(_amm, _trader, size(newOrders))) ++ updateOrderIdStr(makeString(newOrders, ","), _amm, _trader))
341360 }
342361
343362
344363
345364 @Callable(i)
346365 func setContext (_sender) = if ((i.caller != this))
347366 then throw("Only self-call")
348367 else [StringEntry(k_sender, _sender)]
349368
350369
351370
352371 @Callable(i)
353372 func resetContext () = if ((i.caller != this))
354373 then throw("Only self-call")
355374 else [DeleteEntry(k_sender)]
356375
357376
358377
359378 @Callable(i)
360-func initialize (_coordinator) = if (if (initialized())
379+func initialize (_coordinator,_spreadLimit) = if (if (if (initialized())
380+ then true
381+ else if ((0 > _spreadLimit))
382+ then true
383+ else (_spreadLimit > (DECIMAL_UNIT / 10)))
361384 then true
362385 else (i.caller != this))
363386 then throw("Unable to initialize")
364- else [StringEntry(k_coordinatorAddress, toString(addressFromStringValue(_coordinator))), BooleanEntry(k_initialized, true)]
387+ else [StringEntry(k_coordinatorAddress, toString(addressFromStringValue(_coordinator))), IntegerEntry(k_spreadLimit, _spreadLimit), BooleanEntry(k_initialized, true)]
365388
366389
367390
368391 @Callable(i)
369-func createOrder (_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice) = {
392+func changeSettings (_spreadLimit) = if (if ((i.caller != adminAddress()))
393+ then true
394+ else if ((0 > _spreadLimit))
395+ then true
396+ else (_spreadLimit > (DECIMAL_UNIT / 10)))
397+ then throw("Invalid changeSettings params")
398+ else [IntegerEntry(k_spreadLimit, _spreadLimit)]
399+
400+
401+
402+@Callable(i)
403+func createOrder (_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice,_expiration,_priceData) = {
370404 let _trader = toString(i.caller)
371- let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
372- if ((cleanUp == cleanUp))
373- then if ((size(i.payments) > 1))
374- then throw("Invalid createOrder parameters: invalid payment count")
375- else {
376- let $t01427514527 = if ((size(i.payments) == 1))
377- then $Tuple2(toBase58String(valueOrErrorMessage(i.payments[0].assetId, "Invalid asset id")), i.payments[0].amount)
378- else $Tuple2("", 0)
379- let paymentAssetId = $t01427514527._1
380- let paymentAmount = $t01427514527._2
381- let doCall = invoke(this, "internalCreateOrder", [_trader, _amm, _type, _triggerPrice, _limitPrice, _amountIn, _leverage, _side, _refLink, paymentAssetId, paymentAmount], nil)
382- if ((doCall == doCall))
383- then {
384- let orderId = match doCall {
385- case t: Int =>
386- t
387- case _ =>
388- throw("Invalid ID of created order")
389- }
390- if ((_type == LIMIT))
405+ let updatePrice = invoke(addressFromStringValue(_amm), "updateOracle", [_priceData], nil)
406+ if ((updatePrice == updatePrice))
407+ then {
408+ let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
409+ if ((cleanUp == cleanUp))
410+ then if ((size(i.payments) > 1))
411+ then throw("Invalid createOrder parameters: invalid payment count")
412+ else {
413+ let $t01570315955 = if ((size(i.payments) == 1))
414+ then $Tuple2(toBase58String(valueOrErrorMessage(i.payments[0].assetId, "Invalid asset id")), i.payments[0].amount)
415+ else $Tuple2("", 0)
416+ let paymentAssetId = $t01570315955._1
417+ let paymentAmount = $t01570315955._2
418+ let doCall = invoke(this, "internalCreateOrder", [_trader, _amm, _type, _triggerPrice, _limitPrice, _amountIn, _leverage, _side, _refLink, paymentAssetId, paymentAmount, _expiration], nil)
419+ if ((doCall == doCall))
391420 then {
392- let makeStop = if ((_stopTriggerPrice > 0))
421+ let orderId = match doCall {
422+ case t: Int =>
423+ t
424+ case _ =>
425+ throw("Invalid ID of created order")
426+ }
427+ if ((_type == LIMIT))
393428 then {
394- let doMakeStop = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, STOP, _stopTriggerPrice, _stopLimitPrice], nil)
395- if ((doMakeStop == doMakeStop))
396- then nil
397- else throw("Strict value is not equal to itself.")
429+ let makeStop = if ((_stopTriggerPrice > 0))
430+ then {
431+ let doMakeStop = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, STOP, _stopTriggerPrice, _stopLimitPrice], nil)
432+ if ((doMakeStop == doMakeStop))
433+ then nil
434+ else throw("Strict value is not equal to itself.")
435+ }
436+ else nil
437+ let makeTake = if ((_takeTriggerPrice > 0))
438+ then {
439+ let doMakeTake = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, TAKE, _takeTriggerPrice, _takeLimitPrice], nil)
440+ if ((doMakeTake == doMakeTake))
441+ then nil
442+ else throw("Strict value is not equal to itself.")
443+ }
444+ else nil
445+ (makeStop ++ makeTake)
398446 }
399447 else nil
400- let makeTake = if ((_takeTriggerPrice > 0))
401- then {
402- let doMakeTake = invoke(this, "internalCreateOrderRequest", [_trader, _amm, orderId, TAKE, _takeTriggerPrice, _takeLimitPrice], nil)
403- if ((doMakeTake == doMakeTake))
404- then nil
405- else throw("Strict value is not equal to itself.")
406- }
407- else nil
408- (makeStop ++ makeTake)
409448 }
410- else nil
449+ else throw("Strict value is not equal to itself.")
411450 }
412- else throw("Strict value is not equal to itself.")
413- }
451+ else throw("Strict value is not equal to itself.")
452+ }
414453 else throw("Strict value is not equal to itself.")
415454 }
416455
417456
418457
419458 @Callable(i)
420-func increasePositionWithStopLoss (_amm,_direction,_leverage,_minBaseAssetAmount,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice) = {
459+func increasePositionWithStopLoss (_amm,_direction,_leverage,_minBaseAssetAmount,_refLink,_stopTriggerPrice,_stopLimitPrice,_takeTriggerPrice,_takeLimitPrice,_priceData) = {
421460 let _trader = toString(i.caller)
422461 let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
423462 if ((cleanUp == cleanUp))
424463 then if (if (!(initialized()))
425464 then true
426465 else !(isWhitelist(_amm)))
427466 then throw("Invalid increasePositionWithStopLoss parameters")
428467 else {
429468 let positionSize = getPositionSize(_amm, _trader, _direction)
430469 if ((positionSize != 0))
431470 then throw("Invalid increasePositionWithStopLoss parameters: only new position")
432471 else {
433472 let doSetContext = invoke(this, "setContext", [_trader], nil)
434473 if ((doSetContext == doSetContext))
435474 then {
436- let doClosePosition = invoke(addressFromStringValue(_amm), "increasePosition", [_direction, _leverage, _minBaseAssetAmount, _refLink], i.payments)
437- if ((doClosePosition == doClosePosition))
475+ let doOpenPosition = invoke(addressFromStringValue(_amm), "increasePosition", [_direction, _leverage, _minBaseAssetAmount, _refLink, _priceData], i.payments)
476+ if ((doOpenPosition == doOpenPosition))
438477 then {
439478 let doResetContext = invoke(this, "resetContext", nil, nil)
440479 if ((doResetContext == doResetContext))
441480 then {
442481 let openedPositionSize = getPositionSize(_amm, _trader, _direction)
443482 if ((openedPositionSize == openedPositionSize))
444483 then {
445484 let amountIn = abs(openedPositionSize)
446- let stopLossSide = if ((0 > openedPositionSize))
447- then LONG
448- else SHORT
449- let doCreateStopOrder = if ((_stopTriggerPrice > 0))
485+ if ((amountIn == amountIn))
450486 then {
451- let doCreateStopOrder = invoke(this, "internalCreateOrder", [_trader, _amm, STOP, _stopTriggerPrice, _stopLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0], nil)
452- if ((doCreateStopOrder == doCreateStopOrder))
453- then nil
454- else throw("Strict value is not equal to itself.")
455- }
456- else nil
457- if ((doCreateStopOrder == doCreateStopOrder))
458- then {
459- let doCreateTakeOrder = if ((_takeTriggerPrice > 0))
487+ let stopLossSide = if ((0 > openedPositionSize))
488+ then LONG
489+ else SHORT
490+ let checkAmountIn = if ((0 >= amountIn))
460491 then {
461- let doCreateTakeOrder = invoke(this, "internalCreateOrder", [_trader, _amm, TAKE, _takeTriggerPrice, _takeLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0], nil)
462- if ((doCreateTakeOrder == doCreateTakeOrder))
463- then nil
492+ let data = makeString([_amm, _trader, toString(_direction), toString(_leverage), toString(_minBaseAssetAmount), _refLink, _priceData], ",")
493+ throw(((("Invalid amountIn=" + toString(amountIn)) + " after success increasePosition: ") + data))
494+ }
495+ else 0
496+ if ((checkAmountIn == checkAmountIn))
497+ then {
498+ let doCreateStopOrder = if ((_stopTriggerPrice > 0))
499+ then {
500+ let doCreateStopOrder = invoke(this, "internalCreateOrder", [_trader, _amm, STOP, _stopTriggerPrice, _stopLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0, 0], nil)
501+ if ((doCreateStopOrder == doCreateStopOrder))
502+ then nil
503+ else throw("Strict value is not equal to itself.")
504+ }
505+ else nil
506+ if ((doCreateStopOrder == doCreateStopOrder))
507+ then {
508+ let doCreateTakeOrder = if ((_takeTriggerPrice > 0))
509+ then {
510+ let doCreateTakeOrder = invoke(this, "internalCreateOrder", [_trader, _amm, TAKE, _takeTriggerPrice, _takeLimitPrice, amountIn, 0, stopLossSide, _refLink, "", 0, 0], nil)
511+ if ((doCreateTakeOrder == doCreateTakeOrder))
512+ then nil
513+ else throw("Strict value is not equal to itself.")
514+ }
515+ else nil
516+ if ((doCreateTakeOrder == doCreateTakeOrder))
517+ then nil
518+ else throw("Strict value is not equal to itself.")
519+ }
464520 else throw("Strict value is not equal to itself.")
465521 }
466- else nil
467- if ((doCreateTakeOrder == doCreateTakeOrder))
468- then nil
469522 else throw("Strict value is not equal to itself.")
470523 }
471524 else throw("Strict value is not equal to itself.")
472525 }
473526 else throw("Strict value is not equal to itself.")
474527 }
475528 else throw("Strict value is not equal to itself.")
476529 }
477530 else throw("Strict value is not equal to itself.")
478531 }
479532 else throw("Strict value is not equal to itself.")
480533 }
481534 }
482535 else throw("Strict value is not equal to itself.")
483536 }
484537
485538
486539
487540 @Callable(i)
488541 func internalMarkOrderExecuted (_amm,_trader,_orderId) = if (if (if (!(initialized()))
489542 then true
490543 else !(isWhitelist(_amm)))
491544 then true
492545 else !((i.caller == this)))
493546 then throw("Invalid internalMarkOrderExecuted parameters")
494547 else {
495548 let newTraderOrderCount = (getTraderOrderCount(_amm, _trader) - 1)
496549 ((updateTraderOrderCount(_amm, _trader, newTraderOrderCount) ++ addRemoveOrderId(_orderId, _amm, _trader, false)) ++ markExecuteOrder(_orderId))
497550 }
498551
499552
500553
501554 @Callable(i)
502555 func internalCreateOrderRequest (_trader,_amm,_orderId,_type,_triggerPrice,_limitPrice) = if (if (if (if (if (!(initialized()))
503556 then true
504557 else !(isWhitelist(_amm)))
505558 then true
506559 else (0 >= _triggerPrice))
507560 then true
508561 else (0 > _limitPrice))
509562 then true
510563 else !((i.caller == this)))
511564 then throw("Invalid internalCreateOrderRequest parameters")
512565 else saveOrderRequest(_orderId, _amm, _trader, _type, _triggerPrice, _limitPrice)
513566
514567
515568
516569 @Callable(i)
517-func internalCreateOrder (_trader,_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_paymentAssetId,_paymentAmount) = {
518- let x1 = toString(!(initialized()))
519- let x2 = toString(!(isWhitelist(_amm)))
520- let x3 = toString((0 >= _triggerPrice))
521- let x4 = toString((0 > _limitPrice))
522- let x5 = toString((0 >= _amountIn))
523- let x6 = toString((0 > _leverage))
524- let x7 = toString(!(if ((_side == LONG))
570+func internalCreateOrder (_trader,_amm,_type,_triggerPrice,_limitPrice,_amountIn,_leverage,_side,_refLink,_paymentAssetId,_paymentAmount,_expiration) = if (if (if (if (if (if (if (if (if (if (!(initialized()))
571+ then true
572+ else !(isWhitelist(_amm)))
573+ then true
574+ else (0 >= _triggerPrice))
575+ then true
576+ else (0 > _limitPrice))
577+ then true
578+ else (0 >= _amountIn))
579+ then true
580+ else (0 > _leverage))
581+ then true
582+ else !(if ((_side == LONG))
525583 then true
526584 else (_side == SHORT)))
527- let x8 = toString(!(if (if ((_type == STOP))
585+ then true
586+ else !(if (if ((_type == STOP))
528587 then true
529588 else (_type == TAKE))
530589 then true
531590 else (_type == LIMIT)))
532- let x9 = toString(!((i.caller == this)))
533- let all = makeString([x1, x2, x3, x4, x5, x6, x7, x8, x9], ",")
534- if (if (if (if (if (if (if (if (if (!(initialized()))
535- then true
536- else !(isWhitelist(_amm)))
537- then true
538- else (0 >= _triggerPrice))
539- then true
540- else (0 > _limitPrice))
541- then true
542- else (0 >= _amountIn))
543- then true
544- else (0 > _leverage))
545- then true
546- else !(if ((_side == LONG))
591+ then true
592+ else !((i.caller == this)))
593+ then true
594+ else (0 > _expiration))
595+ then throw("Invalid internalCreateOrder parameters")
596+ else {
597+ let orderId = (currentOrderId() + 1)
598+ let positionDirection = getPositionDirection(_side, _type)
599+ let newTraderOrderCount = (getTraderOrderCount(_amm, _trader) + 1)
600+ let positionSize = getPositionSize(_amm, _trader, positionDirection)
601+ let orderMarketPrice = getMarketPrice(_amm)
602+ let _direction = if (if (if ((positionSize == 0))
547603 then true
548- else (_side == SHORT)))
549- then true
550- else !(if (if ((_type == STOP))
604+ else if ((positionSize > 0))
605+ then (_side == LONG)
606+ else false)
551607 then true
552- else (_type == TAKE))
553- then true
554- else (_type == LIMIT)))
555- then true
556- else !((i.caller == this)))
557- then throw(("Invalid internalCreateOrder parameters " + all))
558- else {
559- let orderId = (currentOrderId() + 1)
560- let positionDirection = getPositionDirection(_side, _type)
561- let newTraderOrderCount = (getTraderOrderCount(_amm, _trader) + 1)
562- let positionSize = getPositionSize(_amm, _trader, positionDirection)
563- let _direction = if (if (if ((positionSize == 0))
608+ else if ((0 > positionSize))
609+ then (_side == SHORT)
610+ else false)
611+ then INCREASE
612+ else DECREASE
613+ if (if ((positionSize == 0))
614+ then if ((_type == STOP))
564615 then true
565- else if ((positionSize > 0))
566- then (_side == LONG)
567- else false)
568- then true
569- else if ((0 > positionSize))
570- then (_side == SHORT)
571- else false)
572- then INCREASE
573- else DECREASE
574- if (if ((positionSize == 0))
575- then if ((_type == STOP))
576- then true
577- else (_type == TAKE)
578- else false)
579- then throw("Can not create STOP/TAKE order: no position")
580- else {
581- let usdnPayment = if ((_direction == INCREASE))
582- then if (if ((_paymentAssetId != toBase58String(quoteAsset())))
616+ else (_type == TAKE)
617+ else false)
618+ then throw("Can not create STOP/TAKE order: no position")
619+ else {
620+ let usdnPayment = if ((_direction == INCREASE))
621+ then if (if ((_paymentAssetId != toBase58String(quoteAsset())))
622+ then true
623+ else (_paymentAmount != _amountIn))
624+ then throw("Invalid createLimitOrder parameters: invalid payment")
625+ else {
626+ let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _paymentAmount)])
627+ if ((stake == stake))
628+ then _paymentAmount
629+ else throw("Strict value is not equal to itself.")
630+ }
631+ else 0
632+ if ((usdnPayment == usdnPayment))
633+ then {
634+ let positionId = if ((positionSize != 0))
635+ then getPositionId(_amm, _trader, positionDirection)
636+ else 0
637+ if (if (if ((_type == STOP))
583638 then true
584- else (_paymentAmount != _amountIn))
585- then throw("Invalid createLimitOrder parameters: invalid payment")
586- else {
587- let stake = invoke(managerAddress(), "deposit", nil, [AttachedPayment(quoteAsset(), _paymentAmount)])
588- if ((stake == stake))
589- then _paymentAmount
590- else throw("Strict value is not equal to itself.")
591- }
592- else 0
593- if ((usdnPayment == usdnPayment))
594- then {
595- let positionId = if ((positionSize != 0))
596- then getPositionId(_amm, _trader, positionDirection)
597- else 0
598- if (if (if ((_type == STOP))
599- then true
600- else (_type == TAKE))
601- then (positionId == 0)
602- else false)
603- then throw("STOP and TAKE order should be assigned to position with id != 0")
604- else if ((newTraderOrderCount > MAX_TRADER_ORDERS_PER_AMM))
605- then throw("Invalid createLimitOrder parameters: order count")
606- else {
607- let changeSet = (((saveOrder(orderId, _amm, _trader, _amountIn, _leverage, _type, _triggerPrice, usdnPayment, _side, _refLink, positionId, _limitPrice) ++ addRemoveOrderId(orderId, _amm, _trader, true)) ++ updateTraderOrderCount(_amm, _trader, newTraderOrderCount)) ++ updateLastOrderId(orderId))
608- $Tuple2(changeSet, orderId)
609- }
610- }
611- else throw("Strict value is not equal to itself.")
612- }
613- }
614- }
639+ else (_type == TAKE))
640+ then (positionId == 0)
641+ else false)
642+ then throw("STOP and TAKE order should be assigned to position with id != 0")
643+ else if ((newTraderOrderCount > MAX_TRADER_ORDERS_PER_AMM))
644+ then throw("Invalid createLimitOrder parameters: order count")
645+ else {
646+ let changeSet = (((saveOrder(orderId, _amm, _trader, _amountIn, _leverage, _type, _triggerPrice, usdnPayment, _side, _refLink, positionId, _limitPrice, _expiration, orderMarketPrice) ++ addRemoveOrderId(orderId, _amm, _trader, true)) ++ updateTraderOrderCount(_amm, _trader, newTraderOrderCount)) ++ updateLastOrderId(orderId))
647+ $Tuple2(changeSet, orderId)
648+ }
649+ }
650+ else throw("Strict value is not equal to itself.")
651+ }
652+ }
615653
616654
617655
618656 @Callable(i)
619657 func cancelOrder (_orderId) = {
620- let $t02378624005 = getOrder(_orderId)
621- let _amm = $t02378624005._1
622- let _trader = $t02378624005._2
623- let _amountIn = $t02378624005._3
624- let _leverage = $t02378624005._4
625- let _type = $t02378624005._5
626- let _triggerPrice = $t02378624005._6
627- let _amountUsdn = $t02378624005._7
628- let _side = $t02378624005._8
629- let _refLink = $t02378624005._9
630- let _positionId = $t02378624005._10
631- let _limitPrice = $t02378624005._11
658+ let $t02559125741 = getOrder(_orderId)
659+ let _amm = $t02559125741._1
660+ let _trader = $t02559125741._2
661+ let _amountIn = $t02559125741._3
662+ let _leverage = $t02559125741._4
663+ let _type = $t02559125741._5
664+ let _triggerPrice = $t02559125741._6
665+ let _amountUsdn = $t02559125741._7
632666 if (if (if (!(initialized()))
633667 then true
634668 else !(isValid(_orderId)))
635669 then true
636670 else !((toString(i.caller) == _trader)))
637671 then throw("Invalid cancelOrder parameters")
638672 else {
639673 let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
640674 if ((cleanUp == cleanUp))
641675 then {
642676 let newTraderOrderCount = (getTraderOrderCount(_amm, _trader) - 1)
643677 let withdraw = if ((_amountUsdn > 0))
644678 then {
645679 let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), _amountUsdn], nil)
646680 if ((unstake == unstake))
647681 then nil
648682 else throw("Strict value is not equal to itself.")
649683 }
650684 else nil
651685 if ((withdraw == withdraw))
652686 then (((markCancelOrder(_orderId) ++ addRemoveOrderId(_orderId, _amm, _trader, false)) ++ updateTraderOrderCount(_amm, _trader, newTraderOrderCount)) ++ (if ((_amountUsdn > 0))
653687 then [ScriptTransfer(i.caller, _amountUsdn, quoteAsset())]
654688 else nil))
655689 else throw("Strict value is not equal to itself.")
656690 }
657691 else throw("Strict value is not equal to itself.")
658692 }
659693 }
660694
661695
662696
663697 @Callable(i)
664-func executeOrder (_orderId) = {
665- let $t02520325422 = getOrder(_orderId)
666- let _amm = $t02520325422._1
667- let _trader = $t02520325422._2
668- let _amountIn = $t02520325422._3
669- let _leverage = $t02520325422._4
670- let _type = $t02520325422._5
671- let _triggerPrice = $t02520325422._6
672- let _amountUsdn = $t02520325422._7
673- let _side = $t02520325422._8
674- let _refLink = $t02520325422._9
675- let _positionId = $t02520325422._10
676- let _limitPrice = $t02520325422._11
677- let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
678- if ((cleanUp == cleanUp))
698+func executeOrder (_orderId,_priceData) = {
699+ let $t02695927221 = getOrder(_orderId)
700+ let _amm = $t02695927221._1
701+ let _trader = $t02695927221._2
702+ let _amountIn = $t02695927221._3
703+ let _leverage = $t02695927221._4
704+ let _type = $t02695927221._5
705+ let _triggerPrice = $t02695927221._6
706+ let _amountUsdn = $t02695927221._7
707+ let _side = $t02695927221._8
708+ let _refLink = $t02695927221._9
709+ let _positionId = $t02695927221._10
710+ let _limitPrice = $t02695927221._11
711+ let _timestamp = $t02695927221._12
712+ let _orderMarketPrice = $t02695927221._13
713+ let updatePrice = invoke(addressFromStringValue(_amm), "updateOracle", [_priceData], nil)
714+ if ((updatePrice == updatePrice))
679715 then {
680- let positionDirection = getPositionDirection(_side, _type)
681- if (if (!(initialized()))
682- then true
683- else !(isValid(_orderId)))
684- then throw("Invalid executeOrder parameters")
685- else {
686- let positionSize = getPositionSize(_amm, _trader, positionDirection)
687- if ((positionSize == positionSize))
688- then {
689- let currentPositionId = if ((positionSize != 0))
690- then getPositionId(_amm, _trader, positionDirection)
691- else 0
692- let $t02592929307 = if ((_type == STOP))
716+ let cleanUp = invoke(this, "cleanUpStaleOrders", [_amm, _trader], nil)
717+ if ((cleanUp == cleanUp))
718+ then {
719+ let positionDirection = getPositionDirection(_side, _type)
720+ if (if (!(initialized()))
721+ then true
722+ else !(isValid(_orderId)))
723+ then throw("Invalid executeOrder parameters")
724+ else {
725+ let positionSize = getPositionSize(_amm, _trader, positionDirection)
726+ if ((positionSize == positionSize))
693727 then {
694- let _positionDirection = if ((positionSize > 0))
695- then LONG
696- else if ((0 > positionSize))
697- then SHORT
698- else throw("Can not execute STOP order: no open position")
699- let marketPrice = getMarketPrice(_amm)
700- let isExecutable = if ((_side == _positionDirection))
701- then throw("Can not execute STOP order: reduce only")
702- else if ((currentPositionId != _positionId))
703- then throw("Can not execute STOP order: position closed")
704- else if ((_positionDirection == LONG))
705- then (_triggerPrice >= marketPrice)
706- else (marketPrice >= _triggerPrice)
707- if (isExecutable)
708- then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false], nil)
709- else throw("Can not execute STOP order: triggerPrice mismatch")
710- }
711- else if ((_type == TAKE))
712- then {
713- let _positionDirection = if ((positionSize > 0))
714- then LONG
715- else if ((0 > positionSize))
716- then SHORT
717- else throw("Can not execute STOP order: no open position")
718- let marketPrice = getMarketPrice(_amm)
719- let isExecutable = if ((_side == _positionDirection))
720- then throw("Can not execute TAKE order: reduce only")
721- else if ((currentPositionId != _positionId))
722- then throw(((("Can not execute TAKE order: position closed " + toString(currentPositionId)) + "!=") + toString(_positionId)))
723- else if ((_positionDirection == LONG))
724- then (marketPrice >= _triggerPrice)
725- else (_triggerPrice >= marketPrice)
726- if (isExecutable)
727- then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false], nil)
728- else throw("Can not execute TAKE order: triggerPrice mismatch")
729- }
730- else if ((_type == LIMIT))
728+ let currentPositionId = if ((positionSize != 0))
729+ then getPositionId(_amm, _trader, positionDirection)
730+ else 0
731+ let $t02783031791 = if ((_type == STOP))
731732 then {
733+ let _positionDirection = if ((positionSize > 0))
734+ then LONG
735+ else if ((0 > positionSize))
736+ then SHORT
737+ else throw("Can not execute STOP order: no open position")
732738 let marketPrice = getMarketPrice(_amm)
733- let spread = if ((_limitPrice == 0))
734- then getSpread(_triggerPrice)
735- else abs((_triggerPrice - _limitPrice))
736- let isExecutable = if ((marketPrice >= (_triggerPrice - spread)))
737- then ((_triggerPrice + spread) >= marketPrice)
738- else false
739+ let isExecutable = if ((_side == _positionDirection))
740+ then throw("Can not execute STOP order: reduce only")
741+ else if ((currentPositionId != _positionId))
742+ then throw("Can not execute STOP order: position closed")
743+ else if ((_positionDirection == LONG))
744+ then (_triggerPrice >= marketPrice)
745+ else (marketPrice >= _triggerPrice)
739746 if (isExecutable)
747+ then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false, _priceData], nil)
748+ else throw("Can not execute STOP order: triggerPrice mismatch")
749+ }
750+ else if ((_type == TAKE))
751+ then {
752+ let _positionDirection = if ((positionSize > 0))
753+ then LONG
754+ else if ((0 > positionSize))
755+ then SHORT
756+ else throw("Can not execute STOP order: no open position")
757+ let marketPrice = getMarketPrice(_amm)
758+ let isExecutable = if ((_side == _positionDirection))
759+ then throw("Can not execute TAKE order: reduce only")
760+ else if ((currentPositionId != _positionId))
761+ then throw(((("Can not execute TAKE order: position closed " + toString(currentPositionId)) + "!=") + toString(_positionId)))
762+ else if ((_positionDirection == LONG))
763+ then (marketPrice >= _triggerPrice)
764+ else (_triggerPrice >= marketPrice)
765+ if (isExecutable)
766+ then $Tuple3("closePosition", [minv(_amountIn, abs(positionSize)), positionDirection, muld(_limitPrice, abs(positionSize)), false, _priceData], nil)
767+ else throw("Can not execute TAKE order: triggerPrice mismatch")
768+ }
769+ else if ((_type == LIMIT))
740770 then {
741- let amountInWithFee = (_amountUsdn - muld(_amountUsdn, getFee(_amm, _trader)))
742- $Tuple3("increasePosition", [_side, _leverage, if ((_limitPrice == 0))
743- then 0
744- else divd(amountInWithFee, _limitPrice), _refLink], [AttachedPayment(quoteAsset(), _amountUsdn)])
771+ let marketPrice = getMarketPrice(_amm)
772+ let isExecutableByStop = if ((_limitPrice == 0))
773+ then true
774+ else if ((_limitPrice > _orderMarketPrice))
775+ then (marketPrice >= _limitPrice)
776+ else (_limitPrice >= marketPrice)
777+ let isExecutableForTriggerPrice = if ((_side == LONG))
778+ then (_triggerPrice >= marketPrice)
779+ else (marketPrice >= _triggerPrice)
780+ let isExecutable = if (isExecutableByStop)
781+ then isExecutableForTriggerPrice
782+ else false
783+ if (isExecutable)
784+ then {
785+ let marginAmount = divd(_amountUsdn, (muld(getFee(_amm, _trader), _leverage) + DECIMAL_UNIT))
786+ let openNotional = muld(marginAmount, _leverage)
787+ let idealMinBaseAssetAmount = divd(openNotional, _triggerPrice)
788+ let minBaseAssetAmount = (idealMinBaseAssetAmount - muld(idealMinBaseAssetAmount, getSpreadLimit()))
789+ $Tuple3("increasePosition", [_side, _leverage, minBaseAssetAmount, _refLink, _priceData], [AttachedPayment(quoteAsset(), _amountUsdn)])
790+ }
791+ else throw("Can not execute LIMIT order: triggerPrice mismatch")
745792 }
746- else throw("Can not execute LIMIT order: triggerPrice mismatch")
793+ else throw(("Invalid order type: " + toString(_type)))
794+ let method = $t02783031791._1
795+ let args = $t02783031791._2
796+ let payments = $t02783031791._3
797+ let withdraw = if ((size(payments) == 1))
798+ then {
799+ let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), payments[0].amount], nil)
800+ if ((unstake == unstake))
801+ then nil
802+ else throw("Strict value is not equal to itself.")
747803 }
748- else throw(("Invalid order type: " + toString(_type)))
749- let method = $t02592929307._1
750- let args = $t02592929307._2
751- let payments = $t02592929307._3
752- let withdraw = if ((size(payments) == 1))
753- then {
754- let unstake = invoke(managerAddress(), "withdraw", [toBase58String(quoteAsset()), payments[0].amount], nil)
755- if ((unstake == unstake))
756- then nil
757- else throw("Strict value is not equal to itself.")
758- }
759- else nil
760- if ((withdraw == withdraw))
761- then {
762- let doSetContext = invoke(this, "setContext", [_trader], nil)
763- if ((doSetContext == doSetContext))
804+ else nil
805+ if ((withdraw == withdraw))
764806 then {
765- let doClosePosition = invoke(addressFromStringValue(_amm), method, args, payments)
766- if ((doClosePosition == doClosePosition))
807+ let doSetContext = invoke(this, "setContext", [_trader], nil)
808+ if ((doSetContext == doSetContext))
767809 then {
768- let doResetContext = invoke(this, "resetContext", nil, nil)
769- if ((doResetContext == doResetContext))
810+ let doClosePosition = invoke(addressFromStringValue(_amm), method, args, payments)
811+ if ((doClosePosition == doClosePosition))
770812 then {
771- let executeOrderRequest = if ((_type == LIMIT))
813+ let doResetContext = invoke(this, "resetContext", nil, nil)
814+ if ((doResetContext == doResetContext))
772815 then {
773- let newPositionSize = getPositionSize(_amm, _trader, positionDirection)
774- if ((newPositionSize == newPositionSize))
816+ let executeOrderRequest = if ((_type == LIMIT))
775817 then {
776- let newPositionId = getPositionId(_amm, _trader, positionDirection)
777- if ((newPositionId == newPositionId))
818+ let newPositionSize = getPositionSize(_amm, _trader, positionDirection)
819+ if ((newPositionSize == newPositionSize))
778820 then {
779- let positionSizeDelta = (newPositionSize - positionSize)
780- let closeOrderSide = if ((newPositionSize > 0))
781- then SHORT
782- else LONG
783- let makeTakeOrder = if (haveOrderRequest(_orderId, TAKE))
821+ let newPositionId = getPositionId(_amm, _trader, positionDirection)
822+ if ((newPositionId == newPositionId))
784823 then {
785- let $t03026930468 = getOrderRequest(_orderId, TAKE)
786- let newOrderId = $t03026930468._1
787- let newAmm = $t03026930468._2
788- let newTrader = $t03026930468._3
789- let newType = $t03026930468._4
790- let newTriggerPrice = $t03026930468._5
791- let newLimitPrice = $t03026930468._6
792- let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0], nil)
793- if ((doCall == doCall))
794- then nil
795- else throw("Strict value is not equal to itself.")
796- }
797- else nil
798- if ((makeTakeOrder == makeTakeOrder))
799- then {
800- let makeStopOrder = if (haveOrderRequest(_orderId, STOP))
824+ let positionSizeDelta = (newPositionSize - positionSize)
825+ let closeOrderSide = if ((newPositionSize > 0))
826+ then SHORT
827+ else LONG
828+ let makeTakeOrder = if (haveOrderRequest(_orderId, TAKE))
801829 then {
802- let $t03102431223 = getOrderRequest(_orderId, STOP)
803- let newOrderId = $t03102431223._1
804- let newAmm = $t03102431223._2
805- let newTrader = $t03102431223._3
806- let newType = $t03102431223._4
807- let newTriggerPrice = $t03102431223._5
808- let newLimitPrice = $t03102431223._6
809- let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0], nil)
830+ let $t03275332952 = getOrderRequest(_orderId, TAKE)
831+ let newOrderId = $t03275332952._1
832+ let newAmm = $t03275332952._2
833+ let newTrader = $t03275332952._3
834+ let newType = $t03275332952._4
835+ let newTriggerPrice = $t03275332952._5
836+ let newLimitPrice = $t03275332952._6
837+ let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0, 0], nil)
810838 if ((doCall == doCall))
811839 then nil
812840 else throw("Strict value is not equal to itself.")
813841 }
814842 else nil
815- if ((makeStopOrder == makeStopOrder))
816- then (makeTakeOrder ++ makeStopOrder)
843+ if ((makeTakeOrder == makeTakeOrder))
844+ then {
845+ let makeStopOrder = if (haveOrderRequest(_orderId, STOP))
846+ then {
847+ let $t03352533724 = getOrderRequest(_orderId, STOP)
848+ let newOrderId = $t03352533724._1
849+ let newAmm = $t03352533724._2
850+ let newTrader = $t03352533724._3
851+ let newType = $t03352533724._4
852+ let newTriggerPrice = $t03352533724._5
853+ let newLimitPrice = $t03352533724._6
854+ let doCall = invoke(this, "internalCreateOrder", [newTrader, newAmm, newType, newTriggerPrice, newLimitPrice, positionSizeDelta, 0, closeOrderSide, "", "", 0, 0], nil)
855+ if ((doCall == doCall))
856+ then nil
857+ else throw("Strict value is not equal to itself.")
858+ }
859+ else nil
860+ if ((makeStopOrder == makeStopOrder))
861+ then (makeTakeOrder ++ makeStopOrder)
862+ else throw("Strict value is not equal to itself.")
863+ }
817864 else throw("Strict value is not equal to itself.")
818865 }
819866 else throw("Strict value is not equal to itself.")
820867 }
821868 else throw("Strict value is not equal to itself.")
822869 }
823- else throw("Strict value is not equal to itself.")
824- }
825- else nil
826- if ((executeOrderRequest == executeOrderRequest))
827- then {
828- let doMarkOrderExecuted = invoke(this, "internalMarkOrderExecuted", [_amm, _trader, _orderId], nil)
829- if ((doMarkOrderExecuted == doMarkOrderExecuted))
830- then nil
870+ else nil
871+ if ((executeOrderRequest == executeOrderRequest))
872+ then {
873+ let doMarkOrderExecuted = invoke(this, "internalMarkOrderExecuted", [_amm, _trader, _orderId], nil)
874+ if ((doMarkOrderExecuted == doMarkOrderExecuted))
875+ then nil
876+ else throw("Strict value is not equal to itself.")
877+ }
831878 else throw("Strict value is not equal to itself.")
832879 }
833880 else throw("Strict value is not equal to itself.")
834881 }
835882 else throw("Strict value is not equal to itself.")
836883 }
837884 else throw("Strict value is not equal to itself.")
838885 }
839886 else throw("Strict value is not equal to itself.")
840887 }
841888 else throw("Strict value is not equal to itself.")
842889 }
843- else throw("Strict value is not equal to itself.")
844890 }
891+ else throw("Strict value is not equal to itself.")
845892 }
846893 else throw("Strict value is not equal to itself.")
847894 }
848895
849896
850897
851898 @Callable(i)
852-func view_canExecuteOrder (_orderId) = {
853- let s = invoke(this, "executeOrder", [_orderId], nil)
899+func view_canExecuteOrder (_orderId,_priceData) = {
900+ let s = invoke(this, "executeOrder", [_orderId, _priceData], nil)
854901 if ((s == s))
855902 then throw("Success")
856903 else throw("Strict value is not equal to itself.")
857904 }
858905
859906
860907 @Verifier(tx)
861908 func verify () = {
862909 let coordinatorStr = getString(this, k_coordinatorAddress)
863910 if (isDefined(coordinatorStr))
864911 then {
865912 let admin = getString(addressFromStringValue(value(coordinatorStr)), k_admin_address)
866913 if (isDefined(admin))
867914 then valueOrElse(getBoolean(addressFromStringValue(value(admin)), ((("status_" + toString(this)) + "_") + toBase58String(tx.id))), false)
868915 else throw("unable to verify: admin not set in coordinator")
869916 }
870917 else sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
871918 }
872919

github/deemru/w8io/169f3d6 
185.35 ms