tx · 3LgE7R4w5wEhwTjRbtCKAPTMBiFBrLTSVD9dsXg3nULF

3N7LYefWk1WhvCsY5A9nTq244SbVPmhCypi:  -0.02000000 Waves

2023.03.01 18:18 [2471169] smart account 3N7LYefWk1WhvCsY5A9nTq244SbVPmhCypi > SELF 0.00000000 Waves

{ "type": 13, "id": "3LgE7R4w5wEhwTjRbtCKAPTMBiFBrLTSVD9dsXg3nULF", "fee": 2000000, "feeAssetId": null, "timestamp": 1677683925717, "version": 2, "chainId": 84, "sender": "3N7LYefWk1WhvCsY5A9nTq244SbVPmhCypi", "senderPublicKey": "4qGVXRFATZNzo2RZZ5UYHXojGyJgSRBwtgJcCU8rJe5A", "proofs": [ "2995Fmrx3YJLXS4qYT4x9jsQtngqvA8pLPTVN69cfZgRwtLrzi3MiX5mJQPWr8qMo3vqza5feRGK2TXXjaRPqjm6", "wWtuaASW4QxikyGfWVrEDmEgpYuJVpeGkHwu6RZzpSUhiRnvmxRcRngurzNaWaAwwFWaFzttBU9kHkXAagxMzwS", "55xp5VfST3Gn76atKtXt4JzULyDeNYMFQvv3wEwG6PQhRj5Zkqa9P9t4ynhiT8zwzxbx2RVPF6A2j9Am1mTuBMMT" ], "script": "base64:BgI5CAISBQoDCAgBEgMKAQESBwoFAQEBAQESAwoBARIECgIBAhIDCgEBEgASBQoDAQEBEgcKBQEBAQEBdQALcmV2aXNpb25OdW0CAAAJTVVMVFNDQUxFAAgAB1NDQUxFMTYAEAADU0VQAgJfXwAFSEFMRjgAgOHrFwAIRVVMRVJYMTYJALYCAQD0quOekdSkMAAEVFdPWAkAtgIBAAIABU1VTFQ2AMCEPQAGTVVMVFg2CQC2AgEAwIQ9AAVNVUxUOACAwtcvAAZNVUxUWDgJALYCAQCAwtcvAAdNVUxUWDEwCQC2AgEAgMivoCUAB01VTFRYMTYJALYCAQCAgIT+pt7hEQAHV0FWRVNJRAkA2QQBAgVXQVZFUwALa1Jlc3VsdElkeEEAAAAPa1Jlc3VsdElkeFBhdWxCAAEAGWtSZXN1bHRJZHhXUmVzZXJ2ZXNJblVzZG4AAgAQa1Jlc3VsdElkeE11bHRCUgADABNrUmVzdWx0SWR4TXVsdFBvd2VyAAQAG2tSZXN1bHRJZHhNdWx0RXhwSW5Qb3dlclN0cgAFABJrUmVzdWx0SWR4TXVsdEtTdHIABgAPbk1ldHJpY0lkeFByaWNlAAAAG25NZXRyaWNJZHhVc2RuTG9ja2VkQmFsYW5jZQABABxuTWV0cmljSWR4V2F2ZXNMb2NrZWRCYWxhbmNlAAIAEW5NZXRyaWNJZHhSZXNlcnZlAAMAF25NZXRyaWNJZHhSZXNlcnZlSW5Vc2RuAAQAFG5NZXRyaWNJZHhVc2RuU3VwcGx5AAUAEW5NZXRyaWNJZHhTdXJwbHVzAAYAGG5NZXRyaWNJZHhTdXJwbHVzUGVyY2VudAAHAAxuTWV0cmljSWR4QlIACAAUbk1ldHJpY0lkeE5zYnRTdXBwbHkACQAXbk1ldHJpY0lkeE1heE5zYnRTdXBwbHkACgAUbk1ldHJpY0lkeFN1cmZTdXBwbHkACwASbk1ldHJpY1VzZG5Vc2R0UGVnAAwAFm5NZXRyaWNDdXJyZW50UHJpY2VBZGoADQARbk1ldHJpY0Jhc2tldEluZm8ADgESa2V5TmV1dHJpbm9BZGRyZXNzAAIdJXMlc19fY29uZmlnX19uZXV0cmlub0FkZHJlc3MBE2tleVN3YXBBbW91bnRBUGFyYW0AAhglcyVzX19jb25maWdfX3N3YXBBUGFyYW0BE2tleVN3YXBBbW91bnRCUGFyYW0AAhglcyVzX19jb25maWdfX3N3YXBCUGFyYW0BF2tleVVzZG5Td2FwQW1vdW50QVBhcmFtAAIcJXMlc19fY29uZmlnX191c2RuU3dhcEFQYXJhbQEXa2V5VXNkblN3YXBBbW91bnRCUGFyYW0AAhwlcyVzX19jb25maWdfX3VzZG5Td2FwQlBhcmFtARNrZXlQcmljZUFkak1pbkNvZWZmAAIeJXMlc19fcHJpY2VBZGpfX21pbkNvZWZmaWNpZW50ARdrZXlQcmljZUFkakFyYlJlZ3VsYXRvcgACHCVzJXNfX3ByaWNlQWRqX19hcmJSZWd1bGF0b3IBEmtleU5ldXRyaW5vQXNzZXRJZAACEW5ldXRyaW5vX2Fzc2V0X2lkAQ5rZXlOc2J0QXNzZXRJZAACDWJvbmRfYXNzZXRfaWQBDmtleVN1cmZBc3NldElkAAINc3VyZl9hc3NldF9pZAERc3dhcHNUaW1lZnJhbWVLRVkAAg9zd2Fwc190aW1lZnJhbWUBGmtleVVzZXJMYXN0UXVpY2tTd2FwSGVpZ2h0AQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgIXdXNlckxhc3RRdWlja1N3YXBIZWlnaHQJAMwIAgULdXNlckFkZHJlc3MFA25pbAUDU0VQAR1rZXlRdWlja1N3YXBVc2VyU3BlbnRJblBlcmlvZAELdXNlckFkZHJlc3MJALkJAgkAzAgCAgQlcyVzCQDMCAICGnF1aWNrU3dhcFVzZXJTcGVudEluUGVyaW9kCQDMCAIFC3VzZXJBZGRyZXNzBQNuaWwFA1NFUAEPYmFza2V0QXNzZXRzS2V5AAIaJXMlc19fY29tbW9uX19iYXNrZXRBc3NldHMBCGtleVByaWNlAAIFcHJpY2UBD2tleVByaWNlQnlBc3NldAEHYXNzZXRJZAkArAICAh4lcyVzJXNfX2NvbW1vbl9fcHJpY2VCeUFzc2V0X18FB2Fzc2V0SWQADVJwZEJhbGFuY2VLZXkCC3JwZF9iYWxhbmNlARFycGRVc2VyQmFsYW5jZUtleQIFb3duZXIHYXNzZXRJZAkAuQkCCQDMCAIFDVJwZEJhbGFuY2VLZXkJAMwIAgUHYXNzZXRJZAkAzAgCBQVvd25lcgUDbmlsAgFfAQ9nZXRTdHJpbmdPckZhaWwCB2FkZHJlc3MDa2V5CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ0IAgUHYWRkcmVzcwUDa2V5CQC5CQIJAMwIAgIKbWFuZGF0b3J5IAkAzAgCCQClCAEFB2FkZHJlc3MJAMwIAgIBLgkAzAgCBQNrZXkJAMwIAgIPIGlzIG5vdCBkZWZpbmVkBQNuaWwCAAEMZ2V0SW50T3JGYWlsAgdhZGRyZXNzA2tleQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFB2FkZHJlc3MFA2tleQkAuQkCCQDMCAICCm1hbmRhdG9yeSAJAMwIAgkApQgBBQdhZGRyZXNzCQDMCAICAS4JAMwIAgUDa2V5CQDMCAICDyBpcyBub3QgZGVmaW5lZAUDbmlsAgABDmdldE51bWJlckJ5S2V5AgdhZGRyZXNzA2tleQkBC3ZhbHVlT3JFbHNlAgkAmggCBQdhZGRyZXNzBQNrZXkAAAENZ2V0Qm9vbE9yRmFpbAIHYWRkcmVzcwNrZXkJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmwgCBQdhZGRyZXNzBQNrZXkJALkJAgkAzAgCAgptYW5kYXRvcnkgCQDMCAIJAKUIAQUHYWRkcmVzcwkAzAgCAgEuCQDMCAIFA2tleQkAzAgCAg8gaXMgbm90IGRlZmluZWQFA25pbAIAARZjb252ZXJ0TmV1dHJpbm9Ub1dhdmVzAgZhbW91bnQFcHJpY2UJAGsDBQZhbW91bnQFBU1VTFQ4BQVwcmljZQEWY29udmVydFdhdmVzVG9OZXV0cmlubwIGYW1vdW50BXByaWNlCQBrAwUGYW1vdW50BQVwcmljZQUFTVVMVDgBBXRvWDE2AgdvcmlnVmFsDW9yaWdTY2FsZU11bHQJALwCAwkAtgIBBQdvcmlnVmFsBQdNVUxUWDE2CQC2AgEFDW9yaWdTY2FsZU11bHQBB2Zyb21YMTYCA3ZhbA9yZXN1bHRTY2FsZU11bHQJAKADAQkAvAIDBQN2YWwJALYCAQUPcmVzdWx0U2NhbGVNdWx0BQdNVUxUWDE2AQlhc0FueUxpc3QBA3ZhbAQHJG1hdGNoMAUDdmFsAwkAAQIFByRtYXRjaDACCUxpc3RbQW55XQQKdmFsQW55THlzdAUHJG1hdGNoMAUKdmFsQW55THlzdAkAAgECG2ZhaWwgdG8gY2FzdCBpbnRvIExpc3RbQW55XQEFYXNJbnQBA3ZhbAQHJG1hdGNoMAUDdmFsAwkAAQIFByRtYXRjaDACA0ludAQGdmFsSW50BQckbWF0Y2gwBQZ2YWxJbnQJAAIBAhVmYWlsIHRvIGNhc3QgaW50byBJbnQBCGFzU3RyaW5nAQN2YWwEByRtYXRjaDAFA3ZhbAMJAAECBQckbWF0Y2gwAgZTdHJpbmcEBnZhbFN0cgUHJG1hdGNoMAUGdmFsU3RyCQACAQIYZmFpbCB0byBjYXN0IGludG8gU3RyaW5nAQ1hc1ByaWNlU1RSVUNUAQF2BAckbWF0Y2gwBQF2AwkAAQIFByRtYXRjaDACFChJbnQsIEludCwgSW50LCBJbnQpBAZzdHJ1Y3QFByRtYXRjaDAFBnN0cnVjdAkAAgECHWZhaWwgdG8gY2FzdCBpbnRvIFByaWNlU1RSVUNUABlJZHhDb250cm9sQ2ZnTmV1dHJpbm9EYXBwAAEAGElkeENvbnRyb2xDZmdBdWN0aW9uRGFwcAACABRJZHhDb250cm9sQ2ZnUnBkRGFwcAADABVJZHhDb250cm9sQ2ZnTWF0aERhcHAABAAcSWR4Q29udHJvbENmZ0xpcXVpZGF0aW9uRGFwcAAFABVJZHhDb250cm9sQ2ZnUmVzdERhcHAABgAdSWR4Q29udHJvbENmZ05vZGVSZWdpc3RyeURhcHAABwAcSWR4Q29udHJvbENmZ05zYnRTdGFraW5nRGFwcAAIABlJZHhDb250cm9sQ2ZnTWVkaWF0b3JEYXBwAAkAHElkeENvbnRyb2xDZmdTdXJmU3Rha2luZ0RhcHAACgAgSWR4Q29udHJvbENmZ0duc2J0Q29udHJvbGxlckRhcHAACwAXSWR4Q29udHJvbENmZ1Jlc3RWMkRhcHAADAAbSWR4Q29udHJvbENmZ0dvdmVybmFuY2VEYXBwAA0AHElkeENvbnRyb2xDZmdQZWdQcm92aWRlckRhcHAADgERa2V5Q29udHJvbEFkZHJlc3MAAhwlcyVzX19jb25maWdfX2NvbnRyb2xBZGRyZXNzAQ1rZXlDb250cm9sQ2ZnAAIRJXNfX2NvbnRyb2xDb25maWcBFHJlYWRDb250cm9sQ2ZnT3JGYWlsAQdjb250cm9sCQC8CQIJAQ9nZXRTdHJpbmdPckZhaWwCBQdjb250cm9sCQENa2V5Q29udHJvbENmZwAFA1NFUAEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgpjb250cm9sQ2ZnA2lkeAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJAJEDAgUKY29udHJvbENmZwUDaWR4CQCsAgICLUNvbnRyb2wgY2ZnIGRvZXNuJ3QgY29udGFpbiBhZGRyZXNzIGF0IGluZGV4IAkApAMBBQNpZHgAD2NvbnRyb2xDb250cmFjdAkBEUBleHRyTmF0aXZlKDEwNjIpAQkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzCQERa2V5Q29udHJvbEFkZHJlc3MAAiMzTjROUzdkNEpvOWE2RjE0TGlGVUtLWVZkVWtrZjJlUDRaeAAKY29udHJvbENmZwkBFHJlYWRDb250cm9sQ2ZnT3JGYWlsAQUPY29udHJvbENvbnRyYWN0ABBuZXV0cmlub0NvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUZSWR4Q29udHJvbENmZ05ldXRyaW5vRGFwcAAPYXVjdGlvbkNvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUYSWR4Q29udHJvbENmZ0F1Y3Rpb25EYXBwABNsaXF1aWRhdGlvbkNvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUcSWR4Q29udHJvbENmZ0xpcXVpZGF0aW9uRGFwcAATdXNkblN0YWtpbmdDb250cmFjdAkBGGdldENvbnRyYWN0QWRkcmVzc09yRmFpbAIFCmNvbnRyb2xDZmcFFElkeENvbnRyb2xDZmdScGREYXBwABNwZWdQcm92aWRlckNvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUcSWR4Q29udHJvbENmZ1BlZ1Byb3ZpZGVyRGFwcAASbmV1dHJpbm9Bc3NldElkU3RyCQEPZ2V0U3RyaW5nT3JGYWlsAgUQbmV1dHJpbm9Db250cmFjdAkBEmtleU5ldXRyaW5vQXNzZXRJZAAAD25ldXRyaW5vQXNzZXRJZAkA2QQBBRJuZXV0cmlub0Fzc2V0SWRTdHIAC25zYnRBc3NldElkCQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFEG5ldXRyaW5vQ29udHJhY3QJAQ5rZXlOc2J0QXNzZXRJZAAAC3N1cmZBc3NldElkCQDZBAEJARFAZXh0ck5hdGl2ZSgxMDUzKQIFD2F1Y3Rpb25Db250cmFjdAkBDmtleVN1cmZBc3NldElkAAEQa2V5QmFsYW5jZUxvY2tlZAACDWJhbGFuY2VfbG9ja18BDnRvdGFsTG9ja2VkS0VZAghzd2FwVHlwZQdhc3NldElkCQC5CQIJAMwIAgIGJXMlcyVzCQDMCAICC2JhbGFuY2VMb2NrCQDMCAIFCHN3YXBUeXBlCQDMCAIFB2Fzc2V0SWQFA25pbAUDU0VQARVrZXlUb2tlbkxvY2tlZEJhbGFuY2UBB2Fzc2V0SWQJAQ50b3RhbExvY2tlZEtFWQICC291dE5ldXRyaW5vBQdhc3NldElkARhrZXlOZXV0cmlub0xvY2tlZEJhbGFuY2UACQEOdG90YWxMb2NrZWRLRVkCAghvdXRNdWx0aQUSbmV1dHJpbm9Bc3NldElkU3RyARtrZXlPbGROZXV0cmlub0xvY2tlZEJhbGFuY2UACQCsAgIJARBrZXlCYWxhbmNlTG9ja2VkAAIIbmV1dHJpbm8BGGtleU9sZFdhdmVzTG9ja2VkQmFsYW5jZQAJAKwCAgkBEGtleUJhbGFuY2VMb2NrZWQAAgV3YXZlcwEZY2FsY1VzZG5PdXRPZk1hcmtldFN1cHBseQAEFm91dE9mTWFya2V0QWRkcmVzc0xpc3QJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUPY29udHJvbENvbnRyYWN0AiUlcyVzX19jb250cm9sX19vdXRPZk1hcmtldEFkZHJlc3NMaXN0AgADCQAAAgUWb3V0T2ZNYXJrZXRBZGRyZXNzTGlzdAIAAAAKAQtiYWxhbmNlc1NVTQIRb3V0T2ZNYXJrZXRTdXBwbHkLbmV4dEFkZHJlc3MJAGQCCQBkAgURb3V0T2ZNYXJrZXRTdXBwbHkJAQt2YWx1ZU9yRWxzZQIJAJoIAgUTdXNkblN0YWtpbmdDb250cmFjdAkBEXJwZFVzZXJCYWxhbmNlS2V5AgULbmV4dEFkZHJlc3MFEm5ldXRyaW5vQXNzZXRJZFN0cgAACQDwBwIJARFAZXh0ck5hdGl2ZSgxMDYyKQEFC25leHRBZGRyZXNzBQ9uZXV0cmlub0Fzc2V0SWQKAAIkbAkAtQkCBRZvdXRPZk1hcmtldEFkZHJlc3NMaXN0BQNTRVAKAAIkcwkAkAMBBQIkbAoABSRhY2MwAAAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBC2JhbGFuY2VzU1VNAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhNMaXN0IHNpemUgZXhjZWVkcyA3CQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcBEXJlYWRDdXJyZW50QXZnUGVnAAgJAQ1hc1ByaWNlU1RSVUNUAQkA/AcEBRNwZWdQcm92aWRlckNvbnRyYWN0AgVwcmljZQkAzAgCAAQFA25pbAUDbmlsAl8yAQthZGp1c3RQcmljZQIHcHJpY2VYNg11c2RuVXNkdFBlZ1g2BAptaW5Db2VmZlg2CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJARNrZXlQcmljZUFkak1pbkNvZWZmAADAzyQEDmFyYlJlZ3VsYXRvclg2CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJARdrZXlQcmljZUFkakFyYlJlZ3VsYXRvcgAA8JMJBA9tYXhQcmljZUNvZWZmWDYJAJYDAQkAzAgCBQptaW5Db2VmZlg2CQDMCAIJAGsDCQBkAgUFTVVMVDYFDmFyYlJlZ3VsYXRvclg2BQ11c2RuVXNkdFBlZ1g2BQVNVUxUNgUDbmlsBAxwcmljZUNvZWZmWDYJAJcDAQkAzAgCBQ9tYXhQcmljZUNvZWZmWDYJAMwIAgUFTVVMVDYFA25pbAkAawMFB3ByaWNlWDYFBU1VTFQ2BQxwcmljZUNvZWZmWDYADGN1cnJlbnRQcmljZQkBEUBleHRyTmF0aXZlKDEwNTApAgUPY29udHJvbENvbnRyYWN0CQEIa2V5UHJpY2UAAA11c2RuVXNkdFBlZ1g2CQERcmVhZEN1cnJlbnRBdmdQZWcAAA9jdXJyZW50UHJpY2VBZGoJAQthZGp1c3RQcmljZQIFDGN1cnJlbnRQcmljZQUNdXNkblVzZHRQZWdYNgAVbmV1dHJpbm9Mb2NrZWRCYWxhbmNlCQBkAgkBC3ZhbHVlT3JFbHNlAgkAmggCBRBuZXV0cmlub0NvbnRyYWN0CQEba2V5T2xkTmV1dHJpbm9Mb2NrZWRCYWxhbmNlAAAACQELdmFsdWVPckVsc2UCCQCaCAIFEG5ldXRyaW5vQ29udHJhY3QJARhrZXlOZXV0cmlub0xvY2tlZEJhbGFuY2UAAAAAEndhdmVzTG9ja2VkQmFsYW5jZQkAZAIJAQt2YWx1ZU9yRWxzZQIJAJoIAgUQbmV1dHJpbm9Db250cmFjdAkBGGtleU9sZFdhdmVzTG9ja2VkQmFsYW5jZQAAAAkBC3ZhbHVlT3JFbHNlAgkAmggCBRBuZXV0cmlub0NvbnRyYWN0CQEVa2V5VG9rZW5Mb2NrZWRCYWxhbmNlAQIFV0FWRVMAAAEHcmVzZXJ2ZQEHYXNzZXRJZAQLdGVtcEFzc2V0SWQJANkEAQUHYXNzZXRJZAMJAAACBQt0ZW1wQXNzZXRJZAUHV0FWRVNJRAkAZQIICQDvBwEFEG5ldXRyaW5vQ29udHJhY3QHcmVndWxhcgUSd2F2ZXNMb2NrZWRCYWxhbmNlCQBlAgkA8AcCBRBuZXV0cmlub0NvbnRyYWN0BQt0ZW1wQXNzZXRJZAkBC3ZhbHVlT3JFbHNlAgkAmggCBRBuZXV0cmlub0NvbnRyYWN0CQEVa2V5VG9rZW5Mb2NrZWRCYWxhbmNlAQUHYXNzZXRJZAAAAQ1yZXNlcnZlc0luVXNkAQdhc3NldElkBAN2b2wJAQdyZXNlcnZlAQUHYXNzZXRJZAQCcHIJAQV2YWx1ZQEJAJoIAgUPY29udHJvbENvbnRyYWN0CQEPa2V5UHJpY2VCeUFzc2V0AQUHYXNzZXRJZAkAawMFA3ZvbAUCcHIFBU1VTFQ4ABluZXV0cmlub091dE9mTWFya2V0U3VwcGx5CQEZY2FsY1VzZG5PdXRPZk1hcmtldFN1cHBseQAAE25ldXRyaW5vVG90YWxTdXBwbHkJAGUCCQBlAgkAZAIFFW5ldXRyaW5vTG9ja2VkQmFsYW5jZQgJAQV2YWx1ZQEJAOwHAQUPbmV1dHJpbm9Bc3NldElkCHF1YW50aXR5CQDwBwIFEG5ldXRyaW5vQ29udHJhY3QFD25ldXRyaW5vQXNzZXRJZAkA8AcCBRNsaXF1aWRhdGlvbkNvbnRyYWN0BQ9uZXV0cmlub0Fzc2V0SWQADm5ldXRyaW5vU3VwcGx5CQBlAgUTbmV1dHJpbm9Ub3RhbFN1cHBseQUZbmV1dHJpbm9PdXRPZk1hcmtldFN1cHBseQEFZ2V0QlIABAZiYXNrZXQJALUJAgkBBXZhbHVlAQkAnQgCBRBuZXV0cmlub0NvbnRyYWN0CQEPYmFza2V0QXNzZXRzS2V5AAUDU0VQCgEFYWRkQnICA2FjYwdhc3NldElkCQBkAgUDYWNjCQENcmVzZXJ2ZXNJblVzZAEFB2Fzc2V0SWQEC3N1bVJlc2VydmVzCgACJGwFBmJhc2tldAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAAAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEFYWRkQnICBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoJAGsDBQtzdW1SZXNlcnZlcwUFTVVMVDYFDm5ldXRyaW5vU3VwcGx5AAJCUgMJAAACBQ5uZXV0cmlub1N1cHBseQAAAAAJAQVnZXRCUgABDWdldEJhc2tldEluZm8ABAxiYXNrZXRBc3NldHMJAQV2YWx1ZQEJAJ0IAgUQbmV1dHJpbm9Db250cmFjdAkBD2Jhc2tldEFzc2V0c0tleQAEBmJhc2tldAkAtQkCBQxiYXNrZXRBc3NldHMFA1NFUAoBCndlaWdodENhbGMCA2FjYwdhc3NldElkBAN2b2wJAQdyZXNlcnZlAQUHYXNzZXRJZAQCcHIJAQV2YWx1ZQEJAJoIAgUPY29udHJvbENvbnRyYWN0CQEPa2V5UHJpY2VCeUFzc2V0AQUHYXNzZXRJZAQEcmVzVQkAawMFA3ZvbAUCcHIFBU1VTFQ4CQCUCgIJAM0IAggFA2FjYwJfMQkAlgoEBQdhc3NldElkBQN2b2wFBHJlc1UFAnByCQBkAggFA2FjYwJfMgUEcmVzVQQBdAoAAiRsBQZiYXNrZXQKAAIkcwkAkAMBBQIkbAoABSRhY2MwCQCUCgIFA25pbAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQp3ZWlnaHRDYWxjAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKCgEHY29tYmluZQIDYWNjBGl0ZW0EBXNoYXJlAwkAAAIIBQF0Al8yAAAFBU1VTFQ2CQBrAwgFBGl0ZW0CXzMFBU1VTFQ2CAUBdAJfMgkAzQgCBQNhY2MJALkJAgkAzAgCCAUEaXRlbQJfMQkAzAgCCQCkAwEFBXNoYXJlCQDMCAIJAKQDAQgFBGl0ZW0CXzQJAMwIAgkApAMBCAUEaXRlbQJfMgUDbmlsAgE6BA13ZWlnaHRlZFByaWNlAwkAAAIFDm5ldXRyaW5vU3VwcGx5AAAAAAkAawMIBQF0Al8yBQVNVUxUNgUObmV1dHJpbm9TdXBwbHkJAJUKAwkAuQkCCgACJGwIBQF0Al8xCgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYxXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQdjb21iaW5lAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYxXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMAkBBSRmMV8yAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAgFfBQ13ZWlnaHRlZFByaWNlCAUBdAJfMgkBaQENY29uc3RydWN0b3JWMQMPbmV1dHJpbm9BZGRyZXNzD25zYnRMb2NrQWRkcmVzcxBzd2FwQW1vdW50QVBhcmFtAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIObm90IGF1dGhvcml6ZWQJAMwIAgkBC1N0cmluZ0VudHJ5AgkBEmtleU5ldXRyaW5vQWRkcmVzcwAFD25ldXRyaW5vQWRkcmVzcwkAzAgCCQEMSW50ZWdlckVudHJ5AgkBE2tleVN3YXBBbW91bnRBUGFyYW0ABRBzd2FwQW1vdW50QVBhcmFtBQNuaWwBaQEVY2FsY1N3YXBMaW1pdFJFQURPTkxZAQtnTnNidEFtb3VudAQVbGltaXRCYXNrZXRUb2tlbkluVXNkAwkAZwIAAAULZ05zYnRBbW91bnQAAAQIYVBhcmFtWDgJALYCAQkBDGdldEludE9yRmFpbAIFBHRoaXMJARNrZXlTd2FwQW1vdW50QVBhcmFtAAQJYlBhcmFtWDE2CQC2AgEJAQxnZXRJbnRPckZhaWwCBQR0aGlzCQETa2V5U3dhcEFtb3VudEJQYXJhbQAECXBvd1BhcnRYOAkAdgYJALYCAQULZ05zYnRBbW91bnQABgUJYlBhcmFtWDE2ABAACAUHQ0VJTElORwkAoAMBCQC8AgMFCGFQYXJhbVg4BQlwb3dQYXJ0WDgFB01VTFRYMTAECWxpbWl0VXNkbgMJAGcCAAAFC2dOc2J0QW1vdW50AAAECGFQYXJhbVg4CQC2AgEJAQxnZXRJbnRPckZhaWwCBQR0aGlzCQEXa2V5VXNkblN3YXBBbW91bnRBUGFyYW0ABAliUGFyYW1YMTYJALYCAQkBDGdldEludE9yRmFpbAIFBHRoaXMJARdrZXlVc2RuU3dhcEFtb3VudEJQYXJhbQAEBGJyWDYJALYCAQUCQlIECXBvd1BhcnRYOAkAdgYJALYCAQULZ05zYnRBbW91bnQABgUJYlBhcmFtWDE2ABAACAUHQ0VJTElORwkAoAMBCQC8AgMJALwCAwUIYVBhcmFtWDgFCXBvd1BhcnRYOAUHTVVMVFgxMAUEYnJYNgUGTVVMVFg2CQCUCgIFA25pbAkAzAgCBRVsaW1pdEJhc2tldFRva2VuSW5Vc2QJAMwIAgAACQDMCAIFCWxpbWl0VXNkbgUDbmlsAWkBEmNhbGN1bGF0ZUtSRUFET05MWQUEd1JhdwR1UmF3BXByaWNlBG1SYXcEc1JhdwQDRVhQCQC2AgEAyfSlAQQIRVhQU0NBTEUABgQBYQkBC3ZhbHVlT3JFbHNlAgkAmggCBQ9hdWN0aW9uQ29udHJhY3QCEG5zYnRDdXJ2ZVBhcmFtX2EAAwQFcGF1bEIJAQt2YWx1ZU9yRWxzZQIJAJoIAgUPYXVjdGlvbkNvbnRyYWN0AhBuc2J0Q3VydmVQYXJhbV9iCQBpAgkAaAIAAwUFTVVMVDYACgQPd1Jlc2VydmVzSW5Vc2RuCQEWY29udmVydFdhdmVzVG9OZXV0cmlubwIFBHdSYXcFBXByaWNlBAZtdWx0QlIJAGsDBQ93UmVzZXJ2ZXNJblVzZG4FBU1VTFQ4BQR1UmF3AwkAZgIFBm11bHRCUgDQ7/fjJwkAAgEJAKwCAgkArAICAgNCUj0JAKQDAQUGbXVsdEJSAicgPiAxMDY3OC41NjQ4MTYlIHdpbGwgb3ZlcmZsb3cgZXhwb25lbnQECW11bHRQb3dlcgkAaAIFAWEJAGUCBQZtdWx0QlIFBU1VTFQ4BA5tdWx0RXhwSW5Qb3dlcgkAdgYFA0VYUAUIRVhQU0NBTEUJALYCAQUJbXVsdFBvd2VyBQlNVUxUU0NBTEUFB1NDQUxFMTYFBERPV04EBW11bHRLCQC8AgMJALYCAQUFcGF1bEIFDm11bHRFeHBJblBvd2VyBQZNVUxUWDYJAJQKAgUDbmlsCQDMCAIFAWEJAMwIAgUFcGF1bEIJAMwIAgUPd1Jlc2VydmVzSW5Vc2RuCQDMCAIFBm11bHRCUgkAzAgCBQltdWx0UG93ZXIJAMwIAgkApgMBBQ5tdWx0RXhwSW5Qb3dlcgkAzAgCCQCmAwEFBW11bHRLBQNuaWwBaQEVY3VydmVGdW5jdGlvblJFQURPTkxZAQt3YXZlc1BheVJhdwQPbmV1dHJpbm9NZXRyaWNzCQEJYXNBbnlMaXN0AQkA/AcEBQR0aGlzAhpjYWxjTmV1dGlub01ldHJpY3NSRUFET05MWQUDbmlsBQNuaWwEBXByaWNlCQEFYXNJbnQBCQCRAwIFD25ldXRyaW5vTWV0cmljcwUPbk1ldHJpY0lkeFByaWNlBAR3UmF3CQEFYXNJbnQBCQCRAwIFD25ldXRyaW5vTWV0cmljcwURbk1ldHJpY0lkeFJlc2VydmUEBHVSYXcJAQVhc0ludAEJAJEDAgUPbmV1dHJpbm9NZXRyaWNzBRRuTWV0cmljSWR4VXNkblN1cHBseQQEc1JhdwkBBWFzSW50AQkAkQMCBQ9uZXV0cmlub01ldHJpY3MFFG5NZXRyaWNJZHhOc2J0U3VwcGx5BARtUmF3CQEFYXNJbnQBCQCRAwIFD25ldXRyaW5vTWV0cmljcwUXbk1ldHJpY0lkeE1heE5zYnRTdXBwbHkECmtDYWxjQXJyYXkJAQlhc0FueUxpc3QBCQD8BwQFBHRoaXMCEmNhbGN1bGF0ZUtSRUFET05MWQkAzAgCBQR3UmF3CQDMCAIFBHVSYXcJAMwIAgUFcHJpY2UJAMwIAgUEbVJhdwkAzAgCBQRzUmF3BQNuaWwFA25pbAQFbXVsdEsJAKcDAQkBCGFzU3RyaW5nAQkAkQMCBQprQ2FsY0FycmF5BRJrUmVzdWx0SWR4TXVsdEtTdHIEB3VzZG5QYXkJARZjb252ZXJ0V2F2ZXNUb05ldXRyaW5vAgULd2F2ZXNQYXlSYXcFBXByaWNlBBBiaWdNYXhOc2J0U3VwcGx5CQC2AgEFBG1SYXcEDWJpZ05zYnRTdXBwbHkJALYCAQkAZQIFBG1SYXcFBHNSYXcEBXN0ZXAxCQC8AgMJALYCAQUHdXNkblBheQUHTVVMVFgxNgUFbXVsdEsEBXN0ZXAyCQC8AgMFBXN0ZXAxBQ1iaWdOc2J0U3VwcGx5BRBiaWdNYXhOc2J0U3VwcGx5BAVzdGVwMwkAoAMBCQC8AgMFDWJpZ05zYnRTdXBwbHkFEGJpZ01heE5zYnRTdXBwbHkJALcCAgUFc3RlcDIFEGJpZ01heE5zYnRTdXBwbHkEDW5zYnRBbW91bnRSYXcJAGUCCQBlAgUEbVJhdwUEc1JhdwUFc3RlcDMJAJQKAgUDbmlsCQDMCAIFDW5zYnRBbW91bnRSYXcJAMwIAgUHdXNkblBheQkAzAgCBQR3UmF3CQDMCAIFBHVSYXcJAMwIAgUEbVJhdwkAzAgCBQRzUmF3CQDMCAIJAQVhc0ludAEJAJEDAgUKa0NhbGNBcnJheQULa1Jlc3VsdElkeEEJAMwIAgkBBWFzSW50AQkAkQMCBQprQ2FsY0FycmF5BQ9rUmVzdWx0SWR4UGF1bEIJAMwIAgkBBWFzSW50AQkAkQMCBQprQ2FsY0FycmF5BRlrUmVzdWx0SWR4V1Jlc2VydmVzSW5Vc2RuCQDMCAIFBXByaWNlCQDMCAIJAQVhc0ludAEJAJEDAgUKa0NhbGNBcnJheQUQa1Jlc3VsdElkeE11bHRCUgkAzAgCCQEFYXNJbnQBCQCRAwIFCmtDYWxjQXJyYXkFE2tSZXN1bHRJZHhNdWx0UG93ZXIJAMwIAgkAkQMCBQprQ2FsY0FycmF5BRtrUmVzdWx0SWR4TXVsdEV4cEluUG93ZXJTdHIJAMwIAgkAkQMCBQprQ2FsY0FycmF5BRJrUmVzdWx0SWR4TXVsdEtTdHIJAMwIAgkApgMBBQVzdGVwMQkAzAgCCQCmAwEFBXN0ZXAyCQDMCAIFBXN0ZXAzBQNuaWwBaQEUc3VyZkZ1bmN0aW9uUkVBRE9OTFkCBmFtb3VudAdhc3NldElkAwkAZgIAAAUGYW1vdW50CQACAQIZYW1vdW50IHNob3VsZCBiZSBwb3NpdGl2ZQQFcHJpY2UFDGN1cnJlbnRQcmljZQQIcHJpY2VCaWcJALYCAQUFcHJpY2UECHJlc2VydmVXCQEHcmVzZXJ2ZQECBVdBVkVTBApyZXNlcnZlQmlnCQC2AgEFCHJlc2VydmVXBAZzdXBwbHkFDm5ldXRyaW5vU3VwcGx5BAlzdXBwbHlCaWcJALYCAQUGc3VwcGx5AwkAZwIFAkJSBQVNVUxUNgkAAgEJAKwCAgkArAICAgVCUiA9IAkApAMBBQJCUgIRLCBjYW5ub3QgYnV5IFNVUkYEC21heFdhdmVzUGF5CQBlAgkAawMFBnN1cHBseQUFTVVMVDgFBXByaWNlBQhyZXNlcnZlVwQKbWF4VXNkblBheQkAZQIFBnN1cHBseQkAawMFCHJlc2VydmVXBQVwcmljZQUFTVVMVDgECXVzZUFtb3VudAMJAAACBQdhc3NldElkBQ9uZXV0cmlub0Fzc2V0SWQDCQBmAgUGYW1vdW50BQptYXhVc2RuUGF5BQptYXhVc2RuUGF5BQZhbW91bnQDCQAAAgUHYXNzZXRJZAUHV0FWRVNJRAMJAGYCBQZhbW91bnQFC21heFdhdmVzUGF5BQttYXhXYXZlc1BheQUGYW1vdW50AAAECWFtb3VudEJpZwkAtgIBBQl1c2VBbW91bnQDCQAAAgUHYXNzZXRJZAUPbmV1dHJpbm9Bc3NldElkCQCUCgIFA25pbAkAzAgCCQCgAwEJALwCAwkAvAIDBQlhbW91bnRCaWcFBk1VTFRYOAUIcHJpY2VCaWcJALgCAgUJc3VwcGx5QmlnCQC6AgIFCWFtb3VudEJpZwUEVFdPWAUKcmVzZXJ2ZUJpZwkAzAgCAAAJAMwIAgUJdXNlQW1vdW50CQDMCAIFCHJlc2VydmVXCQDMCAIFBnN1cHBseQkAzAgCBQJCUgkAzAgCBQhyZXNlcnZlVwkAzAgCCQBlAgUGc3VwcGx5BQl1c2VBbW91bnQJAMwIAgkAawMJARZjb252ZXJ0V2F2ZXNUb05ldXRyaW5vAgUIcmVzZXJ2ZVcFBXByaWNlBQVNVUxUNgkAZQIFBnN1cHBseQUJdXNlQW1vdW50CQDMCAIJAGUCBQZhbW91bnQFCXVzZUFtb3VudAkAzAgCBQVwcmljZQUDbmlsAwkAAAIFB2Fzc2V0SWQFB1dBVkVTSUQJAJQKAgUDbmlsCQDMCAIJAKADAQkAvAIDBQlzdXBwbHlCaWcJAHcGCQC8AgMJALcCAgUKcmVzZXJ2ZUJpZwUJYW1vdW50QmlnBQdNVUxUWDE2BQpyZXNlcnZlQmlnABAFCEVVTEVSWDE2ABAAEAUGSEFMRlVQBQdNVUxUWDE2CQDMCAIFCXVzZUFtb3VudAkAzAgCAAAJAMwIAgUIcmVzZXJ2ZVcJAMwIAgUGc3VwcGx5CQDMCAIFAkJSCQDMCAIJAGQCBQhyZXNlcnZlVwUJdXNlQW1vdW50CQDMCAIFBnN1cHBseQkAzAgCCQBrAwkBFmNvbnZlcnRXYXZlc1RvTmV1dHJpbm8CCQBkAgUIcmVzZXJ2ZVcFCXVzZUFtb3VudAUFcHJpY2UFBU1VTFQ2BQZzdXBwbHkJAMwIAgkAZQIFBmFtb3VudAUJdXNlQW1vdW50CQDMCAIFBXByaWNlBQNuaWwJAAIBAhFVbnN1cHBvcnRlZCBhc3NldAFpASBjYWxjQ29udHJhY3ROc2J0UHJpY2VTWVNSRUFET05MWQEPbnNidFN1cHBseURFTFRBBA9uZXV0cmlub01ldHJpY3MJAQlhc0FueUxpc3QBCQD8BwQFBHRoaXMCGmNhbGNOZXV0aW5vTWV0cmljc1JFQURPTkxZBQNuaWwFA25pbAQFcHJpY2UJAQVhc0ludAEJAJEDAgUPbmV1dHJpbm9NZXRyaWNzBQ9uTWV0cmljSWR4UHJpY2UEBHdSYXcJAQVhc0ludAEJAJEDAgUPbmV1dHJpbm9NZXRyaWNzBRFuTWV0cmljSWR4UmVzZXJ2ZQQEdVJhdwkAZQIJAQVhc0ludAEJAJEDAgUPbmV1dHJpbm9NZXRyaWNzBRRuTWV0cmljSWR4VXNkblN1cHBseQUPbnNidFN1cHBseURFTFRBBARzUmF3CQBkAgkBBWFzSW50AQkAkQMCBQ9uZXV0cmlub01ldHJpY3MFFG5NZXRyaWNJZHhOc2J0U3VwcGx5BQ9uc2J0U3VwcGx5REVMVEEEBG1SYXcJAQVhc0ludAEJAJEDAgUPbmV1dHJpbm9NZXRyaWNzBRduTWV0cmljSWR4TWF4TnNidFN1cHBseQQKa0NhbGNBcnJheQkBCWFzQW55TGlzdAEJAPwHBAUEdGhpcwISY2FsY3VsYXRlS1JFQURPTkxZCQDMCAIFBHdSYXcJAMwIAgUEdVJhdwkAzAgCBQVwcmljZQkAzAgCBQRtUmF3CQDMCAIFBHNSYXcFA25pbAUDbmlsBAhtdWx0S1gxNgkApwMBCQEIYXNTdHJpbmcBCQCRAwIFCmtDYWxjQXJyYXkFEmtSZXN1bHRJZHhNdWx0S1N0cgQMbXVsdFN0ZXAxWDE2CQC8AgMJALYCAQUEbVJhdwUHTVVMVFgxNgkAtgIBCQBlAgUEbVJhdwUEc1JhdwQMbXVsdFN0ZXAyWDE2CQC8AgMFDG11bHRTdGVwMVgxNgUMbXVsdFN0ZXAxWDE2BQdNVUxUWDE2BBVtdWx0TnNidDJ1c2RuUHJpY2VYMTYJALwCAwUIbXVsdEtYMTYFDG11bHRTdGVwMlgxNgUHTVVMVFgxNgQObnNidDJ1c2RuUHJpY2UJAKADAQkAvAIDBRVtdWx0TnNidDJ1c2RuUHJpY2VYMTYJALYCAQUFTVVMVDYFB01VTFRYMTYED25zYnQyd2F2ZXNQcmljZQkAawMFDm5zYnQydXNkblByaWNlBQVNVUxUNgUFcHJpY2UJAJQKAgUDbmlsCQDMCAIFDm5zYnQydXNkblByaWNlCQDMCAIFD25zYnQyd2F2ZXNQcmljZQUDbmlsAWkBGmNhbGNOZXV0aW5vTWV0cmljc1JFQURPTkxZAAQNbnNidFN1cHBseU1BWAgJAQV2YWx1ZQEJAOwHAQULbnNidEFzc2V0SWQIcXVhbnRpdHkECm5zYnRTdXBwbHkJAGUCBQ1uc2J0U3VwcGx5TUFYCQDwBwIFD2F1Y3Rpb25Db250cmFjdAULbnNidEFzc2V0SWQECnN1cmZTdXBwbHkICQEFdmFsdWUBCQDsBwEFC3N1cmZBc3NldElkCHF1YW50aXR5BApiYXNrZXRJbmZvCQENZ2V0QmFza2V0SW5mbwAJAJQKAgUDbmlsCQDMCAIFDGN1cnJlbnRQcmljZQkAzAgCBRVuZXV0cmlub0xvY2tlZEJhbGFuY2UJAMwIAgUSd2F2ZXNMb2NrZWRCYWxhbmNlCQDMCAIJAQdyZXNlcnZlAQIFV0FWRVMJAMwIAgkBDXJlc2VydmVzSW5Vc2QBAgVXQVZFUwkAzAgCBQ5uZXV0cmlub1N1cHBseQkAzAgCBRluZXV0cmlub091dE9mTWFya2V0U3VwcGx5CQDMCAIFE25ldXRyaW5vVG90YWxTdXBwbHkJAMwIAgUCQlIJAMwIAgUKbnNidFN1cHBseQkAzAgCBQ1uc2J0U3VwcGx5TUFYCQDMCAIFCnN1cmZTdXBwbHkJAMwIAgUNdXNkblVzZHRQZWdYNgkAzAgCBQ9jdXJyZW50UHJpY2VBZGoJAMwIAggFCmJhc2tldEluZm8CXzEJAMwIAggFCmJhc2tldEluZm8CXzIJAMwIAggFCmJhc2tldEluZm8CXzMFA25pbAFpASFnZXRVbnN0YWtlQ29taXNzaW9uQW1vdW50UkVBRE9OTFkDBmFtb3VudAtzdGFydEhlaWdodAhoYWxmTGlmZQkAlAoCBQNuaWwJAGsDBQZhbW91bnQJAGwGAAIAAAkAawMJAQEtAQkAZQIFBmhlaWdodAULc3RhcnRIZWlnaHQFBU1VTFQ4BQhoYWxmTGlmZQAIAAgFBkhBTEZVUAUFTVVMVDgBaQETbWVyZ2VTdGFrZXNSRUFET05MWQUHYW1vdW50MQdoZWlnaHQxB2Ftb3VudDIHaGVpZ2h0MghoYWxmTGlmZQQBdwkAawMFB2Ftb3VudDIJAGwGAAIAAAkAawMJAGUCBQdoZWlnaHQyBQdoZWlnaHQxBQVNVUxUOAUIaGFsZkxpZmUACAAIBQZIQUxGVVAFBU1VTFQ4BAF2CQBrAwkAZAIFB2Ftb3VudDEFB2Ftb3VudDIFBU1VTFQ4CQBkAgUHYW1vdW50MQUBdwkAlAoCBQNuaWwJAGQCBQdoZWlnaHQxCQBpAgkAZQIFBUhBTEY4CQBoAgUIaGFsZkxpZmUJAG0GBQF2AAgAAgAAAAgFBkhBTEZVUAUFTVVMVDgBAnR4AQZ2ZXJpZnkABBNwdWJLZXlBZG1pbnNMaXN0U3RyCQC5CQIJAMwIAgIsRXh0RUVLMTlubUtqOW1DcG5XeXZFRUpGWUFUTE1jVkVNdm9oaFVIa3lITm0JAMwIAgIsRXY1cHk1RmZCUVg5Y1pwWUtuZlFyVEI0OUJ5ZjhRbXBaV2VEVlJpbTR5VjcJAMwIAgIsRFV1dUxqWHU5OG5Cd1pjN2Zxd0NUanRBM25uUndnVGJrTVNyNVNVMk5tRFIJAMwIAgIsRFV1dUxqWHU5OG5Cd1pjN2Zxd0NUanRBM25uUndnVGJrTVNyNVNVMk5tRFIFA25pbAUDU0VQBBBwdWJLZXlBZG1pbnNMaXN0CQC1CQIJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUPY29udHJvbENvbnRyYWN0Agwlc19fbXVsdGlzaWcFE3B1YktleUFkbWluc0xpc3RTdHIFA1NFUAQFY291bnQJAGQCCQBkAgkAZAIDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAACQDZBAEJAJEDAgUQcHViS2V5QWRtaW5zTGlzdAAAAAEAAAMJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAEJANkEAQkAkQMCBRBwdWJLZXlBZG1pbnNMaXN0AAEAAQAAAwkA9AMDCAUCdHgJYm9keUJ5dGVzCQCRAwIIBQJ0eAZwcm9vZnMAAgkA2QQBCQCRAwIFEHB1YktleUFkbWluc0xpc3QAAgABAAADCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwADCQDZBAEJAJEDAgUQcHViS2V5QWRtaW5zTGlzdAADAAIAAAkAZwIFBWNvdW50AAPIiXtE", "height": 2471169, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 6G2CLdNyFLGjnGEAwZzNXWWqxS8BbkBpYTRctnuYkUFf Next: oQ7kKpj284T27xovAnJBjUfevnCFzL1KJLA59sr6aK9 Diff:
OldNewDifferences
1-{-# STDLIB_VERSION 5 #-}
1+{-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let revisionNum = ""
216216 func keyControlCfg () = "%s__controlConfig"
217217
218218
219-func readControlCfgOrFail (control) = split(getStringOrFail(control, keyControlCfg()), SEP)
219+func readControlCfgOrFail (control) = split_4C(getStringOrFail(control, keyControlCfg()), SEP)
220220
221221
222222 func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
Full:
OldNewDifferences
1-{-# STDLIB_VERSION 5 #-}
1+{-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let revisionNum = ""
55
66 let MULTSCALE = 8
77
88 let SCALE16 = 16
99
1010 let SEP = "__"
1111
1212 let HALF8 = 50000000
1313
1414 let EULERX16 = toBigInt(27182818284590452)
1515
1616 let TWOX = toBigInt(2)
1717
1818 let MULT6 = 1000000
1919
2020 let MULTX6 = toBigInt(1000000)
2121
2222 let MULT8 = 100000000
2323
2424 let MULTX8 = toBigInt(100000000)
2525
2626 let MULTX10 = toBigInt(10000000000)
2727
2828 let MULTX16 = toBigInt(10000000000000000)
2929
3030 let WAVESID = fromBase58String("WAVES")
3131
3232 let kResultIdxA = 0
3333
3434 let kResultIdxPaulB = 1
3535
3636 let kResultIdxWReservesInUsdn = 2
3737
3838 let kResultIdxMultBR = 3
3939
4040 let kResultIdxMultPower = 4
4141
4242 let kResultIdxMultExpInPowerStr = 5
4343
4444 let kResultIdxMultKStr = 6
4545
4646 let nMetricIdxPrice = 0
4747
4848 let nMetricIdxUsdnLockedBalance = 1
4949
5050 let nMetricIdxWavesLockedBalance = 2
5151
5252 let nMetricIdxReserve = 3
5353
5454 let nMetricIdxReserveInUsdn = 4
5555
5656 let nMetricIdxUsdnSupply = 5
5757
5858 let nMetricIdxSurplus = 6
5959
6060 let nMetricIdxSurplusPercent = 7
6161
6262 let nMetricIdxBR = 8
6363
6464 let nMetricIdxNsbtSupply = 9
6565
6666 let nMetricIdxMaxNsbtSupply = 10
6767
6868 let nMetricIdxSurfSupply = 11
6969
7070 let nMetricUsdnUsdtPeg = 12
7171
7272 let nMetricCurrentPriceAdj = 13
7373
7474 let nMetricBasketInfo = 14
7575
7676 func keyNeutrinoAddress () = "%s%s__config__neutrinoAddress"
7777
7878
7979 func keySwapAmountAParam () = "%s%s__config__swapAParam"
8080
8181
8282 func keySwapAmountBParam () = "%s%s__config__swapBParam"
8383
8484
8585 func keyUsdnSwapAmountAParam () = "%s%s__config__usdnSwapAParam"
8686
8787
8888 func keyUsdnSwapAmountBParam () = "%s%s__config__usdnSwapBParam"
8989
9090
9191 func keyPriceAdjMinCoeff () = "%s%s__priceAdj__minCoefficient"
9292
9393
9494 func keyPriceAdjArbRegulator () = "%s%s__priceAdj__arbRegulator"
9595
9696
9797 func keyNeutrinoAssetId () = "neutrino_asset_id"
9898
9999
100100 func keyNsbtAssetId () = "bond_asset_id"
101101
102102
103103 func keySurfAssetId () = "surf_asset_id"
104104
105105
106106 func swapsTimeframeKEY () = "swaps_timeframe"
107107
108108
109109 func keyUserLastQuickSwapHeight (userAddress) = makeString(["%s%s", "userLastQuickSwapHeight", userAddress], SEP)
110110
111111
112112 func keyQuickSwapUserSpentInPeriod (userAddress) = makeString(["%s%s", "quickSwapUserSpentInPeriod", userAddress], SEP)
113113
114114
115115 func basketAssetsKey () = "%s%s__common__basketAssets"
116116
117117
118118 func keyPrice () = "price"
119119
120120
121121 func keyPriceByAsset (assetId) = ("%s%s%s__common__priceByAsset__" + assetId)
122122
123123
124124 let RpdBalanceKey = "rpd_balance"
125125
126126 func rpdUserBalanceKey (owner,assetId) = makeString([RpdBalanceKey, assetId, owner], "_")
127127
128128
129129 func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
130130
131131
132132 func getIntOrFail (address,key) = valueOrErrorMessage(getInteger(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
133133
134134
135135 func getNumberByKey (address,key) = valueOrElse(getInteger(address, key), 0)
136136
137137
138138 func getBoolOrFail (address,key) = valueOrErrorMessage(getBoolean(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
139139
140140
141141 func convertNeutrinoToWaves (amount,price) = fraction(amount, MULT8, price)
142142
143143
144144 func convertWavesToNeutrino (amount,price) = fraction(amount, price, MULT8)
145145
146146
147147 func toX16 (origVal,origScaleMult) = fraction(toBigInt(origVal), MULTX16, toBigInt(origScaleMult))
148148
149149
150150 func fromX16 (val,resultScaleMult) = toInt(fraction(val, toBigInt(resultScaleMult), MULTX16))
151151
152152
153153 func asAnyList (val) = match val {
154154 case valAnyLyst: List[Any] =>
155155 valAnyLyst
156156 case _ =>
157157 throw("fail to cast into List[Any]")
158158 }
159159
160160
161161 func asInt (val) = match val {
162162 case valInt: Int =>
163163 valInt
164164 case _ =>
165165 throw("fail to cast into Int")
166166 }
167167
168168
169169 func asString (val) = match val {
170170 case valStr: String =>
171171 valStr
172172 case _ =>
173173 throw("fail to cast into String")
174174 }
175175
176176
177177 func asPriceSTRUCT (v) = match v {
178178 case struct: (Int, Int, Int, Int) =>
179179 struct
180180 case _ =>
181181 throw("fail to cast into PriceSTRUCT")
182182 }
183183
184184
185185 let IdxControlCfgNeutrinoDapp = 1
186186
187187 let IdxControlCfgAuctionDapp = 2
188188
189189 let IdxControlCfgRpdDapp = 3
190190
191191 let IdxControlCfgMathDapp = 4
192192
193193 let IdxControlCfgLiquidationDapp = 5
194194
195195 let IdxControlCfgRestDapp = 6
196196
197197 let IdxControlCfgNodeRegistryDapp = 7
198198
199199 let IdxControlCfgNsbtStakingDapp = 8
200200
201201 let IdxControlCfgMediatorDapp = 9
202202
203203 let IdxControlCfgSurfStakingDapp = 10
204204
205205 let IdxControlCfgGnsbtControllerDapp = 11
206206
207207 let IdxControlCfgRestV2Dapp = 12
208208
209209 let IdxControlCfgGovernanceDapp = 13
210210
211211 let IdxControlCfgPegProviderDapp = 14
212212
213213 func keyControlAddress () = "%s%s__config__controlAddress"
214214
215215
216216 func keyControlCfg () = "%s__controlConfig"
217217
218218
219-func readControlCfgOrFail (control) = split(getStringOrFail(control, keyControlCfg()), SEP)
219+func readControlCfgOrFail (control) = split_4C(getStringOrFail(control, keyControlCfg()), SEP)
220220
221221
222222 func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
223223
224224
225225 let controlContract = addressFromStringValue(valueOrElse(getString(this, keyControlAddress()), "3N4NS7d4Jo9a6F14LiFUKKYVdUkkf2eP4Zx"))
226226
227227 let controlCfg = readControlCfgOrFail(controlContract)
228228
229229 let neutrinoContract = getContractAddressOrFail(controlCfg, IdxControlCfgNeutrinoDapp)
230230
231231 let auctionContract = getContractAddressOrFail(controlCfg, IdxControlCfgAuctionDapp)
232232
233233 let liquidationContract = getContractAddressOrFail(controlCfg, IdxControlCfgLiquidationDapp)
234234
235235 let usdnStakingContract = getContractAddressOrFail(controlCfg, IdxControlCfgRpdDapp)
236236
237237 let pegProviderContract = getContractAddressOrFail(controlCfg, IdxControlCfgPegProviderDapp)
238238
239239 let neutrinoAssetIdStr = getStringOrFail(neutrinoContract, keyNeutrinoAssetId())
240240
241241 let neutrinoAssetId = fromBase58String(neutrinoAssetIdStr)
242242
243243 let nsbtAssetId = fromBase58String(getStringValue(neutrinoContract, keyNsbtAssetId()))
244244
245245 let surfAssetId = fromBase58String(getStringValue(auctionContract, keySurfAssetId()))
246246
247247 func keyBalanceLocked () = "balance_lock_"
248248
249249
250250 func totalLockedKEY (swapType,assetId) = makeString(["%s%s%s", "balanceLock", swapType, assetId], SEP)
251251
252252
253253 func keyTokenLockedBalance (assetId) = totalLockedKEY("outNeutrino", assetId)
254254
255255
256256 func keyNeutrinoLockedBalance () = totalLockedKEY("outMulti", neutrinoAssetIdStr)
257257
258258
259259 func keyOldNeutrinoLockedBalance () = (keyBalanceLocked() + "neutrino")
260260
261261
262262 func keyOldWavesLockedBalance () = (keyBalanceLocked() + "waves")
263263
264264
265265 func calcUsdnOutOfMarketSupply () = {
266266 let outOfMarketAddressList = valueOrElse(getString(controlContract, "%s%s__control__outOfMarketAddressList"), "")
267267 if ((outOfMarketAddressList == ""))
268268 then 0
269269 else {
270270 func balancesSUM (outOfMarketSupply,nextAddress) = ((outOfMarketSupply + valueOrElse(getInteger(usdnStakingContract, rpdUserBalanceKey(nextAddress, neutrinoAssetIdStr)), 0)) + assetBalance(addressFromStringValue(nextAddress), neutrinoAssetId))
271271
272272 let $l = split(outOfMarketAddressList, SEP)
273273 let $s = size($l)
274274 let $acc0 = 0
275275 func $f0_1 ($a,$i) = if (($i >= $s))
276276 then $a
277277 else balancesSUM($a, $l[$i])
278278
279279 func $f0_2 ($a,$i) = if (($i >= $s))
280280 then $a
281281 else throw("List size exceeds 7")
282282
283283 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7)
284284 }
285285 }
286286
287287
288288 func readCurrentAvgPeg () = asPriceSTRUCT(invoke(pegProviderContract, "price", [4], nil))._2
289289
290290
291291 func adjustPrice (priceX6,usdnUsdtPegX6) = {
292292 let minCoeffX6 = valueOrElse(getInteger(this, keyPriceAdjMinCoeff()), 600000)
293293 let arbRegulatorX6 = valueOrElse(getInteger(this, keyPriceAdjArbRegulator()), 150000)
294294 let maxPriceCoeffX6 = max([minCoeffX6, fraction((MULT6 + arbRegulatorX6), usdnUsdtPegX6, MULT6)])
295295 let priceCoeffX6 = min([maxPriceCoeffX6, MULT6])
296296 fraction(priceX6, MULT6, priceCoeffX6)
297297 }
298298
299299
300300 let currentPrice = getIntegerValue(controlContract, keyPrice())
301301
302302 let usdnUsdtPegX6 = readCurrentAvgPeg()
303303
304304 let currentPriceAdj = adjustPrice(currentPrice, usdnUsdtPegX6)
305305
306306 let neutrinoLockedBalance = (valueOrElse(getInteger(neutrinoContract, keyOldNeutrinoLockedBalance()), 0) + valueOrElse(getInteger(neutrinoContract, keyNeutrinoLockedBalance()), 0))
307307
308308 let wavesLockedBalance = (valueOrElse(getInteger(neutrinoContract, keyOldWavesLockedBalance()), 0) + valueOrElse(getInteger(neutrinoContract, keyTokenLockedBalance("WAVES")), 0))
309309
310310 func reserve (assetId) = {
311311 let tempAssetId = fromBase58String(assetId)
312312 if ((tempAssetId == WAVESID))
313313 then (wavesBalance(neutrinoContract).regular - wavesLockedBalance)
314314 else (assetBalance(neutrinoContract, tempAssetId) - valueOrElse(getInteger(neutrinoContract, keyTokenLockedBalance(assetId)), 0))
315315 }
316316
317317
318318 func reservesInUsd (assetId) = {
319319 let vol = reserve(assetId)
320320 let pr = value(getInteger(controlContract, keyPriceByAsset(assetId)))
321321 fraction(vol, pr, MULT8)
322322 }
323323
324324
325325 let neutrinoOutOfMarketSupply = calcUsdnOutOfMarketSupply()
326326
327327 let neutrinoTotalSupply = (((neutrinoLockedBalance + value(assetInfo(neutrinoAssetId)).quantity) - assetBalance(neutrinoContract, neutrinoAssetId)) - assetBalance(liquidationContract, neutrinoAssetId))
328328
329329 let neutrinoSupply = (neutrinoTotalSupply - neutrinoOutOfMarketSupply)
330330
331331 func getBR () = {
332332 let basket = split(value(getString(neutrinoContract, basketAssetsKey())), SEP)
333333 func addBr (acc,assetId) = (acc + reservesInUsd(assetId))
334334
335335 let sumReserves = {
336336 let $l = basket
337337 let $s = size($l)
338338 let $acc0 = 0
339339 func $f0_1 ($a,$i) = if (($i >= $s))
340340 then $a
341341 else addBr($a, $l[$i])
342342
343343 func $f0_2 ($a,$i) = if (($i >= $s))
344344 then $a
345345 else throw("List size exceeds 10")
346346
347347 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10)
348348 }
349349 fraction(sumReserves, MULT6, neutrinoSupply)
350350 }
351351
352352
353353 let BR = if ((neutrinoSupply == 0))
354354 then 0
355355 else getBR()
356356
357357 func getBasketInfo () = {
358358 let basketAssets = value(getString(neutrinoContract, basketAssetsKey()))
359359 let basket = split(basketAssets, SEP)
360360 func weightCalc (acc,assetId) = {
361361 let vol = reserve(assetId)
362362 let pr = value(getInteger(controlContract, keyPriceByAsset(assetId)))
363363 let resU = fraction(vol, pr, MULT8)
364364 $Tuple2((acc._1 :+ $Tuple4(assetId, vol, resU, pr)), (acc._2 + resU))
365365 }
366366
367367 let t = {
368368 let $l = basket
369369 let $s = size($l)
370370 let $acc0 = $Tuple2(nil, 0)
371371 func $f0_1 ($a,$i) = if (($i >= $s))
372372 then $a
373373 else weightCalc($a, $l[$i])
374374
375375 func $f0_2 ($a,$i) = if (($i >= $s))
376376 then $a
377377 else throw("List size exceeds 10")
378378
379379 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10)
380380 }
381381 func combine (acc,item) = {
382382 let share = if ((t._2 == 0))
383383 then MULT6
384384 else fraction(item._3, MULT6, t._2)
385385 (acc :+ makeString([item._1, toString(share), toString(item._4), toString(item._2)], ":"))
386386 }
387387
388388 let weightedPrice = if ((neutrinoSupply == 0))
389389 then 0
390390 else fraction(t._2, MULT6, neutrinoSupply)
391391 $Tuple3(makeString({
392392 let $l = t._1
393393 let $s = size($l)
394394 let $acc0 = nil
395395 func $f1_1 ($a,$i) = if (($i >= $s))
396396 then $a
397397 else combine($a, $l[$i])
398398
399399 func $f1_2 ($a,$i) = if (($i >= $s))
400400 then $a
401401 else throw("List size exceeds 10")
402402
403403 $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10)
404404 }, "_"), weightedPrice, t._2)
405405 }
406406
407407
408408 @Callable(i)
409409 func constructorV1 (neutrinoAddress,nsbtLockAddress,swapAmountAParam) = if ((i.caller != this))
410410 then throw("not authorized")
411411 else [StringEntry(keyNeutrinoAddress(), neutrinoAddress), IntegerEntry(keySwapAmountAParam(), swapAmountAParam)]
412412
413413
414414
415415 @Callable(i)
416416 func calcSwapLimitREADONLY (gNsbtAmount) = {
417417 let limitBasketTokenInUsd = if ((0 >= gNsbtAmount))
418418 then 0
419419 else {
420420 let aParamX8 = toBigInt(getIntOrFail(this, keySwapAmountAParam()))
421421 let bParamX16 = toBigInt(getIntOrFail(this, keySwapAmountBParam()))
422422 let powPartX8 = pow(toBigInt(gNsbtAmount), 6, bParamX16, 16, 8, CEILING)
423423 toInt(fraction(aParamX8, powPartX8, MULTX10))
424424 }
425425 let limitUsdn = if ((0 >= gNsbtAmount))
426426 then 0
427427 else {
428428 let aParamX8 = toBigInt(getIntOrFail(this, keyUsdnSwapAmountAParam()))
429429 let bParamX16 = toBigInt(getIntOrFail(this, keyUsdnSwapAmountBParam()))
430430 let brX6 = toBigInt(BR)
431431 let powPartX8 = pow(toBigInt(gNsbtAmount), 6, bParamX16, 16, 8, CEILING)
432432 toInt(fraction(fraction(aParamX8, powPartX8, MULTX10), brX6, MULTX6))
433433 }
434434 $Tuple2(nil, [limitBasketTokenInUsd, 0, limitUsdn])
435435 }
436436
437437
438438
439439 @Callable(i)
440440 func calculateKREADONLY (wRaw,uRaw,price,mRaw,sRaw) = {
441441 let EXP = toBigInt(2718281)
442442 let EXPSCALE = 6
443443 let a = valueOrElse(getInteger(auctionContract, "nsbtCurveParam_a"), 3)
444444 let paulB = valueOrElse(getInteger(auctionContract, "nsbtCurveParam_b"), ((3 * MULT6) / 10))
445445 let wReservesInUsdn = convertWavesToNeutrino(wRaw, price)
446446 let multBR = fraction(wReservesInUsdn, MULT8, uRaw)
447447 if ((multBR > 10678564816))
448448 then throw((("BR=" + toString(multBR)) + " > 10678.564816% will overflow exponent"))
449449 else {
450450 let multPower = (a * (multBR - MULT8))
451451 let multExpInPower = pow(EXP, EXPSCALE, toBigInt(multPower), MULTSCALE, SCALE16, DOWN)
452452 let multK = fraction(toBigInt(paulB), multExpInPower, MULTX6)
453453 $Tuple2(nil, [a, paulB, wReservesInUsdn, multBR, multPower, toString(multExpInPower), toString(multK)])
454454 }
455455 }
456456
457457
458458
459459 @Callable(i)
460460 func curveFunctionREADONLY (wavesPayRaw) = {
461461 let neutrinoMetrics = asAnyList(invoke(this, "calcNeutinoMetricsREADONLY", nil, nil))
462462 let price = asInt(neutrinoMetrics[nMetricIdxPrice])
463463 let wRaw = asInt(neutrinoMetrics[nMetricIdxReserve])
464464 let uRaw = asInt(neutrinoMetrics[nMetricIdxUsdnSupply])
465465 let sRaw = asInt(neutrinoMetrics[nMetricIdxNsbtSupply])
466466 let mRaw = asInt(neutrinoMetrics[nMetricIdxMaxNsbtSupply])
467467 let kCalcArray = asAnyList(invoke(this, "calculateKREADONLY", [wRaw, uRaw, price, mRaw, sRaw], nil))
468468 let multK = parseBigIntValue(asString(kCalcArray[kResultIdxMultKStr]))
469469 let usdnPay = convertWavesToNeutrino(wavesPayRaw, price)
470470 let bigMaxNsbtSupply = toBigInt(mRaw)
471471 let bigNsbtSupply = toBigInt((mRaw - sRaw))
472472 let step1 = fraction(toBigInt(usdnPay), MULTX16, multK)
473473 let step2 = fraction(step1, bigNsbtSupply, bigMaxNsbtSupply)
474474 let step3 = toInt(fraction(bigNsbtSupply, bigMaxNsbtSupply, (step2 + bigMaxNsbtSupply)))
475475 let nsbtAmountRaw = ((mRaw - sRaw) - step3)
476476 $Tuple2(nil, [nsbtAmountRaw, usdnPay, wRaw, uRaw, mRaw, sRaw, asInt(kCalcArray[kResultIdxA]), asInt(kCalcArray[kResultIdxPaulB]), asInt(kCalcArray[kResultIdxWReservesInUsdn]), price, asInt(kCalcArray[kResultIdxMultBR]), asInt(kCalcArray[kResultIdxMultPower]), kCalcArray[kResultIdxMultExpInPowerStr], kCalcArray[kResultIdxMultKStr], toString(step1), toString(step2), step3])
477477 }
478478
479479
480480
481481 @Callable(i)
482482 func surfFunctionREADONLY (amount,assetId) = if ((0 > amount))
483483 then throw("amount should be positive")
484484 else {
485485 let price = currentPrice
486486 let priceBig = toBigInt(price)
487487 let reserveW = reserve("WAVES")
488488 let reserveBig = toBigInt(reserveW)
489489 let supply = neutrinoSupply
490490 let supplyBig = toBigInt(supply)
491491 if ((BR >= MULT6))
492492 then throw((("BR = " + toString(BR)) + ", cannot buy SURF"))
493493 else {
494494 let maxWavesPay = (fraction(supply, MULT8, price) - reserveW)
495495 let maxUsdnPay = (supply - fraction(reserveW, price, MULT8))
496496 let useAmount = if ((assetId == neutrinoAssetId))
497497 then if ((amount > maxUsdnPay))
498498 then maxUsdnPay
499499 else amount
500500 else if ((assetId == WAVESID))
501501 then if ((amount > maxWavesPay))
502502 then maxWavesPay
503503 else amount
504504 else 0
505505 let amountBig = toBigInt(useAmount)
506506 if ((assetId == neutrinoAssetId))
507507 then $Tuple2(nil, [toInt(fraction(fraction(amountBig, MULTX8, priceBig), (supplyBig - (amountBig / TWOX)), reserveBig)), 0, useAmount, reserveW, supply, BR, reserveW, (supply - useAmount), fraction(convertWavesToNeutrino(reserveW, price), MULT6, (supply - useAmount)), (amount - useAmount), price])
508508 else if ((assetId == WAVESID))
509509 then $Tuple2(nil, [toInt(fraction(supplyBig, log(fraction((reserveBig + amountBig), MULTX16, reserveBig), 16, EULERX16, 16, 16, HALFUP), MULTX16)), useAmount, 0, reserveW, supply, BR, (reserveW + useAmount), supply, fraction(convertWavesToNeutrino((reserveW + useAmount), price), MULT6, supply), (amount - useAmount), price])
510510 else throw("Unsupported asset")
511511 }
512512 }
513513
514514
515515
516516 @Callable(i)
517517 func calcContractNsbtPriceSYSREADONLY (nsbtSupplyDELTA) = {
518518 let neutrinoMetrics = asAnyList(invoke(this, "calcNeutinoMetricsREADONLY", nil, nil))
519519 let price = asInt(neutrinoMetrics[nMetricIdxPrice])
520520 let wRaw = asInt(neutrinoMetrics[nMetricIdxReserve])
521521 let uRaw = (asInt(neutrinoMetrics[nMetricIdxUsdnSupply]) - nsbtSupplyDELTA)
522522 let sRaw = (asInt(neutrinoMetrics[nMetricIdxNsbtSupply]) + nsbtSupplyDELTA)
523523 let mRaw = asInt(neutrinoMetrics[nMetricIdxMaxNsbtSupply])
524524 let kCalcArray = asAnyList(invoke(this, "calculateKREADONLY", [wRaw, uRaw, price, mRaw, sRaw], nil))
525525 let multKX16 = parseBigIntValue(asString(kCalcArray[kResultIdxMultKStr]))
526526 let multStep1X16 = fraction(toBigInt(mRaw), MULTX16, toBigInt((mRaw - sRaw)))
527527 let multStep2X16 = fraction(multStep1X16, multStep1X16, MULTX16)
528528 let multNsbt2usdnPriceX16 = fraction(multKX16, multStep2X16, MULTX16)
529529 let nsbt2usdnPrice = toInt(fraction(multNsbt2usdnPriceX16, toBigInt(MULT6), MULTX16))
530530 let nsbt2wavesPrice = fraction(nsbt2usdnPrice, MULT6, price)
531531 $Tuple2(nil, [nsbt2usdnPrice, nsbt2wavesPrice])
532532 }
533533
534534
535535
536536 @Callable(i)
537537 func calcNeutinoMetricsREADONLY () = {
538538 let nsbtSupplyMAX = value(assetInfo(nsbtAssetId)).quantity
539539 let nsbtSupply = (nsbtSupplyMAX - assetBalance(auctionContract, nsbtAssetId))
540540 let surfSupply = value(assetInfo(surfAssetId)).quantity
541541 let basketInfo = getBasketInfo()
542542 $Tuple2(nil, [currentPrice, neutrinoLockedBalance, wavesLockedBalance, reserve("WAVES"), reservesInUsd("WAVES"), neutrinoSupply, neutrinoOutOfMarketSupply, neutrinoTotalSupply, BR, nsbtSupply, nsbtSupplyMAX, surfSupply, usdnUsdtPegX6, currentPriceAdj, basketInfo._1, basketInfo._2, basketInfo._3])
543543 }
544544
545545
546546
547547 @Callable(i)
548548 func getUnstakeComissionAmountREADONLY (amount,startHeight,halfLife) = $Tuple2(nil, fraction(amount, pow(2, 0, fraction(-((height - startHeight)), MULT8, halfLife), 8, 8, HALFUP), MULT8))
549549
550550
551551
552552 @Callable(i)
553553 func mergeStakesREADONLY (amount1,height1,amount2,height2,halfLife) = {
554554 let w = fraction(amount2, pow(2, 0, fraction((height2 - height1), MULT8, halfLife), 8, 8, HALFUP), MULT8)
555555 let v = fraction((amount1 + amount2), MULT8, (amount1 + w))
556556 $Tuple2(nil, (height1 + ((HALF8 - (halfLife * log(v, 8, 2, 0, 8, HALFUP))) / MULT8)))
557557 }
558558
559559
560560 @Verifier(tx)
561561 func verify () = {
562562 let pubKeyAdminsListStr = makeString(["ExtEEK19nmKj9mCpnWyvEEJFYATLMcVEMvohhUHkyHNm", "Ev5py5FfBQX9cZpYKnfQrTB49Byf8QmpZWeDVRim4yV7", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR", "DUuuLjXu98nBwZc7fqwCTjtA3nnRwgTbkMSr5SU2NmDR"], SEP)
563563 let pubKeyAdminsList = split(valueOrElse(getString(controlContract, "%s__multisig"), pubKeyAdminsListStr), SEP)
564564 let count = ((((if (sigVerify(tx.bodyBytes, tx.proofs[0], fromBase58String(pubKeyAdminsList[0])))
565565 then 1
566566 else 0) + (if (sigVerify(tx.bodyBytes, tx.proofs[1], fromBase58String(pubKeyAdminsList[1])))
567567 then 1
568568 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[2], fromBase58String(pubKeyAdminsList[2])))
569569 then 1
570570 else 0)) + (if (sigVerify(tx.bodyBytes, tx.proofs[3], fromBase58String(pubKeyAdminsList[3])))
571571 then 2
572572 else 0))
573573 (count >= 3)
574574 }
575575

github/deemru/w8io/3ef1775 
109.24 ms