tx · 94PoeraAsNjxtnL8ya3Hkev1YB6BHN2NhhKXSgX3CJ9r

3N2mWZGG3ejU8Ypmhs9SRjJqEifx7wvETGe:  -0.01400000 Waves

2021.07.20 16:26 [1621998] smart account 3N2mWZGG3ejU8Ypmhs9SRjJqEifx7wvETGe > SELF 0.00000000 Waves

{ "type": 13, "id": "94PoeraAsNjxtnL8ya3Hkev1YB6BHN2NhhKXSgX3CJ9r", "fee": 1400000, "feeAssetId": null, "timestamp": 1626787629211, "version": 2, "chainId": 84, "sender": "3N2mWZGG3ejU8Ypmhs9SRjJqEifx7wvETGe", "senderPublicKey": "G4GRGfKZznvAxNXzrvWVv9HEKBbeR9VUdnuAGnk7gPAj", "proofs": [ "vP11tjFgeUNc1HSFYQeiQtR8jYkHTVbGF94mkoyeDa62NWiRtA5TE2JH5K23RG4XWW4qWmm8e8DteBHE5V6up7Y", "2sqFLoqusdix3aCm7qDo9zwpeJ1KabZRZH4MgBbJD7CUxMHeX5AnZzQWqb34MrEedga9j9wmXcZtUFU6SoTfLFx2" ], "script": "base64:AAIFAAAAAAAAACoIAhIDCgEEEgcKBQEBCAgEEgMKAQESAwoBARIAEgMKAQESABIAEgMKAQEAAABLAAAAAAd2ZXJzaW9uAgAAAAUxLjAuMAAAAAAKa2V5VmVyc2lvbgIAAAAHdmVyc2lvbgAAAAAJa2V5QWN0aXZlAgAAAAZhY3RpdmUAAAAAC2tleUFzc2V0SWRBAgAAAApBX2Fzc2V0X2lkAAAAAAtrZXlBc3NldElkQgIAAAAKQl9hc3NldF9pZAAAAAALa2V5QmFsYW5jZUECAAAAD0FfYXNzZXRfYmFsYW5jZQAAAAALa2V5QmFsYW5jZUICAAAAD0JfYXNzZXRfYmFsYW5jZQAAAAAPa2V5QmFsYW5jZUluaXRBAgAAAAxBX2Fzc2V0X2luaXQAAAAAD2tleUJhbGFuY2VJbml0QgIAAAAMQl9hc3NldF9pbml0AAAAAA9rZXlTaGFyZUFzc2V0SWQCAAAADnNoYXJlX2Fzc2V0X2lkAAAAABNrZXlTaGFyZUFzc2V0U3VwcGx5AgAAABJzaGFyZV9hc3NldF9zdXBwbHkAAAAADWtleUNvbW1pc3Npb24CAAAACmNvbW1pc3Npb24AAAAAG2tleUNvbW1pc3Npb25TY2FsZURlbGltaXRlcgIAAAAaY29tbWlzc2lvbl9zY2FsZV9kZWxpbWl0ZXIAAAAACGtleUNhdXNlAgAAAA5zaHV0ZG93bl9jYXVzZQAAAAAPa2V5Rmlyc3RIYXJ2ZXN0AgAAAA1maXJzdF9oYXJ2ZXN0AAAAABVrZXlGaXJzdEhhcnZlc3RIZWlnaHQCAAAAFGZpcnN0X2hhcnZlc3RfaGVpZ2h0AAAAAAtrU2hhcmVMaW1pdAIAAAAcc2hhcmVfbGltaXRfb25fZmlyc3RfaGFydmVzdAAAAAALa0Jhc2VQZXJpb2QCAAAAC2Jhc2VfcGVyaW9kAAAAAA1rUGVyaW9kTGVuZ3RoAgAAAA1wZXJpb2RfbGVuZ3RoAAAAAAxrU3RhcnRIZWlnaHQCAAAADHN0YXJ0X2hlaWdodAAAAAATa0ZpcnN0SGFydmVzdEhlaWdodAIAAAAUZmlyc3RfaGFydmVzdF9oZWlnaHQAAAAAD2tleUFkbWluUHViS2V5MQIAAAALYWRtaW5fcHViXzEAAAAAD2tleUFkbWluUHViS2V5MgIAAAALYWRtaW5fcHViXzIAAAAAD2tleUFkbWluUHViS2V5MwIAAAALYWRtaW5fcHViXzMAAAAABm9yYWNsZQkBAAAAB0FkZHJlc3MAAAABAQAAABoBVOlFqh6QLzqu8boO5i6akl8amITh82KzCAEAAAALZ2V0QWRtaW5QdWIAAAABAAAAC2tleUFkbWluUHViBAAAAAckbWF0Y2gwCQAEHQAAAAIFAAAABm9yYWNsZQUAAAALa2V5QWRtaW5QdWIDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABlN0cmluZwQAAAAGc3RyaW5nBQAAAAckbWF0Y2gwCQACWQAAAAEFAAAABnN0cmluZwQAAAAHbm90aGluZwUAAAAHJG1hdGNoMAkAAAIAAAABAgAAABlBZG1pbiBwdWJsaWMga2V5IGlzIGVtcHR5AAAAAAxhZG1pblB1YktleTEJAQAAAAtnZXRBZG1pblB1YgAAAAEFAAAAD2tleUFkbWluUHViS2V5MQAAAAAMYWRtaW5QdWJLZXkyCQEAAAALZ2V0QWRtaW5QdWIAAAABBQAAAA9rZXlBZG1pblB1YktleTIAAAAADGFkbWluUHViS2V5MwkBAAAAC2dldEFkbWluUHViAAAAAQUAAAAPa2V5QWRtaW5QdWJLZXkzAAAAABRhZG1pblB1YktleVN0YXJ0U3RvcAEAAAAgBM915WTQwQ9OZ7kfE2TEGg3qhCRCMX0zkF1AWAqdk0oAAAAAEmFkbWluUHViS2V5U3Rha2luZwEAAAAgBM915WTQwQ9OZ7kfE2TEGg3qhCRCMX0zkF1AWAqdk0oAAAAADXdhbGxldEFkZHJlc3MJAQAAAAdBZGRyZXNzAAAAAQEAAAAaAVTfPUuXiVhZ3hXg5FELVt09el+9tX+FIYkAAAAADXZvdGluZ0FkZHJlc3MJAQAAAAdBZGRyZXNzAAAAAQEAAAAaAVQaNn7fV6KdJ2lc/r/3JtTn+6HDaD4gyn8AAAAADnN0YWtpbmdBZGRyZXNzCQEAAAAHQWRkcmVzcwAAAAEBAAAAGgFUuYnNz0dxupdy7LhFDzOwqkmF0Q14MgJtAAAAABRVU0ROVG9XYXZlc0V4Y2hhbmdlcgkBAAAAB0FkZHJlc3MAAAABAQAAABoBVEy/jZRZdxqtM4EcBkTl9asSMhMMxFs92AAAAAAKYmFzZVBlcmlvZAkBAAAAE3ZhbHVlT3JFcnJvck1lc3NhZ2UAAAACCQAEGgAAAAIFAAAADXZvdGluZ0FkZHJlc3MFAAAAC2tCYXNlUGVyaW9kAgAAABFFbXB0eSBrQmFzZVBlcmlvZAAAAAALc3RhcnRIZWlnaHQJAQAAABN2YWx1ZU9yRXJyb3JNZXNzYWdlAAAAAgkABBoAAAACBQAAAA12b3RpbmdBZGRyZXNzBQAAAAxrU3RhcnRIZWlnaHQCAAAAEkVtcHR5IGtTdGFydEhlaWdodAAAAAAMcGVyaW9kTGVuZ3RoCQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAQaAAAAAgUAAAANdm90aW5nQWRkcmVzcwUAAAANa1BlcmlvZExlbmd0aAIAAAATRW1wdHkga1BlcmlvZExlbmd0aAAAAAAVZmlyc3RIYXJ2ZXN0RW5kUGVyaW9kCQAAZAAAAAIJAABkAAAAAgUAAAAKYmFzZVBlcmlvZAkAAGkAAAACCQAAZQAAAAIFAAAABmhlaWdodAUAAAALc3RhcnRIZWlnaHQFAAAADHBlcmlvZExlbmd0aAAAAAAAAAAAAwAAAAAEVVNETgEAAAAgtiYpwwT1zlORpA5LdSQvZIxRsfrfr1QpvUjSHSqyqtEAAAAAEHN0YWtpbmdGZWVJblVTRE4AAAAAAAAEHrAAAAAACGlzQWN0aXZlCQEAAAARQGV4dHJOYXRpdmUoMTA1MSkAAAACBQAAAAR0aGlzBQAAAAlrZXlBY3RpdmUAAAAAC3N0ckFzc2V0SWRBCQEAAAARQGV4dHJOYXRpdmUoMTA1MykAAAACBQAAAAR0aGlzBQAAAAtrZXlBc3NldElkQQAAAAALc3RyQXNzZXRJZEIJAQAAABFAZXh0ck5hdGl2ZSgxMDUzKQAAAAIFAAAABHRoaXMFAAAAC2tleUFzc2V0SWRCAAAAAAhhc3NldElkQQMJAAAAAAAAAgUAAAALc3RyQXNzZXRJZEECAAAABVdBVkVTBQAAAAR1bml0CQACWQAAAAEFAAAAC3N0ckFzc2V0SWRBAAAAAAhhc3NldElkQgMJAAAAAAAAAgUAAAALc3RyQXNzZXRJZEICAAAABVdBVkVTBQAAAAR1bml0CQACWQAAAAEFAAAAC3N0ckFzc2V0SWRCAAAAAAphc3Nl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c3Rha2VkQW1vdW50VVNETgAAAAAAAAAAAAAAAAAZYWNjb3VudEJhbGFuY2VXaXRoU3Rha2VkQQkAAGQAAAACCQEAAAAOYWNjb3VudEJhbGFuY2UAAAABBQAAAAhhc3NldElkQQMJAAAAAAAAAgUAAAAIYXNzZXRJZEEFAAAABFVTRE4FAAAAEHN0YWtlZEFtb3VudFVTRE4AAAAAAAAAAAAAAAAAGWFjY291bnRCYWxhbmNlV2l0aFN0YWtlZEIJAABkAAAAAgkBAAAADmFjY291bnRCYWxhbmNlAAAAAQUAAAAIYXNzZXRJZEIDCQAAAAAAAAIFAAAACGFzc2V0SWRCBQAAAARVU0ROBQAAABBzdGFrZWRBbW91bnRVU0ROAAAAAAAAAAAAAAAAABBoYXNFbm91Z2hCYWxhbmNlAwkAAGcAAAACBQAAABlhY2NvdW50QmFsYW5jZVdpdGhTdGFrZWRBBQAAAAhiYWxhbmNlQQkAAGcAAAACBQAAABlhY2NvdW50QmFsYW5jZVdpdGhTdGFrZWRCBQAAAAhiYWxhbmNlQgcBAAAADGdldEFzc2V0SW5mbwAAAAEAAAAHYXNzZXRJZAQAAAAHJG1hdGNoMAUAAAAHYXNzZXRJZAMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAKQnl0ZVZlY3RvcgQAAAACaWQFAAAAByRtYXRjaDAEAAAACHN0cmluZ0lkCQACWAAAAAEFAAAAAmlkBAAAAARpbmZvCQEAAAATdmFsdWVPckVycm9yTWVzc2FnZQAAAAIJAAPsAAAAAQUAAAACaWQJAAEsAAAAAgkAASwAAAACAgAAAAZBc3NldCAFAAAACHN0cmluZ0lkAgAAAA4gZG9lc24ndCBleGlzdAkABRUAAAADBQAAAAhzdHJpbmdJZAgFAAAABGluZm8AAAAEbmFtZQgFAAAABGluZm8AAAAIZGVjaW1hbHMDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABFVuaXQEAAAABXdhdmVzBQAAAAckbWF0Y2gwCQAFFQAAAAMCAAAABVdBVkVTAgAAAAVXQVZFUwAAAAAAAAAACAkAAAIAAAABAgAAAAtNYXRjaCBlcnJvcgEAAAAWZ2V0QXNzZXRJbmZvRnJvbVN0cmluZwAAAAEAAAAIYXNzZXRTdHIDCQAAAAAAAAIFAAAACGFzc2V0U3RyAgAAAAVXQVZFUwkABRUAAAADAgAAAAVXQVZFUwIAAAAFV0FWRVMAAAAAAAAAAAgEAAAACHN0cmluZ0lkBQAAAAhhc3NldFN0cgQAAAACaWQJAAJZAAAAAQUAAAAIYXNzZXRTdHIEAAAABGluZm8JAQAAABN2YWx1ZU9yRXJyb3JNZXNzYWdlAAAAAgkAA+wAAAABBQAAAAJpZAkAASwAAAACCQABLAAAAAICAAAABkFzc2V0IAUAAAAIc3RyaW5nSWQCAAAADiBkb2Vzbid0IGV4aXN0CQAFFQAAAAMFAAAACHN0cmluZ0lkCAUAAAAEaW5mbwAAAARuYW1lCAUAAAAEaW5mbwAAAAhkZWNpbWFscwEAAAAHc3VzcGVuZAAAAAEAAAAFY2F1c2UJAARMAAAAAgkBAAAADEJvb2xlYW5FbnRyeQAAAAIFAAAACWtleUFjdGl2ZQcJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAIa2V5Q2F1c2UFAAAABWNhdXNlBQAAAANuaWwBAAAAEGRlZHVjdFN0YWtpbmdGZWUAAAACAAAABmFtb3VudAAAAAdhc3NldElkAwkAAAAAAAACBQAAAAdhc3NldElkBQAAAARVU0ROBAAAAAZyZXN1bHQJAABlAAAAAgUAAAAGYW1vdW50BQAAABBzdGFraW5nRmVlSW5VU0ROAwkAAGcAAAACAAAAAAAAAAAABQAAAAZyZXN1bHQJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACAgAAABRJbnN1ZmZpY2llbnQgYW1vdW50IAkAAaQAAAABBQAAAAZhbW91bnQCAAAAFyB0byBkZWR1Y3Qgc3Rha2luZyBmZWUgCQABpAAAAAEFAAAAEHN0YWtpbmdGZWVJblVTRE4CAAAABiBVU0QtTgUAAAAGcmVzdWx0BQAAAAZhbW91bnQBAAAAIXRocm93SW5zdWZmaWNpZW50QXZhaWxhYmxlQmFsYW5jZQAAAAMAAAAGYW1vdW50AAAACWF2YWlsYWJsZQAAAAlhc3NldE5hbWUJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAIUluc3VmZmljaWVudCBEQXBwIGJhbGFuY2UgdG8gcGF5IAkAAaQAAAABBQAAAAZhbW91bnQCAAAAASAFAAAACWFzc2V0TmFtZQIAAAAcIGR1ZSB0byBzdGFraW5nLiBBdmFpbGFibGU6IAkAAaQAAAABBQAAAAlhdmFpbGFibGUCAAAAASAFAAAACWFzc2V0TmFtZQIAAABALiBQbGVhc2UgY29udGFjdCBzdXBwb3J0IGluIFRlbGVncmFtOiBodHRwczovL3QubWUvc3dvcGZpc3VwcG9ydAEAAAAidGhyb3dJbnN1ZmZpY2llbnRBdmFpbGFibGVCYWxhbmNlcwAAAAIAAAAHYW1vdW50QQAAAAdhbW91bnRCCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAhSW5zdWZmaWNpZW50IERBcHAgYmFsYW5jZSB0byBwYXkgCQABpAAAAAEFAAAAB2Ftb3VudEECAAAAASAFAAAACmFzc2V0TmFtZUECAAAABSBhbmQgCQABpAAAAAEFAAAAB2Ftb3VudEICAAAAASAFAAAACmFzc2V0TmFtZUICAAAAHCBkdWUgdG8gc3Rha2luZy4gQXZhaWxhYmxlOiAJAAGkAAAAAQUAAAARYXZhaWxhYmxlQmFsYW5jZUECAAAAASAFAAAACmFzc2V0TmFtZUECAAAABSBhbmQgCQABpAAAAAEFAAAAEWF2YWlsYWJsZUJhbGFuY2VCAgAAAAEgBQAAAAphc3NldE5hbWVCAgAAAEAuIFBsZWFzZSBjb250YWN0IHN1cHBvcnQgaW4gVGVsZWdyYW06IGh0dHBzOi8vdC5tZS9zd29wZmlzdXBwb3J0AQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAAJAQAAAAdzdXNwZW5kAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAjU3VzcGljaW91cyBzdGF0ZS4gQWN0dWFsIGJhbGFuY2VzOiAJAAGkAAAAAQUAAAAZYWNjb3VudEJhbGFuY2VXaXRoU3Rha2VkQQIAAAABIAUAAAAKYXNzZXROYW1lQQIAAAACLCAJAAGkAAAAAQUAAAAZYWNjb3VudEJhbGFuY2VXaXRoU3Rha2VkQgIAAAABIAUAAAAKYXNzZXROYW1lQgIAAAAJLiBTdGF0ZTogCQABpAAAAAEFAAAACGJhbGFuY2VBAgAAAAEgBQAAAAphc3NldE5hbWVBAgAAAAIsIAkAAaQAAAABBQAAAAhiYWxhbmNlQgIAAAABIAUAAAAKYXNzZXROYW1lQgAAAAkAAAABaQEAAAAEaW5pdAAAAAEAAAAMZmlyc3RIYXJ2ZXN0BAAAAAskdDA2NzA5Njc4NgkABRQAAAACCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAZhbW91bnQICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAAAAB2Fzc2V0SWQEAAAACnBtdEFtb3VudEEIBQAAAAskdDA2NzA5Njc4NgAAAAJfMQQAAAALcG10QXNzZXRJZEEIBQAAAAskdDA2NzA5Njc4NgAAAAJfMgQAAAALJHQwNjc5MTY4NjgJAAUUAAAAAggJAAGRAAAAAggFAAAAAWkAAAAIcGF5bWVudHMAAAAAAAAAAAEAAAAGYW1vdW50CAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAQAAAAdhc3NldElkBAAAAApwbXRBbW91bnRCCAUAAAALJHQwNjc5MTY4NjgAAAACXzEEAAAAC3BtdEFzc2V0SWRCCAUAAAALJHQwNjc5MTY4NjgAAAACXzIEAAAACyR0MDY4NzM2OTUwCQEAAAAMZ2V0QXNzZXRJbmZvAAAAAQUAAAALcG10QXNzZXRJZEEEAAAADnBtdFN0ckFzc2V0SWRBCAUAAAALJHQwNjg3MzY5NTAAAAACXzEEAAAADXBtdEFzc2V0TmFtZUEIBQAAAAskdDA2ODczNjk1MAAAAAJfMgQAAAAMcG10RGVjaW1hbHNBCAUAAAALJHQwNjg3MzY5NTAAAAACXzMEAAAACyR0MDY5NTU3MDMyCQEAAAAMZ2V0QXNzZXRJbmZvAAAAAQUAAAALcG10QXNzZXRJZEIEAAAADnBtdFN0ckFzc2V0SWRCCAUAAAALJHQwNjk1NTcwMzIAAAACXzEEAAAADXBtdEFzc2V0TmFtZUIIBQAAAAskdDA2OTU1NzAzMgAAAAJfMgQAAAAMcG10RGVjaW1hbHNCCAUAAAALJHQwNjk1NTcwMzIAAAACXzMDCQEAAAABIQAAAAEJAQAAAA9jb250YWluc0VsZW1lbnQAAAACCQAETAAAAAIFAAAADGFkbWluUHViS2V5MQkABEwAAAACBQAAAAxhZG1pblB1YktleTIJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkzCQAETAAAAAIFAAAAEmFkbWluUHViS2V5U3Rha2luZwUAAAADbmlsCAUAAAABaQAAAA9jYWxsZXJQdWJsaWNLZXkJAAACAAAAAQIAAAAhT25seSBhZG1pbiBjYW4gY2FsbCB0aGlzIGZ1bmN0aW9uAwkBAAAACWlzRGVmaW5lZAAAAAEJAAQbAAAAAgUAAAAEdGhpcwUAAAAJa2V5QWN0aXZlCQAAAgAAAAECAAAAFkRBcHAgaXMgYWxyZWFkeSBhY3RpdmUDCQAAAAAAAAIFAAAAC3BtdEFzc2V0SWRBBQAAAAtwbXRBc3NldElkQgkAAAIAAAABAgAAABhBc3NldHMgbXVzdCBiZSBkaWZmZXJlbnQEAAAACXNoYXJlTmFtZQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAABcwkAAS8AAAACBQAAAA1wbXRBc3NldE5hbWVBAAAAAAAAAAAHAgAAAAFfCQABLwAAAAIFAAAADXBtdEFzc2V0TmFtZUIAAAAAAAAAAAcEAAAAEHNoYXJlRGVzY3JpcHRpb24JAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgkAASwAAAACAgAAACJTaGFyZVRva2VuIG9mIFN3b3BGaSBwcm90b2NvbCBmb3IgBQAAAA1wbXRBc3NldE5hbWVBAgAAAAUgYW5kIAUAAAANcG10QXNzZXROYW1lQgIAAAAMIGF0IGFkZHJlc3MgCQAEJQAAAAEFAAAABHRoaXMEAAAADXNoYXJlRGVjaW1hbHMJAABpAAAAAgkAAGQAAAACBQAAAAxwbXREZWNpbWFsc0EFAAAADHBtdERlY2ltYWxzQgAAAAAAAAAAAgQAAAAEYXJnMQkAAGwAAAAGBQAAAApwbXRBbW91bnRBBQAAAAxwbXREZWNpbWFsc0EAAAAAAAAAAAUAAAAAAAAAAAEFAAAADHBtdERlY2ltYWxzQQUAAAAERE9XTgQAAAAEYXJnMgkAAGwAAAAGBQAAAApwbXRBbW91bnRCBQAAAAxwbXREZWNpbWFsc0IAAAAAAAAAAAUAAAAAAAAAAAEFAAAADHBtdERlY2ltYWxzQgUAAAAERE9XTgQAAAAEYXJnMwkAAGwAAAAGAAAAAAAAAAAKAAAAAAAAAAAABQAAAA1zaGFyZURlY2ltYWxzAAAAAAAAAAAAAAAAAAAAAAAABQAAAARET1dOBAAAABJzaGFyZUluaXRpYWxTdXBwbHkJAABrAAAAAwUAAAAEYXJnMQUAAAAEYXJnMgUAAAAEYXJnMwQAAAAKc2hhcmVJc3N1ZQkABEIAAAAFBQAAAAlzaGFyZU5hbWUFAAAAEHNoYXJlRGVzY3JpcHRpb24FAAAAEnNoYXJlSW5pdGlhbFN1cHBseQUAAAANc2hhcmVEZWNpbWFscwYEAAAADHNoYXJlSXNzdWVJZAkABDgAAAABBQAAAApzaGFyZUlzc3VlBAAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAKa2V5VmVyc2lvbgUAAAAHdmVyc2lvbgkABEwAAAACCQEAAAAMQm9vbGVhbkVudHJ5AAAAAgUAAAAJa2V5QWN0aXZlBgkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAAtrZXlBc3NldElkQQUAAAAOcG10U3RyQXNzZXRJZEEJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAALa2V5QXNzZXRJZEIFAAAADnBtdFN0ckFzc2V0SWRCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAAKcG10QW1vdW50QQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAALa2V5QmFsYW5jZUIFAAAACnBtdEFtb3VudEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAADWtleUNvbW1pc3Npb24FAAAACmNvbW1pc3Npb24JAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAG2tleUNvbW1pc3Npb25TY2FsZURlbGltaXRlcgUAAAAYY29tbWlzc2lvblNjYWxlRGVsaW1pdGVyCQAETAAAAAIFAAAACnNoYXJlSXNzdWUJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAPa2V5U2hhcmVBc3NldElkCQACWAAAAAEFAAAADHNoYXJlSXNzdWVJZAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAATa2V5U2hhcmVBc3NldFN1cHBseQUAAAASc2hhcmVJbml0aWFsU3VwcGx5CQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAASc2hhcmVJbml0aWFsU3VwcGx5BQAAAAxzaGFyZUlzc3VlSWQFAAAAA25pbAMFAAAADGZpcnN0SGFydmVzdAkABE4AAAACBQAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAADEJvb2xlYW5FbnRyeQAAAAIFAAAAD2tleUZpcnN0SGFydmVzdAUAAAAMZmlyc3RIYXJ2ZXN0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABVrZXlGaXJzdEhhcnZlc3RIZWlnaHQJAABkAAAAAgUAAAALc3RhcnRIZWlnaHQJAABoAAAAAgUAAAAVZmlyc3RIYXJ2ZXN0RW5kUGVyaW9kBQAAAAxwZXJpb2RMZW5ndGgFAAAAA25pbAUAAAAJYmFzZUVudHJ5AAAAAWkBAAAAEWluaXRXaXRoSW5pdFJhdGlvAAAABQAAAAlhbXRBc3NldEEAAAAJYW10QXNzZXRCAAAAC3N0ckFzc2V0SWRBAAAAC3N0ckFzc2V0SWRCAAAADGZpcnN0SGFydmVzdAQAAAALJHQwOTI3NzkzNjQJAQAAABZnZXRBc3NldEluZm9Gcm9tU3RyaW5nAAAAAQUAAAALc3RyQXNzZXRJZEEEAAAADnBtdFN0ckFzc2V0SWRBCAUAAAALJHQwOTI3NzkzNjQAAAACXzEEAAAADXBtdEFzc2V0TmFtZUEIBQAAAAskdDA5Mjc3OTM2NAAAAAJfMgQAAAAMcG10RGVjaW1hbHNBCAUAAAALJHQwOTI3NzkzNjQAAAACXzMEAAAACyR0MDkzNjk5NDU2CQEAAAAWZ2V0QXNzZXRJbmZvRnJvbVN0cmluZwAAAAEFAAAAC3N0ckFzc2V0SWRCBAAAAA5wbXRTdHJBc3NldElkQggFAAAACyR0MDkzNjk5NDU2AAAAAl8xBAAAAA1wbXRBc3NldE5hbWVCCAUAAAALJHQwOTM2OTk0NTYAAAACXzIEAAAADHBtdERlY2ltYWxzQggFAAAACyR0MDkzNjk5NDU2AAAAAl8zAwkBAAAAASEAAAABCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgkABEwAAAACBQAAAAxhZG1pblB1YktleTEJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkyCQAETAAAAAIFAAAADGFkbWluUHViS2V5MwkABEwAAAACBQAAABJhZG1pblB1YktleVN0YWtpbmcFAAAAA25pbAgFAAAAAWkAAAAPY2FsbGVyUHVibGljS2V5CQAAAgAAAAECAAAAIU9ubHkgYWRtaW4gY2FuIGNhbGwgdGhpcyBmdW5jdGlvbgMJAQAAAAlpc0RlZmluZWQAAAABCQAEGwAAAAIFAAAABHRoaXMFAAAACWtleUFjdGl2ZQkAAAIAAAABAgAAABZEQXBwIGlzIGFscmVhZHkgYWN0aXZlAwkAAAAAAAACBQAAAAtzdHJBc3NldElkQQUAAAALc3RyQXNzZXRJZEIJAAACAAAAAQIAAAAYQXNzZXRzIG11c3QgYmUgZGlmZmVyZW50BAAAAAlzaGFyZU5hbWUJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAAXMJAAEvAAAAAgUAAAANcG10QXNzZXROYW1lQQAAAAAAAAAABwIAAAABXwkAAS8AAAACBQAAAA1wbXRBc3NldE5hbWVCAAAAAAAAAAAHBAAAABBzaGFyZURlc2NyaXB0aW9uCQABLAAAAAIJAAEsAAAAAgkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAiU2hhcmVUb2tlbiBvZiBTd29wRmkgcHJvdG9jb2wgZm9yIAUAAAANcG10QXNzZXROYW1lQQIAAAAFIGFuZCAFAAAADXBtdEFzc2V0TmFtZUICAAAADCBhdCBhZGRyZXNzIAkABCUAAAABBQAAAAR0aGlzBAAAAA1zaGFyZURlY2ltYWxzCQAAaQAAAAIJAABkAAAAAgUAAAAMcG10RGVjaW1hbHNBBQAAAAxwbXREZWNpbWFsc0IAAAAAAAAAAAIEAAAAEnNoYXJlSW5pdGlhbFN1cHBseQAAAAAAAAAAAAQAAAAKc2hhcmVJc3N1ZQkABEIAAAAFBQAAAAlzaGFyZU5hbWUFAAAAEHNoYXJlRGVzY3JpcHRpb24FAAAAEnNoYXJlSW5pdGlhbFN1cHBseQUAAAANc2hhcmVEZWNpbWFscwYEAAAADHNoYXJlSXNzdWVJZAkABDgAAAABBQAAAApzaGFyZUlzc3VlBAAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAAKa2V5VmVyc2lvbgUAAAAHdmVyc2lvbgkABEwAAAACCQEAAAAMQm9vbGVhbkVudHJ5AAAAAgUAAAAJa2V5QWN0aXZlBgkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAAtrZXlBc3NldElkQQUAAAAOcG10U3RyQXNzZXRJZEEJAARMAAAAAgkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAALa2V5QXNzZXRJZEIFAAAADnBtdFN0ckFzc2V0SWRCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAA9rZXlCYWxhbmNlSW5pdEEFAAAACWFtdEFzc2V0QQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAPa2V5QmFsYW5jZUluaXRCBQAAAAlhbXRBc3NldEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VBAAAAAAAAAAAACQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQgAAAAAAAAAAAAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAANa2V5Q29tbWlzc2lvbgUAAAAKY29tbWlzc2lvbgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAAba2V5Q29tbWlzc2lvblNjYWxlRGVsaW1pdGVyBQAAABhjb21taXNzaW9uU2NhbGVEZWxpbWl0ZXIJAARMAAAAAgUAAAAKc2hhcmVJc3N1ZQkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACBQAAAA9rZXlTaGFyZUFzc2V0SWQJAAJYAAAAAQUAAAAMc2hhcmVJc3N1ZUlkCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABNrZXlTaGFyZUFzc2V0U3VwcGx5BQAAABJzaGFyZUluaXRpYWxTdXBwbHkFAAAAA25pbAMFAAAADGZpcnN0SGFydmVzdAkABE4AAAACBQAAAAliYXNlRW50cnkJAARMAAAAAgkBAAAADEJvb2xlYW5FbnRyeQAAAAIFAAAAD2tleUZpcnN0SGFydmVzdAUAAAAMZmlyc3RIYXJ2ZXN0CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABVrZXlGaXJzdEhhcnZlc3RIZWlnaHQJAABkAAAAAgUAAAALc3RhcnRIZWlnaHQJAABoAAAAAgUAAAAVZmlyc3RIYXJ2ZXN0RW5kUGVyaW9kBQAAAAxwZXJpb2RMZW5ndGgFAAAAA25pbAUAAAAJYmFzZUVudHJ5AAAAAWkBAAAAGGtlZXBMaW1pdEZvckZpcnN0SGFydmVzdAAAAAEAAAAKc2hhcmVMaW1pdAMJAQAAAAEhAAAAAQUAAAAIaXNBY3RpdmUJAAACAAAAAQIAAAAfREFwcCBpcyBpbmFjdGl2ZSBhdCB0aGlzIG1vbWVudAMJAQAAAAEhAAAAAQkBAAAAD2NvbnRhaW5zRWxlbWVudAAAAAIJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkxCQAETAAAAAIFAAAADGFkbWluUHViS2V5MgkABEwAAAACBQAAAAxhZG1pblB1YktleTMJAARMAAAAAgUAAAASYWRtaW5QdWJLZXlTdGFraW5nBQAAAANuaWwIBQAAAAFpAAAAD2NhbGxlclB1YmxpY0tleQkAAAIAAAABAgAAACFPbmx5IGFkbWluIGNhbiBjYWxsIHRoaXMgZnVuY3Rpb24JAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tTaGFyZUxpbWl0BQAAAApzaGFyZUxpbWl0BQAAAANuaWwAAAABaQEAAAAWcmVwbGVuaXNoV2l0aFR3b1Rva2VucwAAAAEAAAARc2xpcHBhZ2VUb2xlcmFuY2UEAAAAC3BtdEFzc2V0SWRBCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAdhc3NldElkBAAAAAtwbXRBc3NldElkQggJAAGRAAAAAggFAAAAAWkAAAAIcGF5bWVudHMAAAAAAAAAAAEAAAAHYXNzZXRJZAQAAAAKcG10QW1vdW50QQkBAAAAEGRlZHVjdFN0YWtpbmdGZWUAAAACCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAZhbW91bnQFAAAAC3BtdEFzc2V0SWRBBAAAAApwbXRBbW91bnRCCQEAAAAQZGVkdWN0U3Rha2luZ0ZlZQAAAAIICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAABAAAABmFtb3VudAUAAAALcG10QXNzZXRJZEIDAwkAAAAAAAACBQAAAAhiYWxhbmNlQQAAAAAAAAAAAAkAAAAAAAACBQAAAAhiYWxhbmNlQgAAAAAAAAAAAAcEAAAADSR0MDEyMTQxMTIyMTgJAQAAAAxnZXRBc3NldEluZm8AAAABBQAAAAtwbXRBc3NldElkQQQAAAAOcG10U3RyQXNzZXRJZEEIBQAAAA0kdDAxMjE0MTEyMjE4AAAAAl8xBAAAAA1wbXRBc3NldE5hbWVBCAUAAAANJHQwMTIxNDExMjIxOAAAAAJfMgQAAAAMcG10RGVjaW1hbHNBCAUAAAANJHQwMTIxNDExMjIxOAAAAAJfMwQAAAANJHQwMTIyMjcxMjMwNAkBAAAADGdldEFzc2V0SW5mbwAAAAEFAAAAC3BtdEFzc2V0SWRCBAAAAA5wbXRTdHJBc3NldElkQggFAAAADSR0MDEyMjI3MTIzMDQAAAACXzEEAAAADXBtdEFzc2V0TmFtZUIIBQAAAA0kdDAxMjIyNzEyMzA0AAAAAl8yBAAAAAxwbXREZWNpbWFsc0IIBQAAAA0kdDAxMjIyNzEyMzA0AAAAAl8zBAAAAAp0b2tlblJhdGlvCQAAawAAAAMJAABrAAAAAwUAAAAKYXNzZXRJbml0QQUAAAALc2NhbGVWYWx1ZTgFAAAACnBtdEFtb3VudEEFAAAAC3NjYWxlVmFsdWUzCQAAawAAAAMFAAAACmFzc2V0SW5pdEIFAAAAC3NjYWxlVmFsdWU4BQAAAApwbXRBbW91bnRCAwkAAAAAAAACBQAAAAtwbXRBc3NldElkQQUAAAALcG10QXNzZXRJZEIJAAACAAAAAQIAAAAYQXNzZXRzIG11c3QgYmUgZGlmZmVyZW50BAAAAA1zaGFyZURlY2ltYWxzCQAAaQAAAAIJAABkAAAAAgUAAAAMcG10RGVjaW1hbHNBBQAAAAxwbXREZWNpbWFsc0IAAAAAAAAAAAIEAAAAEnNoYXJlSW5pdGlhbFN1cHBseQkAAGsAAAADCQAAbAAAAAYFAAAACnBtdEFtb3VudEEFAAAADHBtdERlY2ltYWxzQQAAAAAAAAAABQAAAAAAAAAAAQUAAAAMcG10RGVjaW1hbHNBBQAAAARET1dOCQAAbAAAAAYFAAAACnBtdEFtb3VudEIFAAAADHBtdERlY2ltYWxzQgAAAAAAAAAABQAAAAAAAAAAAQUAAAAMcG10RGVjaW1hbHNCBQAAAARET1dOCQAAbAAAAAYAAAAAAAAAAAoAAAAAAAAAAAAFAAAADXNoYXJlRGVjaW1hbHMAAAAAAAAAAAAAAAAAAAAAAAAFAAAABERPV04DCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDAwkAAGYAAAACAAAAAAAAAAAABQAAABFzbGlwcGFnZVRvbGVyYW5jZQYJAABmAAAAAgUAAAARc2xpcHBhZ2VUb2xlcmFuY2UFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAKVNsaXBwYWdlIHRvbGVyYW5jZSBtdXN0IGJlIGJldHdlZW4gMCBhbmQgCQABpAAAAAEFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyAgAAABYgaW5jbHVzaXZlbHkuIEFjdHVhbDogCQABpAAAAAEFAAAAEXNsaXBwYWdlVG9sZXJhbmNlAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAgkAAAIAAAABAgAAABxUd28gYXR0YWNoZWQgYXNzZXRzIGV4cGVjdGVkAwMJAABmAAAAAgkAAGkAAAACCQAAaAAAAAIFAAAAC3NjYWxlVmFsdWUzCQAAZQAAAAIFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIFAAAACnRva2VuUmF0aW8GCQAAZgAAAAIFAAAACnRva2VuUmF0aW8JAABpAAAAAgkAAGgAAAACBQAAAAtzY2FsZVZhbHVlMwkAAGQAAAACBQAAABpzbGlwcGFnZVRvbGVyYW5jZURlbGltaXRlcgUAAAARc2xpcHBhZ2VUb2xlcmFuY2UFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyCQAAAgAAAAECAAAAPUluY29ycmVjdCBhc3NldHMgYW1vdW50OiBhbW91bnRzIG11c3QgaGF2ZSB0aGUgY29udHJhY3QgcmF0aW8DAwkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEEFAAAACGFzc2V0SWRBBgkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEIFAAAACGFzc2V0SWRCCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAJUluY29ycmVjdCBhc3NldHMgYXR0YWNoZWQuIEV4cGVjdGVkOiAFAAAAC3N0ckFzc2V0SWRBAgAAAAUgYW5kIAUAAAALc3RyQXNzZXRJZEIDCQAAAAAAAAIFAAAAEnNoYXJlSW5pdGlhbFN1cHBseQAAAAAAAAAAAAkAAAIAAAABAgAAAB1Ub28gc21hbGwgYW1vdW50IHRvIHJlcGxlbmlzaAMJAQAAAAEhAAAAAQUAAAAQaGFzRW5vdWdoQmFsYW5jZQkABE4AAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAKcG10QW1vdW50QQUAAAALcG10QXNzZXRJZEEJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAAApwbXRBbW91bnRCBQAAAAtwbXRBc3NldElkQgUAAAADbmlsCQEAAAARc3VzcGVuZFN1c3BpY2lvdXMAAAAACQAETAAAAAIJAQAAAAdSZWlzc3VlAAAAAwUAAAAMc2hhcmVBc3NldElkBQAAABJzaGFyZUluaXRpYWxTdXBwbHkGCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAAKcG10QW1vdW50QQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAALa2V5QmFsYW5jZUIFAAAACnBtdEFtb3VudEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAE2tleVNoYXJlQXNzZXRTdXBwbHkFAAAAEnNoYXJlSW5pdGlhbFN1cHBseQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAAEnNoYXJlSW5pdGlhbFN1cHBseQUAAAAMc2hhcmVBc3NldElkBQAAAANuaWwEAAAACnRva2VuUmF0aW8JAABrAAAAAwkAAGsAAAADBQAAAAhiYWxhbmNlQQUAAAALc2NhbGVWYWx1ZTgFAAAACnBtdEFtb3VudEEFAAAAC3NjYWxlVmFsdWUzCQAAawAAAAMFAAAACGJhbGFuY2VCBQAAAAtzY2FsZVZhbHVlOAUAAAAKcG10QW1vdW50QgQAAAATcmF0aW9TaGFyZVRva2Vuc0luQQkAAGsAAAADBQAAAApwbXRBbW91bnRBBQAAAAtzY2FsZVZhbHVlOAUAAAAIYmFsYW5jZUEEAAAAE3JhdGlvU2hhcmVUb2tlbnNJbkIJAABrAAAAAwUAAAAKcG10QW1vdW50QgUAAAALc2NhbGVWYWx1ZTgFAAAACGJhbGFuY2VCBAAAABVzaGFyZVRva2VuVG9QYXlBbW91bnQJAABrAAAAAwkAAZcAAAABCQAETAAAAAIFAAAAE3JhdGlvU2hhcmVUb2tlbnNJbkEJAARMAAAAAgUAAAATcmF0aW9TaGFyZVRva2Vuc0luQgUAAAADbmlsBQAAABBzaGFyZUFzc2V0U3VwcGx5BQAAAAtzY2FsZVZhbHVlOAMJAQAAAAEhAAAAAQUAAAAIaXNBY3RpdmUJAAACAAAAAQIAAAAfREFwcCBpcyBpbmFjdGl2ZSBhdCB0aGlzIG1vbWVudAMDCQAAZgAAAAIAAAAAAAAAAAAFAAAAEXNsaXBwYWdlVG9sZXJhbmNlBgkAAGYAAAACBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAApU2xpcHBhZ2UgdG9sZXJhbmNlIG11c3QgYmUgYmV0d2VlbiAwIGFuZCAJAAGkAAAAAQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXICAAAAFiBpbmNsdXNpdmVseS4gQWN0dWFsOiAJAAGkAAAAAQUAAAARc2xpcHBhZ2VUb2xlcmFuY2UDCQEAAAACIT0AAAACCQABkAAAAAEIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAACCQAAAgAAAAECAAAAHFR3byBhdHRhY2hlZCBhc3NldHMgZXhwZWN0ZWQDAwkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEEFAAAACGFzc2V0SWRBBgkBAAAAAiE9AAAAAgUAAAALcG10QXNzZXRJZEIFAAAACGFzc2V0SWRCCQAAAgAAAAEJAAEsAAAAAgkAASwAAAACCQABLAAAAAICAAAAJUluY29ycmVjdCBhc3NldHMgYXR0YWNoZWQuIEV4cGVjdGVkOiAFAAAAC3N0ckFzc2V0SWRBAgAAAAUgYW5kIAUAAAALc3RyQXNzZXRJZEIDAwkAAGYAAAACCQAAaQAAAAIJAABoAAAAAgUAAAALc2NhbGVWYWx1ZTMJAABlAAAAAgUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIFAAAAEXNsaXBwYWdlVG9sZXJhbmNlBQAAABpzbGlwcGFnZVRvbGVyYW5jZURlbGltaXRlcgUAAAAKdG9rZW5SYXRpbwYJAABmAAAAAgUAAAAKdG9rZW5SYXRpbwkAAGkAAAACCQAAaAAAAAIFAAAAC3NjYWxlVmFsdWUzCQAAZAAAAAIFAAAAGnNsaXBwYWdlVG9sZXJhbmNlRGVsaW1pdGVyBQAAABFzbGlwcGFnZVRvbGVyYW5jZQUAAAAac2xpcHBhZ2VUb2xlcmFuY2VEZWxpbWl0ZXIJAAACAAAAAQIAAAA9SW5jb3JyZWN0IGFzc2V0cyBhbW91bnQ6IGFtb3VudHMgbXVzdCBoYXZlIHRoZSBjb250cmFjdCByYXRpbwMJAAAAAAAAAgUAAAAVc2hhcmVUb2tlblRvUGF5QW1vdW50AAAAAAAAAAAACQAAAgAAAAECAAAAHVRvbyBzbWFsbCBhbW91bnQgdG8gcmVwbGVuaXNoAwkBAAAAASEAAAABBQAAABBoYXNFbm91Z2hCYWxhbmNlCQAETgAAAAIJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAAApwbXRBbW91bnRBBQAAAAtwbXRBc3NldElkQQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAACnBtdEFtb3VudEIFAAAAC3BtdEFzc2V0SWRCBQAAAANuaWwJAQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAAJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VBCQAAZAAAAAIFAAAACGJhbGFuY2VBBQAAAApwbXRBbW91bnRBCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQgkAAGQAAAACBQAAAAhiYWxhbmNlQgUAAAAKcG10QW1vdW50QgkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAATa2V5U2hhcmVBc3NldFN1cHBseQkAAGQAAAACBQAAABBzaGFyZUFzc2V0U3VwcGx5BQAAABVzaGFyZVRva2VuVG9QYXlBbW91bnQJAARMAAAAAgkBAAAAB1JlaXNzdWUAAAADBQAAAAxzaGFyZUFzc2V0SWQFAAAAFXNoYXJlVG9rZW5Ub1BheUFtb3VudAYJAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAABVzaGFyZVRva2VuVG9QYXlBbW91bnQFAAAADHNoYXJlQXNzZXRJZAUAAAADbmlsAAAAAWkBAAAACHdpdGhkcmF3AAAAAAQAAAANJHQwMTY3MDMxNjg1MwkABRQAAAACCAkAAZEAAAACCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAAAAAAZhbW91bnQICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAAAAB2Fzc2V0SWQEAAAACXBtdEFtb3VudAgFAAAADSR0MDE2NzAzMTY4NTMAAAACXzEEAAAACnBtdEFzc2V0SWQIBQAAAA0kdDAxNjcwMzE2ODUzAAAAAl8yBAAAAAxhbW91bnRUb1BheUEJAQAAABBkZWR1Y3RTdGFraW5nRmVlAAAAAgkAAGsAAAADBQAAAAlwbXRBbW91bnQFAAAACGJhbGFuY2VBBQAAABBzaGFyZUFzc2V0U3VwcGx5BQAAAAhhc3NldElkQQQAAAAMYW1vdW50VG9QYXlCCQEAAAAQZGVkdWN0U3Rha2luZ0ZlZQAAAAIJAABrAAAAAwUAAAAJcG10QW1vdW50BQAAAAhiYWxhbmNlQgUAAAAQc2hhcmVBc3NldFN1cHBseQUAAAAIYXNzZXRJZEIDCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDCQEAAAACIT0AAAACCQABkAAAAAEIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAABCQAAAgAAAAECAAAAHU9uZSBhdHRhY2hlZCBwYXltZW50IGV4cGVjdGVkAwkBAAAAAiE9AAAAAgUAAAAKcG10QXNzZXRJZAUAAAAMc2hhcmVBc3NldElkCQAAAgAAAAEJAAEsAAAAAgIAAAAkSW5jb3JyZWN0IGFzc2V0IGF0dGFjaGVkLiBFeHBlY3RlZDogCQACWAAAAAEFAAAADHNoYXJlQXNzZXRJZAMJAQAAAAEhAAAAAQUAAAAQaGFzRW5vdWdoQmFsYW5jZQkABE4AAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAJcG10QW1vdW50BQAAAApwbXRBc3NldElkBQAAAANuaWwJAQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAADAwkAAGYAAAACBQAAAAxhbW91bnRUb1BheUEFAAAAEWF2YWlsYWJsZUJhbGFuY2VBBgkAAGYAAAACBQAAAAxhbW91bnRUb1BheUIFAAAAEWF2YWlsYWJsZUJhbGFuY2VCCQEAAAAidGhyb3dJbnN1ZmZpY2llbnRBdmFpbGFibGVCYWxhbmNlcwAAAAIFAAAADGFtb3VudFRvUGF5QQUAAAAMYW1vdW50VG9QYXlCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQkAAGUAAAACBQAAAAhiYWxhbmNlQQUAAAAMYW1vdW50VG9QYXlBCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQgkAAGUAAAACBQAAAAhiYWxhbmNlQgUAAAAMYW1vdW50VG9QYXlCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAABNrZXlTaGFyZUFzc2V0U3VwcGx5CQAAZQAAAAIFAAAAEHNoYXJlQXNzZXRTdXBwbHkFAAAACXBtdEFtb3VudAkABEwAAAACCQEAAAAEQnVybgAAAAIFAAAADHNoYXJlQXNzZXRJZAUAAAAJcG10QW1vdW50CQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAMYW1vdW50VG9QYXlBBQAAAAhhc3NldElkQQkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAADGFtb3VudFRvUGF5QgUAAAAIYXNzZXRJZEIFAAAAA25pbAAAAAFpAQAAAAhleGNoYW5nZQAAAAEAAAASbWluQW1vdW50VG9SZWNlaXZlBAAAAA0kdDAxODA1OTE4MTM0CQAFFAAAAAIICQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAAAAAABmFtb3VudAgJAAGRAAAAAggFAAAAAWkAAAAIcGF5bWVudHMAAAAAAAAAAAAAAAAHYXNzZXRJZAQAAAAJcG10QW1vdW50CAUAAAANJHQwMTgwNTkxODEzNAAAAAJfMQQAAAAKcG10QXNzZXRJZAgFAAAADSR0MDE4MDU5MTgxMzQAAAACXzIKAQAAAA1jYWxjdWxhdGVGZWVzAAAAAgAAAAl0b2tlbkZyb20AAAAHdG9rZW5UbwQAAAAQYW1vdW50V2l0aG91dEZlZQkAAGsAAAADBQAAAAd0b2tlblRvBQAAAAlwbXRBbW91bnQJAABkAAAAAgUAAAAJcG10QW1vdW50BQAAAAl0b2tlbkZyb20EAAAADWFtb3VudFdpdGhGZWUJAABrAAAAAwUAAAAQYW1vdW50V2l0aG91dEZlZQkAAGUAAAACBQAAABhjb21taXNzaW9uU2NhbGVEZWxpbWl0ZXIFAAAACmNvbW1pc3Npb24FAAAAGGNvbW1pc3Npb25TY2FsZURlbGltaXRlcgQAAAAQZ292ZXJuYW5jZVJld2FyZAkAAGsAAAADBQAAABBhbW91bnRXaXRob3V0RmVlBQAAABRjb21taXNzaW9uR292ZXJuYW5jZQUAAAAYY29tbWlzc2lvblNjYWxlRGVsaW1pdGVyAwkAAGYAAAACBQAAABJtaW5BbW91bnRUb1JlY2VpdmUFAAAADWFtb3VudFdpdGhGZWUJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAdQ2FsY3VsYXRlZCBhbW91bnQgdG8gcmVjZWl2ZSAJAAGkAAAAAQUAAAANYW1vdW50V2l0aEZlZQIAAAAgIGlzIGxlc3MgdGhhbiBzcGVjaWZpZWQgbWluaW11bSAJAAGkAAAAAQUAAAASbWluQW1vdW50VG9SZWNlaXZlCQAFFQAAAAMFAAAAEGFtb3VudFdpdGhvdXRGZWUFAAAADWFtb3VudFdpdGhGZWUFAAAAEGdvdmVybmFuY2VSZXdhcmQDCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDAwkAAAAAAAACBQAAAAhiYWxhbmNlQQAAAAAAAAAAAAYJAAAAAAAAAgUAAAAIYmFsYW5jZUIAAAAAAAAAAAAJAAACAAAAAQIAAAAgQ2FuJ3QgZXhjaGFuZ2Ugd2l0aCB6ZXJvIGJhbGFuY2UDCQAAZwAAAAIAAAAAAAAAAAAFAAAAEm1pbkFtb3VudFRvUmVjZWl2ZQkAAAIAAAABCQABLAAAAAICAAAANE1pbmltYWwgYW1vdW50IHRvIHJlY2VpdmUgbXVzdCBiZSBwb3NpdGl2ZS4gQWN0dWFsOiAJAAGkAAAAAQUAAAASbWluQW1vdW50VG9SZWNlaXZlAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAQkAAAIAAAABAgAAAB1PbmUgYXR0YWNoZWQgcGF5bWVudCBleHBlY3RlZAMJAQAAAAEhAAAAAQUAAAAQaGFzRW5vdWdoQmFsYW5jZQkABE4AAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgUAAAAJcG10QW1vdW50BQAAAApwbXRBc3NldElkBQAAAANuaWwJAQAAABFzdXNwZW5kU3VzcGljaW91cwAAAAADCQAAAAAAAAIFAAAACnBtdEFzc2V0SWQFAAAACGFzc2V0SWRBBAAAAAthc3NldElkU2VuZAUAAAAIYXNzZXRJZEIEAAAADSR0MDE5NDA4MTk0OTkJAQAAAA1jYWxjdWxhdGVGZWVzAAAAAgUAAAAIYmFsYW5jZUEFAAAACGJhbGFuY2VCBAAAABBhbW91bnRXaXRob3V0RmVlCAUAAAANJHQwMTk0MDgxOTQ5OQAAAAJfMQQAAAANYW1vdW50V2l0aEZlZQgFAAAADSR0MDE5NDA4MTk0OTkAAAACXzIEAAAAEGdvdmVybmFuY2VSZXdhcmQIBQAAAA0kdDAxOTQwODE5NDk5AAAAAl8zBAAAAAtuZXdCYWxhbmNlQQkAAGQAAAACBQAAAAhiYWxhbmNlQQUAAAAJcG10QW1vdW50BAAAAAtuZXdCYWxhbmNlQgkAAGUAAAACCQAAZQAAAAIFAAAACGJhbGFuY2VCBQAAAA1hbW91bnRXaXRoRmVlBQAAABBnb3Zlcm5hbmNlUmV3YXJkAwMDCQAAAAAAAAIFAAAACGFzc2V0SWRBBQAAAARVU0ROCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudFVTRE4FAAAAC25ld0JhbGFuY2VBBwYDCQAAAAAAAAIFAAAACGFzc2V0SWRCBQAAAARVU0ROCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudFVTRE4FAAAAC25ld0JhbGFuY2VCBwkBAAAAIXRocm93SW5zdWZmaWNpZW50QXZhaWxhYmxlQmFsYW5jZQAAAAMFAAAADWFtb3VudFdpdGhGZWUFAAAAEWF2YWlsYWJsZUJhbGFuY2VCBQAAAAphc3NldE5hbWVCCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAALbmV3QmFsYW5jZUEJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VCBQAAAAtuZXdCYWxhbmNlQgkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAADWFtb3VudFdpdGhGZWUFAAAAC2Fzc2V0SWRTZW5kCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMFAAAADXdhbGxldEFkZHJlc3MFAAAAEGdvdmVybmFuY2VSZXdhcmQFAAAAC2Fzc2V0SWRTZW5kBQAAAANuaWwDCQAAAAAAAAIFAAAACnBtdEFzc2V0SWQFAAAACGFzc2V0SWRCBAAAAAthc3NldElkU2VuZAUAAAAIYXNzZXRJZEEEAAAADSR0MDIwMzY4MjA0NTkJAQAAAA1jYWxjdWxhdGVGZWVzAAAAAgUAAAAIYmFsYW5jZUIFAAAACGJhbGFuY2VBBAAAABBhbW91bnRXaXRob3V0RmVlCAUAAAANJHQwMjAzNjgyMDQ1OQAAAAJfMQQAAAANYW1vdW50V2l0aEZlZQgFAAAADSR0MDIwMzY4MjA0NTkAAAACXzIEAAAAEGdvdmVybmFuY2VSZXdhcmQIBQAAAA0kdDAyMDM2ODIwNDU5AAAAAl8zBAAAAAtuZXdCYWxhbmNlQQkAAGUAAAACCQAAZQAAAAIFAAAACGJhbGFuY2VBBQAAAA1hbW91bnRXaXRoRmVlBQAAABBnb3Zlcm5hbmNlUmV3YXJkBAAAAAtuZXdCYWxhbmNlQgkAAGQAAAACBQAAAAhiYWxhbmNlQgUAAAAJcG10QW1vdW50AwMDCQAAAAAAAAIFAAAACGFzc2V0SWRBBQAAAARVU0ROCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudFVTRE4FAAAAC25ld0JhbGFuY2VBBwYDCQAAAAAAAAIFAAAACGFzc2V0SWRCBQAAAARVU0ROCQAAZwAAAAIFAAAAEHN0YWtlZEFtb3VudFVTRE4FAAAAC25ld0JhbGFuY2VCBwkBAAAAIXRocm93SW5zdWZmaWNpZW50QXZhaWxhYmxlQmFsYW5jZQAAAAMFAAAADWFtb3VudFdpdGhGZWUFAAAAEWF2YWlsYWJsZUJhbGFuY2VBBQAAAAphc3NldE5hbWVBCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQUAAAALbmV3QmFsYW5jZUEJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIFAAAAC2tleUJhbGFuY2VCBQAAAAtuZXdCYWxhbmNlQgkABEwAAAACCQEAAAAOU2NyaXB0VHJhbnNmZXIAAAADCAUAAAABaQAAAAZjYWxsZXIFAAAADWFtb3VudFdpdGhGZWUFAAAAC2Fzc2V0SWRTZW5kCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMFAAAADXdhbGxldEFkZHJlc3MFAAAAEGdvdmVybmFuY2VSZXdhcmQFAAAAC2Fzc2V0SWRTZW5kBQAAAANuaWwJAAACAAAAAQkAASwAAAACCQABLAAAAAIJAAEsAAAAAgIAAAAkSW5jb3JyZWN0IGFzc2V0IGF0dGFjaGVkLiBFeHBlY3RlZDogBQAAAAtzdHJBc3NldElkQQIAAAAEIG9yIAUAAAALc3RyQXNzZXRJZEIAAAABaQEAAAAIc2h1dGRvd24AAAAAAwkBAAAAASEAAAABBQAAAAhpc0FjdGl2ZQkAAAIAAAABCQABLAAAAAICAAAAIkRBcHAgaXMgYWxyZWFkeSBzdXNwZW5kZWQuIENhdXNlOiAJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQdAAAAAgUAAAAEdGhpcwUAAAAIa2V5Q2F1c2UCAAAAGnRoZSBjYXVzZSB3YXNuJ3Qgc3BlY2lmaWVkAwkBAAAAASEAAAABCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgkABEwAAAACBQAAAAxhZG1pblB1YktleTEJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkyCQAETAAAAAIFAAAADGFkbWluUHViS2V5MwkABEwAAAACBQAAABRhZG1pblB1YktleVN0YXJ0U3RvcAUAAAADbmlsCAUAAAABaQAAAA9jYWxsZXJQdWJsaWNLZXkJAAACAAAAAQIAAAAhT25seSBhZG1pbiBjYW4gY2FsbCB0aGlzIGZ1bmN0aW9uCQEAAAAHc3VzcGVuZAAAAAECAAAAD1BhdXNlZCBieSBhZG1pbgAAAAFpAQAAAAhhY3RpdmF0ZQAAAAADBQAAAAhpc0FjdGl2ZQkAAAIAAAABAgAAABZEQXBwIGlzIGFscmVhZHkgYWN0aXZlAwkBAAAAASEAAAABCQEAAAAPY29udGFpbnNFbGVtZW50AAAAAgkABEwAAAACBQAAAAxhZG1pblB1YktleTEJAARMAAAAAgUAAAAMYWRtaW5QdWJLZXkyCQAETAAAAAIFAAAADGFkbWluUHViS2V5MwkABEwAAAACBQAAABRhZG1pblB1YktleVN0YXJ0U3RvcAUAAAADbmlsCAUAAAABaQAAAA9jYWxsZXJQdWJsaWNLZXkJAAACAAAAAQIAAAAhT25seSBhZG1pbiBjYW4gY2FsbCB0aGlzIGZ1bmN0aW9uCQAETAAAAAIJAQAAAAxCb29sZWFuRW50cnkAAAACBQAAAAlrZXlBY3RpdmUGCQAETAAAAAIJAQAAAAtEZWxldGVFbnRyeQAAAAEFAAAACGtleUNhdXNlBQAAAANuaWwAAAABaQEAAAAZdGFrZUludG9BY2NvdW50RXh0cmFGdW5kcwAAAAEAAAALYW1vdW50TGVhdmUEAAAAHXVuY291bnRhYmxlQW1vdW50RW5yb2xsQXNzZXRBCQAAZQAAAAIFAAAAGWFjY291bnRCYWxhbmNlV2l0aFN0YWtlZEEFAAAACGJhbGFuY2VBBAAAAB11bmNvdW50YWJsZUFtb3VudEVucm9sbEFzc2V0QgkAAGUAAAACBQAAABlhY2NvdW50QmFsYW5jZVdpdGhTdGFrZWRCBQAAAAhiYWxhbmNlQgQAAAANYW1vdW50RW5yb2xsQQkAAGUAAAACBQAAAB11bmNvdW50YWJsZUFtb3VudEVucm9sbEFzc2V0QQMJAAAAAAAAAgUAAAAIYXNzZXRJZEEFAAAABHVuaXQFAAAAC2Ftb3VudExlYXZlAAAAAAAAAAAABAAAAA1hbW91bnRFbnJvbGxCCQAAZQAAAAIFAAAAHXVuY291bnRhYmxlQW1vdW50RW5yb2xsQXNzZXRCAwkAAAAAAAACBQAAAAhhc3NldElkQgUAAAAEdW5pdAUAAAALYW1vdW50TGVhdmUAAAAAAAAAAAADCQEAAAABIQAAAAEFAAAACGlzQWN0aXZlCQAAAgAAAAECAAAAH0RBcHAgaXMgaW5hY3RpdmUgYXQgdGhpcyBtb21lbnQDCQEAAAACIT0AAAACCAUAAAABaQAAAAZjYWxsZXIFAAAABHRoaXMJAAACAAAAAQIAAAArT25seSB0aGUgREFwcCBpdHNlbGYgY2FuIGNhbGwgdGhpcyBmdW5jdGlvbgMJAABmAAAAAgAAAAAAAAAAAAUAAAALYW1vdW50TGVhdmUJAAACAAAAAQkAASwAAAACAgAAADNBcmd1bWVudCAnYW1vdW50TGVhdmUnIGNhbm5vdCBiZSBuZWdhdGl2ZS4gQWN0dWFsOiAJAAGkAAAAAQUAAAALYW1vdW50TGVhdmUDAwkAAGYAAAACAAAAAAAAAAAABQAAAB11bmNvdW50YWJsZUFtb3VudEVucm9sbEFzc2V0QQYJAABmAAAAAgAAAAAAAAAAAAUAAAAddW5jb3VudGFibGVBbW91bnRFbnJvbGxBc3NldEIJAQAAAAdzdXNwZW5kAAAAAQIAAAAWRW5yb2xsIGFtb3VudCBuZWdhdGl2ZQMDCQAAZgAAAAIAAAAAAAAAAAAFAAAADWFtb3VudEVucm9sbEEGCQAAZgAAAAIAAAAAAAAAAAAFAAAADWFtb3VudEVucm9sbEIJAAACAAAAAQIAAAAVVG9vIGxhcmdlIGFtb3VudExlYXZlCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACBQAAAAtrZXlCYWxhbmNlQQkAAGQAAAACBQAAAAhiYWxhbmNlQQUAAAANYW1vdW50RW5yb2xsQQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgUAAAALa2V5QmFsYW5jZUIJAABkAAAAAgUAAAAIYmFsYW5jZUIFAAAADWFtb3VudEVucm9sbEIJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAAEsAAAAAgIAAAAMbGFzdF9pbmNvbWVfBQAAAAtzdHJBc3NldElkQQUAAAANYW1vdW50RW5yb2xsQQkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkAASwAAAACAgAAAAxsYXN0X2luY29tZV8FAAAAC3N0ckFzc2V0SWRCBQAAAA1hbW91bnRFbnJvbGxCBQAAAANuaWwAAAABAAAAAnR4AQAAAAZ2ZXJpZnkAAAAABAAAABNtdWx0aVNpZ25lZEJ5QWRtaW5zBAAAABJhZG1pblB1YktleTFTaWduZWQDCQAB9AAAAAMIBQAAAAJ0eAAAAAlib2R5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", "height": 1621998, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 5A3KAhd9gVnuYDTncv9TcBxyQBVPz8YFsLV2FYqPXs4d Next: Ft7FUdNm1ut6ckbVzJVMZGbKUmh2G4ZeHvpnDqpS2giv Diff:
OldNewDifferences
1515
1616 let keyBalanceB = "B_asset_balance"
1717
18+let keyBalanceInitA = "A_asset_init"
19+
20+let keyBalanceInitB = "B_asset_init"
21+
1822 let keyShareAssetId = "share_asset_id"
1923
2024 let keyShareAssetSupply = "share_asset_supply"
2428 let keyCommissionScaleDelimiter = "commission_scale_delimiter"
2529
2630 let keyCause = "shutdown_cause"
31+
32+let keyFirstHarvest = "first_harvest"
33+
34+let keyFirstHarvestHeight = "first_harvest_height"
35+
36+let kShareLimit = "share_limit_on_first_harvest"
37+
38+let kBasePeriod = "base_period"
39+
40+let kPeriodLength = "period_length"
41+
42+let kStartHeight = "start_height"
43+
44+let kFirstHarvestHeight = "first_harvest_height"
2745
2846 let keyAdminPubKey1 = "admin_pub_1"
2947
5169
5270 let adminPubKeyStaking = base58'Kn7NpzaG12dLZgcHf2ipUftU6hbJygmrhFqQYE4B7ZK'
5371
54-let governanceAddress = Address(base58'3NAGTtZz6WpupSN89NZD5rMZwwziZEg4Kx4')
72+let walletAddress = Address(base58'3NAGTtZz6WpupSN89NZD5rMZwwziZEg4Kx4')
5573
56-let stakingAddressUSDN = Address(base58'3N6q7sCGSSLBUXDdjBdYGTJbZGZfhhh8cNg')
74+let votingAddress = Address(base58'3MrJgdL1GniipErHy44YF9idzLaUL2iX5DQ')
5775
58-let USDN = base58'8UrfDVd5GreeUwm7uPk7eYz1eMv376kzR52C6sANPkwS'
76+let stakingAddress = Address(base58'3N6q7sCGSSLBUXDdjBdYGTJbZGZfhhh8cNg')
5977
60-let NSBT = base58'36mg8NZTaFRDygiVwb8uBnLR51hetJruUCZcxhaVcHj9'
78+let USDNToWavesExchanger = Address(base58'3Mvutbqb6ykHHvNCCgELTNzyAMaPTqFMixX')
6179
62-let SWOP = base58'2HAJrwa8q4SxBx9cHYaBTQdBjdk5wwqdof7ccpAx2uhZ'
80+let basePeriod = valueOrErrorMessage(getInteger(votingAddress, kBasePeriod), "Empty kBasePeriod")
6381
64-let swopUSDNtoWAVES = Address(base58'3Mvutbqb6ykHHvNCCgELTNzyAMaPTqFMixX')
82+let startHeight = valueOrErrorMessage(getInteger(votingAddress, kStartHeight), "Empty kStartHeight")
6583
66-let swopUSDNtoNSBT = Address(base58'8iwQUs2Q5bPLYULK5U24DR9msCpRQbVagtRWantBuNkp')
84+let periodLength = valueOrErrorMessage(getInteger(votingAddress, kPeriodLength), "Empty kPeriodLength")
85+
86+let firstHarvestEndPeriod = ((basePeriod + ((height - startHeight) / periodLength)) + 3)
87+
88+let USDN = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
6789
6890 let stakingFeeInUSDN = 270000
6991
131153 }
132154
133155
134-func stakedAmount (stakingAddress,asset) = {
135- let stakedAmountCalculated = match getInteger(stakingAddress, ((("rpd_balance_" + asset) + "_") + toString(this))) {
136- case staked: Int =>
137- staked
138- case nothing: Unit =>
139- 0
140- case _ =>
141- throw("Match error")
142- }
143- stakedAmountCalculated
144- }
156+let stakedAmountUSDN = match getInteger(stakingAddress, ((("rpd_balance_" + toBase58String(USDN)) + "_") + toString(this))) {
157+ case staked: Int =>
158+ staked
159+ case nothing: Unit =>
160+ 0
161+ case _ =>
162+ throw("Match error")
163+}
145164
165+let assetInitA = getIntegerValue(this, keyBalanceInitA)
146166
147-let stakedAmountA = stakedAmount(stakingAddressUSDN, strAssetIdA)
167+let assetInitB = getIntegerValue(this, keyBalanceInitB)
148168
149-let stakedAmountB = stakedAmount(stakingAddressUSDN, strAssetIdB)
169+let availableBalanceA = (balanceA - (if ((assetIdA == USDN))
170+ then stakedAmountUSDN
171+ else 0))
150172
151-let availableBalanceA = (balanceA - stakedAmountA)
173+let availableBalanceB = (balanceB - (if ((assetIdB == USDN))
174+ then stakedAmountUSDN
175+ else 0))
152176
153-let availableBalanceB = (balanceB - stakedAmountB)
177+let accountBalanceWithStakedA = (accountBalance(assetIdA) + (if ((assetIdA == USDN))
178+ then stakedAmountUSDN
179+ else 0))
154180
155-let accountBalanceWithStakedA = (accountBalance(assetIdA) + stakedAmountA)
156-
157-let accountBalanceWithStakedB = (accountBalance(assetIdB) + stakedAmountB)
181+let accountBalanceWithStakedB = (accountBalance(assetIdB) + (if ((assetIdB == USDN))
182+ then stakedAmountUSDN
183+ else 0))
158184
159185 let hasEnoughBalance = if ((accountBalanceWithStakedA >= balanceA))
160186 then (accountBalanceWithStakedB >= balanceB)
172198 }
173199
174200
175-func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)]
201+func getAssetInfoFromString (assetStr) = if ((assetStr == "WAVES"))
202+ then $Tuple3("WAVES", "WAVES", 8)
203+ else {
204+ let stringId = assetStr
205+ let id = fromBase58String(assetStr)
206+ let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist"))
207+ $Tuple3(stringId, info.name, info.decimals)
208+ }
176209
177210
178-func stakedTokenCount () = {
179- let isStakedA = if ((stakedAmountA > 0))
180- then 1
181- else 0
182- let isStakedB = if ((stakedAmountB > 0))
183- then 1
184- else 0
185- (isStakedA + isStakedB)
186- }
211+func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)]
187212
188213
189214 func deductStakingFee (amount,assetId) = if ((assetId == USDN))
190215 then {
191- let result = (amount - (stakedTokenCount() * stakingFeeInUSDN))
216+ let result = (amount - stakingFeeInUSDN)
192217 if ((0 >= result))
193218 then throw((((("Insufficient amount " + toString(amount)) + " to deduct staking fee ") + toString(stakingFeeInUSDN)) + " USD-N"))
194219 else result
206231
207232
208233 @Callable(i)
209-func init () = {
210- let $t059456022 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
211- let pmtAmountA = $t059456022._1
212- let pmtAssetIdA = $t059456022._2
213- let $t060276104 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
214- let pmtAmountB = $t060276104._1
215- let pmtAssetIdB = $t060276104._2
216- let $t061096186 = getAssetInfo(pmtAssetIdA)
217- let pmtStrAssetIdA = $t061096186._1
218- let pmtAssetNameA = $t061096186._2
219- let pmtDecimalsA = $t061096186._3
220- let $t061916268 = getAssetInfo(pmtAssetIdB)
221- let pmtStrAssetIdB = $t061916268._1
222- let pmtAssetNameB = $t061916268._2
223- let pmtDecimalsB = $t061916268._3
224- if (isDefined(getBoolean(this, keyActive)))
225- then throw("DApp is already active")
226- else if ((pmtAssetIdA == pmtAssetIdB))
227- then throw("Assets must be different")
228- else {
229- let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
230- let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
231- let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
232- let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, DOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, DOWN), pow(10, 0, shareDecimals, 0, 0, DOWN))
233- let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
234- let shareIssueId = calculateAssetId(shareIssue)
235-[StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
236- }
234+func init (firstHarvest) = {
235+ let $t067096786 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
236+ let pmtAmountA = $t067096786._1
237+ let pmtAssetIdA = $t067096786._2
238+ let $t067916868 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
239+ let pmtAmountB = $t067916868._1
240+ let pmtAssetIdB = $t067916868._2
241+ let $t068736950 = getAssetInfo(pmtAssetIdA)
242+ let pmtStrAssetIdA = $t068736950._1
243+ let pmtAssetNameA = $t068736950._2
244+ let pmtDecimalsA = $t068736950._3
245+ let $t069557032 = getAssetInfo(pmtAssetIdB)
246+ let pmtStrAssetIdB = $t069557032._1
247+ let pmtAssetNameB = $t069557032._2
248+ let pmtDecimalsB = $t069557032._3
249+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
250+ then throw("Only admin can call this function")
251+ else if (isDefined(getBoolean(this, keyActive)))
252+ then throw("DApp is already active")
253+ else if ((pmtAssetIdA == pmtAssetIdB))
254+ then throw("Assets must be different")
255+ else {
256+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
257+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
258+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
259+ let arg1 = pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, DOWN)
260+ let arg2 = pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, DOWN)
261+ let arg3 = pow(10, 0, shareDecimals, 0, 0, DOWN)
262+ let shareInitialSupply = fraction(arg1, arg2, arg3)
263+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
264+ let shareIssueId = calculateAssetId(shareIssue)
265+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
266+ if (firstHarvest)
267+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
268+ else baseEntry
269+ }
237270 }
271+
272+
273+
274+@Callable(i)
275+func initWithInitRatio (amtAssetA,amtAssetB,strAssetIdA,strAssetIdB,firstHarvest) = {
276+ let $t092779364 = getAssetInfoFromString(strAssetIdA)
277+ let pmtStrAssetIdA = $t092779364._1
278+ let pmtAssetNameA = $t092779364._2
279+ let pmtDecimalsA = $t092779364._3
280+ let $t093699456 = getAssetInfoFromString(strAssetIdB)
281+ let pmtStrAssetIdB = $t093699456._1
282+ let pmtAssetNameB = $t093699456._2
283+ let pmtDecimalsB = $t093699456._3
284+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
285+ then throw("Only admin can call this function")
286+ else if (isDefined(getBoolean(this, keyActive)))
287+ then throw("DApp is already active")
288+ else if ((strAssetIdA == strAssetIdB))
289+ then throw("Assets must be different")
290+ else {
291+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
292+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
293+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
294+ let shareInitialSupply = 0
295+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
296+ let shareIssueId = calculateAssetId(shareIssue)
297+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceInitA, amtAssetA), IntegerEntry(keyBalanceInitB, amtAssetB), IntegerEntry(keyBalanceA, 0), IntegerEntry(keyBalanceB, 0), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply)]
298+ if (firstHarvest)
299+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
300+ else baseEntry
301+ }
302+ }
303+
304+
305+
306+@Callable(i)
307+func keepLimitForFirstHarvest (shareLimit) = if (!(isActive))
308+ then throw("DApp is inactive at this moment")
309+ else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
310+ then throw("Only admin can call this function")
311+ else [IntegerEntry(kShareLimit, shareLimit)]
238312
239313
240314
244318 let pmtAssetIdB = i.payments[1].assetId
245319 let pmtAmountA = deductStakingFee(i.payments[0].amount, pmtAssetIdA)
246320 let pmtAmountB = deductStakingFee(i.payments[1].amount, pmtAssetIdB)
247- let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
248- let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
249- let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
250- let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
251- if (!(isActive))
252- then throw("DApp is inactive at this moment")
253- else if (if ((0 > slippageTolerance))
254- then true
255- else (slippageTolerance > slippageToleranceDelimiter))
256- then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
257- else if ((size(i.payments) != 2))
258- then throw("Two attached assets expected")
259- else if (if ((pmtAssetIdA != assetIdA))
321+ if (if ((balanceA == 0))
322+ then (balanceB == 0)
323+ else false)
324+ then {
325+ let $t01214112218 = getAssetInfo(pmtAssetIdA)
326+ let pmtStrAssetIdA = $t01214112218._1
327+ let pmtAssetNameA = $t01214112218._2
328+ let pmtDecimalsA = $t01214112218._3
329+ let $t01222712304 = getAssetInfo(pmtAssetIdB)
330+ let pmtStrAssetIdB = $t01222712304._1
331+ let pmtAssetNameB = $t01222712304._2
332+ let pmtDecimalsB = $t01222712304._3
333+ let tokenRatio = fraction(fraction(assetInitA, scaleValue8, pmtAmountA), scaleValue3, fraction(assetInitB, scaleValue8, pmtAmountB))
334+ if ((pmtAssetIdA == pmtAssetIdB))
335+ then throw("Assets must be different")
336+ else {
337+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
338+ let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, DOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, DOWN), pow(10, 0, shareDecimals, 0, 0, DOWN))
339+ if (!(isActive))
340+ then throw("DApp is inactive at this moment")
341+ else if (if ((0 > slippageTolerance))
342+ then true
343+ else (slippageTolerance > slippageToleranceDelimiter))
344+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
345+ else if ((size(i.payments) != 2))
346+ then throw("Two attached assets expected")
347+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
348+ then true
349+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
350+ then throw("Incorrect assets amount: amounts must have the contract ratio")
351+ else if (if ((pmtAssetIdA != assetIdA))
352+ then true
353+ else (pmtAssetIdB != assetIdB))
354+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
355+ else if ((shareInitialSupply == 0))
356+ then throw("Too small amount to replenish")
357+ else if (!(hasEnoughBalance))
358+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
359+ else [Reissue(shareAssetId, shareInitialSupply, true), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareAssetId)]
360+ }
361+ }
362+ else {
363+ let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
364+ let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
365+ let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
366+ let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
367+ if (!(isActive))
368+ then throw("DApp is inactive at this moment")
369+ else if (if ((0 > slippageTolerance))
260370 then true
261- else (pmtAssetIdB != assetIdB))
262- then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
263- else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
264- then true
265- else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
266- then throw("Incorrect assets amount: amounts must have the contract ratio")
267- else if ((shareTokenToPayAmount == 0))
268- then throw("Too small amount to replenish")
269- else if (!(hasEnoughBalance))
270- then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
271- else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
371+ else (slippageTolerance > slippageToleranceDelimiter))
372+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
373+ else if ((size(i.payments) != 2))
374+ then throw("Two attached assets expected")
375+ else if (if ((pmtAssetIdA != assetIdA))
376+ then true
377+ else (pmtAssetIdB != assetIdB))
378+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
379+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
380+ then true
381+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
382+ then throw("Incorrect assets amount: amounts must have the contract ratio")
383+ else if ((shareTokenToPayAmount == 0))
384+ then throw("Too small amount to replenish")
385+ else if (!(hasEnoughBalance))
386+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
387+ else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
388+ }
272389 }
273390
274391
275392
276393 @Callable(i)
277394 func withdraw () = {
278- let $t01033010480 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
279- let pmtAmount = $t01033010480._1
280- let pmtAssetId = $t01033010480._2
395+ let $t01670316853 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
396+ let pmtAmount = $t01670316853._1
397+ let pmtAssetId = $t01670316853._2
281398 let amountToPayA = deductStakingFee(fraction(pmtAmount, balanceA, shareAssetSupply), assetIdA)
282399 let amountToPayB = deductStakingFee(fraction(pmtAmount, balanceB, shareAssetSupply), assetIdB)
283400 if (!(isActive))
299416
300417 @Callable(i)
301418 func exchange (minAmountToReceive) = {
302- let $t01168711762 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
303- let pmtAmount = $t01168711762._1
304- let pmtAssetId = $t01168711762._2
419+ let $t01805918134 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
420+ let pmtAmount = $t01805918134._1
421+ let pmtAssetId = $t01805918134._2
305422 func calculateFees (tokenFrom,tokenTo) = {
306423 let amountWithoutFee = fraction(tokenTo, pmtAmount, (pmtAmount + tokenFrom))
307424 let amountWithFee = fraction(amountWithoutFee, (commissionScaleDelimiter - commission), commissionScaleDelimiter)
313430
314431 if (!(isActive))
315432 then throw("DApp is inactive at this moment")
316- else if ((0 >= minAmountToReceive))
317- then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
318- else if ((size(i.payments) != 1))
319- then throw("One attached payment expected")
320- else if (!(hasEnoughBalance))
321- then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
322- else if ((pmtAssetId == assetIdA))
323- then {
324- let assetIdSend = assetIdB
325- let $t01293813029 = calculateFees(balanceA, balanceB)
326- let amountWithoutFee = $t01293813029._1
327- let amountWithFee = $t01293813029._2
328- let governanceReward = $t01293813029._3
329- let newBalanceA = (balanceA + pmtAmount)
330- let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
331- if (if ((stakedAmountA >= newBalanceA))
332- then true
333- else (stakedAmountB >= newBalanceB))
334- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
335- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
336- }
337- else if ((pmtAssetId == assetIdB))
433+ else if (if ((balanceA == 0))
434+ then true
435+ else (balanceB == 0))
436+ then throw("Can't exchange with zero balance")
437+ else if ((0 >= minAmountToReceive))
438+ then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
439+ else if ((size(i.payments) != 1))
440+ then throw("One attached payment expected")
441+ else if (!(hasEnoughBalance))
442+ then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
443+ else if ((pmtAssetId == assetIdA))
338444 then {
339- let assetIdSend = assetIdA
340- let $t01385213943 = calculateFees(balanceB, balanceA)
341- let amountWithoutFee = $t01385213943._1
342- let amountWithFee = $t01385213943._2
343- let governanceReward = $t01385213943._3
344- let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
345- let newBalanceB = (balanceB + pmtAmount)
346- if (if ((stakedAmountA >= newBalanceA))
445+ let assetIdSend = assetIdB
446+ let $t01940819499 = calculateFees(balanceA, balanceB)
447+ let amountWithoutFee = $t01940819499._1
448+ let amountWithFee = $t01940819499._2
449+ let governanceReward = $t01940819499._3
450+ let newBalanceA = (balanceA + pmtAmount)
451+ let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
452+ if (if (if ((assetIdA == USDN))
453+ then (stakedAmountUSDN >= newBalanceA)
454+ else false)
347455 then true
348- else (stakedAmountB >= newBalanceB))
349- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
350- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
456+ else if ((assetIdB == USDN))
457+ then (stakedAmountUSDN >= newBalanceB)
458+ else false)
459+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
460+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
351461 }
352- else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
462+ else if ((pmtAssetId == assetIdB))
463+ then {
464+ let assetIdSend = assetIdA
465+ let $t02036820459 = calculateFees(balanceB, balanceA)
466+ let amountWithoutFee = $t02036820459._1
467+ let amountWithFee = $t02036820459._2
468+ let governanceReward = $t02036820459._3
469+ let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
470+ let newBalanceB = (balanceB + pmtAmount)
471+ if (if (if ((assetIdA == USDN))
472+ then (stakedAmountUSDN >= newBalanceA)
473+ else false)
474+ then true
475+ else if ((assetIdB == USDN))
476+ then (stakedAmountUSDN >= newBalanceB)
477+ else false)
478+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
479+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
480+ }
481+ else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
353482 }
354483
355484
416545 }
417546 match tx {
418547 case inv: InvokeScriptTransaction =>
419- if (if (if (if (sigVerify(inv.bodyBytes, inv.proofs[0], adminPubKeyStaking))
420- then (inv.dApp == stakingAddressUSDN)
548+ let callTakeIntoAccount = if ((inv.dApp == this))
549+ then (inv.function == "takeIntoAccountExtraFunds")
550+ else false
551+ let callStaking = if ((inv.dApp == stakingAddress))
552+ then if (if (if ((inv.function == "lockNeutrino"))
553+ then (size(inv.payments) == 1)
554+ else false)
555+ then (inv.payments[0].assetId == USDN)
556+ else false)
557+ then true
558+ else if ((inv.function == "unlockNeutrino"))
559+ then (size(inv.payments) == 0)
560+ else false
561+ else false
562+ let exchangeToWaves = if (if (if ((inv.dApp == USDNToWavesExchanger))
563+ then (inv.function == "exchange")
564+ else false)
565+ then (assetIdA == USDN)
421566 else false)
422567 then true
423- else if (if (if (sigVerify(inv.bodyBytes, inv.proofs[0], adminPubKeyStaking))
424- then (inv.function == "exchange")
425- else false)
568+ else if (if ((assetIdB == USDN))
426569 then (size(inv.payments) == 1)
427570 else false)
428- then if (if (if (if (if ((assetIdA == NSBT))
429- then (assetIdB == USDN)
430- else false)
431- then (inv.dApp == swopUSDNtoWAVES)
432- else false)
433- then (inv.payments[0].assetId == unit)
434- else false)
435- then if ((inv.payments[0].assetId == unit))
436- then true
437- else (inv.payments[0].assetId == USDN)
438- else false)
439- then true
440- else if (if ((assetIdA == NSBT))
441- then (assetIdB == SWOP)
442- else false)
443- then if (if ((inv.dApp == swopUSDNtoNSBT))
444- then (inv.payments[0].assetId == USDN)
445- else false)
446- then true
447- else if ((inv.dApp == swopUSDNtoWAVES))
448- then if ((inv.payments[0].assetId == unit))
449- then true
450- else (inv.payments[0].assetId == USDN)
451- else false
452- else false
453- else false)
571+ then (inv.payments[0].assetId == USDN)
572+ else false
573+ let signedByAdmin = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1))
454574 then true
455- else if (if (sigVerify(inv.bodyBytes, inv.proofs[0], adminPubKeyStaking))
456- then (inv.dApp == this)
457- else false)
458- then (inv.function == "takeIntoAccountExtraFunds")
459- else false)
575+ else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey2))
576+ then true
577+ else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey3))
578+ then true
579+ else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKeyStaking)
580+ if (if (if (if (callTakeIntoAccount)
581+ then true
582+ else callStaking)
583+ then true
584+ else exchangeToWaves)
585+ then signedByAdmin
586+ else false)
460587 then true
461588 else multiSignedByAdmins
462589 case _ =>
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let version = "1.0.0"
55
66 let keyVersion = "version"
77
88 let keyActive = "active"
99
1010 let keyAssetIdA = "A_asset_id"
1111
1212 let keyAssetIdB = "B_asset_id"
1313
1414 let keyBalanceA = "A_asset_balance"
1515
1616 let keyBalanceB = "B_asset_balance"
1717
18+let keyBalanceInitA = "A_asset_init"
19+
20+let keyBalanceInitB = "B_asset_init"
21+
1822 let keyShareAssetId = "share_asset_id"
1923
2024 let keyShareAssetSupply = "share_asset_supply"
2125
2226 let keyCommission = "commission"
2327
2428 let keyCommissionScaleDelimiter = "commission_scale_delimiter"
2529
2630 let keyCause = "shutdown_cause"
31+
32+let keyFirstHarvest = "first_harvest"
33+
34+let keyFirstHarvestHeight = "first_harvest_height"
35+
36+let kShareLimit = "share_limit_on_first_harvest"
37+
38+let kBasePeriod = "base_period"
39+
40+let kPeriodLength = "period_length"
41+
42+let kStartHeight = "start_height"
43+
44+let kFirstHarvestHeight = "first_harvest_height"
2745
2846 let keyAdminPubKey1 = "admin_pub_1"
2947
3048 let keyAdminPubKey2 = "admin_pub_2"
3149
3250 let keyAdminPubKey3 = "admin_pub_3"
3351
3452 let oracle = Address(base58'3NBBWfzZtZtszaXbitTKnrB2xXwv26Bn7H9')
3553
3654 func getAdminPub (keyAdminPub) = match getString(oracle, keyAdminPub) {
3755 case string: String =>
3856 fromBase58String(string)
3957 case nothing =>
4058 throw("Admin public key is empty")
4159 }
4260
4361
4462 let adminPubKey1 = getAdminPub(keyAdminPubKey1)
4563
4664 let adminPubKey2 = getAdminPub(keyAdminPubKey2)
4765
4866 let adminPubKey3 = getAdminPub(keyAdminPubKey3)
4967
5068 let adminPubKeyStartStop = base58'Kn7NpzaG12dLZgcHf2ipUftU6hbJygmrhFqQYE4B7ZK'
5169
5270 let adminPubKeyStaking = base58'Kn7NpzaG12dLZgcHf2ipUftU6hbJygmrhFqQYE4B7ZK'
5371
54-let governanceAddress = Address(base58'3NAGTtZz6WpupSN89NZD5rMZwwziZEg4Kx4')
72+let walletAddress = Address(base58'3NAGTtZz6WpupSN89NZD5rMZwwziZEg4Kx4')
5573
56-let stakingAddressUSDN = Address(base58'3N6q7sCGSSLBUXDdjBdYGTJbZGZfhhh8cNg')
74+let votingAddress = Address(base58'3MrJgdL1GniipErHy44YF9idzLaUL2iX5DQ')
5775
58-let USDN = base58'8UrfDVd5GreeUwm7uPk7eYz1eMv376kzR52C6sANPkwS'
76+let stakingAddress = Address(base58'3N6q7sCGSSLBUXDdjBdYGTJbZGZfhhh8cNg')
5977
60-let NSBT = base58'36mg8NZTaFRDygiVwb8uBnLR51hetJruUCZcxhaVcHj9'
78+let USDNToWavesExchanger = Address(base58'3Mvutbqb6ykHHvNCCgELTNzyAMaPTqFMixX')
6179
62-let SWOP = base58'2HAJrwa8q4SxBx9cHYaBTQdBjdk5wwqdof7ccpAx2uhZ'
80+let basePeriod = valueOrErrorMessage(getInteger(votingAddress, kBasePeriod), "Empty kBasePeriod")
6381
64-let swopUSDNtoWAVES = Address(base58'3Mvutbqb6ykHHvNCCgELTNzyAMaPTqFMixX')
82+let startHeight = valueOrErrorMessage(getInteger(votingAddress, kStartHeight), "Empty kStartHeight")
6583
66-let swopUSDNtoNSBT = Address(base58'8iwQUs2Q5bPLYULK5U24DR9msCpRQbVagtRWantBuNkp')
84+let periodLength = valueOrErrorMessage(getInteger(votingAddress, kPeriodLength), "Empty kPeriodLength")
85+
86+let firstHarvestEndPeriod = ((basePeriod + ((height - startHeight) / periodLength)) + 3)
87+
88+let USDN = base58'DG2xFkPdDwKUoBkzGAhQtLpSGzfXLiCYPEzeKH2Ad24p'
6789
6890 let stakingFeeInUSDN = 270000
6991
7092 let isActive = getBooleanValue(this, keyActive)
7193
7294 let strAssetIdA = getStringValue(this, keyAssetIdA)
7395
7496 let strAssetIdB = getStringValue(this, keyAssetIdB)
7597
7698 let assetIdA = if ((strAssetIdA == "WAVES"))
7799 then unit
78100 else fromBase58String(strAssetIdA)
79101
80102 let assetIdB = if ((strAssetIdB == "WAVES"))
81103 then unit
82104 else fromBase58String(strAssetIdB)
83105
84106 let assetNameA = match assetIdA {
85107 case id: ByteVector =>
86108 value(assetInfo(id)).name
87109 case waves: Unit =>
88110 "WAVES"
89111 case _ =>
90112 throw("Match error")
91113 }
92114
93115 let assetNameB = match assetIdB {
94116 case id: ByteVector =>
95117 value(assetInfo(id)).name
96118 case waves: Unit =>
97119 "WAVES"
98120 case _ =>
99121 throw("Match error")
100122 }
101123
102124 let balanceA = getIntegerValue(this, keyBalanceA)
103125
104126 let balanceB = getIntegerValue(this, keyBalanceB)
105127
106128 let shareAssetId = fromBase58String(getStringValue(this, keyShareAssetId))
107129
108130 let shareAssetSupply = getIntegerValue(this, keyShareAssetSupply)
109131
110132 let commission = 3000
111133
112134 let commissionGovernance = 1200
113135
114136 let commissionScaleDelimiter = 1000000
115137
116138 let scaleValue3 = 1000
117139
118140 let scaleValue8 = 100000000
119141
120142 let slippageToleranceDelimiter = 1000
121143
122144 let scaleValue8Digits = 8
123145
124146 func accountBalance (assetId) = match assetId {
125147 case id: ByteVector =>
126148 assetBalance(this, id)
127149 case waves: Unit =>
128150 wavesBalance(this).available
129151 case _ =>
130152 throw("Match error")
131153 }
132154
133155
134-func stakedAmount (stakingAddress,asset) = {
135- let stakedAmountCalculated = match getInteger(stakingAddress, ((("rpd_balance_" + asset) + "_") + toString(this))) {
136- case staked: Int =>
137- staked
138- case nothing: Unit =>
139- 0
140- case _ =>
141- throw("Match error")
142- }
143- stakedAmountCalculated
144- }
156+let stakedAmountUSDN = match getInteger(stakingAddress, ((("rpd_balance_" + toBase58String(USDN)) + "_") + toString(this))) {
157+ case staked: Int =>
158+ staked
159+ case nothing: Unit =>
160+ 0
161+ case _ =>
162+ throw("Match error")
163+}
145164
165+let assetInitA = getIntegerValue(this, keyBalanceInitA)
146166
147-let stakedAmountA = stakedAmount(stakingAddressUSDN, strAssetIdA)
167+let assetInitB = getIntegerValue(this, keyBalanceInitB)
148168
149-let stakedAmountB = stakedAmount(stakingAddressUSDN, strAssetIdB)
169+let availableBalanceA = (balanceA - (if ((assetIdA == USDN))
170+ then stakedAmountUSDN
171+ else 0))
150172
151-let availableBalanceA = (balanceA - stakedAmountA)
173+let availableBalanceB = (balanceB - (if ((assetIdB == USDN))
174+ then stakedAmountUSDN
175+ else 0))
152176
153-let availableBalanceB = (balanceB - stakedAmountB)
177+let accountBalanceWithStakedA = (accountBalance(assetIdA) + (if ((assetIdA == USDN))
178+ then stakedAmountUSDN
179+ else 0))
154180
155-let accountBalanceWithStakedA = (accountBalance(assetIdA) + stakedAmountA)
156-
157-let accountBalanceWithStakedB = (accountBalance(assetIdB) + stakedAmountB)
181+let accountBalanceWithStakedB = (accountBalance(assetIdB) + (if ((assetIdB == USDN))
182+ then stakedAmountUSDN
183+ else 0))
158184
159185 let hasEnoughBalance = if ((accountBalanceWithStakedA >= balanceA))
160186 then (accountBalanceWithStakedB >= balanceB)
161187 else false
162188
163189 func getAssetInfo (assetId) = match assetId {
164190 case id: ByteVector =>
165191 let stringId = toBase58String(id)
166192 let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist"))
167193 $Tuple3(stringId, info.name, info.decimals)
168194 case waves: Unit =>
169195 $Tuple3("WAVES", "WAVES", 8)
170196 case _ =>
171197 throw("Match error")
172198 }
173199
174200
175-func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)]
201+func getAssetInfoFromString (assetStr) = if ((assetStr == "WAVES"))
202+ then $Tuple3("WAVES", "WAVES", 8)
203+ else {
204+ let stringId = assetStr
205+ let id = fromBase58String(assetStr)
206+ let info = valueOrErrorMessage(assetInfo(id), (("Asset " + stringId) + " doesn't exist"))
207+ $Tuple3(stringId, info.name, info.decimals)
208+ }
176209
177210
178-func stakedTokenCount () = {
179- let isStakedA = if ((stakedAmountA > 0))
180- then 1
181- else 0
182- let isStakedB = if ((stakedAmountB > 0))
183- then 1
184- else 0
185- (isStakedA + isStakedB)
186- }
211+func suspend (cause) = [BooleanEntry(keyActive, false), StringEntry(keyCause, cause)]
187212
188213
189214 func deductStakingFee (amount,assetId) = if ((assetId == USDN))
190215 then {
191- let result = (amount - (stakedTokenCount() * stakingFeeInUSDN))
216+ let result = (amount - stakingFeeInUSDN)
192217 if ((0 >= result))
193218 then throw((((("Insufficient amount " + toString(amount)) + " to deduct staking fee ") + toString(stakingFeeInUSDN)) + " USD-N"))
194219 else result
195220 }
196221 else amount
197222
198223
199224 func throwInsufficientAvailableBalance (amount,available,assetName) = throw((((((((("Insufficient DApp balance to pay " + toString(amount)) + " ") + assetName) + " due to staking. Available: ") + toString(available)) + " ") + assetName) + ". Please contact support in Telegram: https://t.me/swopfisupport"))
200225
201226
202227 func throwInsufficientAvailableBalances (amountA,amountB) = throw((((((((((((((((("Insufficient DApp balance to pay " + toString(amountA)) + " ") + assetNameA) + " and ") + toString(amountB)) + " ") + assetNameB) + " due to staking. Available: ") + toString(availableBalanceA)) + " ") + assetNameA) + " and ") + toString(availableBalanceB)) + " ") + assetNameB) + ". Please contact support in Telegram: https://t.me/swopfisupport"))
203228
204229
205230 func suspendSuspicious () = suspend(((((((((((((((("Suspicious state. Actual balances: " + toString(accountBalanceWithStakedA)) + " ") + assetNameA) + ", ") + toString(accountBalanceWithStakedB)) + " ") + assetNameB) + ". State: ") + toString(balanceA)) + " ") + assetNameA) + ", ") + toString(balanceB)) + " ") + assetNameB))
206231
207232
208233 @Callable(i)
209-func init () = {
210- let $t059456022 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
211- let pmtAmountA = $t059456022._1
212- let pmtAssetIdA = $t059456022._2
213- let $t060276104 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
214- let pmtAmountB = $t060276104._1
215- let pmtAssetIdB = $t060276104._2
216- let $t061096186 = getAssetInfo(pmtAssetIdA)
217- let pmtStrAssetIdA = $t061096186._1
218- let pmtAssetNameA = $t061096186._2
219- let pmtDecimalsA = $t061096186._3
220- let $t061916268 = getAssetInfo(pmtAssetIdB)
221- let pmtStrAssetIdB = $t061916268._1
222- let pmtAssetNameB = $t061916268._2
223- let pmtDecimalsB = $t061916268._3
224- if (isDefined(getBoolean(this, keyActive)))
225- then throw("DApp is already active")
226- else if ((pmtAssetIdA == pmtAssetIdB))
227- then throw("Assets must be different")
228- else {
229- let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
230- let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
231- let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
232- let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, DOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, DOWN), pow(10, 0, shareDecimals, 0, 0, DOWN))
233- let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
234- let shareIssueId = calculateAssetId(shareIssue)
235-[StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
236- }
234+func init (firstHarvest) = {
235+ let $t067096786 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
236+ let pmtAmountA = $t067096786._1
237+ let pmtAssetIdA = $t067096786._2
238+ let $t067916868 = $Tuple2(i.payments[1].amount, i.payments[1].assetId)
239+ let pmtAmountB = $t067916868._1
240+ let pmtAssetIdB = $t067916868._2
241+ let $t068736950 = getAssetInfo(pmtAssetIdA)
242+ let pmtStrAssetIdA = $t068736950._1
243+ let pmtAssetNameA = $t068736950._2
244+ let pmtDecimalsA = $t068736950._3
245+ let $t069557032 = getAssetInfo(pmtAssetIdB)
246+ let pmtStrAssetIdB = $t069557032._1
247+ let pmtAssetNameB = $t069557032._2
248+ let pmtDecimalsB = $t069557032._3
249+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
250+ then throw("Only admin can call this function")
251+ else if (isDefined(getBoolean(this, keyActive)))
252+ then throw("DApp is already active")
253+ else if ((pmtAssetIdA == pmtAssetIdB))
254+ then throw("Assets must be different")
255+ else {
256+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
257+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
258+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
259+ let arg1 = pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, DOWN)
260+ let arg2 = pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, DOWN)
261+ let arg3 = pow(10, 0, shareDecimals, 0, 0, DOWN)
262+ let shareInitialSupply = fraction(arg1, arg2, arg3)
263+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
264+ let shareIssueId = calculateAssetId(shareIssue)
265+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareIssueId)]
266+ if (firstHarvest)
267+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
268+ else baseEntry
269+ }
237270 }
271+
272+
273+
274+@Callable(i)
275+func initWithInitRatio (amtAssetA,amtAssetB,strAssetIdA,strAssetIdB,firstHarvest) = {
276+ let $t092779364 = getAssetInfoFromString(strAssetIdA)
277+ let pmtStrAssetIdA = $t092779364._1
278+ let pmtAssetNameA = $t092779364._2
279+ let pmtDecimalsA = $t092779364._3
280+ let $t093699456 = getAssetInfoFromString(strAssetIdB)
281+ let pmtStrAssetIdB = $t093699456._1
282+ let pmtAssetNameB = $t093699456._2
283+ let pmtDecimalsB = $t093699456._3
284+ if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
285+ then throw("Only admin can call this function")
286+ else if (isDefined(getBoolean(this, keyActive)))
287+ then throw("DApp is already active")
288+ else if ((strAssetIdA == strAssetIdB))
289+ then throw("Assets must be different")
290+ else {
291+ let shareName = ((("s" + take(pmtAssetNameA, 7)) + "_") + take(pmtAssetNameB, 7))
292+ let shareDescription = ((((("ShareToken of SwopFi protocol for " + pmtAssetNameA) + " and ") + pmtAssetNameB) + " at address ") + toString(this))
293+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
294+ let shareInitialSupply = 0
295+ let shareIssue = Issue(shareName, shareDescription, shareInitialSupply, shareDecimals, true)
296+ let shareIssueId = calculateAssetId(shareIssue)
297+ let baseEntry = [StringEntry(keyVersion, version), BooleanEntry(keyActive, true), StringEntry(keyAssetIdA, pmtStrAssetIdA), StringEntry(keyAssetIdB, pmtStrAssetIdB), IntegerEntry(keyBalanceInitA, amtAssetA), IntegerEntry(keyBalanceInitB, amtAssetB), IntegerEntry(keyBalanceA, 0), IntegerEntry(keyBalanceB, 0), IntegerEntry(keyCommission, commission), IntegerEntry(keyCommissionScaleDelimiter, commissionScaleDelimiter), shareIssue, StringEntry(keyShareAssetId, toBase58String(shareIssueId)), IntegerEntry(keyShareAssetSupply, shareInitialSupply)]
298+ if (firstHarvest)
299+ then (baseEntry ++ [BooleanEntry(keyFirstHarvest, firstHarvest), IntegerEntry(keyFirstHarvestHeight, (startHeight + (firstHarvestEndPeriod * periodLength)))])
300+ else baseEntry
301+ }
302+ }
303+
304+
305+
306+@Callable(i)
307+func keepLimitForFirstHarvest (shareLimit) = if (!(isActive))
308+ then throw("DApp is inactive at this moment")
309+ else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStaking], i.callerPublicKey)))
310+ then throw("Only admin can call this function")
311+ else [IntegerEntry(kShareLimit, shareLimit)]
238312
239313
240314
241315 @Callable(i)
242316 func replenishWithTwoTokens (slippageTolerance) = {
243317 let pmtAssetIdA = i.payments[0].assetId
244318 let pmtAssetIdB = i.payments[1].assetId
245319 let pmtAmountA = deductStakingFee(i.payments[0].amount, pmtAssetIdA)
246320 let pmtAmountB = deductStakingFee(i.payments[1].amount, pmtAssetIdB)
247- let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
248- let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
249- let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
250- let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
251- if (!(isActive))
252- then throw("DApp is inactive at this moment")
253- else if (if ((0 > slippageTolerance))
254- then true
255- else (slippageTolerance > slippageToleranceDelimiter))
256- then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
257- else if ((size(i.payments) != 2))
258- then throw("Two attached assets expected")
259- else if (if ((pmtAssetIdA != assetIdA))
321+ if (if ((balanceA == 0))
322+ then (balanceB == 0)
323+ else false)
324+ then {
325+ let $t01214112218 = getAssetInfo(pmtAssetIdA)
326+ let pmtStrAssetIdA = $t01214112218._1
327+ let pmtAssetNameA = $t01214112218._2
328+ let pmtDecimalsA = $t01214112218._3
329+ let $t01222712304 = getAssetInfo(pmtAssetIdB)
330+ let pmtStrAssetIdB = $t01222712304._1
331+ let pmtAssetNameB = $t01222712304._2
332+ let pmtDecimalsB = $t01222712304._3
333+ let tokenRatio = fraction(fraction(assetInitA, scaleValue8, pmtAmountA), scaleValue3, fraction(assetInitB, scaleValue8, pmtAmountB))
334+ if ((pmtAssetIdA == pmtAssetIdB))
335+ then throw("Assets must be different")
336+ else {
337+ let shareDecimals = ((pmtDecimalsA + pmtDecimalsB) / 2)
338+ let shareInitialSupply = fraction(pow(pmtAmountA, pmtDecimalsA, 5, 1, pmtDecimalsA, DOWN), pow(pmtAmountB, pmtDecimalsB, 5, 1, pmtDecimalsB, DOWN), pow(10, 0, shareDecimals, 0, 0, DOWN))
339+ if (!(isActive))
340+ then throw("DApp is inactive at this moment")
341+ else if (if ((0 > slippageTolerance))
342+ then true
343+ else (slippageTolerance > slippageToleranceDelimiter))
344+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
345+ else if ((size(i.payments) != 2))
346+ then throw("Two attached assets expected")
347+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
348+ then true
349+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
350+ then throw("Incorrect assets amount: amounts must have the contract ratio")
351+ else if (if ((pmtAssetIdA != assetIdA))
352+ then true
353+ else (pmtAssetIdB != assetIdB))
354+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
355+ else if ((shareInitialSupply == 0))
356+ then throw("Too small amount to replenish")
357+ else if (!(hasEnoughBalance))
358+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
359+ else [Reissue(shareAssetId, shareInitialSupply, true), IntegerEntry(keyBalanceA, pmtAmountA), IntegerEntry(keyBalanceB, pmtAmountB), IntegerEntry(keyShareAssetSupply, shareInitialSupply), ScriptTransfer(i.caller, shareInitialSupply, shareAssetId)]
360+ }
361+ }
362+ else {
363+ let tokenRatio = fraction(fraction(balanceA, scaleValue8, pmtAmountA), scaleValue3, fraction(balanceB, scaleValue8, pmtAmountB))
364+ let ratioShareTokensInA = fraction(pmtAmountA, scaleValue8, balanceA)
365+ let ratioShareTokensInB = fraction(pmtAmountB, scaleValue8, balanceB)
366+ let shareTokenToPayAmount = fraction(min([ratioShareTokensInA, ratioShareTokensInB]), shareAssetSupply, scaleValue8)
367+ if (!(isActive))
368+ then throw("DApp is inactive at this moment")
369+ else if (if ((0 > slippageTolerance))
260370 then true
261- else (pmtAssetIdB != assetIdB))
262- then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
263- else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
264- then true
265- else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
266- then throw("Incorrect assets amount: amounts must have the contract ratio")
267- else if ((shareTokenToPayAmount == 0))
268- then throw("Too small amount to replenish")
269- else if (!(hasEnoughBalance))
270- then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
271- else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
371+ else (slippageTolerance > slippageToleranceDelimiter))
372+ then throw(((("Slippage tolerance must be between 0 and " + toString(slippageToleranceDelimiter)) + " inclusively. Actual: ") + toString(slippageTolerance)))
373+ else if ((size(i.payments) != 2))
374+ then throw("Two attached assets expected")
375+ else if (if ((pmtAssetIdA != assetIdA))
376+ then true
377+ else (pmtAssetIdB != assetIdB))
378+ then throw(((("Incorrect assets attached. Expected: " + strAssetIdA) + " and ") + strAssetIdB))
379+ else if (if ((((scaleValue3 * (slippageToleranceDelimiter - slippageTolerance)) / slippageToleranceDelimiter) > tokenRatio))
380+ then true
381+ else (tokenRatio > ((scaleValue3 * (slippageToleranceDelimiter + slippageTolerance)) / slippageToleranceDelimiter)))
382+ then throw("Incorrect assets amount: amounts must have the contract ratio")
383+ else if ((shareTokenToPayAmount == 0))
384+ then throw("Too small amount to replenish")
385+ else if (!(hasEnoughBalance))
386+ then ([ScriptTransfer(i.caller, pmtAmountA, pmtAssetIdA), ScriptTransfer(i.caller, pmtAmountB, pmtAssetIdB)] ++ suspendSuspicious())
387+ else [IntegerEntry(keyBalanceA, (balanceA + pmtAmountA)), IntegerEntry(keyBalanceB, (balanceB + pmtAmountB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply + shareTokenToPayAmount)), Reissue(shareAssetId, shareTokenToPayAmount, true), ScriptTransfer(i.caller, shareTokenToPayAmount, shareAssetId)]
388+ }
272389 }
273390
274391
275392
276393 @Callable(i)
277394 func withdraw () = {
278- let $t01033010480 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
279- let pmtAmount = $t01033010480._1
280- let pmtAssetId = $t01033010480._2
395+ let $t01670316853 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
396+ let pmtAmount = $t01670316853._1
397+ let pmtAssetId = $t01670316853._2
281398 let amountToPayA = deductStakingFee(fraction(pmtAmount, balanceA, shareAssetSupply), assetIdA)
282399 let amountToPayB = deductStakingFee(fraction(pmtAmount, balanceB, shareAssetSupply), assetIdB)
283400 if (!(isActive))
284401 then throw("DApp is inactive at this moment")
285402 else if ((size(i.payments) != 1))
286403 then throw("One attached payment expected")
287404 else if ((pmtAssetId != shareAssetId))
288405 then throw(("Incorrect asset attached. Expected: " + toBase58String(shareAssetId)))
289406 else if (!(hasEnoughBalance))
290407 then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
291408 else if (if ((amountToPayA > availableBalanceA))
292409 then true
293410 else (amountToPayB > availableBalanceB))
294411 then throwInsufficientAvailableBalances(amountToPayA, amountToPayB)
295412 else [IntegerEntry(keyBalanceA, (balanceA - amountToPayA)), IntegerEntry(keyBalanceB, (balanceB - amountToPayB)), IntegerEntry(keyShareAssetSupply, (shareAssetSupply - pmtAmount)), Burn(shareAssetId, pmtAmount), ScriptTransfer(i.caller, amountToPayA, assetIdA), ScriptTransfer(i.caller, amountToPayB, assetIdB)]
296413 }
297414
298415
299416
300417 @Callable(i)
301418 func exchange (minAmountToReceive) = {
302- let $t01168711762 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
303- let pmtAmount = $t01168711762._1
304- let pmtAssetId = $t01168711762._2
419+ let $t01805918134 = $Tuple2(i.payments[0].amount, i.payments[0].assetId)
420+ let pmtAmount = $t01805918134._1
421+ let pmtAssetId = $t01805918134._2
305422 func calculateFees (tokenFrom,tokenTo) = {
306423 let amountWithoutFee = fraction(tokenTo, pmtAmount, (pmtAmount + tokenFrom))
307424 let amountWithFee = fraction(amountWithoutFee, (commissionScaleDelimiter - commission), commissionScaleDelimiter)
308425 let governanceReward = fraction(amountWithoutFee, commissionGovernance, commissionScaleDelimiter)
309426 if ((minAmountToReceive > amountWithFee))
310427 then throw(((("Calculated amount to receive " + toString(amountWithFee)) + " is less than specified minimum ") + toString(minAmountToReceive)))
311428 else $Tuple3(amountWithoutFee, amountWithFee, governanceReward)
312429 }
313430
314431 if (!(isActive))
315432 then throw("DApp is inactive at this moment")
316- else if ((0 >= minAmountToReceive))
317- then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
318- else if ((size(i.payments) != 1))
319- then throw("One attached payment expected")
320- else if (!(hasEnoughBalance))
321- then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
322- else if ((pmtAssetId == assetIdA))
323- then {
324- let assetIdSend = assetIdB
325- let $t01293813029 = calculateFees(balanceA, balanceB)
326- let amountWithoutFee = $t01293813029._1
327- let amountWithFee = $t01293813029._2
328- let governanceReward = $t01293813029._3
329- let newBalanceA = (balanceA + pmtAmount)
330- let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
331- if (if ((stakedAmountA >= newBalanceA))
332- then true
333- else (stakedAmountB >= newBalanceB))
334- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
335- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
336- }
337- else if ((pmtAssetId == assetIdB))
433+ else if (if ((balanceA == 0))
434+ then true
435+ else (balanceB == 0))
436+ then throw("Can't exchange with zero balance")
437+ else if ((0 >= minAmountToReceive))
438+ then throw(("Minimal amount to receive must be positive. Actual: " + toString(minAmountToReceive)))
439+ else if ((size(i.payments) != 1))
440+ then throw("One attached payment expected")
441+ else if (!(hasEnoughBalance))
442+ then ([ScriptTransfer(i.caller, pmtAmount, pmtAssetId)] ++ suspendSuspicious())
443+ else if ((pmtAssetId == assetIdA))
338444 then {
339- let assetIdSend = assetIdA
340- let $t01385213943 = calculateFees(balanceB, balanceA)
341- let amountWithoutFee = $t01385213943._1
342- let amountWithFee = $t01385213943._2
343- let governanceReward = $t01385213943._3
344- let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
345- let newBalanceB = (balanceB + pmtAmount)
346- if (if ((stakedAmountA >= newBalanceA))
445+ let assetIdSend = assetIdB
446+ let $t01940819499 = calculateFees(balanceA, balanceB)
447+ let amountWithoutFee = $t01940819499._1
448+ let amountWithFee = $t01940819499._2
449+ let governanceReward = $t01940819499._3
450+ let newBalanceA = (balanceA + pmtAmount)
451+ let newBalanceB = ((balanceB - amountWithFee) - governanceReward)
452+ if (if (if ((assetIdA == USDN))
453+ then (stakedAmountUSDN >= newBalanceA)
454+ else false)
347455 then true
348- else (stakedAmountB >= newBalanceB))
349- then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
350- else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(governanceAddress, governanceReward, assetIdSend)]
456+ else if ((assetIdB == USDN))
457+ then (stakedAmountUSDN >= newBalanceB)
458+ else false)
459+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceB, assetNameB)
460+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
351461 }
352- else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
462+ else if ((pmtAssetId == assetIdB))
463+ then {
464+ let assetIdSend = assetIdA
465+ let $t02036820459 = calculateFees(balanceB, balanceA)
466+ let amountWithoutFee = $t02036820459._1
467+ let amountWithFee = $t02036820459._2
468+ let governanceReward = $t02036820459._3
469+ let newBalanceA = ((balanceA - amountWithFee) - governanceReward)
470+ let newBalanceB = (balanceB + pmtAmount)
471+ if (if (if ((assetIdA == USDN))
472+ then (stakedAmountUSDN >= newBalanceA)
473+ else false)
474+ then true
475+ else if ((assetIdB == USDN))
476+ then (stakedAmountUSDN >= newBalanceB)
477+ else false)
478+ then throwInsufficientAvailableBalance(amountWithFee, availableBalanceA, assetNameA)
479+ else [IntegerEntry(keyBalanceA, newBalanceA), IntegerEntry(keyBalanceB, newBalanceB), ScriptTransfer(i.caller, amountWithFee, assetIdSend), ScriptTransfer(walletAddress, governanceReward, assetIdSend)]
480+ }
481+ else throw(((("Incorrect asset attached. Expected: " + strAssetIdA) + " or ") + strAssetIdB))
353482 }
354483
355484
356485
357486 @Callable(i)
358487 func shutdown () = if (!(isActive))
359488 then throw(("DApp is already suspended. Cause: " + valueOrElse(getString(this, keyCause), "the cause wasn't specified")))
360489 else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStartStop], i.callerPublicKey)))
361490 then throw("Only admin can call this function")
362491 else suspend("Paused by admin")
363492
364493
365494
366495 @Callable(i)
367496 func activate () = if (isActive)
368497 then throw("DApp is already active")
369498 else if (!(containsElement([adminPubKey1, adminPubKey2, adminPubKey3, adminPubKeyStartStop], i.callerPublicKey)))
370499 then throw("Only admin can call this function")
371500 else [BooleanEntry(keyActive, true), DeleteEntry(keyCause)]
372501
373502
374503
375504 @Callable(i)
376505 func takeIntoAccountExtraFunds (amountLeave) = {
377506 let uncountableAmountEnrollAssetA = (accountBalanceWithStakedA - balanceA)
378507 let uncountableAmountEnrollAssetB = (accountBalanceWithStakedB - balanceB)
379508 let amountEnrollA = (uncountableAmountEnrollAssetA - (if ((assetIdA == unit))
380509 then amountLeave
381510 else 0))
382511 let amountEnrollB = (uncountableAmountEnrollAssetB - (if ((assetIdB == unit))
383512 then amountLeave
384513 else 0))
385514 if (!(isActive))
386515 then throw("DApp is inactive at this moment")
387516 else if ((i.caller != this))
388517 then throw("Only the DApp itself can call this function")
389518 else if ((0 > amountLeave))
390519 then throw(("Argument 'amountLeave' cannot be negative. Actual: " + toString(amountLeave)))
391520 else if (if ((0 > uncountableAmountEnrollAssetA))
392521 then true
393522 else (0 > uncountableAmountEnrollAssetB))
394523 then suspend("Enroll amount negative")
395524 else if (if ((0 > amountEnrollA))
396525 then true
397526 else (0 > amountEnrollB))
398527 then throw("Too large amountLeave")
399528 else [IntegerEntry(keyBalanceA, (balanceA + amountEnrollA)), IntegerEntry(keyBalanceB, (balanceB + amountEnrollB)), IntegerEntry(("last_income_" + strAssetIdA), amountEnrollA), IntegerEntry(("last_income_" + strAssetIdB), amountEnrollB)]
400529 }
401530
402531
403532 @Verifier(tx)
404533 func verify () = {
405534 let multiSignedByAdmins = {
406535 let adminPubKey1Signed = if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1))
407536 then 1
408537 else 0
409538 let adminPubKey2Signed = if (sigVerify(tx.bodyBytes, tx.proofs[1], adminPubKey2))
410539 then 1
411540 else 0
412541 let adminPubKey3Signed = if (sigVerify(tx.bodyBytes, tx.proofs[2], adminPubKey3))
413542 then 1
414543 else 0
415544 (((adminPubKey1Signed + adminPubKey2Signed) + adminPubKey3Signed) >= 2)
416545 }
417546 match tx {
418547 case inv: InvokeScriptTransaction =>
419- if (if (if (if (sigVerify(inv.bodyBytes, inv.proofs[0], adminPubKeyStaking))
420- then (inv.dApp == stakingAddressUSDN)
548+ let callTakeIntoAccount = if ((inv.dApp == this))
549+ then (inv.function == "takeIntoAccountExtraFunds")
550+ else false
551+ let callStaking = if ((inv.dApp == stakingAddress))
552+ then if (if (if ((inv.function == "lockNeutrino"))
553+ then (size(inv.payments) == 1)
554+ else false)
555+ then (inv.payments[0].assetId == USDN)
556+ else false)
557+ then true
558+ else if ((inv.function == "unlockNeutrino"))
559+ then (size(inv.payments) == 0)
560+ else false
561+ else false
562+ let exchangeToWaves = if (if (if ((inv.dApp == USDNToWavesExchanger))
563+ then (inv.function == "exchange")
564+ else false)
565+ then (assetIdA == USDN)
421566 else false)
422567 then true
423- else if (if (if (sigVerify(inv.bodyBytes, inv.proofs[0], adminPubKeyStaking))
424- then (inv.function == "exchange")
425- else false)
568+ else if (if ((assetIdB == USDN))
426569 then (size(inv.payments) == 1)
427570 else false)
428- then if (if (if (if (if ((assetIdA == NSBT))
429- then (assetIdB == USDN)
430- else false)
431- then (inv.dApp == swopUSDNtoWAVES)
432- else false)
433- then (inv.payments[0].assetId == unit)
434- else false)
435- then if ((inv.payments[0].assetId == unit))
436- then true
437- else (inv.payments[0].assetId == USDN)
438- else false)
439- then true
440- else if (if ((assetIdA == NSBT))
441- then (assetIdB == SWOP)
442- else false)
443- then if (if ((inv.dApp == swopUSDNtoNSBT))
444- then (inv.payments[0].assetId == USDN)
445- else false)
446- then true
447- else if ((inv.dApp == swopUSDNtoWAVES))
448- then if ((inv.payments[0].assetId == unit))
449- then true
450- else (inv.payments[0].assetId == USDN)
451- else false
452- else false
453- else false)
571+ then (inv.payments[0].assetId == USDN)
572+ else false
573+ let signedByAdmin = if (if (if (sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey1))
454574 then true
455- else if (if (sigVerify(inv.bodyBytes, inv.proofs[0], adminPubKeyStaking))
456- then (inv.dApp == this)
457- else false)
458- then (inv.function == "takeIntoAccountExtraFunds")
459- else false)
575+ else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey2))
576+ then true
577+ else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKey3))
578+ then true
579+ else sigVerify(tx.bodyBytes, tx.proofs[0], adminPubKeyStaking)
580+ if (if (if (if (callTakeIntoAccount)
581+ then true
582+ else callStaking)
583+ then true
584+ else exchangeToWaves)
585+ then signedByAdmin
586+ else false)
460587 then true
461588 else multiSignedByAdmins
462589 case _ =>
463590 multiSignedByAdmins
464591 }
465592 }
466593

github/deemru/w8io/169f3d6 
72.53 ms