2022.11.04 12:57 [3367595] smart account 3PKpsc1TNquw4HAF62pWK8ka1DBz9vyEBkt > SELF 0.00000000 Waves

{ "type": 13, "id": "Csjjuen4yVAGgeGHXHaBKWrVd1skPCLggSBQKokYJXCg", "fee": 3600000, "feeAssetId": null, "timestamp": 1667555628992, "version": 1, "sender": "3PKpsc1TNquw4HAF62pWK8ka1DBz9vyEBkt", "senderPublicKey": "8ZLAVkwG8sZ97hmBdGRkQpYTp5En9xgEYX4PeupqxpFw", "proofs": [ "4ocN6s1KJga18zyjFZWqF9M33zR2aTojh42KRWiXTBJNUAcKeZmMdf79w2qUgnn8jNnSryj82zZMP1RmSop8vDUV" ], "script": "base64:AAIFAAAAAAAAACgIAhIJCgcBAQEBAQgBEgASBAoCCAgSBAoCCAgSBAoCCAgSAwoBCBIAAAAAQAEAAAAPZ2V0U3RyaW5nT3JGYWlsAAAAAQAAAANrZXkJAQAAABN2YWx1ZU9yRXJyb3JNZXNzYWdlAAAAAgkABCIAAAABBQAAAANrZXkJAAEsAAAAAgkAASwAAAACAgAAAA9tYW5kYXRvcnkgdGhpcy4FAAAAA2tleQIAAAAPIGlzIG5vdCBkZWZpbmVkAAAAAANTRVACAAAAAl9fAAAAAAhCVUZTQ0FMRQkAATYAAAABAA3gtrOnZAAAAAAAAAZzY2FsZTgAAAAAAAX14QABAAAACHRocm93RXJyAAAAAQAAAANtc2cJAAACAAAAAQkABLkAAAACCQAETAAAAAICAAAACWlkby5yaWRlOgkABEwAAAACBQAAAANtc2cFAAAAA25pbAIAAAABIAEAAAAdY29udmVydFByaWNlQXNzZXRJbnRvSWRvQXNzZXQAAAAFAAAAEHByaWNlQXNzZXRBbW91bnQAAAAOcHJpY2VBc3NldE1VTFQAAAAFcHJpY2UAAAAJcHJpY2VNVUxUAAAADGlkb0Fzc2V0TVVMVAQAAAAPYlByaWNlQXNzZXRNVUxUCQABNgAAAAEFAAAADnByaWNlQXNzZXRNVUxUBAAAAA1iSWRvQXNzZXRNVUxUCQABNgAAAAEFAAAADGlkb0Fzc2V0TVVMVAQAAAAOYlByaWNlQXNzZXRCVUYJAAE8AAAAAwkAATYAAAABBQAAABBwcmljZUFzc2V0QW1vdW50BQAAAAhCVUZTQ0FMRQUAAAAPYlByaWNlQXNzZXRNVUxUBAAAAA9iQW1vdW50QXNzZXRCVUYJAAE8AAAAAwUAAAAOYlByaWNlQXNzZXRCVUYJAAE2AAAAAQUAAAAJcHJpY2VNVUxUCQABNgAAAAEFAAAABXByaWNlCQABoAAAAAEJAAE8AAAAAwUAAAAPYkFtb3VudEFzc2V0QlVGCQABNgAAAAEFAAAADGlkb0Fzc2V0TVVMVAUAAAAIQlVGU0NBTEUAAAAADklkeENmZ0lkb1N0YXJ0AAAAAAAAAAABAAAAABFJZHhDZmdJZG9EdXJhdGlvbgAAAAAAAAAAAgAAAAAQSWR4Q2ZnQ2xhaW1TdGFydAAAAAAAAAAAAwAAAAATSWR4Q2ZnQ2xhaW1EdXJhdGlvbgAAAAAAAAAABAAAAAALSWR4Q2ZnUHJpY2UAAAAAAAAAAAUAAAAAD0lkeENmZ1ByaWNlTXVsdAAAAAAAAAAABgAAAAAQSWR4Q2ZnSWRvQXNzZXRJZAAAAAAAAAAABwAAAAASSWR4Q2ZnSWRvQXNzZXRNdWx0AAAAAAAAAAAIAAAAABJJZHhDZmdQcmljZUFzc2V0SWQAAAAAAAAAAAkAAAAAFElkeENmZ1ByaWNlQXNzZXRNdWx0AAAAAAAAAAAKAAAAABVJZHhDZmdNaW5JbnZlc3RBbW91bnQAAAAAAAAAAAsBAAAADWZyb21hdENvbmZpZ1MAAAAMAAAACGlkb1N0YXJ0AAAAC2lkb0R1cmF0aW9uAAAACmNsYWltU3RhcnQAAAANY2xhaW1EdXJhdGlvbgAAAAVwcmljZQAAAAlwcmljZU11bHQAAAAMaWRvQXNzZXRJZDU4AAAADGlkb0Fzc2V0TXVsdAAAAA5wcmljZUFzc2V0SWQ1OAAAAA5wcmljZUFzc2V0TXVsdAAAAA9taW5JbnZlc3RBbW91bnQAAAATdG90YWxJZG9Bc3NldFRvU2VsbAkABLkAAAACCQAETAAAAAICAAAAGCVkJWQlZCVkJWQlZCVzJWQlcyVkJWQlZAkABEwAAAACBQAAAAhpZG9TdGFydAkABEwAAAACBQAAAAtpZG9EdXJhdGlvbgkABEwAAAACBQAAAApjbGFpbVN0YXJ0CQAETAAAAAIFAAAADWNsYWltRHVyYXRpb24JAARMAAAAAgUAAAAFcHJpY2UJAARMAAAAAgUAAAAJcHJpY2VNdWx0CQAETAAAAAIFAAAADGlkb0Fzc2V0SWQ1OAkABEwAAAACBQAAAAxpZG9Bc3NldE11bHQJAARMAAAAAgUAAAAOcHJpY2VBc3NldElkNTgJAARMAAAAAgUAAAAOcHJpY2VBc3NldE11bHQJAARMAAAAAgUAAAAPbWluSW52ZXN0QW1vdW50CQAETAAAAAIFAAAAE3RvdGFsSWRvQXNzZXRUb1NlbGwFAAAAA25pbAUAAAADU0VQAQAAAAxmcm9tYXRDb25maWcAAAAMAAAACGlkb1N0YXJ0AAAAC2lkb0R1cmF0aW9uAAAACmNsYWltU3RhcnQAAAANY2xhaW1EdXJhdGlvbgAAAAVwcmljZQAAAAlwcmljZU11bHQAAAAMaWRvQXNzZXRJZDU4AAAADGlkb0Fzc2V0TXVsdAAAAA5wcmljZUFzc2V0SWQ1OAAAAA5wcmljZUFzc2V0TXVsdAAAAA9taW5JbnZlc3RBbW91bnQAAAATdG90YWxJZG9Bc3NldFRvU2VsbAkBAAAADWZyb21hdENvbmZpZ1MAAAAMCQABpAAAAAEFAAAACGlkb1N0YXJ0CQABpAAAAAEFAAAAC2lkb0R1cmF0aW9uCQABpAAAAAEFAAAACmNsYWltU3RhcnQJAAGkAAAAAQUAAAANY2xhaW1EdXJhdGlvbgkAAaQAAAABBQAAAAVwcmljZQkAAaQAAAABBQAAAAlwcmljZU11bHQFAAAADGlkb0Fzc2V0SWQ1OAkAAaQAAAABBQAAAAxpZG9Bc3NldE11bHQFAAAADnByaWNlQXNzZXRJZDU4CQABpAAAAAEFAAAADnByaWNlQXNzZXRNdWx0CQABpAAAAAEFAAAAD21pbkludmVzdEFtb3VudAkAAaQAAAABBQAAABN0b3RhbElkb0Fzc2V0VG9TZWxsAAAAABFJZHhJbnZUb3RhbEFtb3VudAAAAAAAAAAAAQAAAAAVSWR4SW52UmVtYWluaW5nQW1vdW50AAAAAAAAAAACAAAAAB1JZHhJbnZDbGFpbWVkUHJpY2VBc3NldEFtb3VudAAAAAAAAAAAAwAAAAAbSWR4SW52Q2xhaW1lZElkb0Fzc2V0QW1vdW50AAAAAAAAAAAEAAAAABdJZHhJbnZMYXN0Q2xhaW1lZEhlaWdodAAAAAAAAAAABQEAAAAPZm9ybWF0SW52ZXN0b3JTAAAABQAAAAt0b3RhbEFtb3VudAAAAA9yZW1haW5pbmdBbW91bnQAAAAXY2xhaW1lZFByaWNlQXNzZXRBbW91bnQAAAAVY2xhaW1lZElkb0Fzc2V0QW1vdW50AAAAEWxhc3RDbGFpbWVkSGVpZ2h0CQAEuQAAAAIJAARMAAAAAgIAAAAKJWQlZCVkJWQlZAkABEwAAAACBQAAAAt0b3RhbEFtb3VudAkABEwAAAACBQAAAA9yZW1haW5pbmdBbW91bnQJAARMAAAAAgUAAAAXY2xhaW1lZFByaWNlQXNzZXRBbW91bnQJAARMAAAAAgUAAAAVY2xhaW1lZElkb0Fzc2V0QW1vdW50CQAETAAAAAIFAAAAEWxhc3RDbGFpbWVkSGVpZ2h0BQAAAANuaWwFAAAAA1NFUAEAAAAOZm9ybWF0SW52ZXN0b3IAAAAFAAAAC3RvdGFsQW1vdW50AAAAD3JlbWFpbmluZ0Ftb3VudAAAABdjbGFpbWVkUHJpY2VBc3NldEFtb3VudAAAABVjbGFpbWVkSWRvQXNzZXRBbW91bnQAAAARbGFzdENsYWltZWRIZWlnaHQJAQAAAA9mb3JtYXRJbnZlc3RvclMAAAAFCQABpAAAAAEFAAAAC3RvdGFsQW1vdW50CQABpAAAAAEFAAAAD3JlbWFpbmluZ0Ftb3VudAkAAaQAAAABBQAAABdjbGFpbWVkUHJpY2VBc3NldEFtb3VudAkAAaQAAAABBQAAABVjbGFpbWVkSWRvQXNzZXRBbW91bnQJAAGkAAAAAQUAAAARbGFzdENsYWltZWRIZWlnaHQBAAAAE2Zvcm1hdEhpc3RvcnlSZWNvcmQAAAACAAAAEHByaWNlQXNzZXRBbW91bnQAAAAOaWRvQXNzZXRBbW91bnQJAAS5AAAAAgkABEwAAAACAgAAAAglZCVkJWQlZAkABEwAAAACCQABpAAAAAEFAAAABmhlaWdodAkABEwAAAACCQABpAAAAAEIBQAAAAlsYXN0QmxvY2sAAAAJdGltZXN0YW1wCQAETAAAAAIJAAGkAAAAAQUAAAAQcHJpY2VBc3NldEFtb3VudAkABEwAAAACCQABpAAAAAEFAAAADmlkb0Fzc2V0QW1vdW50BQAAAANuaWwFAAAAA1NFUAEAAAAJa2V5Q29uZmlnAAAAAAIAAAAKJXNfX2NvbmZpZwEAAAALa2V5SW52ZXN0b3IAAAABAAAAC3VzZXJBZGRyZXNzCQABLAAAAAICAAAABCVzX18FAAAAC3VzZXJBZGRyZXNzAQAAAAlrZXlUb3RhbHMAAAAAAgAAAAolc19fdG90YWxzAQAAABlrZXlPcGVyYXRpb25IaXN0b3J5UmVjb3JkAAAAAwAAAAR0eXBlAAAAC3VzZXJBZGRyZXNzAAAABnR4SWQ1OAkABLkAAAACCQAETAAAAAICAAAAESVzJXMlcyVzX19oaXN0b3J5CQAETAAAAAIFAAAABHR5cGUJAARMAAAAAgUAAAALdXNlckFkZHJlc3MJAARMAAAAAgUA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ZXkBAAAAGmtleVBlbmRpbmdNYW5hZ2VyUHVibGljS2V5AAAAAAIAAAAbJXNfX3BlbmRpbmdNYW5hZ2VyUHVibGljS2V5AQAAAA9yZWFkQ29uZmlnQXJyYXkAAAAACQAEtQAAAAIJAQAAAA9nZXRTdHJpbmdPckZhaWwAAAABCQEAAAAJa2V5Q29uZmlnAAAAAAUAAAADU0VQAQAAACNyZWFkVG90YWxzQXJyYXlPckRlZmF1bHRCeUN1c3RvbUtleQAAAAEAAAAJY3VzdG9tS2V5CQAEtQAAAAIJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQiAAAAAQUAAAAJY3VzdG9tS2V5CQEAAAAPZm9ybWF0SW52ZXN0b3JTAAAABQIAAAABMAIAAAABMAIAAAABMAIAAAABMAIAAAABMAUAAAADU0VQAQAAABhyZWFkVG90YWxzQXJyYXlPckRlZmF1bHQAAAAACQEAAAAjcmVhZFRvdGFsc0FycmF5T3JEZWZhdWx0QnlDdXN0b21LZXkAAAABCQEAAAAJa2V5VG90YWxzAAAAAAEAAAAacmVhZEludmVzdG9yQXJyYXlPckRlZmF1bHQAAAABAAAAC3VzZXJBZGRyZXNzCQEAAAAjcmVhZFRvdGFsc0FycmF5T3JEZWZhdWx0QnlDdXN0b21LZXkAAAABCQEAAAALa2V5SW52ZXN0b3IAAAABBQAAAAt1c2VyQWRkcmVzcwEAAAAXcmVhZEludmVzdG9yQXJyYXlPckZhaWwAAAABAAAAC3VzZXJBZGRyZXNzCQAEtQAAAAIJAQAAAA9nZXRTdHJpbmdPckZhaWwAAAABCQEAAAALa2V5SW52ZXN0b3IAAAABBQAAAAt1c2VyQWRkcmVzcwUAAAADU0VQAAAAABVJZHhEaWZmVG90YWxJbmNyZW1lbnQAAAAAAAAAAAAAAAAAJElkeERpZmZSZW1haW5pbmdQcmljZUFtb3VudEluY3JlbWVudAAAAAAAAAAAAQAAAAAiSWR4RGlmZkNsYWltZWRQcmljZUFtb3VudEluY3JlbWVudAAAAAAAAAAAAgAAAAAlSWR4RGlmZkNsYWltZWRJZG9Bc3NldEFtb3VudEluY3JlbWVudAAAAAAAAAAAAwEAAAALVG90YWxzRW50cnkAAAAFAAAAA2tleQAAAAlvcmlnQXJyYXkAAAANaW5jcmVtZW50RGlmZgAAABRuZXdMYXN0Q2xhaW1lZEhlaWdodAAAABFwcmljZUFzc2V0QmFsYW5jZQQAAAALdG90YWxBbW91bnQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlvcmlnQXJyYXkFAAAAEUlkeEludlRvdGFsQW1vdW50BAAAAA9yZW1haW5pbmdBbW91bnQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlvcmlnQXJyYXkFAAAAFUlkeEludlJlbWFpbmluZ0Ftb3VudAQAAAAXY2xhaW1lZFByaWNlQXNzZXRBbW91bnQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlvcmlnQXJyYXkFAAAAHUlkeEludkNsYWltZWRQcmljZUFzc2V0QW1vdW50BAAAABVjbGFpbWVkSWRvQXNzZXRBbW91bnQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlvcmlnQXJyYXkFAAAAG0lkeEludkNsYWltZWRJZG9Bc3NldEFtb3VudAQAAAARbGFzdENsYWltZWRIZWlnaHQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAlvcmlnQXJyYXkFAAAAF0lkeEludkxhc3RDbGFpbWVkSGVpZ2h0BAAAAA5uZXdUb3RhbEFtb3VudAkAAGQAAAACBQAAAAt0b3RhbEFtb3VudAkAAZEAAAACBQAAAA1pbmNyZW1lbnREaWZmBQAAABVJZHhEaWZmVG90YWxJbmNyZW1lbnQEAAAAEm5ld1JlbWFpbmluZ0Ftb3VudAkAAGQAAAACBQAAAA9yZW1haW5pbmdBbW91bnQJAAGRAAAAAgUAAAANaW5jcmVtZW50RGlmZgUAAAAkSWR4RGlmZlJlbWFpbmluZ1ByaWNlQW1vdW50SW5jcmVtZW50BAAAAAhjZmdBcnJheQkBAAAAD3JlYWRDb25maWdBcnJheQAAAAAEAAAADnByaWNlQXNzZXRJZDU4CQABkQAAAAIFAAAACGNmZ0FycmF5BQAAABJJZHhDZmdQcmljZUFzc2V0SWQEAAAAEnByaWNlQXNzZXREZWNpbWFscwgJAQAAAAV2YWx1ZQAAAAEJAAPsAAAAAQkAAlkAAAABBQAAAA5wcmljZUFzc2V0SWQ1OAAAAAhkZWNpbWFscwQAAAAjcHJpY2VBc3NldEJhbGFuY2VQcmljZUFzc2V0RGVjaW1hbHMJAABrAAAAAwUAAAARcHJpY2VBc3NldEJhbGFuY2UFAAAABnNjYWxlOAkAAGwAAAAGAAAAAAAAAAAKAAAAAAAAAAAABQAAABJwcmljZUFzc2V0RGVjaW1hbHMAAAAAAAAAAAAAAAAAAAAAAAAFAAAABERPV04EAAAAGm5ld0NsYWltZWRQcmljZUFzc2V0QW1vdW50CQAAZQAAAAIJAABkAAAAAgUAAAAXY2xhaW1lZFByaWNlQXNzZXRBbW91bnQJAAGRAAAAAgUAAAANaW5jcmVtZW50RGlmZgUAAAAiSWR4RGlmZkNsYWltZWRQcmljZUFtb3VudEluY3JlbWVudAUAAAARcHJpY2VBc3NldEJhbGFuY2UEAAAAGG5ld0NsYWltZWRJZG9Bc3NldEFtb3VudAkAAGQAAAACCQAAZAAAAAIFAAAAFWNsYWltZWRJZG9Bc3NldEFtb3VudAkAAZEAAAACBQAAAA1pbmNyZW1lbnREaWZmBQAAACVJZHhEaWZmQ2xhaW1lZElkb0Fzc2V0QW1vdW50SW5jcmVtZW50BQAAACNwcmljZUFzc2V0QmFsYW5jZVByaWNlQXNzZXREZWNpbWFscwMJAABmAAAAAgAAAAAAAAAAAAUAAAASbmV3UmVtYWluaW5nQW1vdW50CQAAAgAAAAECAAAADGludmFsaWQgbWF0aAkBAAAAC1N0cmluZ0VudHJ5AAAAAgUAAAADa2V5CQEAAAAOZm9ybWF0SW52ZXN0b3IAAAAFBQAAAA5uZXdUb3RhbEFtb3VudAUAAAASbmV3UmVtYWluaW5nQW1vdW50BQAAABpuZXdDbGFpbWVkUHJpY2VBc3NldEFtb3VudAUAAAAYbmV3Q2xhaW1lZElkb0Fzc2V0QW1vdW50BQAAABRuZXdMYXN0Q2xhaW1lZEhlaWdodAEAAAAbSW52ZXN0T3BlcmF0aW9uSGlzdG9yeUVudHJ5AAAABAAAAAt1c2VyQWRkcmVzcwAAABBwcmljZUFzc2V0QW1vdW50AAAADmlkb0Fzc2V0QW1vdW50AAAABHR4SWQJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABlrZXlPcGVyYXRpb25IaXN0b3J5UmVjb3JkAAAAAwIAAAAGaW52ZXN0BQAAAAt1c2VyQWRkcmVzcwkAAlgAAAABBQAAAAR0eElkCQEAAAATZm9ybWF0SGlzdG9yeVJlY29yZAAAAAIFAAAAEHByaWNlQXNzZXRBbW91bnQFAAAADmlkb0Fzc2V0QW1vdW50AQAAABpDbGFpbU9wZXJhdGlvbkhpc3RvcnlFbnRyeQAAAAQAAAALdXNlckFkZHJlc3MAAAAQcHJpY2VBc3NldEFtb3VudAAAAA5pZG9Bc3NldEFtb3VudAAAAAR0eElkCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAAZa2V5T3BlcmF0aW9uSGlzdG9yeVJlY29yZAAAAAMCAAAABWNsYWltBQAAAAt1c2VyQWRkcmVzcwkAAlgAAAABBQAAAAR0eElkCQEAAAATZm9ybWF0SGlzdG9yeVJlY29yZAAAAAIFAAAAEHByaWNlQXNzZXRBbW91bnQFAAAADmlkb0Fzc2V0QW1vdW50AQAAAA1pbnRlcm5hbENsYWltAAAAAwAAABBjbGFpbWVkQXNzZXRJZDU4AAAAC3VzZXJBZGRyZXNzAAAABHR4SWQEAAAACGNmZ0FycmF5CQEAAAAPcmVhZENvbmZpZ0FycmF5AAAAAAQAAAAKY2xhaW1TdGFydAkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAACGNmZ0FycmF5BQAAABBJZHhDZmdDbGFpbVN0YXJ0BAAAAA1jbGFpbUR1cmF0aW9uCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAE0lkeENmZ0NsYWltRHVyYXRpb24EAAAACGNsYWltRW5kCQAAZAAAAAIFAAAACmNsYWltU3RhcnQFAAAADWNsYWltRHVyYXRpb24EAAAABXByaWNlCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAC0lkeENmZ1ByaWNlBAAAAAlwcmljZU11bHQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAAPSWR4Q2ZnUHJpY2VNdWx0BAAAAAxpZG9Bc3NldElkNTgJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAEElkeENmZ0lkb0Fzc2V0SWQEAAAACmlkb0Fzc2V0SWQJAAJZAAAAAQUAAAAMaWRvQXNzZXRJZDU4BAAAAAxpZG9Bc3NldE11bHQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAASSWR4Q2ZnSWRvQXNzZXRNdWx0BAAAAA5wcmljZUFzc2V0SWQ1OAkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAASSWR4Q2ZnUHJpY2VBc3NldElkBAAAAAxwcmljZUFzc2V0SWQJAAJZAAAAAQUAAAAOcHJpY2VBc3NldElkNTgEAAAADnByaWNlQXNzZXRNdWx0CQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAFElkeENmZ1ByaWNlQXNzZXRNdWx0BAAAAA11c2VyQWRkcmVzczU4CQAEJQAAAAEFAAAAC3VzZXJBZGRyZXNzBAAAAA9vcmlnSW52ZXN0QXJyYXkJAQAAABdyZWFkSW52ZXN0b3JBcnJheU9yRmFpbAAAAAEFAAAADXVzZXJBZGRyZXNzNTgEAAAAEWludmVzdFRvdGFsQW1vdW50CQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAPb3JpZ0ludmVzdEFycmF5BQAAABFJZHhJbnZUb3RhbEFtb3VudAQAAAAaaW52ZXN0TGFzdENsYWltZWRIZWlnaHRUTVAJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAA9vcmlnSW52ZXN0QXJyYXkFAAAAF0lkeEludkxhc3RDbGFpbWVkSGVpZ2h0BAAAABdpbnZlc3RMYXN0Q2xhaW1lZEhlaWdodAMJAABnAAAAAgUAAAAKY2xhaW1TdGFydAUAAAAaaW52ZXN0TGFzdENsYWltZWRIZWlnaHRUTVAFAAAACmNsYWltU3RhcnQFAAAAGmludmVzdExhc3RDbGFpbWVkSGVpZ2h0VE1QBAAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAMJAABmAAAAAgUAAAAGaGVpZ2h0BQAAAAhjbGFpbUVuZAUAAAAIY2xhaW1FbmQDCQAAZgAAAAIFAAAACmNsYWltU3RhcnQFAAAABmhlaWdodAUAAAAKY2xhaW1TdGFydAUAAAAGaGVpZ2h0BAAAAA5jbGFpbWluZ0Jsb2NrcwkAAGUAAAACBQAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAUAAAAXaW52ZXN0TGFzdENsYWltZWRIZWlnaHQEAAAAGGNsYWltaW5nUHJpY2VBc3NldEFtb3VudAkAAGsAAAADBQAAABFpbnZlc3RUb3RhbEFtb3VudAUAAAAOY2xhaW1pbmdCbG9ja3MFAAAADWNsYWltRHVyYXRpb24EAAAAFmNsYWltaW5nSWRvQXNzZXRBbW91bnQJAQAAAB1jb252ZXJ0UHJpY2VBc3NldEludG9JZG9Bc3NldAAAAAUFAAAAGGNsYWltaW5nUHJpY2VBc3NldEFtb3VudAUAAAAOcHJpY2VBc3NldE11bHQFAAAABXByaWNlBQAAAAlwcmljZU11bHQFAAAADGlkb0Fzc2V0TXVsdAQAAAAGaXNVU0ROCQAAAAAAAAIFAAAAEGNsYWltZWRBc3NldElkNTgFAAAADnByaWNlQXNzZXRJZDU4BAAAABNpc1VTRE5DbGFpbURpc2FibGVkCQEAAAALdmFsdWVPckVsc2UAAAACCQAEIAAAAAEJAQAAABRrZXlVU0ROQ2xhaW1EaXNhYmxlZAAAAAAHBAAAAAZjaGVja3MJAARMAAAAAgMJAQAAAAEhAAAAAQMFAAAABmlzVVNETgUAAAATaXNVU0ROQ2xhaW1EaXNhYmxlZAcGCQAAAgAAAAECAAAAFlVTRE4gY2xhaW0gaXMgZGlzYWJsZWQFAAAAA25pbAMJAAAAAAAAAgUAAAAGY2hlY2tzBQAAAAZjaGVja3MDCQAAAAAAAAIFAAAAEGNsYWltZWRBc3NldElkNTgFAAAADnByaWNlQXNzZXRJZDU4CQAFGAAAAAYJAARMAAAAAgAAAAAAAAAAAAkABEwAAAACCQEAAAABLQAAAAEFAAAAGGNsYWltaW5nUHJpY2VBc3NldEFtb3VudAkABEwAAAACBQAAABhjbGFpbWluZ1ByaWNlQXNzZXRBbW91bnQJAARMAAAAAgAAAAAAAAAAAAUAAAADbmlsBQAAABhjbGFpbWluZ1ByaWNlQXNzZXRBbW91bnQFAAAADHByaWNlQXNzZXRJZAUAAAAPb3JpZ0ludmVzdEFycmF5BQAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAkABEwAAAACBQAAABhjbGFpbWluZ1ByaWNlQXNzZXRBbW91bnQJAARMAAAAAgUAAAAWY2xhaW1pbmdJZG9Bc3NldEFtb3VudAUAAAADbmlsAwkAAAAAAAACBQAAABBjbGFpbWVkQXNzZXRJZDU4BQAAAAxpZG9Bc3NldElkNTgJAAUYAAAABgkABEwAAAACAAAAAAAAAAAACQAETAAAAAIJAQAAAAEtAAAAAQUAAAAYY2xhaW1pbmdQcmljZUFzc2V0QW1vdW50CQAETAAAAAIAAAAAAAAAAAAJAARMAAAAAgUAAAAWY2xhaW1pbmdJZG9Bc3NldEFtb3VudAUAAAADbmlsBQAAABZjbGFpbWluZ0lkb0Fzc2V0QW1vdW50BQAAAAppZG9Bc3NldElkBQAAAA9vcmlnSW52ZXN0QXJyYXkFAAAAFG5ld0NsYWltUGVyaW9kSGVpZ2h0CQAETAAAAAIFAAAAGGNsYWltaW5nUHJpY2VBc3NldEFtb3VudAkABEwAAAACBQAAABZjbGFpbWluZ0lkb0Fzc2V0QW1vdW50BQAAAANuaWwJAAACAAAAAQkAASwAAAACAgAAABV1bnN1cHBvcnRlZCBhc3NldElkOiAFAAAAEGNsYWltZWRBc3NldElkNTgJAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQAAAA9pbnRlcm5hbENsYWltVjIAAAAEAAAADnByaWNlQXNzZXRJZDU4AAAADXVzZXJBZGRyZXNzNTgAAAAJb3V0QW1vdW50AAAAGXRvdGFsVXNlckF2YWlsYWJsZVRvQ2xhaW0EAAAAHnRvdGFsUGVyaW9kUHJpY2VBc3NldEFsbG93YW5jZQkBAAAABXZhbHVlAAAAAQkABB8AAAABCQEAAAAXa2V5VG90YWxQZXJpb2RBbGxvd2FuY2UAAAABBQAAAA5wcmljZUFzc2V0SWQ1OAQAAAAddXNlclBlcmlvZFByaWNlQXNzZXRBbGxvd2FuY2UJAQAAAAV2YWx1ZQAAAAEJAAQfAAAAAQkBAAAAFmtleVVzZXJQZXJpb2RBbGxvd2FuY2UAAAABBQAAAA5wcmljZUFzc2V0SWQ1OAQAAAAMcGVyaW9kTGVuZ3RoCQEAAAAFdmFsdWUAAAABCQAEHwAAAAEJAQAAAA9rZXlQZXJpb2RMZW5ndGgAAAAABAAAAA1jdXJyZW50UGVyaW9kCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHwAAAAEJAQAAABBrZXlDdXJyZW50UGVyaW9kAAAAAAAAAAAAAAAAAAQAAAAbemVyb1BlcmlvZEVuZEhlaWdoSXNEZWZpbmVkCQEAAAAJaXNEZWZpbmVkAAAAAQkABB8AAAABCQEAAAASa2V5UGVyaW9kRW5kSGVpZ2h0AAAAAQAAAAAAAAAAAAQAAAANJHQwMTIyNDMxNDExMwMJAABmAAAAAgUAAAANY3VycmVudFBlcmlvZAAAAAAAAAAAAAQAAAAVbGFzdFBlcmlvZFN0YXJ0SGVpZ2h0CQEAAAAFdmFsdWUAAAABCQAEHwAAAAEJAQAAABRrZXlQZXJpb2RTdGFydEhlaWdodAAAAAEFAAAADWN1cnJlbnRQZXJpb2QEAAAAE2xhc3RQZXJpb2RFbmRIZWlnaHQJAQAAAAV2YWx1ZQAAAAEJAAQfAAAAAQkBAAAAEmtleVBlcmlvZEVuZEhlaWdodAAAAAEFAAAADWN1cnJlbnRQZXJpb2QEAAAADSR0MDEyNTAzMTMyMDgDCQAAZgAAAAIFAAAABmhlaWdodAUAAAATbGFzdFBlcmlvZEVuZEhlaWdodAQAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QJAABkAAAAAgUAAAANY3VycmVudFBlcmlvZAAAAAAAAAAAAQQAAAALcGVyaW9kU3RhcnQDCQAAZgAAAAIFAAAABmhlaWdodAkAAGQAAAACBQAAABNsYXN0UGVyaW9kRW5kSGVpZ2h0BQAAAAxwZXJpb2RMZW5ndGgEAAAAF2Jsb2Nrc1RvTGFzdFBlcmlvZFN0YXJ0CQAAagAAAAIJAABlAAAAAgUAAAAGaGVpZ2h0BQAAABNsYXN0UGVyaW9kRW5kSGVpZ2h0BQAAAAxwZXJpb2RMZW5ndGgDCQAAAAAAAAIFAAAAF2Jsb2Nrc1RvTGFzdFBlcmlvZFN0YXJ0AAAAAAAAAAAACQAAZAAAAAIJAABlAAAAAgUAAAAGaGVpZ2h0BQAAAAxwZXJpb2RMZW5ndGgAAAAAAAAAAAEJAABlAAAAAgUAAAAGaGVpZ2h0BQAAABdibG9ja3NUb0xhc3RQZXJpb2RTdGFydAkAAGQAAAACBQAAABNsYXN0UGVyaW9kRW5kSGVpZ2h0AAAAAAAAAAABBAAAAAlwZXJpb2RFbmQJAABlAAAAAgkAAGQAAAACBQAAAAtwZXJpb2RTdGFydAUAAAAMcGVyaW9kTGVuZ3RoAAAAAAAAAAABCQAFFQAAAAMFAAAAFHVwZGF0ZWRDdXJyZW50UGVyaW9kBQAAAAtwZXJpb2RTdGFydAUAAAAJcGVyaW9kRW5kCQAFFQAAAAMFAAAADWN1cnJlbnRQZXJpb2QFAAAAFWxhc3RQZXJpb2RTdGFydEhlaWdodAUAAAATbGFzdFBlcmlvZEVuZEhlaWdodAQAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QIBQAAAA0kdDAxMjUwMzEzMjA4AAAAAl8xBAAAAAtwZXJpb2RTdGFydAgFAAAADSR0MDEyNTAzMTMyMDgAAAACXzIEAAAACXBlcmlvZEVuZAgFAAAADSR0MDEyNTAzMTMyMDgAAAACXzMJAAUVAAAAAwUAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QFAAAAC3BlcmlvZFN0YXJ0BQAAAAlwZXJpb2RFbmQDBQAAABt6ZXJvUGVyaW9kRW5kSGVpZ2hJc0RlZmluZWQEAAAAFXplcm9QZXJpb2RTdGFydEhlaWdodAkBAAAABXZhbHVlAAAAAQkABB8AAAABCQEAAAAUa2V5UGVyaW9kU3RhcnRIZWlnaHQAAAABAAAAAAAAAAAABAAAABN6ZXJvUGVyaW9kRW5kSGVpZ2h0CQEAAAAFdmFsdWUAAAABCQAEHwAAAAEJAQAAABJrZXlQZXJpb2RFbmRIZWlnaHQAAAABAAAAAAAAAAAABAAAAA0kdDAxMzQ2MzEzODQ1AwkAAGYAAAACBQAAAAZoZWlnaHQFAAAAE3plcm9QZXJpb2RFbmRIZWlnaHQEAAAAFHVwZGF0ZWRDdXJyZW50UGVyaW9kCQAAZAAAAAIFAAAADWN1cnJlbnRQZXJpb2QAAAAAAAAAAAEEAAAAC3BlcmlvZFN0YXJ0CQAAZAAAAAIFAAAAE3plcm9QZXJpb2RFbmRIZWlnaHQAAAAAAAAAAAEEAAAACXBlcmlvZEVuZAkAAGUAAAACCQAAZAAAAAIFAAAAC3BlcmlvZFN0YXJ0BQAAAAxwZXJpb2RMZW5ndGgAAAAAAAAAAAEJAAUVAAAAAwUAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QFAAAAC3BlcmlvZFN0YXJ0BQAAAAlwZXJpb2RFbmQJAAUVAAAAAwUAAAANY3VycmVudFBlcmlvZAUAAAAVemVyb1BlcmlvZFN0YXJ0SGVpZ2h0BQAAABN6ZXJvUGVyaW9kRW5kSGVpZ2h0BAAAABR1cGRhdGVkQ3VycmVudFBlcmlvZAgFAAAADSR0MDEzNDYzMTM4NDUAAAACXzEEAAAAC3BlcmlvZFN0YXJ0CAUAAAANJHQwMTM0NjMxMzg0NQAAAAJfMgQAAAAJcGVyaW9kRW5kCAUAAAANJHQwMTM0NjMxMzg0NQAAAAJfMwkABRUAAAADBQAAABR1cGRhdGVkQ3VycmVudFBlcmlvZAUAAAALcGVyaW9kU3RhcnQFAAAACXBlcmlvZEVuZAkABRUAAAADBQAAAA1jdXJyZW50UGVyaW9kCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHwAAAAEJAQAAABRrZXlQZXJpb2RTdGFydEhlaWdodAAAAAEFAAAADWN1cnJlbnRQZXJpb2QFAAAABmhlaWdodAkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABB8AAAABCQEAAAASa2V5UGVyaW9kRW5kSGVpZ2h0AAAAAQUAAAANY3VycmVudFBlcmlvZAkAAGUAAAACCQAAZAAAAAIFAAAABmhlaWdodAUAAAAMcGVyaW9kTGVuZ3RoAAAAAAAAAAABBAAAABR1cGRhdGVkQ3VycmVudFBlcmlvZAgFAAAADSR0MDEyMjQzMTQxMTMAAAACXzEEAAAAC3BlcmlvZFN0YXJ0CAUAAAANJHQwMTIyNDMxNDExMwAAAAJfMgQAAAAJcGVyaW9kRW5kCAUAAAANJHQwMTIyNDMxNDExMwAAAAJfMwQAAAAbcGVyaW9kVG90YWxBdmFpbGFibGVUb0NsYWltCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHwAAAAEJAQAAAB5rZXlQZXJpb2RUb3RhbEF2YWlsYWJsZVRvQ2xhaW0AAAACBQAAAA5wcmljZUFzc2V0SWQ1OAUAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QFAAAAHnRvdGFsUGVyaW9kUHJpY2VBc3NldEFsbG93YW5jZQQAAAAacGVyaW9kVXNlckF2YWlsYWJsZVRvQ2xhaW0JAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQfAAAAAQkBAAAAHWtleVBlcmlvZFVzZXJBdmFpbGFibGVUb0NsYWltAAAAAwUAAAAOcHJpY2VBc3NldElkNTgFAAAAFHVwZGF0ZWRDdXJyZW50UGVyaW9kBQAAAA11c2VyQWRkcmVzczU4BQAAAB11c2VyUGVyaW9kUHJpY2VBc3NldEFsbG93YW5jZQQAAAARcHJpY2VBc3NldEJhbGFuY2UJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQfAAAAAQkBAAAAFGtleVByaWNlQXNzZXRCYWxhbmNlAAAAAQUAAAANdXNlckFkZHJlc3M1OAAAAAAAAAAAAAQAAAAZcGVyaW9kTWluQXZhaWxhYmxlVG9DbGFpbQkAAZcAAAABCQAETAAAAAIJAABkAAAAAgUAAAAJb3V0QW1vdW50BQAAABFwcmljZUFzc2V0QmFsYW5jZQkABEwAAAACBQAAABtwZXJpb2RUb3RhbEF2YWlsYWJsZVRvQ2xhaW0JAARMAAAAAgUAAAAacGVyaW9kVXNlckF2YWlsYWJsZVRvQ2xhaW0FAAAAA25pbAQAAAAcdXNkdFByaWNlQXNzZXRBbGxvd2FibGVSYXRpbwkBAAAABXZhbHVlAAAAAQkABB8AAAABCQEAAAAfa2V5VXNkdFByaWNlQXNzZXRBbGxvd2FibGVSYXRpbwAAAAAEAAAAG3B1dE9uZVRrblYyUHJpY2VBc3NldEFtb3VudAUAAAAGc2NhbGU4BAAAAA0kdDAxNDg2ODE1MTIxCgAAAAABQAkAA/wAAAAECQEAAAARQGV4dHJOYXRpdmUoMTA2MikAAAABCQEAAAAFdmFsdWUAAAABCQAEIgAAAAEJAQAAABtrZXlVc2R0UHJpY2VBc3NldFN0YWJsZVBvb2wAAAAAAgAAACFwdXRPbmVUa25WMldpdGhvdXRUYWtlRmVlUkVBRE9OTFkJAARMAAAAAgUAAAAbcHV0T25lVGtuVjJQcmljZUFzc2V0QW1vdW50CQAETAAAAAIFAAAADnByaWNlQXNzZXRJZDU4BQAAAANuaWwFAAAAA25pbAMJAAABAAAAAgUAAAABQAIAAAAPKEludCwgSW50LCBJbnQpBQAAAAFACQAAAgAAAAECAAAAJENvdWxkbid0IGNhc3QgQW55IHRvIChJbnQsIEludCwgSW50KQMJAAAAAAAAAgUAAAANJHQwMTQ4NjgxNTEyMQUAAAANJHQwMTQ4NjgxNTEyMQQAAAAFYm9udXMIBQAAAA0kdDAxNDg2ODE1MTIxAAAAAl8zBAAAAAlmZWVBbW91bnQIBQAAAA0kdDAxNDg2ODE1MTIxAAAAAl8yBAAAAAhscEFtb3VudAgFAAAADSR0MDE0ODY4MTUxMjEAAAACXzEEAAAAC3VzZHRBc3NldElkCQEAAAAFdmFsdWUAAAABCQAEIgAAAAEJAQAAAA5rZXlVc2R0QXNzZXRJZAAAAAAEAAAADSR0MDE1MTgzMTU0MDEKAAAAAAFACQAD/AAAAAQJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEJAQAAAAV2YWx1ZQAAAAEJAAQiAAAAAQkBAAAAG2tleVVzZHRQcmljZUFzc2V0U3RhYmxlUG9vbAAAAAACAAAAE2dldE9uZVRrblYyUkVBRE9OTFkJAARMAAAAAgUAAAALdXNkdEFzc2V0SWQJAARMAAAAAgUAAAAIbHBBbW91bnQFAAAAA25pbAUAAAADbmlsAwkAAAEAAAACBQAAAAFAAgAAAAooSW50LCBJbnQpBQAAAAFACQAAAgAAAAECAAAAH0NvdWxkbid0IGNhc3QgQW55IHRvIChJbnQsIEludCkDCQAAAAAAAAIFAAAADSR0MDE1MTgzMTU0MDEFAAAADSR0MDE1MTgzMTU0MDEEAAAAFGdldE9uZVRrblYyRmVlQW1vdW50CAUAAAANJHQwMTUxODMxNTQwMQAAAAJfMgQAAAAKdXNkdEFtb3VudAgFAAAADSR0MDE1MTgzMTU0MDEAAAACXzEEAAAAGmN1cnJlbnRVc2R0UHJpY2VBc3NldFJhdGlvCQAAawAAAAMFAAAAG3B1dE9uZVRrblYyUHJpY2VBc3NldEFtb3VudAUAAAAGc2NhbGU4BQAAAAp1c2R0QW1vdW50BAAAABNlbmRQZXJpb2RCbG9ja3NMZWZ0CQAAZQAAAAIFAAAACXBlcmlvZEVuZAUAAAAGaGVpZ2h0CQAFHAAAAAoFAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0FAAAAG3BlcmlvZFRvdGFsQXZhaWxhYmxlVG9DbGFpbQUAAAAacGVyaW9kVXNlckF2YWlsYWJsZVRvQ2xhaW0FAAAAGXRvdGFsVXNlckF2YWlsYWJsZVRvQ2xhaW0FAAAAHHVzZHRQcmljZUFzc2V0QWxsb3dhYmxlUmF0aW8FAAAAGmN1cnJlbnRVc2R0UHJpY2VBc3NldFJhdGlvBQAAABNlbmRQZXJpb2RCbG9ja3NMZWZ0BQAAABR1cGRhdGVkQ3VycmVudFBlcmlvZAUAAAALcGVyaW9kU3RhcnQFAAAACXBlcmlvZEVuZAkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQAAABZtYW5hZ2VyUHVibGljS2V5T3JVbml0AAAAAAQAAAAHJG1hdGNoMAkABCIAAAABCQEAAAATa2V5TWFuYWdlclB1YmxpY0tleQAAAAADCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABlN0cmluZwQAAAABcwUAAAAHJG1hdGNoMAkAAlkAAAABBQAAAAFzAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAARVbml0BQAAAAR1bml0CQAAAgAAAAECAAAAC01hdGNoIGVycm9yAQAAAB1wZW5kaW5nTWFuYWdlclB1YmxpY0tleU9yVW5pdAAAAAAEAAAAByRtYXRjaDAJAAQiAAAAAQkBAAAAGmtleVBlbmRpbmdNYW5hZ2VyUHVibGljS2V5AAAAAAMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAGU3RyaW5nBAAAAAFzBQAAAAckbWF0Y2gwCQACWQAAAAEFAAAAAXMDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAABFVuaXQFAAAABHVuaXQJAAACAAAAAQIAAAALTWF0Y2ggZXJyb3IBAAAAC211c3RNYW5hZ2VyAAAAAQAAAAFpBAAAAAJwZAkAAAIAAAABAgAAABFQZXJtaXNzaW9uIGRlbmllZAQAAAAHJG1hdGNoMAkBAAAAFm1hbmFnZXJQdWJsaWNLZXlPclVuaXQAAAAAAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAApCeXRlVmVj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AAAAC1RvdGFsc0VudHJ5AAAABQkBAAAACWtleVRvdGFscwAAAAAFAAAAD29yaWdUb3RhbHNBcnJheQUAAAAKdG90YWxzRGlmZgUAAAAKY2xhaW1TdGFydAAAAAAAAAAAAAUAAAADbmlsAAAAAWkBAAAABmludmVzdAAAAAAEAAAACGNmZ0FycmF5CQEAAAAPcmVhZENvbmZpZ0FycmF5AAAAAAQAAAAIaWRvU3RhcnQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAAOSWR4Q2ZnSWRvU3RhcnQEAAAAC2lkb0R1cmF0aW9uCQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAEUlkeENmZ0lkb0R1cmF0aW9uBAAAAAZpZG9FbmQJAABkAAAAAgUAAAAIaWRvU3RhcnQFAAAAC2lkb0R1cmF0aW9uBAAAAApjbGFpbVN0YXJ0CQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAEElkeENmZ0NsYWltU3RhcnQEAAAADWNsYWltRHVyYXRpb24JAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAATSWR4Q2ZnQ2xhaW1EdXJhdGlvbgQAAAAFcHJpY2UJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAALSWR4Q2ZnUHJpY2UEAAAACXByaWNlTXVsdAkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAACGNmZ0FycmF5BQAAAA9JZHhDZmdQcmljZU11bHQEAAAADGlkb0Fzc2V0SWQ1OAkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAAQSWR4Q2ZnSWRvQXNzZXRJZAQAAAAKaWRvQXNzZXRJZAkAAlkAAAABBQAAAAxpZG9Bc3NldElkNTgEAAAADGlkb0Fzc2V0TXVsdAkBAAAADXBhcnNlSW50VmFsdWUAAAABCQABkQAAAAIFAAAACGNmZ0FycmF5BQAAABJJZHhDZmdJZG9Bc3NldE11bHQEAAAADnByaWNlQXNzZXRJZDU4CQABkQAAAAIFAAAACGNmZ0FycmF5BQAAABJJZHhDZmdQcmljZUFzc2V0SWQEAAAADHByaWNlQXNzZXRJZAkAAlkAAAABBQAAAA5wcmljZUFzc2V0SWQ1OAQAAAAOcHJpY2VBc3NldE11bHQJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAAUSWR4Q2ZnUHJpY2VBc3NldE11bHQEAAAADm1pbkl2ZXN0QW1vdW50CQEAAAANcGFyc2VJbnRWYWx1ZQAAAAEJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAFUlkeENmZ01pbkludmVzdEFtb3VudAQAAAALdXNlckFkZHJlc3MJAAQlAAAAAQgFAAAAAWkAAAAGY2FsbGVyAwkAAGYAAAACBQAAAAhpZG9TdGFydAUAAAAGaGVpZ2h0CQAAAgAAAAECAAAAHGlkbyBoYXMgbm90IGJlZW4gc3RhcnRlZCB5ZXQDCQAAZgAAAAIFAAAABmhlaWdodAUAAAAGaWRvRW5kCQAAAgAAAAECAAAAGmlkbyBoYXMgYmVlbiBhbHJlYWR5IGVuZGVkAwkBAAAAAiE9AAAAAgkAAZAAAAABCAUAAAABaQAAAAhwYXltZW50cwAAAAAAAAAAAQkAAAIAAAABAgAAAB1leGFjdGx5IDEgcGF5bWVudCBpcyBleHBlY3RlZAQAAAADcG10CQEAAAAFdmFsdWUAAAABCQABkQAAAAIIBQAAAAFpAAAACHBheW1lbnRzAAAAAAAAAAAABAAAAApwbXRBc3NldElkCQEAAAAFdmFsdWUAAAABCAUAAAADcG10AAAAB2Fzc2V0SWQEAAAACXBtdEFtb3VudAgFAAAAA3BtdAAAAAZhbW91bnQDCQEAAAACIT0AAAACBQAAAApwbXRBc3NldElkBQAAAAxwcmljZUFzc2V0SWQJAAACAAAAAQkAASwAAAACCQABLAAAAAICAAAAGmludmFsaWQgcGF5bWVudCBhc3NldCBpZDogCQACWAAAAAEFAAAACnBtdEFzc2V0SWQCAAAADCBpcyBleHBlY3RlZAQAAAARb3JpZ0ludmVzdG9yQXJyYXkJAQAAABpyZWFkSW52ZXN0b3JBcnJheU9yRGVmYXVsdAAAAAEFAAAAC3VzZXJBZGRyZXNzBAAAAA9vcmlnVG90YWxzQXJyYXkJAQAAABhyZWFkVG90YWxzQXJyYXlPckRlZmF1bHQAAAAABAAAABNuZXdQcmljZVRvdGFsQW1vdW50CQAAZAAAAAIJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACBQAAAA9vcmlnVG90YWxzQXJyYXkFAAAAEUlkeEludlRvdGFsQW1vdW50BQAAAAlwbXRBbW91bnQEAAAAFnJlcXVpcmVkSWRvQXNzZXRBbW91bnQJAABoAAAAAgUAAAATbmV3UHJpY2VUb3RhbEFtb3VudAAAAAAAAAAAZAMJAABmAAAAAgUAAAAWcmVxdWlyZWRJZG9Bc3NldEFtb3VudAkAA/AAAAACBQAAAAR0aGlzBQAAAAppZG9Bc3NldElkCQAAAgAAAAECAAAAOUlETyBhc3NldCBoYXMgYmVlbiAtIHNvbGQgY29uc2lkZXIgdG8gdXNlIHNtYWxsZXIgcGF5bWVudAQAAAAKdG90YWxzRGlmZgkABEwAAAACBQAAAAlwbXRBbW91bnQJAARMAAAAAgUAAAAJcG10QW1vdW50CQAETAAAAAIAAAAAAAAAAAAJAARMAAAAAgAAAAAAAAAAAAUAAAADbmlsCQAETAAAAAIJAQAAAAtUb3RhbHNFbnRyeQAAAAUJAQAAAAtrZXlJbnZlc3RvcgAAAAEFAAAAC3VzZXJBZGRyZXNzBQAAABFvcmlnSW52ZXN0b3JBcnJheQUAAAAKdG90YWxzRGlmZgUAAAAKY2xhaW1TdGFydAAAAAAAAAAAAAkABEwAAAACCQEAAAALVG90YWxzRW50cnkAAAAFCQEAAAAJa2V5VG90YWxzAAAAAAUAAAAPb3JpZ1RvdGFsc0FycmF5BQAAAAp0b3RhbHNEaWZmBQAAAApjbGFpbVN0YXJ0AAAAAAAAAAAACQAETAAAAAIJAQAAABtJbnZlc3RPcGVyYXRpb25IaXN0b3J5RW50cnkAAAAEBQAAAAt1c2VyQWRkcmVzcwUAAAAJcG10QW1vdW50AAAAAAAAAAAACAUAAAABaQAAAA10cmFuc2FjdGlvbklkBQAAAANuaWwAAAABaQEAAAAFY2xhaW0AAAACAAAAEGNsYWltZWRBc3NldElkNTgAAAANdXNlckFkZHJlc3M1OAQAAAAPY2FsbGVyQWRkcmVzczU4CQAEJQAAAAEIBQAAAAFpAAAABmNhbGxlcgMJAQAAAAIhPQAAAAIFAAAADXVzZXJBZGRyZXNzNTgFAAAAD2NhbGxlckFkZHJlc3M1OAkAAAIAAAABAgAAAA5ub3QgYXV0aG9yaXplZAQAAAAIY2ZnQXJyYXkJAQAAAA9yZWFkQ29uZmlnQXJyYXkAAAAABAAAAA5wcmljZUFzc2V0SWQ1OAkAAZEAAAACBQAAAAhjZmdBcnJheQUAAAASSWR4Q2ZnUHJpY2VBc3NldElkBAAAABBjbGFpbVJlc3VsdFR1cGxlCQEAAAANaW50ZXJuYWxDbGFpbQAAAAMFAAAAEGNsYWltZWRBc3NldElkNTgIBQAAAAFpAAAABmNhbGxlcggFAAAAAWkAAAANdHJhbnNhY3Rpb25JZAQAAAAKdG90YWxzRGlmZggFAAAAEGNsYWltUmVzdWx0VHVwbGUAAAACXzEEAAAACW91dEFtb3VudAgFAAAAEGNsYWltUmVzdWx0VHVwbGUAAAACXzIEAAAACm91dEFzc2V0SWQIBQAAABBjbGFpbVJlc3VsdFR1cGxlAAAAAl8zBAAAAA9vcmlnSW52ZXN0QXJyYXkIBQAAABBjbGFpbVJlc3VsdFR1cGxlAAAAAl80BAAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAgFAAAAEGNsYWltUmVzdWx0VHVwbGUAAAACXzUEAAAAGmNsYWltZWRQcmljZUFtb3VudEZyb21EaWZmCQABkQAAAAIFAAAACnRvdGFsc0RpZmYFAAAAIklkeERpZmZDbGFpbWVkUHJpY2VBbW91bnRJbmNyZW1lbnQEAAAAHWNsYWltZWRJZG9Bc3NldEFtb3VudEZyb21EaWZmCQABkQAAAAIFAAAACnRvdGFsc0RpZmYFAAAAJUlkeERpZmZDbGFpbWVkSWRvQXNzZXRBbW91bnRJbmNyZW1lbnQEAAAAEXByaWNlQXNzZXRCYWxhbmNlCQEAAAALdmFsdWVPckVsc2UAAAACCQAEHwAAAAEJAQAAABRrZXlQcmljZUFzc2V0QmFsYW5jZQAAAAEFAAAADXVzZXJBZGRyZXNzNTgAAAAAAAAAAAAEAAAAEnByaWNlQXNzZXREZWNpbWFscwgJAQAAAAV2YWx1ZQAAAAEJAAPsAAAAAQkAAlkAAAABBQAAAA5wcmljZUFzc2V0SWQ1OAAAAAhkZWNpbWFscwQAAAAHZW50cmllcwMJAAAAAAAAAgUAAAAQY2xhaW1lZEFzc2V0SWQ1OAUAAAAOcHJpY2VBc3NldElkNTgEAAAADSR0MDIxMjQ4MjE2OTIJAQAAAA9pbnRlcm5hbENsYWltVjIAAAAEBQAAAA5wcmljZUFzc2V0SWQ1OAUAAAANdXNlckFkZHJlc3M1OAUAAAAJb3V0QW1vdW50CQABkQAAAAIFAAAACnRvdGFsc0RpZmYFAAAAIklkeERpZmZDbGFpbWVkUHJpY2VBbW91bnRJbmNyZW1lbnQEAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0IBQAAAA0kdDAyMTI0ODIxNjkyAAAAAl8xBAAAABtwZXJpb2RUb3RhbEF2YWlsYWJsZVRvQ2xhaW0IBQAAAA0kdDAyMTI0ODIxNjkyAAAAAl8yBAAAABpwZXJpb2RVc2VyQXZhaWxhYmxlVG9DbGFpbQgFAAAADSR0MDIxMjQ4MjE2OTIAAAACXzMEAAAAGXRvdGFsVXNlckF2YWlsYWJsZVRvQ2xhaW0IBQAAAA0kdDAyMTI0ODIxNjkyAAAAAl80BAAAABx1c2R0UHJpY2VBc3NldEFsbG93YWJsZVJhdGlvCAUAAAANJHQwMjEyNDgyMTY5MgAAAAJfNQQAAAAaY3VycmVudFVzZHRQcmljZUFzc2V0UmF0aW8IBQAAAA0kdDAyMTI0ODIxNjkyAAAAAl82BAAAABNlbmRQZXJpb2RCbG9ja3NMZWZ0CAUAAAANJHQwMjEyNDgyMTY5MgAAAAJfNwQAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QIBQAAAA0kdDAyMTI0ODIxNjkyAAAAAl84BAAAAAtwZXJpb2RTdGFydAgFAAAADSR0MDIxMjQ4MjE2OTIAAAACXzkEAAAACXBlcmlvZEVuZAgFAAAADSR0MDIxMjQ4MjE2OTIAAAADXzEwBAAAAAZjaGVja3MJAARMAAAAAgMJAABmAAAAAgUAAAAacGVyaW9kVXNlckF2YWlsYWJsZVRvQ2xhaW0AAAAAAAAAAAAGCQEAAAAIdGhyb3dFcnIAAAABAgAAADp1bmF2YWlsYWJsZSB0byBjbGFpbSBiZWNhdXNlIHVzZXIgcGVyaW9kIGFsbG93YW5jZSByZWFjaGVkCQAETAAAAAIDCQAAZgAAAAIFAAAAG3BlcmlvZFRvdGFsQXZhaWxhYmxlVG9DbGFpbQAAAAAAAAAAAAYJAQAAAAh0aHJvd0VycgAAAAECAAAAO3VuYXZhaWxhYmxlIHRvIGNsYWltIGJlY2F1c2UgdG90YWwgcGVyaW9kIGFsbG93YW5jZSByZWFjaGVkCQAETAAAAAIDCQAAZgAAAAIFAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0AAAAAAAAAAAAGCQEAAAAIdGhyb3dFcnIAAAABAgAAABBub3RoaW5nIHRvIGNsYWltCQAETAAAAAIDCQAAZgAAAAIFAAAAHHVzZHRQcmljZUFzc2V0QWxsb3dhYmxlUmF0aW8FAAAAGmN1cnJlbnRVc2R0UHJpY2VBc3NldFJhdGlvBgkBAAAACHRocm93RXJyAAAAAQIAAABPdW5hdmFpbGFibGUgdG8gY2xhaW0gYmVjYXVzZSB1c2RuIHByaWNlIGxvd2VyIHRoYW4gdXNkdFByaWNlQXNzZXRBbGxvd2FibGVSYXRpbwUAAAADbmlsAwkAAAAAAAACBQAAAAZjaGVja3MFAAAABmNoZWNrcwQAAAAidXBkYXRlZFBlcmlvZFRvdGFsQXZhaWxhYmxlVG9DbGFpbQkAAGUAAAACBQAAABtwZXJpb2RUb3RhbEF2YWlsYWJsZVRvQ2xhaW0FAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0EAAAAIXVwZGF0ZWRQZXJpb2RVc2VyQXZhaWxhYmxlVG9DbGFpbQkAAGUAAAACBQAAABpwZXJpb2RVc2VyQXZhaWxhYmxlVG9DbGFpbQUAAAAZcGVyaW9kTWluQXZhaWxhYmxlVG9DbGFpbQQAAAAHZW50cmllcwMJAABnAAAAAgUAAAARcHJpY2VBc3NldEJhbGFuY2UFAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0JAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAABlwZXJpb2RNaW5BdmFpbGFibGVUb0NsYWltBQAAAApvdXRBc3NldElkCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAUa2V5UHJpY2VBc3NldEJhbGFuY2UAAAABBQAAAA11c2VyQWRkcmVzczU4CQAAZQAAAAIFAAAAEXByaWNlQXNzZXRCYWxhbmNlBQAAABlwZXJpb2RNaW5BdmFpbGFibGVUb0NsYWltBQAAAANuaWwEAAAAGHVwZGF0ZWRQcmljZUFzc2V0QmFsYW5jZQkAAGUAAAACCQAAZAAAAAIFAAAAEXByaWNlQXNzZXRCYWxhbmNlBQAAAAlvdXRBbW91bnQFAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0JAARMAAAAAgkBAAAADlNjcmlwdFRyYW5zZmVyAAAAAwgFAAAAAWkAAAAGY2FsbGVyBQAAABlwZXJpb2RNaW5BdmFpbGFibGVUb0NsYWltBQAAAApvdXRBc3NldElkCQAETAAAAAIJAQAAAAtUb3RhbHNFbnRyeQAAAAUJAQAAAAtrZXlJbnZlc3RvcgAAAAEFAAAADXVzZXJBZGRyZXNzNTgFAAAAD29yaWdJbnZlc3RBcnJheQUAAAAKdG90YWxzRGlmZgUAAAAUbmV3Q2xhaW1QZXJpb2RIZWlnaHQAAAAAAAAAAAAJAARMAAAAAgkBAAAAC1RvdGFsc0VudHJ5AAAABQkBAAAACWtleVRvdGFscwAAAAAJAQAAABhyZWFkVG90YWxzQXJyYXlPckRlZmF1bHQAAAAABQAAAAp0b3RhbHNEaWZmBQAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAAAAAAAAAAAAAkABEwAAAACCQEAAAAMSW50ZWdlckVudHJ5AAAAAgkBAAAAFGtleVByaWNlQXNzZXRCYWxhbmNlAAAAAQUAAAANdXNlckFkZHJlc3M1OAUAAAAYdXBkYXRlZFByaWNlQXNzZXRCYWxhbmNlBQAAAANuaWwJAAUUAAAAAgkABE4AAAACCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAQa2V5Q3VycmVudFBlcmlvZAAAAAAFAAAAFHVwZGF0ZWRDdXJyZW50UGVyaW9kCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAUa2V5UGVyaW9kU3RhcnRIZWlnaHQAAAABBQAAABR1cGRhdGVkQ3VycmVudFBlcmlvZAUAAAALcGVyaW9kU3RhcnQJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAQAAABJrZXlQZXJpb2RFbmRIZWlnaHQAAAABBQAAABR1cGRhdGVkQ3VycmVudFBlcmlvZAUAAAAJcGVyaW9kRW5kCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAea2V5UGVyaW9kVG90YWxBdmFpbGFibGVUb0NsYWltAAAAAgUAAAAOcHJpY2VBc3NldElkNTgFAAAAFHVwZGF0ZWRDdXJyZW50UGVyaW9kBQAAACJ1cGRhdGVkUGVyaW9kVG90YWxBdmFpbGFibGVUb0NsYWltCQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQEAAAAda2V5UGVyaW9kVXNlckF2YWlsYWJsZVRvQ2xhaW0AAAADBQAAAA5wcmljZUFzc2V0SWQ1OAUAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QFAAAADXVzZXJBZGRyZXNzNTgFAAAAIXVwZGF0ZWRQZXJpb2RVc2VyQXZhaWxhYmxlVG9DbGFpbQkABEwAAAACCQEAAAAaQ2xhaW1PcGVyYXRpb25IaXN0b3J5RW50cnkAAAAEBQAAAA11c2VyQWRkcmVzczU4BQAAABlwZXJpb2RNaW5BdmFpbGFibGVUb0NsYWltBQAAAB1jbGFpbWVkSWRvQXNzZXRBbW91bnRGcm9tRGlmZggFAAAAAWkAAAANdHJhbnNhY3Rpb25JZAUAAAADbmlsBQAAAAdlbnRyaWVzBQAAAAR1bml0CQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgQAAAAccHJpY2VBc3NldEJhbGFuY2VJZG9EZWNpbWFscwkAAGsAAAADBQAAABFwcmljZUFzc2V0QmFsYW5jZQUAAAAGc2NhbGU4CQAAbAAAAAYAAAAAAAAAAAoAAAAAAAAAAAAFAAAAEnByaWNlQXNzZXREZWNpbWFscwAAAAAAAAAAAAAAAAAAAAAAAAUAAAAERE9XTgkABRQAAAACCQAETAAAAAIJAQAAAA5TY3JpcHRUcmFuc2ZlcgAAAAMIBQAAAAFpAAAABmNhbGxlcgkAAGQAAAACBQAAAAlvdXRBbW91bnQFAAAAHHByaWNlQXNzZXRCYWxhbmNlSWRvRGVjaW1hbHMFAAAACm91dEFzc2V0SWQJAARMAAAAAgkBAAAADEludGVnZXJFbnRyeQAAAAIJAQAAABRrZXlQcmljZUFzc2V0QmFsYW5jZQAAAAEFAAAADXVzZXJBZGRyZXNzNTgAAAAAAAAAAAAJAARMAAAAAgkBAAAAC1RvdGFsc0VudHJ5AAAABQkBAAAAC2tleUludmVzdG9yAAAAAQUAAAANdXNlckFkZHJlc3M1OAUAAAAPb3JpZ0ludmVzdEFycmF5BQAAAAp0b3RhbHNEaWZmBQAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAUAAAARcHJpY2VBc3NldEJhbGFuY2UJAARMAAAAAgkBAAAAC1RvdGFsc0VudHJ5AAAABQkBAAAACWtleVRvdGFscwAAAAAJAQAAABhyZWFkVG90YWxzQXJyYXlPckRlZmF1bHQAAAAABQAAAAp0b3RhbHNEaWZmBQAAABRuZXdDbGFpbVBlcmlvZEhlaWdodAUAAAARcHJpY2VBc3NldEJhbGFuY2UJAARMAAAAAgkBAAAAGkNsYWltT3BlcmF0aW9uSGlzdG9yeUVudHJ5AAAABAUAAAANdXNlckFkZHJlc3M1OAUAAAAaY2xhaW1lZFByaWNlQW1vdW50RnJvbURpZmYJAABkAAAAAgUAAAAdY2xhaW1lZElkb0Fzc2V0QW1vdW50RnJvbURpZmYFAAAAHHByaWNlQXNzZXRCYWxhbmNlSWRvRGVjaW1hbHMIBQAAAAFpAAAADXRyYW5zYWN0aW9uSWQFAAAAA25pbAUAAAAEdW5pdAUAAAAHZW50cmllcwAAAAFpAQAAAA1jbGFpbVJFQURPTkxZAAAAAgAAABBjbGFpbWVkQXNzZXRJZDU4AAAADXVzZXJBZGRyZXNzNTgEAAAAEGNsYWltUmVzdWx0VHVwbGUJAQAAAA1pbnRlcm5hbENsYWltAAAAAwUAAAAQY2xhaW1lZEFzc2V0SWQ1OAkBAAAAEUBleHRyTmF0aXZlKDEwNjIpAAAAAQUAAAANdXNlckFkZHJlc3M1OAkAAlkAAAABAgAAAAAEAAAACnRvdGFsc0RpZmYIBQAAABBjbGFpbVJlc3VsdFR1cGxlAAAAAl8xBAAAAAlvdXRBbW91bnQIBQAAABBjbGFpbVJlc3VsdFR1cGxlAAAAAl8yBAAAAApvdXRBc3NldElkCAUAAAAQY2xhaW1SZXN1bHRUdXBsZQAAAAJfMwQAAAAPb3JpZ0ludmVzdEFycmF5CAUAAAAQY2xhaW1SZXN1bHRUdXBsZQAAAAJfNAQAAAAUbmV3Q2xhaW1QZXJpb2RIZWlnaHQIBQAAABBjbGFpbVJlc3VsdFR1cGxlAAAAAl81BAAAABVhdmFpbGFibGVUb0NsYWltQXJyYXkIBQAAABBjbGFpbVJlc3VsdFR1cGxlAAAAAl82BAAAABthdmFpbGFibGVQcmljZUFtb3VudFRvQ2xhaW0JAAGRAAAAAgUAAAAVYXZhaWxhYmxlVG9DbGFpbUFycmF5AAAAAAAAAAAABAAAABlhdmFpbGFibGVJZG9BbW91bnRUb0NsYWltCQABkQAAAAIFAAAAFWF2YWlsYWJsZVRvQ2xhaW1BcnJheQAAAAAAAAAAAQkABRQAAAACBQAAAANuaWwJAAS5AAAAAgkABEwAAAACAgAAAAYlcyVkJWQJAARMAAAAAgUAAAANdXNlckFkZHJlc3M1OAkABEwAAAACCQABpAAAAAEFAAAAG2F2YWlsYWJsZVByaWNlQW1vdW50VG9DbGFpbQkABEwAAAACCQABpAAAAAEFAAAAGWF2YWlsYWJsZUlkb0Ftb3VudFRvQ2xhaW0FAAAAA25pbAUAAAADU0VQAAAAAWkBAAAAD2NsYWltVjJSRUFET05MWQAAAAIAAAAQY2xhaW1lZEFzc2V0SWQ1OAAAAA11c2VyQWRkcmVzczU4BAAAABBjbGFpbVJlc3VsdFR1cGxlCQEAAAANaW50ZXJuYWxDbGFpbQAAAAMFAAAAEGNsYWltZWRBc3NldElkNTgJAQAAABFAZXh0ck5hdGl2ZSgxMDYyKQAAAAEFAAAADXVzZXJBZGRyZXNzNTgJAAJZAAAAAQIAAAAABAAAAAp0b3RhbHNEaWZmCAUAAAAQY2xhaW1SZXN1bHRUdXBsZQAAAAJfMQQAAAAJb3V0QW1vdW50CAUAAAAQY2xhaW1SZXN1bHRUdXBsZQAAAAJfMgQAAAAKb3V0QXNzZXRJZAgFAAAAEGNsYWltUmVzdWx0VHVwbGUAAAACXzMEAAAAD29yaWdJbnZlc3RBcnJheQgFAAAAEGNsYWltUmVzdWx0VHVwbGUAAAACXzQEAAAAFG5ld0NsYWltUGVyaW9kSGVpZ2h0CAUAAAAQY2xhaW1SZXN1bHRUdXBsZQAAAAJfNQQAAAAVYXZhaWxhYmxlVG9DbGFpbUFycmF5CAUAAAAQY2xhaW1SZXN1bHRUdXBsZQAAAAJfNgQAAAAbYXZhaWxhYmxlUHJpY2VBbW91bnRUb0NsYWltCQABkQAAAAIFAAAAFWF2YWlsYWJsZVRvQ2xhaW1BcnJheQAAAAAAAAAAAAQAAAAZYXZhaWxhYmxlSWRvQW1vdW50VG9DbGFpbQkAAZEAAAACBQAAABVhdmFpbGFibGVUb0NsYWltQXJyYXkAAAAAAAAAAAEEAAAACGNmZ0FycmF5CQEAAAAPcmVhZENvbmZpZ0FycmF5AAAAAAQAAAAOcHJpY2VBc3NldElkNTgJAAGRAAAAAgUAAAAIY2ZnQXJyYXkFAAAAEklkeENmZ1ByaWNlQXNzZXRJZAQAAAARcHJpY2VBc3NldEJhbGFuY2UJAQAAAAt2YWx1ZU9yRWxzZQAAAAIJAAQfAAAAAQkBAAAAFGtleVByaWNlQXNzZXRCYWxhbmNlAAAAAQUAAAANdXNlckFkZHJlc3M1OAAAAAAAAAAAAAQAAAAccHJpY2VBc3NldEJhbGFuY2VJZG9EZWNpbWFscwkAAGgAAAACBQAAABFwcmljZUFzc2V0QmFsYW5jZQAAAAAAAAAAZAQAAAAuYXZhaWxhYmxlSWRvQW1vdW50VG9DbGFpbVdpdGhQcmljZUFzc2V0QmFsYW5jZQkAAGQAAAACBQAAABlhdmFpbGFibGVJZG9BbW91bnRUb0NsYWltBQAAABxwcmljZUFzc2V0QmFsYW5jZUlkb0RlY2ltYWxzBAAAAA0kdDAyNzk4NTI4Mzk3CQEAAAAPaW50ZXJuYWxDbGFpbVYyAAAABAUAAAAOcHJpY2VBc3NldElkNTgFAAAADXVzZXJBZGRyZXNzNTgFAAAACW91dEFtb3VudAkAAZEAAAACBQAAAAp0b3RhbHNEaWZmBQAAACJJZHhEaWZmQ2xhaW1lZFByaWNlQW1vdW50SW5jcmVtZW50BAAAABlwZXJpb2RNaW5BdmFpbGFibGVUb0NsYWltCAUAAAANJHQwMjc5ODUyODM5NwAAAAJfMQQAAAAbcGVyaW9kVG90YWxBdmFpbGFibGVUb0NsYWltCAUAAAANJHQwMjc5ODUyODM5NwAAAAJfMgQAAAAacGVyaW9kVXNlckF2YWlsYWJsZVRvQ2xhaW0IBQAAAA0kdDAyNzk4NTI4Mzk3AAAAAl8zBAAAABl0b3RhbFVzZXJBdmFpbGFibGVUb0NsYWltCAUAAAANJHQwMjc5ODUyODM5NwAAAAJfNAQAAAAcdXNkdFByaWNlQXNzZXRBbGxvd2FibGVSYXRpbwgFAAAADSR0MDI3OTg1MjgzOTcAAAACXzUEAAAAGmN1cnJlbnRVc2R0UHJpY2VBc3NldFJhdGlvCAUAAAANJHQwMjc5ODUyODM5NwAAAAJfNgQAAAATZW5kUGVyaW9kQmxvY2tzTGVmdAgFAAAADSR0MDI3OTg1MjgzOTcAAAACXzcEAAAAFHVwZGF0ZWRDdXJyZW50UGVyaW9kCAUAAAANJHQwMjc5ODUyODM5NwAAAAJfOAQAAAALcGVyaW9kU3RhcnQIBQAAAA0kdDAyNzk4NTI4Mzk3AAAAAl85BAAAAAlwZXJpb2RFbmQIBQAAAA0kdDAyNzk4NTI4Mzk3AAAAA18xMAQAAAAWY3VycmVudFBlcmlvZEVuZEhlaWdodAkBAAAAC3ZhbHVlT3JFbHNlAAAAAgkABB8AAAABCQEAAAASa2V5UGVyaW9kRW5kSGVpZ2h0AAAAAQUAAAAUdXBkYXRlZEN1cnJlbnRQZXJpb2QAAAAAAAAAAAAEAAAAGnVzZXJUb3RhbFByaWNlQXNzZXRDbGFpbWVkCQAAZQAAAAIJAQAAAA1wYXJzZUludFZhbHVlAAAAAQkAAZEAAAACCQEAAAAXcmVhZEludmVzdG9yQXJyYXlPckZhaWwAAAABBQAAAA11c2VyQWRkcmVzczU4BQAAAB1JZHhJbnZDbGFpbWVkUHJpY2VBc3NldEFtb3VudAUAAAARcHJpY2VBc3NldEJhbGFuY2UEAAAADHJlc3VsdFN0cmluZwMJAABmAAAAAgUAAAAGaGVpZ2h0BQAAABZjdXJyZW50UGVyaW9kRW5kSGVpZ2h0BAAAAAxwZXJpb2RMZW5naHQJAQAAAAV2YWx1ZQAAAAEJAAQfAAAAAQkBAAAAD2tleVBlcmlvZExlbmd0aAAAAAAEAAAAE3VzZXJQZXJpb2RBbGxvd2FuY2UJAQAAAAV2YWx1ZQAAAAEJAAQfAAAAAQkBAAAAFmtleVVzZXJQZXJpb2RBbGxvd2FuY2UAAAABBQAAAA5wcmljZUFzc2V0SWQ1OAQAAAAUdG90YWxQZXJpb2RBbGxvd2FuY2UJAQAAAAV2YWx1ZQAAAAEJAAQfAAAAAQkBAAAAF2tleVRvdGFsUGVyaW9kQWxsb3dhbmNlAAAAAQUAAAAOcHJpY2VBc3NldElkNTgJAAS5AAAAAgkABEwAAAACAgAAAAwlZCVkJWQlZCVkJWQJAARMAAAAAgkAAaQAAAABBQAAAC5hdmFpbGFibGVJZG9BbW91bnRUb0NsYWltV2l0aFByaWNlQXNzZXRCYWxhbmNlCQAETAAAAAIJAAGkAAAAAQUAAAATdXNlclBlcmlvZEFsbG93YW5jZQkABEwAAAACCQABpAAAAAEFAAAAFHRvdGFsUGVyaW9kQWxsb3dhbmNlCQAETAAAAAIJAAGkAAAAAQUAAAAcdXNkdFByaWNlQXNzZXRBbGxvd2FibGVSYXRpbwkABEwAAAACCQABpAAAAAEFAAAAGmN1cnJlbnRVc2R0UHJpY2VBc3NldFJhdGlvCQAETAAAAAIJAAGkAAAAAQUAAAAMcGVyaW9kTGVuZ2h0CQAETAAAAAIJAAGkAAAAAQUAAAAadXNlclRvdGFsUHJpY2VBc3NldENsYWltZWQFAAAAA25pbAUAAAADU0VQCQAEuQAAAAIJAARMAAAAAgIAAAAMJWQlZCVkJWQlZCVkCQAETAAAAAIJAAGkAAAAAQUAAAAuYXZhaWxhYmxlSWRvQW1vdW50VG9DbGFpbVdpdGhQcmljZUFzc2V0QmFsYW5jZQkABEwAAAACCQABpAAAAAEFAAAAGXBlcmlvZE1pbkF2YWlsYWJsZVRvQ2xhaW0JAARMAAAAAgkAAaQAAAABBQAAABtwZXJpb2RUb3RhbEF2YWlsYWJsZVRvQ2xhaW0JAARMAAAAAgkAAaQAAAABBQAAABx1c2R0UHJpY2VBc3NldEFsbG93YWJsZVJhdGlvCQAETAAAAAIJAAGkAAAAAQUAAAAaY3VycmVudFVzZHRQcmljZUFzc2V0UmF0aW8JAARMAAAAAgkAAaQAAAABBQAAABNlbmRQZXJpb2RCbG9ja3NMZWZ0CQAETAAAAAIJAAGkAAAAAQUAAAAadXNlclRvdGFsUHJpY2VBc3NldENsYWltZWQFAAAAA25pbAUAAAADU0VQCQAFFAAAAAIFAAAAA25pbAUAAAAMcmVzdWx0U3RyaW5nAAAAAWkBAAAACnNldE1hbmFnZXIAAAABAAAAF3BlbmRpbmdNYW5hZ2VyUHVibGljS2V5BAAAAAtjaGVja0NhbGxlcgkBAAAAC211c3RNYW5hZ2VyAAAAAQUAAAABaQMJAAAAAAAAAgUAAAALY2hlY2tDYWxsZXIFAAAAC2NoZWNrQ2FsbGVyBAAAABVjaGVja01hbmFnZXJQdWJsaWNLZXkJAAJZAAAAAQUAAAAXcGVuZGluZ01hbmFnZXJQdWJsaWNLZXkDCQAAAAAAAAIFAAAAFWNoZWNrTWFuYWdlclB1YmxpY0tleQUAAAAVY2hlY2tNYW5hZ2VyUHVibGljS2V5CQAETAAAAAIJAQAAAAtTdHJpbmdFbnRyeQAAAAIJAQAAABprZXlQZW5kaW5nTWFuYWdlclB1YmxpY0tleQAAAAAFAAAAF3BlbmRpbmdNYW5hZ2VyUHVibGljS2V5BQAAAANuaWwJAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgAAAAFpAQAAAA5jb25maXJtTWFuYWdlcgAAAAAEAAAAAnBtCQEAAAAdcGVuZGluZ01hbmFnZXJQdWJsaWNLZXlPclVuaXQAAAAABAAAAAVoYXNQTQMJAQAAAAlpc0RlZmluZWQAAAABBQAAAAJwbQYJAAACAAAAAQIAAAASTm8gcGVuZGluZyBtYW5hZ2VyAwkAAAAAAAACBQAAAAVoYXNQTQUAAAAFaGFzUE0EAAAAB2NoZWNrUE0DCQAAAAAAAAIIBQAAAAFpAAAAD2NhbGxlclB1YmxpY0tleQkBAAAABXZhbHVlAAAAAQUAAAACcG0GCQAAAgAAAAECAAAAG1lvdSBhcmUgbm90IHBlbmRpbmcgbWFuYWdlcgMJAAAAAAAAAgUAAAAHY2hlY2tQTQUAAAAHY2hlY2tQTQkABEwAAAACCQEAAAALU3RyaW5nRW50cnkAAAACCQEAAAATa2V5TWFuYWdlclB1YmxpY0tleQAAAAAJAAJYAAAAAQkBAAAABXZhbHVlAAAAAQUAAAACcG0JAARMAAAAAgkBAAAAC0RlbGV0ZUVudHJ5AAAAAQkBAAAAGmtleVBlbmRpbmdNYW5hZ2VyUHVibGljS2V5AAAAAAUAAAADbmlsCQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4AAAABAAAAAnR4AQAAAAZ2ZXJpZnkAAAAABAAAAA90YXJnZXRQdWJsaWNLZXkEAAAAByRtYXRjaDAJAQAAABZtYW5hZ2VyUHVibGljS2V5T3JVbml0AAAAAAMJAAABAAAAAgUAAAAHJG1hdGNoMAIAAAAKQnl0ZVZlY3RvcgQAAAACcGsFAAAAByRtYXRjaDAFAAAAAnBrAwkAAAEAAAACBQAAAAckbWF0Y2gwAgAAAARVbml0CAUAAAACdHgAAAAPc2VuZGVyUHVibGljS2V5CQAAAgAAAAECAAAAC01hdGNoIGVycm9yCQAB9AAAAAMIBQAAAAJ0eAAAAAlib2R5Qnl0ZXMJAAGRAAAAAggFAAAAAnR4AAAABnByb29mcwAAAAAAAAAAAAUAAAAPdGFyZ2V0UHVibGljS2V5FGzCgQ==", "chainId": 87, "height": 3367595, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: EGeaSLDkNzExqjrkhW9zzf251C7hHhswkmJUudEoovXH Next: VvfAGWVUx2QHgVhyAJ8TmdPuiLMWo4BfUwerfT7fDCY Diff:
OldNewDifferences
77 let SEP = "__"
88
99 let BUFSCALE = toBigInt(1000000000000000000)
10+
11+let scale8 = 100000000
12+
13+func throwErr (msg) = throw(makeString(["ido.ride:", msg], " "))
14+
1015
1116 func convertPriceAssetIntoIdoAsset (priceAssetAmount,priceAssetMULT,price,priceMULT,idoAssetMULT) = {
1217 let bPriceAssetMULT = toBigInt(priceAssetMULT)
8287 func keyUSDNClaimEndHeight () = "%s__usdnClaimEndHeight"
8388
8489
90+func keyPeriodLength () = makeString(["%s", "periodLength"], SEP)
91+
92+
93+func keyCurrentPeriod () = makeString(["%s", "currentPeriod"], SEP)
94+
95+
96+func keyPeriodStartHeight (periodNum) = makeString(["%s%s", "periodStartHeight", toString(periodNum)], SEP)
97+
98+
99+func keyPeriodEndHeight (periodNum) = makeString(["%s%s", "periodEndHeight", toString(periodNum)], SEP)
100+
101+
102+func keyUsdtPriceAssetAllowableRatio () = makeString(["%s", "usdtPriceAssetAllowableRatio"], SEP)
103+
104+
105+func keyTotalPeriodAllowance (assetId) = makeString(["%s%s", "totalPeriodAllowance", assetId], SEP)
106+
107+
108+func keyUserPeriodAllowance (assetId) = makeString(["%s%s", "userPeriodAllowance", assetId], SEP)
109+
110+
111+func keyPeriodTotalAvailableToClaim (assetId,periodNum) = makeString(["%s%s%s", "periodTotalAvailableToClaim", assetId, toString(periodNum)], SEP)
112+
113+
114+func keyPeriodUserAvailableToClaim (assetId,periodNum,userAddress) = makeString(["%s%s%s%s", "periodUserAvailableToClaim", assetId, toString(periodNum), userAddress], SEP)
115+
116+
117+func keyUsdtPriceAssetStablePool () = makeString(["%s", "usdtPriceAssetStablePool"], SEP)
118+
119+
120+func keyUsdtAssetId () = makeString(["%s", "usdtAssetId"], SEP)
121+
122+
123+func keyPriceAssetBalance (address) = makeString(["%s%s", "priceAssetBalance", address], SEP)
124+
125+
85126 func keyManagerPublicKey () = "%s__managerPublicKey"
86127
87128
111152
112153 let IdxDiffClaimedIdoAssetAmountIncrement = 3
113154
114-func TotalsEntry (key,origArray,incrementDiff,newLastClaimedHeight) = {
155+func TotalsEntry (key,origArray,incrementDiff,newLastClaimedHeight,priceAssetBalance) = {
115156 let totalAmount = parseIntValue(origArray[IdxInvTotalAmount])
116157 let remainingAmount = parseIntValue(origArray[IdxInvRemainingAmount])
117158 let claimedPriceAssetAmount = parseIntValue(origArray[IdxInvClaimedPriceAssetAmount])
119160 let lastClaimedHeight = parseIntValue(origArray[IdxInvLastClaimedHeight])
120161 let newTotalAmount = (totalAmount + incrementDiff[IdxDiffTotalIncrement])
121162 let newRemainingAmount = (remainingAmount + incrementDiff[IdxDiffRemainingPriceAmountIncrement])
122- let newClaimedPriceAssetAmount = (claimedPriceAssetAmount + incrementDiff[IdxDiffClaimedPriceAmountIncrement])
123- let newClaimedIdoAssetAmount = (claimedIdoAssetAmount + incrementDiff[IdxDiffClaimedIdoAssetAmountIncrement])
163+ let cfgArray = readConfigArray()
164+ let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
165+ let priceAssetDecimals = value(assetInfo(fromBase58String(priceAssetId58))).decimals
166+ let priceAssetBalancePriceAssetDecimals = fraction(priceAssetBalance, scale8, pow(10, 0, priceAssetDecimals, 0, 0, DOWN))
167+ let newClaimedPriceAssetAmount = ((claimedPriceAssetAmount + incrementDiff[IdxDiffClaimedPriceAmountIncrement]) - priceAssetBalance)
168+ let newClaimedIdoAssetAmount = ((claimedIdoAssetAmount + incrementDiff[IdxDiffClaimedIdoAssetAmountIncrement]) + priceAssetBalancePriceAssetDecimals)
124169 if ((0 > newRemainingAmount))
125170 then throw("invalid math")
126171 else StringEntry(key, formatInvestor(newTotalAmount, newRemainingAmount, newClaimedPriceAssetAmount, newClaimedIdoAssetAmount, newLastClaimedHeight))
163208 let claimingIdoAssetAmount = convertPriceAssetIntoIdoAsset(claimingPriceAssetAmount, priceAssetMult, price, priceMult, idoAssetMult)
164209 let isUSDN = (claimedAssetId58 == priceAssetId58)
165210 let isUSDNClaimDisabled = valueOrElse(getBoolean(keyUSDNClaimDisabled()), false)
166- let isUSDNClaimEnded = match getInteger(keyUSDNClaimEndHeight()) {
167- case end: Int =>
168- (height > end)
169- case _: Unit =>
170- false
171- case _ =>
172- throw("Match error")
173- }
174211 let checks = [if (!(if (isUSDN)
175212 then isUSDNClaimDisabled
176213 else false))
177214 then true
178- else throw("USDN claim is disabled"), if (!(if (isUSDN)
179- then isUSDNClaimEnded
180- else false))
181- then true
182- else throw("USDN claim is ended")]
215+ else throw("USDN claim is disabled")]
183216 if ((checks == checks))
184217 then if ((claimedAssetId58 == priceAssetId58))
185218 then $Tuple6([0, -(claimingPriceAssetAmount), claimingPriceAssetAmount, 0], claimingPriceAssetAmount, priceAssetId, origInvestArray, newClaimPeriodHeight, [claimingPriceAssetAmount, claimingIdoAssetAmount])
186219 else if ((claimedAssetId58 == idoAssetId58))
187220 then $Tuple6([0, -(claimingPriceAssetAmount), 0, claimingIdoAssetAmount], claimingIdoAssetAmount, idoAssetId, origInvestArray, newClaimPeriodHeight, [claimingPriceAssetAmount, claimingIdoAssetAmount])
188221 else throw(("unsupported assetId: " + claimedAssetId58))
222+ else throw("Strict value is not equal to itself.")
223+ }
224+
225+
226+func internalClaimV2 (priceAssetId58,userAddress58,outAmount,totalUserAvailableToClaim) = {
227+ let totalPeriodPriceAssetAllowance = value(getInteger(keyTotalPeriodAllowance(priceAssetId58)))
228+ let userPeriodPriceAssetAllowance = value(getInteger(keyUserPeriodAllowance(priceAssetId58)))
229+ let periodLength = value(getInteger(keyPeriodLength()))
230+ let currentPeriod = valueOrElse(getInteger(keyCurrentPeriod()), 0)
231+ let zeroPeriodEndHeighIsDefined = isDefined(getInteger(keyPeriodEndHeight(0)))
232+ let $t01224314113 = if ((currentPeriod > 0))
233+ then {
234+ let lastPeriodStartHeight = value(getInteger(keyPeriodStartHeight(currentPeriod)))
235+ let lastPeriodEndHeight = value(getInteger(keyPeriodEndHeight(currentPeriod)))
236+ let $t01250313208 = if ((height > lastPeriodEndHeight))
237+ then {
238+ let updatedCurrentPeriod = (currentPeriod + 1)
239+ let periodStart = if ((height > (lastPeriodEndHeight + periodLength)))
240+ then {
241+ let blocksToLastPeriodStart = ((height - lastPeriodEndHeight) % periodLength)
242+ if ((blocksToLastPeriodStart == 0))
243+ then ((height - periodLength) + 1)
244+ else (height - blocksToLastPeriodStart)
245+ }
246+ else (lastPeriodEndHeight + 1)
247+ let periodEnd = ((periodStart + periodLength) - 1)
248+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
249+ }
250+ else $Tuple3(currentPeriod, lastPeriodStartHeight, lastPeriodEndHeight)
251+ let updatedCurrentPeriod = $t01250313208._1
252+ let periodStart = $t01250313208._2
253+ let periodEnd = $t01250313208._3
254+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
255+ }
256+ else if (zeroPeriodEndHeighIsDefined)
257+ then {
258+ let zeroPeriodStartHeight = value(getInteger(keyPeriodStartHeight(0)))
259+ let zeroPeriodEndHeight = value(getInteger(keyPeriodEndHeight(0)))
260+ let $t01346313845 = if ((height > zeroPeriodEndHeight))
261+ then {
262+ let updatedCurrentPeriod = (currentPeriod + 1)
263+ let periodStart = (zeroPeriodEndHeight + 1)
264+ let periodEnd = ((periodStart + periodLength) - 1)
265+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
266+ }
267+ else $Tuple3(currentPeriod, zeroPeriodStartHeight, zeroPeriodEndHeight)
268+ let updatedCurrentPeriod = $t01346313845._1
269+ let periodStart = $t01346313845._2
270+ let periodEnd = $t01346313845._3
271+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
272+ }
273+ else $Tuple3(currentPeriod, valueOrElse(getInteger(keyPeriodStartHeight(currentPeriod)), height), valueOrElse(getInteger(keyPeriodEndHeight(currentPeriod)), ((height + periodLength) - 1)))
274+ let updatedCurrentPeriod = $t01224314113._1
275+ let periodStart = $t01224314113._2
276+ let periodEnd = $t01224314113._3
277+ let periodTotalAvailableToClaim = valueOrElse(getInteger(keyPeriodTotalAvailableToClaim(priceAssetId58, updatedCurrentPeriod)), totalPeriodPriceAssetAllowance)
278+ let periodUserAvailableToClaim = valueOrElse(getInteger(keyPeriodUserAvailableToClaim(priceAssetId58, updatedCurrentPeriod, userAddress58)), userPeriodPriceAssetAllowance)
279+ let priceAssetBalance = valueOrElse(getInteger(keyPriceAssetBalance(userAddress58)), 0)
280+ let periodMinAvailableToClaim = min([(outAmount + priceAssetBalance), periodTotalAvailableToClaim, periodUserAvailableToClaim])
281+ let usdtPriceAssetAllowableRatio = value(getInteger(keyUsdtPriceAssetAllowableRatio()))
282+ let putOneTknV2PriceAssetAmount = scale8
283+ let $t01486815121 = {
284+ let @ = invoke(addressFromStringValue(value(getString(keyUsdtPriceAssetStablePool()))), "putOneTknV2WithoutTakeFeeREADONLY", [putOneTknV2PriceAssetAmount, priceAssetId58], nil)
285+ if ($isInstanceOf(@, "(Int, Int, Int)"))
286+ then @
287+ else throw("Couldn't cast Any to (Int, Int, Int)")
288+ }
289+ if (($t01486815121 == $t01486815121))
290+ then {
291+ let bonus = $t01486815121._3
292+ let feeAmount = $t01486815121._2
293+ let lpAmount = $t01486815121._1
294+ let usdtAssetId = value(getString(keyUsdtAssetId()))
295+ let $t01518315401 = {
296+ let @ = invoke(addressFromStringValue(value(getString(keyUsdtPriceAssetStablePool()))), "getOneTknV2READONLY", [usdtAssetId, lpAmount], nil)
297+ if ($isInstanceOf(@, "(Int, Int)"))
298+ then @
299+ else throw("Couldn't cast Any to (Int, Int)")
300+ }
301+ if (($t01518315401 == $t01518315401))
302+ then {
303+ let getOneTknV2FeeAmount = $t01518315401._2
304+ let usdtAmount = $t01518315401._1
305+ let currentUsdtPriceAssetRatio = fraction(putOneTknV2PriceAssetAmount, scale8, usdtAmount)
306+ let endPeriodBlocksLeft = (periodEnd - height)
307+ $Tuple10(periodMinAvailableToClaim, periodTotalAvailableToClaim, periodUserAvailableToClaim, totalUserAvailableToClaim, usdtPriceAssetAllowableRatio, currentUsdtPriceAssetRatio, endPeriodBlocksLeft, updatedCurrentPeriod, periodStart, periodEnd)
308+ }
309+ else throw("Strict value is not equal to itself.")
310+ }
189311 else throw("Strict value is not equal to itself.")
190312 }
191313
250372 let priceAssetMult = pow(10, 0, priceAssetInfo.decimals, 0, 0, DOWN)
251373 let origTotalsArray = readTotalsArrayOrDefault()
252374 let totalsDiff = [0, 0, 0, 0]
253-[StringEntry(keyConfig(), fromatConfig(idoStart, idoDuration, claimStart, claimDuration, price, priceMult, idoAssetId58, idoAssetMult, priceAssetId58, priceAssetMult, minInvestAmount, pmt.amount)), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart)]
375+[StringEntry(keyConfig(), fromatConfig(idoStart, idoDuration, claimStart, claimDuration, price, priceMult, idoAssetId58, idoAssetMult, priceAssetId58, priceAssetMult, minInvestAmount, pmt.amount)), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart, 0)]
254376 }
255377 }
256378
295417 then throw("IDO asset has been - sold consider to use smaller payment")
296418 else {
297419 let totalsDiff = [pmtAmount, pmtAmount, 0, 0]
298-[TotalsEntry(keyInvestor(userAddress), origInvestorArray, totalsDiff, claimStart), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart), InvestOperationHistoryEntry(userAddress, pmtAmount, 0, i.transactionId)]
420+[TotalsEntry(keyInvestor(userAddress), origInvestorArray, totalsDiff, claimStart, 0), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart, 0), InvestOperationHistoryEntry(userAddress, pmtAmount, 0, i.transactionId)]
299421 }
300422 }
301423 }
309431 if ((userAddress58 != callerAddress58))
310432 then throw("not authorized")
311433 else {
434+ let cfgArray = readConfigArray()
435+ let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
312436 let claimResultTuple = internalClaim(claimedAssetId58, i.caller, i.transactionId)
313437 let totalsDiff = claimResultTuple._1
314438 let outAmount = claimResultTuple._2
317441 let newClaimPeriodHeight = claimResultTuple._5
318442 let claimedPriceAmountFromDiff = totalsDiff[IdxDiffClaimedPriceAmountIncrement]
319443 let claimedIdoAssetAmountFromDiff = totalsDiff[IdxDiffClaimedIdoAssetAmountIncrement]
320- $Tuple2([ScriptTransfer(i.caller, outAmount, outAssetId), TotalsEntry(keyInvestor(userAddress58), origInvestArray, totalsDiff, newClaimPeriodHeight), TotalsEntry(keyTotals(), readTotalsArrayOrDefault(), totalsDiff, newClaimPeriodHeight), ClaimOperationHistoryEntry(userAddress58, claimedPriceAmountFromDiff, claimedIdoAssetAmountFromDiff, i.transactionId)], unit)
444+ let priceAssetBalance = valueOrElse(getInteger(keyPriceAssetBalance(userAddress58)), 0)
445+ let priceAssetDecimals = value(assetInfo(fromBase58String(priceAssetId58))).decimals
446+ let entries = if ((claimedAssetId58 == priceAssetId58))
447+ then {
448+ let $t02124821692 = internalClaimV2(priceAssetId58, userAddress58, outAmount, totalsDiff[IdxDiffClaimedPriceAmountIncrement])
449+ let periodMinAvailableToClaim = $t02124821692._1
450+ let periodTotalAvailableToClaim = $t02124821692._2
451+ let periodUserAvailableToClaim = $t02124821692._3
452+ let totalUserAvailableToClaim = $t02124821692._4
453+ let usdtPriceAssetAllowableRatio = $t02124821692._5
454+ let currentUsdtPriceAssetRatio = $t02124821692._6
455+ let endPeriodBlocksLeft = $t02124821692._7
456+ let updatedCurrentPeriod = $t02124821692._8
457+ let periodStart = $t02124821692._9
458+ let periodEnd = $t02124821692._10
459+ let checks = [if ((periodUserAvailableToClaim > 0))
460+ then true
461+ else throwErr("unavailable to claim because user period allowance reached"), if ((periodTotalAvailableToClaim > 0))
462+ then true
463+ else throwErr("unavailable to claim because total period allowance reached"), if ((periodMinAvailableToClaim > 0))
464+ then true
465+ else throwErr("nothing to claim"), if ((usdtPriceAssetAllowableRatio > currentUsdtPriceAssetRatio))
466+ then true
467+ else throwErr("unavailable to claim because usdn price lower than usdtPriceAssetAllowableRatio")]
468+ if ((checks == checks))
469+ then {
470+ let updatedPeriodTotalAvailableToClaim = (periodTotalAvailableToClaim - periodMinAvailableToClaim)
471+ let updatedPeriodUserAvailableToClaim = (periodUserAvailableToClaim - periodMinAvailableToClaim)
472+ let entries = if ((priceAssetBalance >= periodMinAvailableToClaim))
473+ then [ScriptTransfer(i.caller, periodMinAvailableToClaim, outAssetId), IntegerEntry(keyPriceAssetBalance(userAddress58), (priceAssetBalance - periodMinAvailableToClaim))]
474+ else {
475+ let updatedPriceAssetBalance = ((priceAssetBalance + outAmount) - periodMinAvailableToClaim)
476+[ScriptTransfer(i.caller, periodMinAvailableToClaim, outAssetId), TotalsEntry(keyInvestor(userAddress58), origInvestArray, totalsDiff, newClaimPeriodHeight, 0), TotalsEntry(keyTotals(), readTotalsArrayOrDefault(), totalsDiff, newClaimPeriodHeight, 0), IntegerEntry(keyPriceAssetBalance(userAddress58), updatedPriceAssetBalance)]
477+ }
478+ $Tuple2(([IntegerEntry(keyCurrentPeriod(), updatedCurrentPeriod), IntegerEntry(keyPeriodStartHeight(updatedCurrentPeriod), periodStart), IntegerEntry(keyPeriodEndHeight(updatedCurrentPeriod), periodEnd), IntegerEntry(keyPeriodTotalAvailableToClaim(priceAssetId58, updatedCurrentPeriod), updatedPeriodTotalAvailableToClaim), IntegerEntry(keyPeriodUserAvailableToClaim(priceAssetId58, updatedCurrentPeriod, userAddress58), updatedPeriodUserAvailableToClaim), ClaimOperationHistoryEntry(userAddress58, periodMinAvailableToClaim, claimedIdoAssetAmountFromDiff, i.transactionId)] ++ entries), unit)
479+ }
480+ else throw("Strict value is not equal to itself.")
481+ }
482+ else {
483+ let priceAssetBalanceIdoDecimals = fraction(priceAssetBalance, scale8, pow(10, 0, priceAssetDecimals, 0, 0, DOWN))
484+ $Tuple2([ScriptTransfer(i.caller, (outAmount + priceAssetBalanceIdoDecimals), outAssetId), IntegerEntry(keyPriceAssetBalance(userAddress58), 0), TotalsEntry(keyInvestor(userAddress58), origInvestArray, totalsDiff, newClaimPeriodHeight, priceAssetBalance), TotalsEntry(keyTotals(), readTotalsArrayOrDefault(), totalsDiff, newClaimPeriodHeight, priceAssetBalance), ClaimOperationHistoryEntry(userAddress58, claimedPriceAmountFromDiff, (claimedIdoAssetAmountFromDiff + priceAssetBalanceIdoDecimals), i.transactionId)], unit)
485+ }
486+ entries
321487 }
322488 }
323489
335501 let availablePriceAmountToClaim = availableToClaimArray[0]
336502 let availableIdoAmountToClaim = availableToClaimArray[1]
337503 $Tuple2(nil, makeString(["%s%d%d", userAddress58, toString(availablePriceAmountToClaim), toString(availableIdoAmountToClaim)], SEP))
504+ }
505+
506+
507+
508+@Callable(i)
509+func claimV2READONLY (claimedAssetId58,userAddress58) = {
510+ let claimResultTuple = internalClaim(claimedAssetId58, addressFromStringValue(userAddress58), fromBase58String(""))
511+ let totalsDiff = claimResultTuple._1
512+ let outAmount = claimResultTuple._2
513+ let outAssetId = claimResultTuple._3
514+ let origInvestArray = claimResultTuple._4
515+ let newClaimPeriodHeight = claimResultTuple._5
516+ let availableToClaimArray = claimResultTuple._6
517+ let availablePriceAmountToClaim = availableToClaimArray[0]
518+ let availableIdoAmountToClaim = availableToClaimArray[1]
519+ let cfgArray = readConfigArray()
520+ let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
521+ let priceAssetBalance = valueOrElse(getInteger(keyPriceAssetBalance(userAddress58)), 0)
522+ let priceAssetBalanceIdoDecimals = (priceAssetBalance * 100)
523+ let availableIdoAmountToClaimWithPriceAssetBalance = (availableIdoAmountToClaim + priceAssetBalanceIdoDecimals)
524+ let $t02798528397 = internalClaimV2(priceAssetId58, userAddress58, outAmount, totalsDiff[IdxDiffClaimedPriceAmountIncrement])
525+ let periodMinAvailableToClaim = $t02798528397._1
526+ let periodTotalAvailableToClaim = $t02798528397._2
527+ let periodUserAvailableToClaim = $t02798528397._3
528+ let totalUserAvailableToClaim = $t02798528397._4
529+ let usdtPriceAssetAllowableRatio = $t02798528397._5
530+ let currentUsdtPriceAssetRatio = $t02798528397._6
531+ let endPeriodBlocksLeft = $t02798528397._7
532+ let updatedCurrentPeriod = $t02798528397._8
533+ let periodStart = $t02798528397._9
534+ let periodEnd = $t02798528397._10
535+ let currentPeriodEndHeight = valueOrElse(getInteger(keyPeriodEndHeight(updatedCurrentPeriod)), 0)
536+ let userTotalPriceAssetClaimed = (parseIntValue(readInvestorArrayOrFail(userAddress58)[IdxInvClaimedPriceAssetAmount]) - priceAssetBalance)
537+ let resultString = if ((height > currentPeriodEndHeight))
538+ then {
539+ let periodLenght = value(getInteger(keyPeriodLength()))
540+ let userPeriodAllowance = value(getInteger(keyUserPeriodAllowance(priceAssetId58)))
541+ let totalPeriodAllowance = value(getInteger(keyTotalPeriodAllowance(priceAssetId58)))
542+ makeString(["%d%d%d%d%d%d", toString(availableIdoAmountToClaimWithPriceAssetBalance), toString(userPeriodAllowance), toString(totalPeriodAllowance), toString(usdtPriceAssetAllowableRatio), toString(currentUsdtPriceAssetRatio), toString(periodLenght), toString(userTotalPriceAssetClaimed)], SEP)
543+ }
544+ else makeString(["%d%d%d%d%d%d", toString(availableIdoAmountToClaimWithPriceAssetBalance), toString(periodMinAvailableToClaim), toString(periodTotalAvailableToClaim), toString(usdtPriceAssetAllowableRatio), toString(currentUsdtPriceAssetRatio), toString(endPeriodBlocksLeft), toString(userTotalPriceAssetClaimed)], SEP)
545+ $Tuple2(nil, resultString)
338546 }
339547
340548
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 5 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 func getStringOrFail (key) = valueOrErrorMessage(getString(key), (("mandatory this." + key) + " is not defined"))
55
66
77 let SEP = "__"
88
99 let BUFSCALE = toBigInt(1000000000000000000)
10+
11+let scale8 = 100000000
12+
13+func throwErr (msg) = throw(makeString(["ido.ride:", msg], " "))
14+
1015
1116 func convertPriceAssetIntoIdoAsset (priceAssetAmount,priceAssetMULT,price,priceMULT,idoAssetMULT) = {
1217 let bPriceAssetMULT = toBigInt(priceAssetMULT)
1318 let bIdoAssetMULT = toBigInt(idoAssetMULT)
1419 let bPriceAssetBUF = fraction(toBigInt(priceAssetAmount), BUFSCALE, bPriceAssetMULT)
1520 let bAmountAssetBUF = fraction(bPriceAssetBUF, toBigInt(priceMULT), toBigInt(price))
1621 toInt(fraction(bAmountAssetBUF, toBigInt(idoAssetMULT), BUFSCALE))
1722 }
1823
1924
2025 let IdxCfgIdoStart = 1
2126
2227 let IdxCfgIdoDuration = 2
2328
2429 let IdxCfgClaimStart = 3
2530
2631 let IdxCfgClaimDuration = 4
2732
2833 let IdxCfgPrice = 5
2934
3035 let IdxCfgPriceMult = 6
3136
3237 let IdxCfgIdoAssetId = 7
3338
3439 let IdxCfgIdoAssetMult = 8
3540
3641 let IdxCfgPriceAssetId = 9
3742
3843 let IdxCfgPriceAssetMult = 10
3944
4045 let IdxCfgMinInvestAmount = 11
4146
4247 func fromatConfigS (idoStart,idoDuration,claimStart,claimDuration,price,priceMult,idoAssetId58,idoAssetMult,priceAssetId58,priceAssetMult,minInvestAmount,totalIdoAssetToSell) = makeString(["%d%d%d%d%d%d%s%d%s%d%d%d", idoStart, idoDuration, claimStart, claimDuration, price, priceMult, idoAssetId58, idoAssetMult, priceAssetId58, priceAssetMult, minInvestAmount, totalIdoAssetToSell], SEP)
4348
4449
4550 func fromatConfig (idoStart,idoDuration,claimStart,claimDuration,price,priceMult,idoAssetId58,idoAssetMult,priceAssetId58,priceAssetMult,minInvestAmount,totalIdoAssetToSell) = fromatConfigS(toString(idoStart), toString(idoDuration), toString(claimStart), toString(claimDuration), toString(price), toString(priceMult), idoAssetId58, toString(idoAssetMult), priceAssetId58, toString(priceAssetMult), toString(minInvestAmount), toString(totalIdoAssetToSell))
4651
4752
4853 let IdxInvTotalAmount = 1
4954
5055 let IdxInvRemainingAmount = 2
5156
5257 let IdxInvClaimedPriceAssetAmount = 3
5358
5459 let IdxInvClaimedIdoAssetAmount = 4
5560
5661 let IdxInvLastClaimedHeight = 5
5762
5863 func formatInvestorS (totalAmount,remainingAmount,claimedPriceAssetAmount,claimedIdoAssetAmount,lastClaimedHeight) = makeString(["%d%d%d%d%d", totalAmount, remainingAmount, claimedPriceAssetAmount, claimedIdoAssetAmount, lastClaimedHeight], SEP)
5964
6065
6166 func formatInvestor (totalAmount,remainingAmount,claimedPriceAssetAmount,claimedIdoAssetAmount,lastClaimedHeight) = formatInvestorS(toString(totalAmount), toString(remainingAmount), toString(claimedPriceAssetAmount), toString(claimedIdoAssetAmount), toString(lastClaimedHeight))
6267
6368
6469 func formatHistoryRecord (priceAssetAmount,idoAssetAmount) = makeString(["%d%d%d%d", toString(height), toString(lastBlock.timestamp), toString(priceAssetAmount), toString(idoAssetAmount)], SEP)
6570
6671
6772 func keyConfig () = "%s__config"
6873
6974
7075 func keyInvestor (userAddress) = ("%s__" + userAddress)
7176
7277
7378 func keyTotals () = "%s__totals"
7479
7580
7681 func keyOperationHistoryRecord (type,userAddress,txId58) = makeString(["%s%s%s%s__history", type, userAddress, txId58], SEP)
7782
7883
7984 func keyUSDNClaimDisabled () = "%s__usdnClaimDisabled"
8085
8186
8287 func keyUSDNClaimEndHeight () = "%s__usdnClaimEndHeight"
8388
8489
90+func keyPeriodLength () = makeString(["%s", "periodLength"], SEP)
91+
92+
93+func keyCurrentPeriod () = makeString(["%s", "currentPeriod"], SEP)
94+
95+
96+func keyPeriodStartHeight (periodNum) = makeString(["%s%s", "periodStartHeight", toString(periodNum)], SEP)
97+
98+
99+func keyPeriodEndHeight (periodNum) = makeString(["%s%s", "periodEndHeight", toString(periodNum)], SEP)
100+
101+
102+func keyUsdtPriceAssetAllowableRatio () = makeString(["%s", "usdtPriceAssetAllowableRatio"], SEP)
103+
104+
105+func keyTotalPeriodAllowance (assetId) = makeString(["%s%s", "totalPeriodAllowance", assetId], SEP)
106+
107+
108+func keyUserPeriodAllowance (assetId) = makeString(["%s%s", "userPeriodAllowance", assetId], SEP)
109+
110+
111+func keyPeriodTotalAvailableToClaim (assetId,periodNum) = makeString(["%s%s%s", "periodTotalAvailableToClaim", assetId, toString(periodNum)], SEP)
112+
113+
114+func keyPeriodUserAvailableToClaim (assetId,periodNum,userAddress) = makeString(["%s%s%s%s", "periodUserAvailableToClaim", assetId, toString(periodNum), userAddress], SEP)
115+
116+
117+func keyUsdtPriceAssetStablePool () = makeString(["%s", "usdtPriceAssetStablePool"], SEP)
118+
119+
120+func keyUsdtAssetId () = makeString(["%s", "usdtAssetId"], SEP)
121+
122+
123+func keyPriceAssetBalance (address) = makeString(["%s%s", "priceAssetBalance", address], SEP)
124+
125+
85126 func keyManagerPublicKey () = "%s__managerPublicKey"
86127
87128
88129 func keyPendingManagerPublicKey () = "%s__pendingManagerPublicKey"
89130
90131
91132 func readConfigArray () = split(getStringOrFail(keyConfig()), SEP)
92133
93134
94135 func readTotalsArrayOrDefaultByCustomKey (customKey) = split(valueOrElse(getString(customKey), formatInvestorS("0", "0", "0", "0", "0")), SEP)
95136
96137
97138 func readTotalsArrayOrDefault () = readTotalsArrayOrDefaultByCustomKey(keyTotals())
98139
99140
100141 func readInvestorArrayOrDefault (userAddress) = readTotalsArrayOrDefaultByCustomKey(keyInvestor(userAddress))
101142
102143
103144 func readInvestorArrayOrFail (userAddress) = split(getStringOrFail(keyInvestor(userAddress)), SEP)
104145
105146
106147 let IdxDiffTotalIncrement = 0
107148
108149 let IdxDiffRemainingPriceAmountIncrement = 1
109150
110151 let IdxDiffClaimedPriceAmountIncrement = 2
111152
112153 let IdxDiffClaimedIdoAssetAmountIncrement = 3
113154
114-func TotalsEntry (key,origArray,incrementDiff,newLastClaimedHeight) = {
155+func TotalsEntry (key,origArray,incrementDiff,newLastClaimedHeight,priceAssetBalance) = {
115156 let totalAmount = parseIntValue(origArray[IdxInvTotalAmount])
116157 let remainingAmount = parseIntValue(origArray[IdxInvRemainingAmount])
117158 let claimedPriceAssetAmount = parseIntValue(origArray[IdxInvClaimedPriceAssetAmount])
118159 let claimedIdoAssetAmount = parseIntValue(origArray[IdxInvClaimedIdoAssetAmount])
119160 let lastClaimedHeight = parseIntValue(origArray[IdxInvLastClaimedHeight])
120161 let newTotalAmount = (totalAmount + incrementDiff[IdxDiffTotalIncrement])
121162 let newRemainingAmount = (remainingAmount + incrementDiff[IdxDiffRemainingPriceAmountIncrement])
122- let newClaimedPriceAssetAmount = (claimedPriceAssetAmount + incrementDiff[IdxDiffClaimedPriceAmountIncrement])
123- let newClaimedIdoAssetAmount = (claimedIdoAssetAmount + incrementDiff[IdxDiffClaimedIdoAssetAmountIncrement])
163+ let cfgArray = readConfigArray()
164+ let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
165+ let priceAssetDecimals = value(assetInfo(fromBase58String(priceAssetId58))).decimals
166+ let priceAssetBalancePriceAssetDecimals = fraction(priceAssetBalance, scale8, pow(10, 0, priceAssetDecimals, 0, 0, DOWN))
167+ let newClaimedPriceAssetAmount = ((claimedPriceAssetAmount + incrementDiff[IdxDiffClaimedPriceAmountIncrement]) - priceAssetBalance)
168+ let newClaimedIdoAssetAmount = ((claimedIdoAssetAmount + incrementDiff[IdxDiffClaimedIdoAssetAmountIncrement]) + priceAssetBalancePriceAssetDecimals)
124169 if ((0 > newRemainingAmount))
125170 then throw("invalid math")
126171 else StringEntry(key, formatInvestor(newTotalAmount, newRemainingAmount, newClaimedPriceAssetAmount, newClaimedIdoAssetAmount, newLastClaimedHeight))
127172 }
128173
129174
130175 func InvestOperationHistoryEntry (userAddress,priceAssetAmount,idoAssetAmount,txId) = StringEntry(keyOperationHistoryRecord("invest", userAddress, toBase58String(txId)), formatHistoryRecord(priceAssetAmount, idoAssetAmount))
131176
132177
133178 func ClaimOperationHistoryEntry (userAddress,priceAssetAmount,idoAssetAmount,txId) = StringEntry(keyOperationHistoryRecord("claim", userAddress, toBase58String(txId)), formatHistoryRecord(priceAssetAmount, idoAssetAmount))
134179
135180
136181 func internalClaim (claimedAssetId58,userAddress,txId) = {
137182 let cfgArray = readConfigArray()
138183 let claimStart = parseIntValue(cfgArray[IdxCfgClaimStart])
139184 let claimDuration = parseIntValue(cfgArray[IdxCfgClaimDuration])
140185 let claimEnd = (claimStart + claimDuration)
141186 let price = parseIntValue(cfgArray[IdxCfgPrice])
142187 let priceMult = parseIntValue(cfgArray[IdxCfgPriceMult])
143188 let idoAssetId58 = cfgArray[IdxCfgIdoAssetId]
144189 let idoAssetId = fromBase58String(idoAssetId58)
145190 let idoAssetMult = parseIntValue(cfgArray[IdxCfgIdoAssetMult])
146191 let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
147192 let priceAssetId = fromBase58String(priceAssetId58)
148193 let priceAssetMult = parseIntValue(cfgArray[IdxCfgPriceAssetMult])
149194 let userAddress58 = toString(userAddress)
150195 let origInvestArray = readInvestorArrayOrFail(userAddress58)
151196 let investTotalAmount = parseIntValue(origInvestArray[IdxInvTotalAmount])
152197 let investLastClaimedHeightTMP = parseIntValue(origInvestArray[IdxInvLastClaimedHeight])
153198 let investLastClaimedHeight = if ((claimStart >= investLastClaimedHeightTMP))
154199 then claimStart
155200 else investLastClaimedHeightTMP
156201 let newClaimPeriodHeight = if ((height > claimEnd))
157202 then claimEnd
158203 else if ((claimStart > height))
159204 then claimStart
160205 else height
161206 let claimingBlocks = (newClaimPeriodHeight - investLastClaimedHeight)
162207 let claimingPriceAssetAmount = fraction(investTotalAmount, claimingBlocks, claimDuration)
163208 let claimingIdoAssetAmount = convertPriceAssetIntoIdoAsset(claimingPriceAssetAmount, priceAssetMult, price, priceMult, idoAssetMult)
164209 let isUSDN = (claimedAssetId58 == priceAssetId58)
165210 let isUSDNClaimDisabled = valueOrElse(getBoolean(keyUSDNClaimDisabled()), false)
166- let isUSDNClaimEnded = match getInteger(keyUSDNClaimEndHeight()) {
167- case end: Int =>
168- (height > end)
169- case _: Unit =>
170- false
171- case _ =>
172- throw("Match error")
173- }
174211 let checks = [if (!(if (isUSDN)
175212 then isUSDNClaimDisabled
176213 else false))
177214 then true
178- else throw("USDN claim is disabled"), if (!(if (isUSDN)
179- then isUSDNClaimEnded
180- else false))
181- then true
182- else throw("USDN claim is ended")]
215+ else throw("USDN claim is disabled")]
183216 if ((checks == checks))
184217 then if ((claimedAssetId58 == priceAssetId58))
185218 then $Tuple6([0, -(claimingPriceAssetAmount), claimingPriceAssetAmount, 0], claimingPriceAssetAmount, priceAssetId, origInvestArray, newClaimPeriodHeight, [claimingPriceAssetAmount, claimingIdoAssetAmount])
186219 else if ((claimedAssetId58 == idoAssetId58))
187220 then $Tuple6([0, -(claimingPriceAssetAmount), 0, claimingIdoAssetAmount], claimingIdoAssetAmount, idoAssetId, origInvestArray, newClaimPeriodHeight, [claimingPriceAssetAmount, claimingIdoAssetAmount])
188221 else throw(("unsupported assetId: " + claimedAssetId58))
222+ else throw("Strict value is not equal to itself.")
223+ }
224+
225+
226+func internalClaimV2 (priceAssetId58,userAddress58,outAmount,totalUserAvailableToClaim) = {
227+ let totalPeriodPriceAssetAllowance = value(getInteger(keyTotalPeriodAllowance(priceAssetId58)))
228+ let userPeriodPriceAssetAllowance = value(getInteger(keyUserPeriodAllowance(priceAssetId58)))
229+ let periodLength = value(getInteger(keyPeriodLength()))
230+ let currentPeriod = valueOrElse(getInteger(keyCurrentPeriod()), 0)
231+ let zeroPeriodEndHeighIsDefined = isDefined(getInteger(keyPeriodEndHeight(0)))
232+ let $t01224314113 = if ((currentPeriod > 0))
233+ then {
234+ let lastPeriodStartHeight = value(getInteger(keyPeriodStartHeight(currentPeriod)))
235+ let lastPeriodEndHeight = value(getInteger(keyPeriodEndHeight(currentPeriod)))
236+ let $t01250313208 = if ((height > lastPeriodEndHeight))
237+ then {
238+ let updatedCurrentPeriod = (currentPeriod + 1)
239+ let periodStart = if ((height > (lastPeriodEndHeight + periodLength)))
240+ then {
241+ let blocksToLastPeriodStart = ((height - lastPeriodEndHeight) % periodLength)
242+ if ((blocksToLastPeriodStart == 0))
243+ then ((height - periodLength) + 1)
244+ else (height - blocksToLastPeriodStart)
245+ }
246+ else (lastPeriodEndHeight + 1)
247+ let periodEnd = ((periodStart + periodLength) - 1)
248+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
249+ }
250+ else $Tuple3(currentPeriod, lastPeriodStartHeight, lastPeriodEndHeight)
251+ let updatedCurrentPeriod = $t01250313208._1
252+ let periodStart = $t01250313208._2
253+ let periodEnd = $t01250313208._3
254+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
255+ }
256+ else if (zeroPeriodEndHeighIsDefined)
257+ then {
258+ let zeroPeriodStartHeight = value(getInteger(keyPeriodStartHeight(0)))
259+ let zeroPeriodEndHeight = value(getInteger(keyPeriodEndHeight(0)))
260+ let $t01346313845 = if ((height > zeroPeriodEndHeight))
261+ then {
262+ let updatedCurrentPeriod = (currentPeriod + 1)
263+ let periodStart = (zeroPeriodEndHeight + 1)
264+ let periodEnd = ((periodStart + periodLength) - 1)
265+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
266+ }
267+ else $Tuple3(currentPeriod, zeroPeriodStartHeight, zeroPeriodEndHeight)
268+ let updatedCurrentPeriod = $t01346313845._1
269+ let periodStart = $t01346313845._2
270+ let periodEnd = $t01346313845._3
271+ $Tuple3(updatedCurrentPeriod, periodStart, periodEnd)
272+ }
273+ else $Tuple3(currentPeriod, valueOrElse(getInteger(keyPeriodStartHeight(currentPeriod)), height), valueOrElse(getInteger(keyPeriodEndHeight(currentPeriod)), ((height + periodLength) - 1)))
274+ let updatedCurrentPeriod = $t01224314113._1
275+ let periodStart = $t01224314113._2
276+ let periodEnd = $t01224314113._3
277+ let periodTotalAvailableToClaim = valueOrElse(getInteger(keyPeriodTotalAvailableToClaim(priceAssetId58, updatedCurrentPeriod)), totalPeriodPriceAssetAllowance)
278+ let periodUserAvailableToClaim = valueOrElse(getInteger(keyPeriodUserAvailableToClaim(priceAssetId58, updatedCurrentPeriod, userAddress58)), userPeriodPriceAssetAllowance)
279+ let priceAssetBalance = valueOrElse(getInteger(keyPriceAssetBalance(userAddress58)), 0)
280+ let periodMinAvailableToClaim = min([(outAmount + priceAssetBalance), periodTotalAvailableToClaim, periodUserAvailableToClaim])
281+ let usdtPriceAssetAllowableRatio = value(getInteger(keyUsdtPriceAssetAllowableRatio()))
282+ let putOneTknV2PriceAssetAmount = scale8
283+ let $t01486815121 = {
284+ let @ = invoke(addressFromStringValue(value(getString(keyUsdtPriceAssetStablePool()))), "putOneTknV2WithoutTakeFeeREADONLY", [putOneTknV2PriceAssetAmount, priceAssetId58], nil)
285+ if ($isInstanceOf(@, "(Int, Int, Int)"))
286+ then @
287+ else throw("Couldn't cast Any to (Int, Int, Int)")
288+ }
289+ if (($t01486815121 == $t01486815121))
290+ then {
291+ let bonus = $t01486815121._3
292+ let feeAmount = $t01486815121._2
293+ let lpAmount = $t01486815121._1
294+ let usdtAssetId = value(getString(keyUsdtAssetId()))
295+ let $t01518315401 = {
296+ let @ = invoke(addressFromStringValue(value(getString(keyUsdtPriceAssetStablePool()))), "getOneTknV2READONLY", [usdtAssetId, lpAmount], nil)
297+ if ($isInstanceOf(@, "(Int, Int)"))
298+ then @
299+ else throw("Couldn't cast Any to (Int, Int)")
300+ }
301+ if (($t01518315401 == $t01518315401))
302+ then {
303+ let getOneTknV2FeeAmount = $t01518315401._2
304+ let usdtAmount = $t01518315401._1
305+ let currentUsdtPriceAssetRatio = fraction(putOneTknV2PriceAssetAmount, scale8, usdtAmount)
306+ let endPeriodBlocksLeft = (periodEnd - height)
307+ $Tuple10(periodMinAvailableToClaim, periodTotalAvailableToClaim, periodUserAvailableToClaim, totalUserAvailableToClaim, usdtPriceAssetAllowableRatio, currentUsdtPriceAssetRatio, endPeriodBlocksLeft, updatedCurrentPeriod, periodStart, periodEnd)
308+ }
309+ else throw("Strict value is not equal to itself.")
310+ }
189311 else throw("Strict value is not equal to itself.")
190312 }
191313
192314
193315 func managerPublicKeyOrUnit () = match getString(keyManagerPublicKey()) {
194316 case s: String =>
195317 fromBase58String(s)
196318 case _: Unit =>
197319 unit
198320 case _ =>
199321 throw("Match error")
200322 }
201323
202324
203325 func pendingManagerPublicKeyOrUnit () = match getString(keyPendingManagerPublicKey()) {
204326 case s: String =>
205327 fromBase58String(s)
206328 case _: Unit =>
207329 unit
208330 case _ =>
209331 throw("Match error")
210332 }
211333
212334
213335 func mustManager (i) = {
214336 let pd = throw("Permission denied")
215337 match managerPublicKeyOrUnit() {
216338 case pk: ByteVector =>
217339 if ((i.callerPublicKey == pk))
218340 then true
219341 else pd
220342 case _: Unit =>
221343 if ((i.caller == this))
222344 then true
223345 else pd
224346 case _ =>
225347 throw("Match error")
226348 }
227349 }
228350
229351
230352 @Callable(i)
231353 func constructor (idoStart,idoDuration,claimStart,claimDuration,price,priceAssetId58,minInvestAmount) = {
232354 let priceMult = ((100 * 1000) * 1000)
233355 let idoEnd = (idoStart + idoDuration)
234356 if (isDefined(getString(keyConfig())))
235357 then throw("already initialized")
236358 else if (("3PMEHLx1j6zerarZTYfsGqDeeZqQoMpxq5S" != toString(i.caller)))
237359 then throw("not authorized")
238360 else if ((size(i.payments) != 1))
239361 then throw("exactly 1 payment must be attached")
240362 else if ((idoEnd >= claimStart))
241363 then throw("claimStart must be greater than idoEnd")
242364 else {
243365 let pmt = value(i.payments[0])
244366 let idoAssetId = value(pmt.assetId)
245367 let idoAssetInfo = valueOrErrorMessage(assetInfo(idoAssetId), "fail to load ido asset info")
246368 let idoAssetId58 = toBase58String(idoAssetId)
247369 let idoAssetMult = pow(10, 0, idoAssetInfo.decimals, 0, 0, DOWN)
248370 let priceAssetId = fromBase58String(priceAssetId58)
249371 let priceAssetInfo = valueOrErrorMessage(assetInfo(priceAssetId), "fail to load price asset info")
250372 let priceAssetMult = pow(10, 0, priceAssetInfo.decimals, 0, 0, DOWN)
251373 let origTotalsArray = readTotalsArrayOrDefault()
252374 let totalsDiff = [0, 0, 0, 0]
253-[StringEntry(keyConfig(), fromatConfig(idoStart, idoDuration, claimStart, claimDuration, price, priceMult, idoAssetId58, idoAssetMult, priceAssetId58, priceAssetMult, minInvestAmount, pmt.amount)), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart)]
375+[StringEntry(keyConfig(), fromatConfig(idoStart, idoDuration, claimStart, claimDuration, price, priceMult, idoAssetId58, idoAssetMult, priceAssetId58, priceAssetMult, minInvestAmount, pmt.amount)), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart, 0)]
254376 }
255377 }
256378
257379
258380
259381 @Callable(i)
260382 func invest () = {
261383 let cfgArray = readConfigArray()
262384 let idoStart = parseIntValue(cfgArray[IdxCfgIdoStart])
263385 let idoDuration = parseIntValue(cfgArray[IdxCfgIdoDuration])
264386 let idoEnd = (idoStart + idoDuration)
265387 let claimStart = parseIntValue(cfgArray[IdxCfgClaimStart])
266388 let claimDuration = parseIntValue(cfgArray[IdxCfgClaimDuration])
267389 let price = parseIntValue(cfgArray[IdxCfgPrice])
268390 let priceMult = parseIntValue(cfgArray[IdxCfgPriceMult])
269391 let idoAssetId58 = cfgArray[IdxCfgIdoAssetId]
270392 let idoAssetId = fromBase58String(idoAssetId58)
271393 let idoAssetMult = parseIntValue(cfgArray[IdxCfgIdoAssetMult])
272394 let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
273395 let priceAssetId = fromBase58String(priceAssetId58)
274396 let priceAssetMult = parseIntValue(cfgArray[IdxCfgPriceAssetMult])
275397 let minIvestAmount = parseIntValue(cfgArray[IdxCfgMinInvestAmount])
276398 let userAddress = toString(i.caller)
277399 if ((idoStart > height))
278400 then throw("ido has not been started yet")
279401 else if ((height > idoEnd))
280402 then throw("ido has been already ended")
281403 else if ((size(i.payments) != 1))
282404 then throw("exactly 1 payment is expected")
283405 else {
284406 let pmt = value(i.payments[0])
285407 let pmtAssetId = value(pmt.assetId)
286408 let pmtAmount = pmt.amount
287409 if ((pmtAssetId != priceAssetId))
288410 then throw((("invalid payment asset id: " + toBase58String(pmtAssetId)) + " is expected"))
289411 else {
290412 let origInvestorArray = readInvestorArrayOrDefault(userAddress)
291413 let origTotalsArray = readTotalsArrayOrDefault()
292414 let newPriceTotalAmount = (parseIntValue(origTotalsArray[IdxInvTotalAmount]) + pmtAmount)
293415 let requiredIdoAssetAmount = (newPriceTotalAmount * 100)
294416 if ((requiredIdoAssetAmount > assetBalance(this, idoAssetId)))
295417 then throw("IDO asset has been - sold consider to use smaller payment")
296418 else {
297419 let totalsDiff = [pmtAmount, pmtAmount, 0, 0]
298-[TotalsEntry(keyInvestor(userAddress), origInvestorArray, totalsDiff, claimStart), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart), InvestOperationHistoryEntry(userAddress, pmtAmount, 0, i.transactionId)]
420+[TotalsEntry(keyInvestor(userAddress), origInvestorArray, totalsDiff, claimStart, 0), TotalsEntry(keyTotals(), origTotalsArray, totalsDiff, claimStart, 0), InvestOperationHistoryEntry(userAddress, pmtAmount, 0, i.transactionId)]
299421 }
300422 }
301423 }
302424 }
303425
304426
305427
306428 @Callable(i)
307429 func claim (claimedAssetId58,userAddress58) = {
308430 let callerAddress58 = toString(i.caller)
309431 if ((userAddress58 != callerAddress58))
310432 then throw("not authorized")
311433 else {
434+ let cfgArray = readConfigArray()
435+ let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
312436 let claimResultTuple = internalClaim(claimedAssetId58, i.caller, i.transactionId)
313437 let totalsDiff = claimResultTuple._1
314438 let outAmount = claimResultTuple._2
315439 let outAssetId = claimResultTuple._3
316440 let origInvestArray = claimResultTuple._4
317441 let newClaimPeriodHeight = claimResultTuple._5
318442 let claimedPriceAmountFromDiff = totalsDiff[IdxDiffClaimedPriceAmountIncrement]
319443 let claimedIdoAssetAmountFromDiff = totalsDiff[IdxDiffClaimedIdoAssetAmountIncrement]
320- $Tuple2([ScriptTransfer(i.caller, outAmount, outAssetId), TotalsEntry(keyInvestor(userAddress58), origInvestArray, totalsDiff, newClaimPeriodHeight), TotalsEntry(keyTotals(), readTotalsArrayOrDefault(), totalsDiff, newClaimPeriodHeight), ClaimOperationHistoryEntry(userAddress58, claimedPriceAmountFromDiff, claimedIdoAssetAmountFromDiff, i.transactionId)], unit)
444+ let priceAssetBalance = valueOrElse(getInteger(keyPriceAssetBalance(userAddress58)), 0)
445+ let priceAssetDecimals = value(assetInfo(fromBase58String(priceAssetId58))).decimals
446+ let entries = if ((claimedAssetId58 == priceAssetId58))
447+ then {
448+ let $t02124821692 = internalClaimV2(priceAssetId58, userAddress58, outAmount, totalsDiff[IdxDiffClaimedPriceAmountIncrement])
449+ let periodMinAvailableToClaim = $t02124821692._1
450+ let periodTotalAvailableToClaim = $t02124821692._2
451+ let periodUserAvailableToClaim = $t02124821692._3
452+ let totalUserAvailableToClaim = $t02124821692._4
453+ let usdtPriceAssetAllowableRatio = $t02124821692._5
454+ let currentUsdtPriceAssetRatio = $t02124821692._6
455+ let endPeriodBlocksLeft = $t02124821692._7
456+ let updatedCurrentPeriod = $t02124821692._8
457+ let periodStart = $t02124821692._9
458+ let periodEnd = $t02124821692._10
459+ let checks = [if ((periodUserAvailableToClaim > 0))
460+ then true
461+ else throwErr("unavailable to claim because user period allowance reached"), if ((periodTotalAvailableToClaim > 0))
462+ then true
463+ else throwErr("unavailable to claim because total period allowance reached"), if ((periodMinAvailableToClaim > 0))
464+ then true
465+ else throwErr("nothing to claim"), if ((usdtPriceAssetAllowableRatio > currentUsdtPriceAssetRatio))
466+ then true
467+ else throwErr("unavailable to claim because usdn price lower than usdtPriceAssetAllowableRatio")]
468+ if ((checks == checks))
469+ then {
470+ let updatedPeriodTotalAvailableToClaim = (periodTotalAvailableToClaim - periodMinAvailableToClaim)
471+ let updatedPeriodUserAvailableToClaim = (periodUserAvailableToClaim - periodMinAvailableToClaim)
472+ let entries = if ((priceAssetBalance >= periodMinAvailableToClaim))
473+ then [ScriptTransfer(i.caller, periodMinAvailableToClaim, outAssetId), IntegerEntry(keyPriceAssetBalance(userAddress58), (priceAssetBalance - periodMinAvailableToClaim))]
474+ else {
475+ let updatedPriceAssetBalance = ((priceAssetBalance + outAmount) - periodMinAvailableToClaim)
476+[ScriptTransfer(i.caller, periodMinAvailableToClaim, outAssetId), TotalsEntry(keyInvestor(userAddress58), origInvestArray, totalsDiff, newClaimPeriodHeight, 0), TotalsEntry(keyTotals(), readTotalsArrayOrDefault(), totalsDiff, newClaimPeriodHeight, 0), IntegerEntry(keyPriceAssetBalance(userAddress58), updatedPriceAssetBalance)]
477+ }
478+ $Tuple2(([IntegerEntry(keyCurrentPeriod(), updatedCurrentPeriod), IntegerEntry(keyPeriodStartHeight(updatedCurrentPeriod), periodStart), IntegerEntry(keyPeriodEndHeight(updatedCurrentPeriod), periodEnd), IntegerEntry(keyPeriodTotalAvailableToClaim(priceAssetId58, updatedCurrentPeriod), updatedPeriodTotalAvailableToClaim), IntegerEntry(keyPeriodUserAvailableToClaim(priceAssetId58, updatedCurrentPeriod, userAddress58), updatedPeriodUserAvailableToClaim), ClaimOperationHistoryEntry(userAddress58, periodMinAvailableToClaim, claimedIdoAssetAmountFromDiff, i.transactionId)] ++ entries), unit)
479+ }
480+ else throw("Strict value is not equal to itself.")
481+ }
482+ else {
483+ let priceAssetBalanceIdoDecimals = fraction(priceAssetBalance, scale8, pow(10, 0, priceAssetDecimals, 0, 0, DOWN))
484+ $Tuple2([ScriptTransfer(i.caller, (outAmount + priceAssetBalanceIdoDecimals), outAssetId), IntegerEntry(keyPriceAssetBalance(userAddress58), 0), TotalsEntry(keyInvestor(userAddress58), origInvestArray, totalsDiff, newClaimPeriodHeight, priceAssetBalance), TotalsEntry(keyTotals(), readTotalsArrayOrDefault(), totalsDiff, newClaimPeriodHeight, priceAssetBalance), ClaimOperationHistoryEntry(userAddress58, claimedPriceAmountFromDiff, (claimedIdoAssetAmountFromDiff + priceAssetBalanceIdoDecimals), i.transactionId)], unit)
485+ }
486+ entries
321487 }
322488 }
323489
324490
325491
326492 @Callable(i)
327493 func claimREADONLY (claimedAssetId58,userAddress58) = {
328494 let claimResultTuple = internalClaim(claimedAssetId58, addressFromStringValue(userAddress58), fromBase58String(""))
329495 let totalsDiff = claimResultTuple._1
330496 let outAmount = claimResultTuple._2
331497 let outAssetId = claimResultTuple._3
332498 let origInvestArray = claimResultTuple._4
333499 let newClaimPeriodHeight = claimResultTuple._5
334500 let availableToClaimArray = claimResultTuple._6
335501 let availablePriceAmountToClaim = availableToClaimArray[0]
336502 let availableIdoAmountToClaim = availableToClaimArray[1]
337503 $Tuple2(nil, makeString(["%s%d%d", userAddress58, toString(availablePriceAmountToClaim), toString(availableIdoAmountToClaim)], SEP))
504+ }
505+
506+
507+
508+@Callable(i)
509+func claimV2READONLY (claimedAssetId58,userAddress58) = {
510+ let claimResultTuple = internalClaim(claimedAssetId58, addressFromStringValue(userAddress58), fromBase58String(""))
511+ let totalsDiff = claimResultTuple._1
512+ let outAmount = claimResultTuple._2
513+ let outAssetId = claimResultTuple._3
514+ let origInvestArray = claimResultTuple._4
515+ let newClaimPeriodHeight = claimResultTuple._5
516+ let availableToClaimArray = claimResultTuple._6
517+ let availablePriceAmountToClaim = availableToClaimArray[0]
518+ let availableIdoAmountToClaim = availableToClaimArray[1]
519+ let cfgArray = readConfigArray()
520+ let priceAssetId58 = cfgArray[IdxCfgPriceAssetId]
521+ let priceAssetBalance = valueOrElse(getInteger(keyPriceAssetBalance(userAddress58)), 0)
522+ let priceAssetBalanceIdoDecimals = (priceAssetBalance * 100)
523+ let availableIdoAmountToClaimWithPriceAssetBalance = (availableIdoAmountToClaim + priceAssetBalanceIdoDecimals)
524+ let $t02798528397 = internalClaimV2(priceAssetId58, userAddress58, outAmount, totalsDiff[IdxDiffClaimedPriceAmountIncrement])
525+ let periodMinAvailableToClaim = $t02798528397._1
526+ let periodTotalAvailableToClaim = $t02798528397._2
527+ let periodUserAvailableToClaim = $t02798528397._3
528+ let totalUserAvailableToClaim = $t02798528397._4
529+ let usdtPriceAssetAllowableRatio = $t02798528397._5
530+ let currentUsdtPriceAssetRatio = $t02798528397._6
531+ let endPeriodBlocksLeft = $t02798528397._7
532+ let updatedCurrentPeriod = $t02798528397._8
533+ let periodStart = $t02798528397._9
534+ let periodEnd = $t02798528397._10
535+ let currentPeriodEndHeight = valueOrElse(getInteger(keyPeriodEndHeight(updatedCurrentPeriod)), 0)
536+ let userTotalPriceAssetClaimed = (parseIntValue(readInvestorArrayOrFail(userAddress58)[IdxInvClaimedPriceAssetAmount]) - priceAssetBalance)
537+ let resultString = if ((height > currentPeriodEndHeight))
538+ then {
539+ let periodLenght = value(getInteger(keyPeriodLength()))
540+ let userPeriodAllowance = value(getInteger(keyUserPeriodAllowance(priceAssetId58)))
541+ let totalPeriodAllowance = value(getInteger(keyTotalPeriodAllowance(priceAssetId58)))
542+ makeString(["%d%d%d%d%d%d", toString(availableIdoAmountToClaimWithPriceAssetBalance), toString(userPeriodAllowance), toString(totalPeriodAllowance), toString(usdtPriceAssetAllowableRatio), toString(currentUsdtPriceAssetRatio), toString(periodLenght), toString(userTotalPriceAssetClaimed)], SEP)
543+ }
544+ else makeString(["%d%d%d%d%d%d", toString(availableIdoAmountToClaimWithPriceAssetBalance), toString(periodMinAvailableToClaim), toString(periodTotalAvailableToClaim), toString(usdtPriceAssetAllowableRatio), toString(currentUsdtPriceAssetRatio), toString(endPeriodBlocksLeft), toString(userTotalPriceAssetClaimed)], SEP)
545+ $Tuple2(nil, resultString)
338546 }
339547
340548
341549
342550 @Callable(i)
343551 func setManager (pendingManagerPublicKey) = {
344552 let checkCaller = mustManager(i)
345553 if ((checkCaller == checkCaller))
346554 then {
347555 let checkManagerPublicKey = fromBase58String(pendingManagerPublicKey)
348556 if ((checkManagerPublicKey == checkManagerPublicKey))
349557 then [StringEntry(keyPendingManagerPublicKey(), pendingManagerPublicKey)]
350558 else throw("Strict value is not equal to itself.")
351559 }
352560 else throw("Strict value is not equal to itself.")
353561 }
354562
355563
356564
357565 @Callable(i)
358566 func confirmManager () = {
359567 let pm = pendingManagerPublicKeyOrUnit()
360568 let hasPM = if (isDefined(pm))
361569 then true
362570 else throw("No pending manager")
363571 if ((hasPM == hasPM))
364572 then {
365573 let checkPM = if ((i.callerPublicKey == value(pm)))
366574 then true
367575 else throw("You are not pending manager")
368576 if ((checkPM == checkPM))
369577 then [StringEntry(keyManagerPublicKey(), toBase58String(value(pm))), DeleteEntry(keyPendingManagerPublicKey())]
370578 else throw("Strict value is not equal to itself.")
371579 }
372580 else throw("Strict value is not equal to itself.")
373581 }
374582
375583
376584 @Verifier(tx)
377585 func verify () = {
378586 let targetPublicKey = match managerPublicKeyOrUnit() {
379587 case pk: ByteVector =>
380588 pk
381589 case _: Unit =>
382590 tx.senderPublicKey
383591 case _ =>
384592 throw("Match error")
385593 }
386594 sigVerify(tx.bodyBytes, tx.proofs[0], targetPublicKey)
387595 }
388596

github/deemru/w8io/786bc32 
91.30 ms