tx · 8hraZNZWs8RcDguguFv4reGvojcGJg3AfgCgkfoMqnoR

3N5yarEiTQccnnuerogYT3BxM5Zc5bRgDZy:  -0.04000000 Waves

2022.08.09 20:03 [2177196] smart account 3N5yarEiTQccnnuerogYT3BxM5Zc5bRgDZy > SELF 0.00000000 Waves

{ "type": 13, "id": "8hraZNZWs8RcDguguFv4reGvojcGJg3AfgCgkfoMqnoR", "fee": 4000000, "feeAssetId": null, "timestamp": 1660064651387, "version": 1, "sender": "3N5yarEiTQccnnuerogYT3BxM5Zc5bRgDZy", "senderPublicKey": "FK9j3YFWVChXKESTn33fMZz1JseCNMMqHhfB7smxJo4e", "proofs": [ "48nu2R95fp6PtbrY3f8oLxPSWBo8VJdhnLK9oBcXdU7RC8ae8P1VMBxLcxUk4S8EserMiDTRMNxLhnu7Y5WGMw6Q" ], "script": "base64:BgIqCAISBQoDAQgIEgASABIDCgEBEgASABIAEgMKAQgSBAoCCAESAwoBARIAUAAJc2VwYXJhdG9yAgJfXwADU0VQAgJfXwAFTVVMVDYAwIQ9AAVNVUxUOACAwtcvAAZNVUxUWDYJALYCAQUFTVVMVDYABk1VTFRYOAkAtgIBBQVNVUxUOAAHTVVMVFgxOAkAtgIBAICAkLu61q3wDQAKV0FWRVNJRFNUUgIFV0FWRVMAB1dBVkVTSUQJANkEAQUKV0FWRVNJRFNUUgAZSWR4Q29udHJvbENmZ05ldXRyaW5vRGFwcAABABhJZHhDb250cm9sQ2ZnQXVjdGlvbkRhcHAAAgAUSWR4Q29udHJvbENmZ1JwZERhcHAAAwAVSWR4Q29udHJvbENmZ01hdGhEYXBwAAQAHElkeENvbnRyb2xDZmdMaXF1aWRhdGlvbkRhcHAABQAVSWR4Q29udHJvbENmZ1Jlc3REYXBwAAYAHUlkeENvbnRyb2xDZmdOb2RlUmVnaXN0cnlEYXBwAAcAHElkeENvbnRyb2xDZmdOc2J0U3Rha2luZ0RhcHAACAAZSWR4Q29udHJvbENmZ01lZGlhdG9yRGFwcAAJABxJZHhDb250cm9sQ2ZnU3VyZlN0YWtpbmdEYXBwAAoAIElkeENvbnRyb2xDZmdHbnNidENvbnRyb2xsZXJEYXBwAAsBD2dldFN0cmluZ09yRmFpbAIHYWRkcmVzcwNrZXkJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQdhZGRyZXNzBQNrZXkJAKwCAgkArAICCQCsAgIJAKwCAgIKbWFuZGF0b3J5IAkApQgBBQdhZGRyZXNzAgEuBQNrZXkCDyBpcyBub3QgZGVmaW5lZAERa2V5Q29udHJvbEFkZHJlc3MAAhwlcyVzX19jb25maWdfX2NvbnRyb2xBZGRyZXNzAQ1rZXlDb250cm9sQ2ZnAAIRJXNfX2NvbnRyb2xDb25maWcBFWtleUduc2J0RnJvbVN1cmZDb2VmZgACHSVzJXNfX2NmZ19fZ25zYnRGcm9tU3VyZkNvZWZmARRyZWFkQ29udHJvbENmZ09yRmFpbAEHY29udHJvbAkAtQkCCQEPZ2V0U3RyaW5nT3JGYWlsAgUHY29udHJvbAkBDWtleUNvbnRyb2xDZmcABQNTRVABGGdldENvbnRyYWN0QWRkcmVzc09yRmFpbAIKY29udHJvbENmZwNpZHgJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBCQCRAwIFCmNvbnRyb2xDZmcFA2lkeAkArAICAi1Db250cm9sIGNmZyBkb2Vzbid0IGNvbnRhaW4gYWRkcmVzcyBhdCBpbmRleCAJAKQDAQUDaWR4AA9jb250cm9sQ29udHJhY3QJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwkBEWtleUNvbnRyb2xBZGRyZXNzAAIjM040TlM3ZDRKbzlhNkYxNExpRlVLS1lWZFVra2YyZVA0WngACmNvbnRyb2xDZmcJARRyZWFkQ29udHJvbENmZ09yRmFpbAEFD2NvbnRyb2xDb250cmFjdAAMbWF0aENvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUVSWR4Q29udHJvbENmZ01hdGhEYXBwABBuZXV0cmlub0NvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUZSWR4Q29udHJvbENmZ05ldXRyaW5vRGFwcAAPYXVjdGlvbkNvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUYSWR4Q29udHJvbENmZ0F1Y3Rpb25EYXBwABJnbnNidEZyb21TdXJmQ29lZmYJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBFWtleUduc2J0RnJvbVN1cmZDb2VmZgAArAIBDGtleUJvbmRBc3NldAACDWJvbmRfYXNzZXRfaWQBGWtleUF1Y3Rpb25Db250cmFjdEFkZHJlc3MAAhBhdWN0aW9uX2NvbnRyYWN0ARBrZXlNaW5Mb2NrQW1vdW50AAIRJXNfX21pbkxvY2tBbW91bnQBEGtleVN0YWtlZEFzc2V0SWQAAhElc19fc3Rha2VkQXNzZXRJZAEWa2V5TG9ja1BhcmFtVXNlckFtb3VudAELdXNlckFkZHJlc3MJALkJAgkAzAgCAgYlcyVzJXMJAMwIAgILcGFyYW1CeVVzZXIJAMwIAgkApQgBBQt1c2VyQWRkcmVzcwkAzAgCAgZhbW91bnQFA25pbAUJc2VwYXJhdG9yARZrZXlMb2NrUGFyYW1TdGFydEJsb2NrAQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAgtwYXJhbUJ5VXNlcgkAzAgCCQClCAEFC3VzZXJBZGRyZXNzCQDMCAICBXN0YXJ0BQNuaWwFCXNlcGFyYXRvcgEma2V5TG9ja1BhcmFtVm90aW5nUG93ZXJFZmZlY3RpdmVIZWlnaHQBC3VzZXJBZGRyZXNzCQC5CQIJAMwIAgIGJXMlcyVzCQDMCAICC3BhcmFtQnlVc2VyCQDMCAIJAKUIAQULdXNlckFkZHJlc3MJAMwIAgIRdnBFZmZlY3RpdmVIZWlnaHQFA25pbAUJc2VwYXJhdG9yARBrZXlIaXN0b3J5UmVjb3JkAwR0eXBlC3VzZXJBZGRyZXNzBHR4SWQJALkJAgkAzAgCAgglcyVzJXMlcwkAzAgCAgdoaXN0b3J5CQDMCAIFBHR5cGUJAMwIAgkApQgBBQt1c2VyQWRkcmVzcwkAzAgCCQDYBAEFBHR4SWQFA25pbAUJc2VwYXJhdG9yARdrZXlMb2NrUGFyYW1Ub3RhbEFtb3VudAAJALkJAgkAzAgCAgQlcyVzCQDMCAICBXN0YXRzCQDMCAICEWFjdGl2ZVRvdGFsTG9ja2VkBQNuaWwFCXNlcGFyYXRvcgESa2V5U3RhdHNMb2Nrc0NvdW50AAkAuQkCCQDMCAICBCVzJXMJAMwIAgIFc3RhdHMJAMwIAgIKbG9ja3NDb3VudAUDbmlsBQlzZXBhcmF0b3IBEmtleVN0YXRzVXNlcnNDb3VudAAJALkJAgkAzAgCAgQlcyVzCQDMCAICBXN0YXRzCQDMCAICEGFjdGl2ZVVzZXJzQ291bnQFA25pbAUJc2VwYXJhdG9yAQ1rZXlOZXh0UGVyaW9kAAIOJXNfX25leHRQZXJpb2QBGGtleVN1cHBvcnRlZFJld2FyZEFzc2V0cwACFXN1cHBvcnRlZFJld2FyZEFzc2V0cwERa2V5RGVwb3NpdE51bUxhc3QACQC5CQIJAMwIAgIGJXMlcyVzCQDMCAICA2RlcAkAzAgCAgdsYXN0TnVtBQNuaWwFCXNlcGFyYXRvcgEba2V5VXNlclJld2FyZEZyb21EZXBvc2l0TnVtAQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAhF1c2VyUndkRnJvbURlcE51bQkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQlzZXBhcmF0b3IBFWtleVJld2FyZFBlck5zYnRTdW1BdAIKZGVwb3NpdE51bQN0a24JALkJAgkAzAgCAgQlcyVkCQDMCAICFXJ3ZFBlck5zYnRTdW1CeURlcE51bQkAzAgCCQCkAwEFCmRlcG9zaXROdW0JAMwIAgUDdGtuBQNuaWwFCXNlcGFyYXRvcgEJa2V5UmV3YXJkAgt1c2VyQWRkcmVzcwN0a24JALkJAgkAzAgCAgYlcyVzJXMJAMwIAgIDcndkCQDMCAIFC3VzZXJBZGRyZXNzCQDMCAIFA3RrbgUDbmlsBQlzZXBhcmF0b3IBCmtleUNsYWltZWQCC3VzZXJBZGRyZXNzA3RrbgkAuQkCCQDMCAICBiVzJXMlcwkAzAgCAgNjbG0JAMwIAgULdXNlckFkZHJlc3MJAMwIAgUDdGtuBQNuaWwFCXNlcGFyYXRvcgEXa2V5Tm90RGlzdHJpYnV0ZWRSZXdhcmQBA3RrbgkAuQkCCQDMCAICBCVzJXMJAMwIAgIObm90RGlzdHJpYnV0ZWQJAMwIAgUDdGtuBQNuaWwFCXNlcGFyYXRvcgEFdG9YMTgCB29yaWdWYWwIb3JpZ011bHQJALwCAwkAtgIBBQdvcmlnVmFsBQdNVUxUWDE4BQhvcmlnTXVsdAEMZ2V0SW50T3JaZXJvAQNrZXkJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUDa2V5AAABDGdldEludE9yRWxzZQIDa2V5CmRlZmF1bHRWYWwJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUDa2V5BQpkZWZhdWx0VmFsAQxnZXRJbnRPckZhaWwBA2tleQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFBHRoaXMFA2tleQkArAICCQCsAgICD01hbmRhdG9yeSB0aGlzLgUDa2V5Ag8gaXMgbm90IGRlZmluZWQBDGdldFN0ck9yRWxzZQIDa2V5CmRlZmF1bHRWYWwJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwUDa2V5BQpkZWZhdWx0VmFsAQ90b0FkZHJlc3NPckZhaWwBCmFkZHJlc3NTdHIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkApggBBQphZGRyZXNzU3RyCQCsAgICIWNvdWxkbid0IHBhcnNlIHBhc3NlZCBhZGRyZXNzU3RyPQUKYWRkcmVzc1N0cgELdG9Bc3NldFZlY3QBCGFzc2V0U3RyAwkAAAIFCGFzc2V0U3RyBQpXQVZFU0lEU1RSBQR1bml0CQDZBAEFCGFzc2V0U3RyAQVhc0ludAEDdmFsBAckbWF0Y2gwBQN2YWwDCQABAgUHJG1hdGNoMAIDSW50BAZ2YWxJbnQFByRtYXRjaDAFBnZhbEludAkAAgECFWZhaWwgdG8gY2FzdCBpbnRvIEludAESYXNTd2FwUGFyYW1zU1RSVUNUAQF2BAckbWF0Y2gwBQF2AwkAAQIFByRtYXRjaDACGShJbnQsIEludCwgSW50LCBJbnQsIEludCkEBnN0cnVjdAUHJG1hdGNoMAUGc3RydWN0CQACAQIVZmFpbCB0byBjYXN0IGludG8gSW50ARNmb3JtYXRIaXN0b3J5UmVjb3JkBAlvbGRBbW91bnQIb2xkU3RhcnQJbmV3QW1vdW50CG5ld1N0YXJ0CQC5CQIJAMwIAgIMJWQlZCVkJWQlZCVkCQDMCAIJAKQDAQgFCWxhc3RCbG9jawZoZWlnaHQJAMwIAgkApAMBCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAkAzAgCCQCkAwEFCW9sZEFtb3VudAkAzAgCCQCkAwEFCG9sZFN0YXJ0CQDMCAIJAKQDAQUJbmV3QW1vdW50CQDMCAIJAKQDAQUIbmV3U3RhcnQFA25pbAUJc2VwYXJhdG9yARhmb3JtYXRDbGFpbUhpc3RvcnlSZWNvcmQCBHVzZXIOY2xhaW1lZFJld2FyZHMJALkJAgkAzAgCAgglcyVkJWQlcwkAzAgCBQR1c2VyCQDMCAIJAKQDAQgFCWxhc3RCbG9jawZoZWlnaHQJAMwIAgkApAMBCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAkAzAgCBQ5jbGFpbWVkUmV3YXJkcwUDbmlsBQlzZXBhcmF0b3IBEkhpc3RvcnlSZWNvcmRFbnRyeQcEdHlwZQt1c2VyQWRkcmVzcwR0eElkCW9sZEFtb3VudAhvbGRTdGFydAluZXdBbW91bnQIbmV3U3RhcnQJAQtTdHJpbmdFbnRyeQIJARBrZXlIaXN0b3J5UmVjb3JkAwUEdHlwZQULdXNlckFkZHJlc3MFBHR4SWQJARNmb3JtYXRIaXN0b3J5UmVjb3JkBAUJb2xkQW1vdW50BQhvbGRTdGFydAUJbmV3QW1vdW50BQhuZXdTdGFydAERQ2xhaW1IaXN0b3J5RW50cnkDC3VzZXJBZGRyZXNzBHR4SWQOY2xhaW1lZFJld2FyZHMJAQtTdHJpbmdFbnRyeQIJARBrZXlIaXN0b3J5UmVjb3JkAwIFY2xhaW0FC3VzZXJBZGRyZXNzBQR0eElkCQEYZm9ybWF0Q2xhaW1IaXN0b3J5UmVjb3JkAgkApQgBBQt1c2VyQWRkcmVzcwUOY2xhaW1lZFJld2FyZHMBC1N0YXRzUmVzdWx0Aw50b3RhbExvY2tlZEluYwxsb2NrQ291bnRJbmMNdXNlcnNDb3VudEluYwQKbG9ja3NDb3VudAkBDGdldEludE9yWmVybwEJARJrZXlTdGF0c0xvY2tzQ291bnQABAp1c2Vyc0NvdW50CQEMZ2V0SW50T3JaZXJvAQkBEmtleVN0YXRzVXNlcnNDb3VudAAEC3RvdGFsQW1vdW50CQEMZ2V0SW50T3JaZXJvAQkBF2tleUxvY2tQYXJhbVRvdGFsQW1vdW50AAQOdG90YWxBbW91bnROZXcJAGQCBQt0b3RhbEFtb3VudAUOdG90YWxMb2NrZWRJbmMJAJUKAwkAzAgCCQEMSW50ZWdlckVudHJ5AgkBEmtleVN0YXRzTG9ja3NDb3VudAAJAGQCBQpsb2Nrc0NvdW50BQxsb2NrQ291bnRJbmMJAMwIAgkBDEludGVnZXJFbnRyeQIJARJrZXlTdGF0c1VzZXJzQ291bnQACQBkAgUKdXNlcnNDb3VudAUNdXNlcnNDb3VudEluYwkAzAgCCQEMSW50ZWdlckVudHJ5AgkBF2tleUxvY2tQYXJhbVRvdGFsQW1vdW50AAUOdG90YWxBbW91bnROZXcFA25pbAULdG90YWxBbW91bnQFDnRvdGFsQW1vdW50TmV3AQ9Mb2NrUGFyYW1zRW50cnkDC3VzZXJBZGRyZXNzBmFtb3VudBp2b3RpbmdQb3dlckVmZmVjdGl2ZUhlaWdodAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBFmtleUxvY2tQYXJhbVVzZXJBbW91bnQBBQt1c2VyQWRkcmVzcwUGYW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEWa2V5TG9ja1BhcmFtU3RhcnRCbG9jawEFC3VzZXJBZGRyZXNzBRp2b3RpbmdQb3dlckVmZmVjdGl2ZUhlaWdodAUDbmlsAQ9nZXRQYXJhbXNPckZhaWwACQCUCgIJANkEAQkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMJARBrZXlTdGFrZWRBc3NldElkAAkBDGdldEludE9yRmFpbAEJARBrZXlNaW5Mb2NrQW1vdW50AAEMaXNBY3RpdmVVc2VyAQt1c2VyQWRkcmVzcwkAZgIJAQxnZXRJbnRPckVsc2UCCQEWa2V5TG9ja1BhcmFtVXNlckFtb3VudAEFC3VzZXJBZGRyZXNzAAAAAAETZ2V0VXNlclBhcmFtc09yVW5pdAELdXNlckFkZHJlc3MDCQEMaXNBY3RpdmVVc2VyAQULdXNlckFkZHJlc3MJAJUKAwcJAQxnZXRJbnRPckZhaWwBCQEWa2V5TG9ja1BhcmFtVXNlckFtb3VudAEFC3VzZXJBZGRyZXNzCQEMZ2V0SW50T3JGYWlsAQkBFmtleUxvY2tQYXJhbVN0YXJ0QmxvY2sBBQt1c2VyQWRkcmVzcwUEdW5pdAETZ2V0VXNlclBhcmFtc09yRmFpbAELdXNlckFkZHJlc3MJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkBE2dldFVzZXJQYXJhbXNPclVuaXQBBQt1c2VyQWRkcmVzcwkArAICCQCsAgICBVVzZXIgCQClCAEFC3VzZXJBZGRyZXNzAg8gaXMgbm90IGRlZmluZWQAEnN1cHBvcnRlZEFzc2V0c1N0cgkBDGdldFN0ck9yRWxzZQIJARhrZXlTdXBwb3J0ZWRSZXdhcmRBc3NldHMAAgAAE3N1cHBvcnRlZEFzc2V0c0xpc3QJALUJAgUSc3VwcG9ydGVkQXNzZXRzU3RyAgFfAQpjYWxjUmV3YXJkBQt1c2VyQWRkcmVzcwdhc3NldElkDXN0YWtlZEFtb3VudFgOZGVwb3NpdE51bVVzZXIOZGVwb3NpdE51bUxhc3QEF3Jld2FyZFBlck5zYnRTdW1MYXN0S0VZCQEVa2V5UmV3YXJkUGVyTnNidFN1bUF0AgUOZGVwb3NpdE51bUxhc3QFB2Fzc2V0SWQECnN1bUxhc3RYMTgJAKcDAQkBDGdldFN0ck9yRWxzZQIJARVrZXlSZXdhcmRQZXJOc2J0U3VtQXQCBQ5kZXBvc2l0TnVtTGFzdAUHYXNzZXRJZAIBMAQKc3VtVXNlclgxOAkApwMBCQEMZ2V0U3RyT3JFbHNlAgkBFWtleVJld2FyZFBlck5zYnRTdW1BdAIFDmRlcG9zaXROdW1Vc2VyBQdhc3NldElkAgEwBBFyZXdhcmREeW5hbWljUGFydAkAoAMBCQC8AgMJALgCAgUKc3VtTGFzdFgxOAUKc3VtVXNlclgxOAUNc3Rha2VkQW1vdW50WAUHTVVMVFgxOAQTcmV3YXJkQ2FjaGVkUGFydEtFWQkBCWtleVJld2FyZAIFC3VzZXJBZGRyZXNzBQdhc3NldElkBBByZXdhcmRDYWNoZWRQYXJ0CQEMZ2V0SW50T3JFbHNlAgUTcmV3YXJkQ2FjaGVkUGFydEtFWQAACQCWCgQJAGQCBRByZXdhcmRDYWNoZWRQYXJ0BRFyZXdhcmREeW5hbWljUGFydAUQcmV3YXJkQ2FjaGVkUGFydAURcmV3YXJkRHluYW1pY1BhcnQFE3Jld2FyZENhY2hlZFBhcnRLRVkBDVJld2FyZEVudHJpZXMDCWlzTmV3VXNlcgt1c2VyQWRkcmVzcwxzdGFrZWRBbW91bnQEDXN0YWtlZEFtb3VudFgJALYCAQUMc3Rha2VkQW1vdW50BBt1c2VyUmV3YXJkRnJvbURlcG9zaXROdW1LRVkJARtrZXlVc2VyUmV3YXJkRnJvbURlcG9zaXROdW0BBQt1c2VyQWRkcmVzcwQOZGVwb3NpdE51bVVzZXIJAQxnZXRJbnRPckVsc2UCBRt1c2VyUmV3YXJkRnJvbURlcG9zaXROdW1LRVkA////////////AQQOZGVwb3NpdE51bUxhc3QJAQxnZXRJbnRPckVsc2UCCQERa2V5RGVwb3NpdE51bUxhc3QAAP///////////wEKARtmb3JFYWNoQXNzZXRDYWNoZVVzZXJSZXdhcmQCBWFjY3VtBWFzc2V0BAskdDA4OTQyOTA3NwkBCmNhbGNSZXdhcmQFBQt1c2VyQWRkcmVzcwUFYXNzZXQFDXN0YWtlZEFtb3VudFgFDmRlcG9zaXROdW1Vc2VyBQ5kZXBvc2l0TnVtTGFzdAQLcmV3YXJkVG90YWwIBQskdDA4OTQyOTA3NwJfMQQGY2FjaGVkCAULJHQwODk0MjkwNzcCXzIEB2R5bmFtaWMIBQskdDA4OTQyOTA3NwJfMwQTcmV3YXJkQ2FjaGVkUGFydEtFWQgFCyR0MDg5NDI5MDc3Al80CQDNCAIFBWFjY3VtCQEMSW50ZWdlckVudHJ5AgUTcmV3YXJkQ2FjaGVkUGFydEtFWQULcmV3YXJkVG90YWwDAwkAAAIFDmRlcG9zaXROdW1MYXN0AP///////////wEJAAACBQ5kZXBvc2l0TnVtVXNlcgD///////////8BBwUDbmlsAwMJAAACBQ5kZXBvc2l0TnVtTGFzdAD///////////8BCQBmAgUOZGVwb3NpdE51bVVzZXIA////////////AQcJAAIBAi9pbnZhbGlkIGRlcG9zaXROdW1MYXN0IGFuZCBkZXBvc2l0TnVtVXNlciBzdGF0ZQMDCQBmAgUOZGVwb3NpdE51bUxhc3QA////////////AQkAZwIFDmRlcG9zaXROdW1Vc2VyAP///////////wEHAwUJaXNOZXdVc2VyCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRt1c2VyUmV3YXJkRnJvbURlcG9zaXROdW1LRVkFDmRlcG9zaXROdW1MYXN0BQNuaWwJAM0IAgoAAiRsBRNzdXBwb3J0ZWRBc3NldHNMaXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJARtmb3JFYWNoQXNzZXRDYWNoZVVzZXJSZXdhcmQCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoJAQxJbnRlZ2VyRW50cnkCBRt1c2VyUmV3YXJkRnJvbURlcG9zaXROdW1LRVkFDmRlcG9zaXROdW1MYXN0CQACAQkArAICCQCsAgIJAKwCAgIkdW5jb3ZlcmVkIGNvbmRpdGlvbjogZGVwb3NpdE51bUxhc3Q9CQCkAwEFDmRlcG9zaXROdW1MYXN0AhAgZGVwb3NpdE51bVVzZXI9CQCkAwEFDmRlcG9zaXROdW1Vc2VyASJJbmNyZW1lbnROb3REaXN0cmlidXRlZFJld2FyZEVudHJ5AgN0a24JYW1vdW50SW5jBBdub3REaXN0cmlidXRlZFJld2FyZEtFWQkBF2tleU5vdERpc3RyaWJ1dGVkUmV3YXJkAQUDdGtuBBRub3REaXN0cmlidXRlZFJld2FyZAkBDGdldEludE9yRWxzZQIFF25vdERpc3RyaWJ1dGVkUmV3YXJkS0VZAAAJAMwIAgkBDEludGVnZXJFbnRyeQIFF25vdERpc3RyaWJ1dGVkUmV3YXJkS0VZCQBkAgUUbm90RGlzdHJpYnV0ZWRSZXdhcmQFCWFtb3VudEluYwUDbmlsAQtzdXJmVG9HbnNidAEHc3VyZkFtdAkAaQIFB3N1cmZBbXQFEmduc2J0RnJvbVN1cmZDb2VmZgEfbWVyZ2VWb3RpbmdQb3dlckVmZmVjdGl2ZUhlaWdodAQQcXVhcmFudGluZVBlcmlvZBF2cEVmZmVjdGl2ZUhlaWdodAlzdGFrZWRBbXQMc3Rha2VkQW10TkVXBA9yZW1haW5pbmdUb1dhaXQJAGUCBRF2cEVmZmVjdGl2ZUhlaWdodAUGaGVpZ2h0AwkAZwIAAAUPcmVtYWluaW5nVG9XYWl0CQBkAgUGaGVpZ2h0BRBxdWFyYW50aW5lUGVyaW9kBA1hbHJlYWR5V2FpdGVkCQBlAgUQcXVhcmFudGluZVBlcmlvZAUPcmVtYWluaW5nVG9XYWl0BANrWDgDCQECIT0CBQxzdGFrZWRBbXRORVcAAAkAawMFCXN0YWtlZEFtdAUFTVVMVDgFDHN0YWtlZEFtdE5FVwURdnBFZmZlY3RpdmVIZWlnaHQJAGUCCQBkAgUQcXVhcmFudGluZVBlcmlvZAUGaGVpZ2h0CQBrAwUNYWxyZWFkeVdhaXRlZAUDa1g4BQVNVUxUOAEKbWVyZ2VTdGFrZQILdXNlckFkZHJlc3MLYW1vdW50VG9BZGQEDSR0MDEyNzMwMTI4NDAJAQt2YWx1ZU9yRWxzZQIJARNnZXRVc2VyUGFyYW1zT3JVbml0AQULdXNlckFkZHJlc3MJAJUKAwYAAAAABAlpc05ld1VzZXIIBQ0kdDAxMjczMDEyODQwAl8xBAxzdGFrZWRBbW91bnQIBQ0kdDAxMjczMDEyODQwAl8yBBF2cEVmZmVjdGl2ZUhlaWdodAgFDSR0MDEyNzMwMTI4NDACXzMED3N0YWtlZEFtb3VudE5FVwMFCWlzTmV3VXNlcgULYW1vdW50VG9BZGQJAGQCBQthbW91bnRUb0FkZAUMc3Rha2VkQW1vdW50BBBxdWFyYW50aW5lUGVyaW9kCQBoAgCgCwAOBBR2cEVmZmVjdGl2ZUhlaWdodE5FVwMFCWlzTmV3VXNlcgkAZAIFEHF1YXJhbnRpbmVQZXJpb2QFBmhlaWdodAkBH21lcmdlVm90aW5nUG93ZXJFZmZlY3RpdmVIZWlnaHQEBRBxdWFyYW50aW5lUGVyaW9kBRF2cEVmZmVjdGl2ZUhlaWdodAUMc3Rha2VkQW1vdW50BQ9zdGFrZWRBbW91bnRORVcJAJcKBQUJaXNOZXdVc2VyBQxzdGFrZWRBbW91bnQFEXZwRWZmZWN0aXZlSGVpZ2h0BQ9zdGFrZWRBbW91bnRORVcFFHZwRWZmZWN0aXZlSGVpZ2h0TkVXAQtjb21tb25TdGFrZQILdXNlckFkZHJlc3MBaQQNJHQwMTMzMjYxMzM4MAkBD2dldFBhcmFtc09yRmFpbAAEDXN0YWtlZEFzc2V0SWQIBQ0kdDAxMzMyNjEzMzgwAl8xBA1taW5Mb2NrQW1vdW50CAUNJHQwMTMzMjYxMzM4MAJfMgMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQkAAgECFUludmFsaWQgcGF5bWVudHMgc2l6ZQQHcGF5bWVudAkAkQMCCAUBaQhwYXltZW50cwAABAZhbW91bnQIBQdwYXltZW50BmFtb3VudAQTaW52YWxpZEFzc2V0TWVzc2FnZQkArAICCQCsAgICD0ludmFsaWQgYXNzZXQuIAkA2AQBBQ1zdGFrZWRBc3NldElkAgwgaXMgZXhwZWN0ZWQEB2Fzc2V0SWQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAggFB3BheW1lbnQHYXNzZXRJZAUTaW52YWxpZEFzc2V0TWVzc2FnZQMJAQIhPQIFB2Fzc2V0SWQFDXN0YWtlZEFzc2V0SWQJAAIBBRNpbnZhbGlkQXNzZXRNZXNzYWdlBA51c2VyQWRkcmVzc1N0cgkApQgBBQt1c2VyQWRkcmVzcwQKbWVyZ2VkRGF0YQkBCm1lcmdlU3Rha2UCBQt1c2VyQWRkcmVzcwUGYW1vdW50BAlpc05ld1VzZXIIBQptZXJnZWREYXRhAl8xBAxzdGFrZWRBbW91bnQIBQptZXJnZWREYXRhAl8yBBF2cEVmZmVjdGl2ZUhlaWdodAgFCm1lcmdlZERhdGECXzMED3N0YWtlZEFtb3VudE5FVwgFCm1lcmdlZERhdGECXzQEFHZwRWZmZWN0aXZlSGVpZ2h0TkVXCAUKbWVyZ2VkRGF0YQJfNQMJAGYCBQ1taW5Mb2NrQW1vdW50BQ9zdGFrZWRBbW91bnRORVcJAAIBCQCsAgICE01pbiBsb2NrIGFtb3VudCBpcyAJAKQDAQUNbWluTG9ja0Ftb3VudAQNJHQwMTQxNzkxNDI4MQkBC1N0YXRzUmVzdWx0AwUGYW1vdW50AAEDBQlpc05ld1VzZXIAAQAABAxzdGF0c0VudHJpZXMIBQ0kdDAxNDE3OTE0MjgxAl8xBAt0b3RhbFN0YWtlZAgFDSR0MDE0MTc5MTQyODECXzIEDnRvdGFsU3Rha2VkTmV3CAUNJHQwMTQxNzkxNDI4MQJfMwkAzggCCQDOCAIJAM4IAgkAzAgCCQESSGlzdG9yeVJlY29yZEVudHJ5BwIFc3Rha2UFC3VzZXJBZGRyZXNzCAUBaQ10cmFuc2FjdGlvbklkBQxzdGFrZWRBbW91bnQFEXZwRWZmZWN0aXZlSGVpZ2h0BQ9zdGFrZWRBbW91bnRORVcFFHZwRWZmZWN0aXZlSGVpZ2h0TkVXBQNuaWwJAQ1SZXdhcmRFbnRyaWVzAwUJaXNOZXdVc2VyBQ51c2VyQWRkcmVzc1N0cgUMc3Rha2VkQW1vdW50CQEPTG9ja1BhcmFtc0VudHJ5AwULdXNlckFkZHJlc3MFD3N0YWtlZEFtb3VudE5FVwUUdnBFZmZlY3RpdmVIZWlnaHRORVcFDHN0YXRzRW50cmllcwELY29tbW9uQ2xhaW0CC3VzZXJBZGRyZXNzAWkEDnVzZXJBZGRyZXNzU3RyCQClCAEFC3VzZXJBZGRyZXNzAwkAZgIJAJADAQgFAWkIcGF5bWVudHMAAAkAAgECGXBheW1lbnRzIGFyZSBub3QgYWNjZXB0ZWQEDSR0MDE0NzYwMTQ4NjUJAQt2YWx1ZU9yRWxzZQIJARNnZXRVc2VyUGFyYW1zT3JVbml0AQULdXNlckFkZHJlc3MJAJUKAwYAAAAABAlpc05ld1VzZXIIBQ0kdDAxNDc2MDE0ODY1Al8xBAxzdGFrZWRBbW91bnQIBQ0kdDAxNDc2MDE0ODY1Al8yBAxzdGFraW5nU3RhcnQIBQ0kdDAxNDc2MDE0ODY1Al8zBA1zdGFrZWRBbW91bnRYCQC2AgEFDHN0YWtlZEFtb3VudAQbdXNlclJld2FyZEZyb21EZXBvc2l0TnVtS0VZCQEba2V5VXNlclJld2FyZEZyb21EZXBvc2l0TnVtAQUOdXNlckFkZHJlc3NTdHIEDmRlcG9zaXROdW1Vc2VyCQEMZ2V0SW50T3JFbHNlAgUbdXNlclJld2FyZEZyb21EZXBvc2l0TnVtS0VZAP///////////wEEDmRlcG9zaXROdW1MYXN0CQEMZ2V0SW50T3JFbHNlAgkBEWtleURlcG9zaXROdW1MYXN0AAD///////////8BCgEfZm9yRWFjaEFzc2V0Q2FsY1VuY2xhaW1lZFJld2FyZAIFYWNjdW0FYXNzZXQEDSR0MDE1MjM2MTUzNzQJAQpjYWxjUmV3YXJkBQUOdXNlckFkZHJlc3NTdHIFBWFzc2V0BQ1zdGFrZWRBbW91bnRYBQ5kZXBvc2l0TnVtVXNlcgUOZGVwb3NpdE51bUxhc3QEC3Jld2FyZFRvdGFsCAUNJHQwMTUyMzYxNTM3NAJfMQQGY2FjaGVkCAUNJHQwMTUyMzYxNTM3NAJfMgQHZHluYW1pYwgFDSR0MDE1MjM2MTUzNzQCXzMEE3Jld2FyZENhY2hlZFBhcnRLRVkIBQ0kdDAxNTIzNjE1Mzc0Al80BApjbGFpbWVkS0VZCQEKa2V5Q2xhaW1lZAIFDnVzZXJBZGRyZXNzU3RyBQVhc3NldAQNJHQwMTU0MzQxNTQ3MQUFYWNjdW0EBGRhdGEIBQ0kdDAxNTQzNDE1NDcxAl8xBBFjbGFpbWVkQW10QnlBc3NldAgFDSR0MDE1NDM0MTU0NzECXzIEB25ld1BhcnQJALkJAgkAzAgCBQVhc3NldAkAzAgCCQCkAwEFC3Jld2FyZFRvdGFsBQNuaWwCAToEFGNsYWltZWRBbXRCeUFzc2V0TmV3CQC5CQIJAMwIAgURY2xhaW1lZEFtdEJ5QXNzZXQJAMwIAgUHbmV3UGFydAUDbmlsAgFfAwkAZwIAAAULcmV3YXJkVG90YWwJAJQKAgUEZGF0YQUUY2xhaW1lZEFtdEJ5QXNzZXROZXcJAJQKAgkAzQgCCQDNCAIJAM0IAgUEZGF0YQkBDlNjcmlwdFRyYW5zZmVyAwULdXNlckFkZHJlc3MFC3Jld2FyZFRvdGFsCQELdG9Bc3NldFZlY3QBBQVhc3NldAkBDEludGVnZXJFbnRyeQIFCmNsYWltZWRLRVkJAGQCCQELdmFsdWVPckVsc2UCCQCfCAEFCmNsYWltZWRLRVkAAAULcmV3YXJkVG90YWwJAQxJbnRlZ2VyRW50cnkCBRNyZXdhcmRDYWNoZWRQYXJ0S0VZAAAFFGNsYWltZWRBbXRCeUFzc2V0TmV3BA0kdDAxNTkzMTE2MDQ1CgACJGwFE3N1cHBvcnRlZEFzc2V0c0xpc3QKAAIkcwkAkAMBBQIkbAoABSRhY2MwCQCUCgIFA25pbAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAR9mb3JFYWNoQXNzZXRDYWxjVW5jbGFpbWVkUmV3YXJkAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKBAl0cmFuc2ZlcnMIBQ0kdDAxNTkzMTE2MDQ1Al8xBBdjbGFpbWVkQW10QnlBc3NldFJlc3VsdAgFDSR0MDE1OTMxMTYwNDUCXzIDCQBnAgAACQCQAwEFCXRyYW5zZmVycwUDbmlsCQDNCAIJAM0IAgUJdHJhbnNmZXJzCQEMSW50ZWdlckVudHJ5AgUbdXNlclJld2FyZEZyb21EZXBvc2l0TnVtS0VZBQ5kZXBvc2l0TnVtTGFzdAkBEUNsYWltSGlzdG9yeUVudHJ5AwULdXNlckFkZHJlc3MIBQFpDXRyYW5zYWN0aW9uSWQJALACAgUXY2xhaW1lZEFtdEJ5QXNzZXRSZXN1bHQAAQsBaQELY29uc3RydWN0b3IDDW1pbkxvY2tBbW91bnQVc3VwcG9ydGVkUmV3YXJkQXNzZXRzDXN0YWtlZEFzc2V0SWQDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAhFQZXJtaXNzaW9uIGRlbmllZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBEGtleU1pbkxvY2tBbW91bnQABQ1taW5Mb2NrQW1vdW50CQDMCAIJAQtTdHJpbmdFbnRyeQIJARhrZXlTdXBwb3J0ZWRSZXdhcmRBc3NldHMABRVzdXBwb3J0ZWRSZXdhcmRBc3NldHMJAMwIAgkBC1N0cmluZ0VudHJ5AgkBEGtleVN0YWtlZEFzc2V0SWQABQ1zdGFrZWRBc3NldElkBQNuaWwBaQEFc3Rha2UACQELY29tbW9uU3Rha2UCCAUBaQZjYWxsZXIFAWkBaQETc3Rha2VCeU9yaWdpbkNhbGxlcgAJAQtjb21tb25TdGFrZQIIBQFpDG9yaWdpbkNhbGxlcgUBaQFpAQd1bnN0YWtlAQZhbW91bnQDCQECIT0CCQCQAwEIBQFpCHBheW1lbnRzAAAJAAIBAiN1bnN0YWtlIGRvZXNuJ3QgcmVxdWlyZSBhbnkgcGF5bWVudAQLdXNlckFkZHJlc3MIBQFpBmNhbGxlcgQOdXNlckFkZHJlc3NTdHIJAKUIAQULdXNlckFkZHJlc3MEDSR0MDE2OTU0MTcwMDgJAQ9nZXRQYXJhbXNPckZhaWwABA1zdGFrZWRBc3NldElkCAUNJHQwMTY5NTQxNzAwOAJfMQQNbWluTG9ja0Ftb3VudAgFDSR0MDE2OTU0MTcwMDgCXzIEDSR0MDE3MDExMTcwOTUJARNnZXRVc2VyUGFyYW1zT3JGYWlsAQULdXNlckFkZHJlc3MECWlzTmV3VXNlcggFDSR0MDE3MDExMTcwOTUCXzEEDHN0YWtlZEFtb3VudAgFDSR0MDE3MDExMTcwOTUCXzIEEXZwRWZmZWN0aXZlSGVpZ2h0CAUNJHQwMTcwMTExNzA5NQJfMwQQc3dhcFBhcmFtc1NUUlVDVAkBEmFzU3dhcFBhcmFtc1NUUlVDVAEJAP0HBAUQbmV1dHJpbm9Db250cmFjdAIbc3dhcFBhcmFtc0J5VXNlclNZU1JFQURPTkxZCQDMCAIFDnVzZXJBZGRyZXNzU3RyCQDMCAIAAAUDbmlsBQNuaWwEDHN3YXBMaW1pdE1heAgFEHN3YXBQYXJhbXNTVFJVQ1QCXzEEDnN3YXBMaW1pdFNwZW50CAUQc3dhcFBhcmFtc1NUUlVDVAJfMgQOYmxja3MyTG10UmVzZXQIBRBzd2FwUGFyYW1zU1RSVUNUAl8zAwkAZgIFDnN3YXBMaW1pdFNwZW50AAAJAAIBCQCsAgIJAKwCAgItWW91IGhhdmUgYWxyZWFkeSBtYWRlIGEgc3dhcCBvcGVyYXRpb24uIFdhaXQgCQCkAwEJAGQCBQZoZWlnaHQFDmJsY2tzMkxtdFJlc2V0AhIgaGVpZ2h0IHRvIHVuc3Rha2UDCQBnAgAABQxzdGFrZWRBbW91bnQJAAIBAhJOb3RoaW5nIHRvIHVuc3Rha2UDCQBmAgUGYW1vdW50BQxzdGFrZWRBbW91bnQJAAIBCQCsAgIJAKwCAgkArAICAgpSZXF1ZXN0ZWQgCQCkAwEFBmFtb3VudAISLCBidXQgc3Rha2VkIG9ubHkgCQCkAwEFDHN0YWtlZEFtb3VudAQPc3Rha2VkQW1vdW50TkVXCQBlAgUMc3Rha2VkQW1vdW50BQZhbW91bnQEDSR0MDE3NzY1MTc5MjMJAQtTdGF0c1Jlc3VsdAMJAQEtAQUGYW1vdW50AwkAAAIFBmFtb3VudAUMc3Rha2VkQW1vdW50AP///////////wEAAAMJAAACBQZhbW91bnQFDHN0YWtlZEFtb3VudAD///////////8BAAAEDHN0YXRzRW50cmllcwgFDSR0MDE3NzY1MTc5MjMCXzEEC3RvdGFsU3Rha2VkCAUNJHQwMTc3NjUxNzkyMwJfMgQOdG90YWxTdGFrZWROZXcIBQ0kdDAxNzc2NTE3OTIzAl8zCQDOCAIJAM4IAgkAzggCCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMFC3VzZXJBZGRyZXNzBQZhbW91bnQFDXN0YWtlZEFzc2V0SWQJAMwIAgkBEkhpc3RvcnlSZWNvcmRFbnRyeQcCB3Vuc3Rha2UFC3VzZXJBZGRyZXNzCAUBaQ10cmFuc2FjdGlvbklkBQxzdGFrZWRBbW91bnQFEXZwRWZmZWN0aXZlSGVpZ2h0BQ9zdGFrZWRBbW91bnRORVcFEXZwRWZmZWN0aXZlSGVpZ2h0BQNuaWwJAQ1SZXdhcmRFbnRyaWVzAwcFDnVzZXJBZGRyZXNzU3RyBQxzdGFrZWRBbW91bnQJAQ9Mb2NrUGFyYW1zRW50cnkDBQt1c2VyQWRkcmVzcwUPc3Rha2VkQW1vdW50TkVXBRF2cEVmZmVjdGl2ZUhlaWdodAUMc3RhdHNFbnRyaWVzAWkBB2RlcG9zaXQAAwkBAiE9AgkAkAMBCAUBaQhwYXltZW50cwABCQACAQIfZXhhY3QgMSBwYXltZW50IGlzIGFsbG93ZWQgb25seQQDcG10CQCRAwIIBQFpCHBheW1lbnRzAAAEBmFtb3VudAgFA3BtdAZhbW91bnQECnBtdEFzc2V0SWQJAQt2YWx1ZU9yRWxzZQIIBQNwbXQHYXNzZXRJZAUHV0FWRVNJRAQNcG10QXNzZXRJZFN0cgkA2AQBBQpwbXRBc3NldElkBAhwbXRNdWx0WAMJAAACBQpwbXRBc3NldElkBQdXQVZFU0lEBQZNVUxUWDgFBk1VTFRYNgQHYW1vdW50WAkAtgIBBQZhbW91bnQEC3RvdGFsU3Rha2VkCQEMZ2V0SW50T3JFbHNlAgkBF2tleUxvY2tQYXJhbVRvdGFsQW1vdW50AAAABAx0b3RhbFN0YWtlZFgJALYCAQULdG90YWxTdGFrZWQDCQBmAgAABQt0b3RhbFN0YWtlZAkAAgECG1RPRE86IGNhc2UgaXMgbm90IHN1cHBvcnRlZAMJAAACBQt0b3RhbFN0YWtlZAAACQEiSW5jcmVtZW50Tm90RGlzdHJpYnV0ZWRSZXdhcmRFbnRyeQIFDXBtdEFzc2V0SWRTdHIFBmFtb3VudAQQcmV3YXJkUGVyTnNidFgxOAkAvAIDBQdhbW91bnRYBQdNVUxUWDE4BQx0b3RhbFN0YWtlZFgEEWRlcG9zaXROdW1MYXN0S0VZCQERa2V5RGVwb3NpdE51bUxhc3QABA5kZXBvc2l0TnVtTGFzdAkBDGdldEludE9yRWxzZQIFEWRlcG9zaXROdW1MYXN0S0VZAP///////////wEEDWRlcG9zaXROdW1OZXcJAGQCBQ5kZXBvc2l0TnVtTGFzdAABAwkBASEBCQEIY29udGFpbnMCBRJzdXBwb3J0ZWRBc3NldHNTdHIFDXBtdEFzc2V0SWRTdHIJAAIBCQCsAgIJAKwCAgUSc3VwcG9ydGVkQXNzZXRzU3RyAhEgZG9lc24ndCBjb250YWluIAUNcG10QXNzZXRJZFN0cgoBF3JlZnJlc2hSZXdhcmRQZXJOc2J0U1VNAgVhY2N1bQluZXh0QXNzZXQEFnJld2FyZFBlck5zYnRTdW1OZXdLRVkJARVrZXlSZXdhcmRQZXJOc2J0U3VtQXQCBQ1kZXBvc2l0TnVtTmV3BQluZXh0QXNzZXQECnN1bUxhc3RTdHIJAQxnZXRTdHJPckVsc2UCCQEVa2V5UmV3YXJkUGVyTnNidFN1bUF0AgUOZGVwb3NpdE51bUxhc3QFCW5leHRBc3NldAIBMAkAzQgCBQVhY2N1bQMJAAACBQluZXh0QXNzZXQFDXBtdEFzc2V0SWRTdHIJAQtTdHJpbmdFbnRyeQIFFnJld2FyZFBlck5zYnRTdW1OZXdLRVkJAKYDAQkAtwICCQCnAwEFCnN1bUxhc3RTdHIFEHJld2FyZFBlck5zYnRYMTgJAQtTdHJpbmdFbnRyeQIFFnJld2FyZFBlck5zYnRTdW1OZXdLRVkFCnN1bUxhc3RTdHIJAM0IAgoAAiRsBRNzdXBwb3J0ZWRBc3NldHNMaXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJARdyZWZyZXNoUmV3YXJkUGVyTnNidFNVTQIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgkBDEludGVnZXJFbnRyeQIFEWRlcG9zaXROdW1MYXN0S0VZBQ1kZXBvc2l0TnVtTmV3AWkBDGNsYWltUmV3YXJkcwAJAQtjb21tb25DbGFpbQIIBQFpBmNhbGxlcgUBaQFpARpjbGFpbVJld2FyZHNCeU9yaWdpbkNhbGxlcgAJAQtjb21tb25DbGFpbQIIBQFpDG9yaWdpbkNhbGxlcgUBaQFpARh1bmNsYWltZWRSZXdhcmRzUkVBRE9OTFkBDnVzZXJBZGRyZXNzU3RyCgEWZm9yRWFjaEFzc2V0WmVyb1Jld2FyZAIFYWNjdW0FYXNzZXQJAKwCAgkArAICBQVhY2N1bQkAuQkCCQDMCAIFBWFzc2V0CQDMCAICATAJAMwIAgIBMAUDbmlsAgE6AgFfBBJ1bmNsYWltZWRSZXdhcmRTdHIDCQAAAgUOdXNlckFkZHJlc3NTdHICAAoAAiRsBRNzdXBwb3J0ZWRBc3NldHNMaXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAIACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJARZmb3JFYWNoQXNzZXRaZXJvUmV3YXJkAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAxMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKBAt1c2VyQWRkcmVzcwkBEUBleHRyTmF0aXZlKDEwNjIpAQUOdXNlckFkZHJlc3NTdHIEDSR0MDIwNDk3MjA2MDIJAQt2YWx1ZU9yRWxzZQIJARNnZXRVc2VyUGFyYW1zT3JVbml0AQULdXNlckFkZHJlc3MJAJUKAwYAAAAABAlpc05ld1VzZXIIBQ0kdDAyMDQ5NzIwNjAyAl8xBAxzdGFrZWRBbW91bnQIBQ0kdDAyMDQ5NzIwNjAyAl8yBAxzdGFraW5nU3RhcnQIBQ0kdDAyMDQ5NzIwNjAyAl8zBA1zdGFrZWRBbW91bnRYCQC2AgEFDHN0YWtlZEFtb3VudAQbdXNlclJld2FyZEZyb21EZXBvc2l0TnVtS0VZCQEba2V5VXNlclJld2FyZEZyb21EZXBvc2l0TnVtAQUOdXNlckFkZHJlc3NTdHIEDmRlcG9zaXROdW1Vc2VyCQEMZ2V0SW50T3JFbHNlAgUbdXNlclJld2FyZEZyb21EZXBvc2l0TnVtS0VZAP///////////wEEDmRlcG9zaXROdW1MYXN0CQEMZ2V0SW50T3JFbHNlAgkBEWtleURlcG9zaXROdW1MYXN0AAD///////////8BCgEfZm9yRWFjaEFzc2V0Q2FsY1VuY2xhaW1lZFJld2FyZAIFYWNjdW0FYXNzZXQEDSR0MDIwOTQ4MjEwODYJAQpjYWxjUmV3YXJkBQUOdXNlckFkZHJlc3NTdHIFBWFzc2V0BQ1zdGFrZWRBbW91bnRYBQ5kZXBvc2l0TnVtVXNlcgUOZGVwb3NpdE51bUxhc3QEC3Jld2FyZFRvdGFsCAUNJHQwMjA5NDgyMTA4NgJfMQQGY2FjaGVkCAUNJHQwMjA5NDgyMTA4NgJfMgQHZHluYW1pYwgFDSR0MDIwOTQ4MjEwODYCXzMEE3Jld2FyZENhY2hlZFBhcnRLRVkIBQ0kdDAyMDk0ODIxMDg2Al80BAdjbGFpbWVkCQELdmFsdWVPckVsc2UCCQCfCAEJAQprZXlDbGFpbWVkAgUOdXNlckFkZHJlc3NTdHIFBWFzc2V0AAAJAKwCAgkArAICBQVhY2N1bQkAuQkCCQDMCAIFBWFzc2V0CQDMCAIJAKQDAQULcmV3YXJkVG90YWwJAMwIAgkApAMBBQdjbGFpbWVkBQNuaWwCAToCAV8KAAIkbAUTc3VwcG9ydGVkQXNzZXRzTGlzdAoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEfZm9yRWFjaEFzc2V0Q2FsY1VuY2xhaW1lZFJld2FyZAIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgkAlAoCBQNuaWwJALMCAgUSdW5jbGFpbWVkUmV3YXJkU3RyAAEBaQEWc3VyZlN0YWtpbmdTWVNSRUFET05MWQIVdXNlckFkZHJlc3NTdHJPckVtcHR5CHN1cmZEaWZmBAlzdXJmVG90YWwJAQxnZXRJbnRPckVsc2UCCQEXa2V5TG9ja1BhcmFtVG90YWxBbW91bnQAAAAEEmduc2J0RnJvbVN1cmZUb3RhbAkBC3N1cmZUb0duc2J0AQUJc3VyZlRvdGFsAwkAAAIFFXVzZXJBZGRyZXNzU3RyT3JFbXB0eQIACQCUCgIFA25pbAkAzAgCAAAJAMwIAgUJc3VyZlRvdGFsCQDMCAIAAAkAzAgCBRJnbnNidEZyb21TdXJmVG90YWwJAMwIAgAACQDMCAIFBmhlaWdodAkAzAgCBQZoZWlnaHQFA25pbAQLdXNlckFkZHJlc3MJAQ90b0FkZHJlc3NPckZhaWwBBRV1c2VyQWRkcmVzc1N0ck9yRW1wdHkECm1lcmdlZERhdGEJAQptZXJnZVN0YWtlAgULdXNlckFkZHJlc3MFCHN1cmZEaWZmBAlpc05ld1VzZXIIBQptZXJnZWREYXRhAl8xBAxzdGFrZWRBbW91bnQIBQptZXJnZWREYXRhAl8yBBF2cEVmZmVjdGl2ZUhlaWdodAgFCm1lcmdlZERhdGECXzMED3N0YWtlZEFtb3VudE5FVwgFCm1lcmdlZERhdGECXzQEFHZwRWZmZWN0aXZlSGVpZ2h0TkVXCAUKbWVyZ2VkRGF0YQJfNQQIc3VyZlVzZXIFDHN0YWtlZEFtb3VudAQRZ25zYnRGcm9tU3VyZlVzZXIJAQtzdXJmVG9HbnNidAEFCHN1cmZVc2VyCQCUCgIFA25pbAkAzAgCBQhzdXJmVXNlcgkAzAgCBQlzdXJmVG90YWwJAMwIAgURZ25zYnRGcm9tU3VyZlVzZXIJAMwIAgUSZ25zYnRGcm9tU3VyZlRvdGFsCQDMCAIFEXZwRWZmZWN0aXZlSGVpZ2h0CQDMCAIFFHZwRWZmZWN0aXZlSGVpZ2h0TkVXBQNuaWwBaQEYZ25zYnRGcm9tU3VyZlNZU1JFQURPTkxZAQdzdXJmQW10CQCUCgIFA25pbAkBC3N1cmZUb0duc2J0AQUHc3VyZkFtdAFpARFjb25maWdTWVNSRUFET05MWQAECm1pbkxvY2tBbXQJARFAZXh0ck5hdGl2ZSgxMDU1KQEJARBrZXlNaW5Mb2NrQW1vdW50AAQgc3VyZlZvdGluZ1Bvd2VyUmVzdHJpY3RpdmVQZXJpb2QJAGgCAKALAA4EG2JyVG9TdGFydFN1cmZUcmFuc2Zvcm1hdGlvbgCwmEYJAJQKAgUDbmlsCQDMCAIFCm1pbkxvY2tBbXQJAMwIAgkAaAIFEmduc2J0RnJvbVN1cmZDb2VmZgUFTVVMVDYJAMwIAgUgc3VyZlZvdGluZ1Bvd2VyUmVzdHJpY3RpdmVQZXJpb2QJAMwIAgUbYnJUb1N0YXJ0U3VyZlRyYW5zZm9ybWF0aW9uBQNuaWwAODVa1g==", "chainId": 84, "height": 2177196, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 2UYAH4ZPNW5g4fQtxyKBtSWvgRJwXpEbWG1k3BUafkJA Next: 295jyYfSD5FqVAecBXQbuLR7xsT5M9YLR34GvwjYCkfH Diff:
OldNewDifferences
425425 let isNewUser = $t01701117095._1
426426 let stakedAmount = $t01701117095._2
427427 let vpEffectiveHeight = $t01701117095._3
428- let swapParamsSTRUCT = asSwapParamsSTRUCT(invoke(neutrinoContract, "swapParamsByUserSYSREADONLY", [userAddressStr, 0], nil))
428+ let swapParamsSTRUCT = asSwapParamsSTRUCT(reentrantInvoke(neutrinoContract, "swapParamsByUserSYSREADONLY", [userAddressStr, 0], nil))
429429 let swapLimitMax = swapParamsSTRUCT._1
430430 let swapLimitSpent = swapParamsSTRUCT._2
431431 let blcks2LmtReset = swapParamsSTRUCT._3
437437 then throw(((("Requested " + toString(amount)) + ", but staked only ") + toString(stakedAmount)))
438438 else {
439439 let stakedAmountNEW = (stakedAmount - amount)
440- let $t01775617914 = StatsResult(-(amount), if ((amount == stakedAmount))
440+ let $t01776517923 = StatsResult(-(amount), if ((amount == stakedAmount))
441441 then -1
442442 else 0, if ((amount == stakedAmount))
443443 then -1
444444 else 0)
445- let statsEntries = $t01775617914._1
446- let totalStaked = $t01775617914._2
447- let totalStakedNew = $t01775617914._3
445+ let statsEntries = $t01776517923._1
446+ let totalStaked = $t01776517923._2
447+ let totalStakedNew = $t01776517923._3
448448 ((([ScriptTransfer(userAddress, amount, stakedAssetId), HistoryRecordEntry("unstake", userAddress, i.transactionId, stakedAmount, vpEffectiveHeight, stakedAmountNEW, vpEffectiveHeight)] ++ RewardEntries(false, userAddressStr, stakedAmount)) ++ LockParamsEntry(userAddress, stakedAmountNEW, vpEffectiveHeight)) ++ statsEntries)
449449 }
450450 }
536536 }
537537 else {
538538 let userAddress = addressFromStringValue(userAddressStr)
539- let $t02048820593 = valueOrElse(getUserParamsOrUnit(userAddress), $Tuple3(true, 0, 0))
540- let isNewUser = $t02048820593._1
541- let stakedAmount = $t02048820593._2
542- let stakingStart = $t02048820593._3
539+ let $t02049720602 = valueOrElse(getUserParamsOrUnit(userAddress), $Tuple3(true, 0, 0))
540+ let isNewUser = $t02049720602._1
541+ let stakedAmount = $t02049720602._2
542+ let stakingStart = $t02049720602._3
543543 let stakedAmountX = toBigInt(stakedAmount)
544544 let userRewardFromDepositNumKEY = keyUserRewardFromDepositNum(userAddressStr)
545545 let depositNumUser = getIntOrElse(userRewardFromDepositNumKEY, -1)
546546 let depositNumLast = getIntOrElse(keyDepositNumLast(), -1)
547547 func forEachAssetCalcUnclaimedReward (accum,asset) = {
548- let $t02093921077 = calcReward(userAddressStr, asset, stakedAmountX, depositNumUser, depositNumLast)
549- let rewardTotal = $t02093921077._1
550- let cached = $t02093921077._2
551- let dynamic = $t02093921077._3
552- let rewardCachedPartKEY = $t02093921077._4
548+ let $t02094821086 = calcReward(userAddressStr, asset, stakedAmountX, depositNumUser, depositNumLast)
549+ let rewardTotal = $t02094821086._1
550+ let cached = $t02094821086._2
551+ let dynamic = $t02094821086._3
552+ let rewardCachedPartKEY = $t02094821086._4
553553 let claimed = valueOrElse(getInteger(keyClaimed(userAddressStr, asset)), 0)
554554 ((accum + makeString([asset, toString(rewardTotal), toString(claimed)], ":")) + "_")
555555 }
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let separator = "__"
55
66 let SEP = "__"
77
88 let MULT6 = 1000000
99
1010 let MULT8 = 100000000
1111
1212 let MULTX6 = toBigInt(MULT6)
1313
1414 let MULTX8 = toBigInt(MULT8)
1515
1616 let MULTX18 = toBigInt(1000000000000000000)
1717
1818 let WAVESIDSTR = "WAVES"
1919
2020 let WAVESID = fromBase58String(WAVESIDSTR)
2121
2222 let IdxControlCfgNeutrinoDapp = 1
2323
2424 let IdxControlCfgAuctionDapp = 2
2525
2626 let IdxControlCfgRpdDapp = 3
2727
2828 let IdxControlCfgMathDapp = 4
2929
3030 let IdxControlCfgLiquidationDapp = 5
3131
3232 let IdxControlCfgRestDapp = 6
3333
3434 let IdxControlCfgNodeRegistryDapp = 7
3535
3636 let IdxControlCfgNsbtStakingDapp = 8
3737
3838 let IdxControlCfgMediatorDapp = 9
3939
4040 let IdxControlCfgSurfStakingDapp = 10
4141
4242 let IdxControlCfgGnsbtControllerDapp = 11
4343
4444 func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), (((("mandatory " + toString(address)) + ".") + key) + " is not defined"))
4545
4646
4747 func keyControlAddress () = "%s%s__config__controlAddress"
4848
4949
5050 func keyControlCfg () = "%s__controlConfig"
5151
5252
5353 func keyGnsbtFromSurfCoeff () = "%s%s__cfg__gnsbtFromSurfCoeff"
5454
5555
5656 func readControlCfgOrFail (control) = split(getStringOrFail(control, keyControlCfg()), SEP)
5757
5858
5959 func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
6060
6161
6262 let controlContract = addressFromStringValue(valueOrElse(getString(this, keyControlAddress()), "3N4NS7d4Jo9a6F14LiFUKKYVdUkkf2eP4Zx"))
6363
6464 let controlCfg = readControlCfgOrFail(controlContract)
6565
6666 let mathContract = getContractAddressOrFail(controlCfg, IdxControlCfgMathDapp)
6767
6868 let neutrinoContract = getContractAddressOrFail(controlCfg, IdxControlCfgNeutrinoDapp)
6969
7070 let auctionContract = getContractAddressOrFail(controlCfg, IdxControlCfgAuctionDapp)
7171
7272 let gnsbtFromSurfCoeff = valueOrElse(getInteger(this, keyGnsbtFromSurfCoeff()), 300)
7373
7474 func keyBondAsset () = "bond_asset_id"
7575
7676
7777 func keyAuctionContractAddress () = "auction_contract"
7878
7979
8080 func keyMinLockAmount () = "%s__minLockAmount"
8181
8282
8383 func keyStakedAssetId () = "%s__stakedAssetId"
8484
8585
8686 func keyLockParamUserAmount (userAddress) = makeString(["%s%s%s", "paramByUser", toString(userAddress), "amount"], separator)
8787
8888
8989 func keyLockParamStartBlock (userAddress) = makeString(["%s%s%s", "paramByUser", toString(userAddress), "start"], separator)
9090
9191
9292 func keyLockParamVotingPowerEffectiveHeight (userAddress) = makeString(["%s%s%s", "paramByUser", toString(userAddress), "vpEffectiveHeight"], separator)
9393
9494
9595 func keyHistoryRecord (type,userAddress,txId) = makeString(["%s%s%s%s", "history", type, toString(userAddress), toBase58String(txId)], separator)
9696
9797
9898 func keyLockParamTotalAmount () = makeString(["%s%s", "stats", "activeTotalLocked"], separator)
9999
100100
101101 func keyStatsLocksCount () = makeString(["%s%s", "stats", "locksCount"], separator)
102102
103103
104104 func keyStatsUsersCount () = makeString(["%s%s", "stats", "activeUsersCount"], separator)
105105
106106
107107 func keyNextPeriod () = "%s__nextPeriod"
108108
109109
110110 func keySupportedRewardAssets () = "supportedRewardAssets"
111111
112112
113113 func keyDepositNumLast () = makeString(["%s%s%s", "dep", "lastNum"], separator)
114114
115115
116116 func keyUserRewardFromDepositNum (userAddress) = makeString(["%s%s%s", "userRwdFromDepNum", userAddress], separator)
117117
118118
119119 func keyRewardPerNsbtSumAt (depositNum,tkn) = makeString(["%s%d", "rwdPerNsbtSumByDepNum", toString(depositNum), tkn], separator)
120120
121121
122122 func keyReward (userAddress,tkn) = makeString(["%s%s%s", "rwd", userAddress, tkn], separator)
123123
124124
125125 func keyClaimed (userAddress,tkn) = makeString(["%s%s%s", "clm", userAddress, tkn], separator)
126126
127127
128128 func keyNotDistributedReward (tkn) = makeString(["%s%s", "notDistributed", tkn], separator)
129129
130130
131131 func toX18 (origVal,origMult) = fraction(toBigInt(origVal), MULTX18, origMult)
132132
133133
134134 func getIntOrZero (key) = valueOrElse(getInteger(this, key), 0)
135135
136136
137137 func getIntOrElse (key,defaultVal) = valueOrElse(getInteger(this, key), defaultVal)
138138
139139
140140 func getIntOrFail (key) = valueOrErrorMessage(getInteger(this, key), (("Mandatory this." + key) + " is not defined"))
141141
142142
143143 func getStrOrElse (key,defaultVal) = valueOrElse(getString(this, key), defaultVal)
144144
145145
146146 func toAddressOrFail (addressStr) = valueOrErrorMessage(addressFromString(addressStr), ("couldn't parse passed addressStr=" + addressStr))
147147
148148
149149 func toAssetVect (assetStr) = if ((assetStr == WAVESIDSTR))
150150 then unit
151151 else fromBase58String(assetStr)
152152
153153
154154 func asInt (val) = match val {
155155 case valInt: Int =>
156156 valInt
157157 case _ =>
158158 throw("fail to cast into Int")
159159 }
160160
161161
162162 func asSwapParamsSTRUCT (v) = match v {
163163 case struct: (Int, Int, Int, Int, Int) =>
164164 struct
165165 case _ =>
166166 throw("fail to cast into Int")
167167 }
168168
169169
170170 func formatHistoryRecord (oldAmount,oldStart,newAmount,newStart) = makeString(["%d%d%d%d%d%d", toString(lastBlock.height), toString(lastBlock.timestamp), toString(oldAmount), toString(oldStart), toString(newAmount), toString(newStart)], separator)
171171
172172
173173 func formatClaimHistoryRecord (user,claimedRewards) = makeString(["%s%d%d%s", user, toString(lastBlock.height), toString(lastBlock.timestamp), claimedRewards], separator)
174174
175175
176176 func HistoryRecordEntry (type,userAddress,txId,oldAmount,oldStart,newAmount,newStart) = StringEntry(keyHistoryRecord(type, userAddress, txId), formatHistoryRecord(oldAmount, oldStart, newAmount, newStart))
177177
178178
179179 func ClaimHistoryEntry (userAddress,txId,claimedRewards) = StringEntry(keyHistoryRecord("claim", userAddress, txId), formatClaimHistoryRecord(toString(userAddress), claimedRewards))
180180
181181
182182 func StatsResult (totalLockedInc,lockCountInc,usersCountInc) = {
183183 let locksCount = getIntOrZero(keyStatsLocksCount())
184184 let usersCount = getIntOrZero(keyStatsUsersCount())
185185 let totalAmount = getIntOrZero(keyLockParamTotalAmount())
186186 let totalAmountNew = (totalAmount + totalLockedInc)
187187 $Tuple3([IntegerEntry(keyStatsLocksCount(), (locksCount + lockCountInc)), IntegerEntry(keyStatsUsersCount(), (usersCount + usersCountInc)), IntegerEntry(keyLockParamTotalAmount(), totalAmountNew)], totalAmount, totalAmountNew)
188188 }
189189
190190
191191 func LockParamsEntry (userAddress,amount,votingPowerEffectiveHeight) = [IntegerEntry(keyLockParamUserAmount(userAddress), amount), IntegerEntry(keyLockParamStartBlock(userAddress), votingPowerEffectiveHeight)]
192192
193193
194194 func getParamsOrFail () = $Tuple2(fromBase58String(getStringOrFail(this, keyStakedAssetId())), getIntOrFail(keyMinLockAmount()))
195195
196196
197197 func isActiveUser (userAddress) = (getIntOrElse(keyLockParamUserAmount(userAddress), 0) > 0)
198198
199199
200200 func getUserParamsOrUnit (userAddress) = if (isActiveUser(userAddress))
201201 then $Tuple3(false, getIntOrFail(keyLockParamUserAmount(userAddress)), getIntOrFail(keyLockParamStartBlock(userAddress)))
202202 else unit
203203
204204
205205 func getUserParamsOrFail (userAddress) = valueOrErrorMessage(getUserParamsOrUnit(userAddress), (("User " + toString(userAddress)) + " is not defined"))
206206
207207
208208 let supportedAssetsStr = getStrOrElse(keySupportedRewardAssets(), "")
209209
210210 let supportedAssetsList = split(supportedAssetsStr, "_")
211211
212212 func calcReward (userAddress,assetId,stakedAmountX,depositNumUser,depositNumLast) = {
213213 let rewardPerNsbtSumLastKEY = keyRewardPerNsbtSumAt(depositNumLast, assetId)
214214 let sumLastX18 = parseBigIntValue(getStrOrElse(keyRewardPerNsbtSumAt(depositNumLast, assetId), "0"))
215215 let sumUserX18 = parseBigIntValue(getStrOrElse(keyRewardPerNsbtSumAt(depositNumUser, assetId), "0"))
216216 let rewardDynamicPart = toInt(fraction((sumLastX18 - sumUserX18), stakedAmountX, MULTX18))
217217 let rewardCachedPartKEY = keyReward(userAddress, assetId)
218218 let rewardCachedPart = getIntOrElse(rewardCachedPartKEY, 0)
219219 $Tuple4((rewardCachedPart + rewardDynamicPart), rewardCachedPart, rewardDynamicPart, rewardCachedPartKEY)
220220 }
221221
222222
223223 func RewardEntries (isNewUser,userAddress,stakedAmount) = {
224224 let stakedAmountX = toBigInt(stakedAmount)
225225 let userRewardFromDepositNumKEY = keyUserRewardFromDepositNum(userAddress)
226226 let depositNumUser = getIntOrElse(userRewardFromDepositNumKEY, -1)
227227 let depositNumLast = getIntOrElse(keyDepositNumLast(), -1)
228228 func forEachAssetCacheUserReward (accum,asset) = {
229229 let $t089429077 = calcReward(userAddress, asset, stakedAmountX, depositNumUser, depositNumLast)
230230 let rewardTotal = $t089429077._1
231231 let cached = $t089429077._2
232232 let dynamic = $t089429077._3
233233 let rewardCachedPartKEY = $t089429077._4
234234 (accum :+ IntegerEntry(rewardCachedPartKEY, rewardTotal))
235235 }
236236
237237 if (if ((depositNumLast == -1))
238238 then (depositNumUser == -1)
239239 else false)
240240 then nil
241241 else if (if ((depositNumLast == -1))
242242 then (depositNumUser > -1)
243243 else false)
244244 then throw("invalid depositNumLast and depositNumUser state")
245245 else if (if ((depositNumLast > -1))
246246 then (depositNumUser >= -1)
247247 else false)
248248 then if (isNewUser)
249249 then [IntegerEntry(userRewardFromDepositNumKEY, depositNumLast)]
250250 else ({
251251 let $l = supportedAssetsList
252252 let $s = size($l)
253253 let $acc0 = nil
254254 func $f0_1 ($a,$i) = if (($i >= $s))
255255 then $a
256256 else forEachAssetCacheUserReward($a, $l[$i])
257257
258258 func $f0_2 ($a,$i) = if (($i >= $s))
259259 then $a
260260 else throw("List size exceeds 10")
261261
262262 $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)
263263 } :+ IntegerEntry(userRewardFromDepositNumKEY, depositNumLast))
264264 else throw(((("uncovered condition: depositNumLast=" + toString(depositNumLast)) + " depositNumUser=") + toString(depositNumUser)))
265265 }
266266
267267
268268 func IncrementNotDistributedRewardEntry (tkn,amountInc) = {
269269 let notDistributedRewardKEY = keyNotDistributedReward(tkn)
270270 let notDistributedReward = getIntOrElse(notDistributedRewardKEY, 0)
271271 [IntegerEntry(notDistributedRewardKEY, (notDistributedReward + amountInc))]
272272 }
273273
274274
275275 func surfToGnsbt (surfAmt) = (surfAmt / gnsbtFromSurfCoeff)
276276
277277
278278 func mergeVotingPowerEffectiveHeight (quarantinePeriod,vpEffectiveHeight,stakedAmt,stakedAmtNEW) = {
279279 let remainingToWait = (vpEffectiveHeight - height)
280280 if ((0 >= remainingToWait))
281281 then (height + quarantinePeriod)
282282 else {
283283 let alreadyWaited = (quarantinePeriod - remainingToWait)
284284 let kX8 = if ((stakedAmtNEW != 0))
285285 then fraction(stakedAmt, MULT8, stakedAmtNEW)
286286 else vpEffectiveHeight
287287 ((quarantinePeriod + height) - fraction(alreadyWaited, kX8, MULT8))
288288 }
289289 }
290290
291291
292292 func mergeStake (userAddress,amountToAdd) = {
293293 let $t01273012840 = valueOrElse(getUserParamsOrUnit(userAddress), $Tuple3(true, 0, 0))
294294 let isNewUser = $t01273012840._1
295295 let stakedAmount = $t01273012840._2
296296 let vpEffectiveHeight = $t01273012840._3
297297 let stakedAmountNEW = if (isNewUser)
298298 then amountToAdd
299299 else (amountToAdd + stakedAmount)
300300 let quarantinePeriod = (1440 * 14)
301301 let vpEffectiveHeightNEW = if (isNewUser)
302302 then (quarantinePeriod + height)
303303 else mergeVotingPowerEffectiveHeight(quarantinePeriod, vpEffectiveHeight, stakedAmount, stakedAmountNEW)
304304 $Tuple5(isNewUser, stakedAmount, vpEffectiveHeight, stakedAmountNEW, vpEffectiveHeightNEW)
305305 }
306306
307307
308308 func commonStake (userAddress,i) = {
309309 let $t01332613380 = getParamsOrFail()
310310 let stakedAssetId = $t01332613380._1
311311 let minLockAmount = $t01332613380._2
312312 if ((size(i.payments) != 1))
313313 then throw("Invalid payments size")
314314 else {
315315 let payment = i.payments[0]
316316 let amount = payment.amount
317317 let invalidAssetMessage = (("Invalid asset. " + toBase58String(stakedAssetId)) + " is expected")
318318 let assetId = valueOrErrorMessage(payment.assetId, invalidAssetMessage)
319319 if ((assetId != stakedAssetId))
320320 then throw(invalidAssetMessage)
321321 else {
322322 let userAddressStr = toString(userAddress)
323323 let mergedData = mergeStake(userAddress, amount)
324324 let isNewUser = mergedData._1
325325 let stakedAmount = mergedData._2
326326 let vpEffectiveHeight = mergedData._3
327327 let stakedAmountNEW = mergedData._4
328328 let vpEffectiveHeightNEW = mergedData._5
329329 if ((minLockAmount > stakedAmountNEW))
330330 then throw(("Min lock amount is " + toString(minLockAmount)))
331331 else {
332332 let $t01417914281 = StatsResult(amount, 1, if (isNewUser)
333333 then 1
334334 else 0)
335335 let statsEntries = $t01417914281._1
336336 let totalStaked = $t01417914281._2
337337 let totalStakedNew = $t01417914281._3
338338 ((([HistoryRecordEntry("stake", userAddress, i.transactionId, stakedAmount, vpEffectiveHeight, stakedAmountNEW, vpEffectiveHeightNEW)] ++ RewardEntries(isNewUser, userAddressStr, stakedAmount)) ++ LockParamsEntry(userAddress, stakedAmountNEW, vpEffectiveHeightNEW)) ++ statsEntries)
339339 }
340340 }
341341 }
342342 }
343343
344344
345345 func commonClaim (userAddress,i) = {
346346 let userAddressStr = toString(userAddress)
347347 if ((size(i.payments) > 0))
348348 then throw("payments are not accepted")
349349 else {
350350 let $t01476014865 = valueOrElse(getUserParamsOrUnit(userAddress), $Tuple3(true, 0, 0))
351351 let isNewUser = $t01476014865._1
352352 let stakedAmount = $t01476014865._2
353353 let stakingStart = $t01476014865._3
354354 let stakedAmountX = toBigInt(stakedAmount)
355355 let userRewardFromDepositNumKEY = keyUserRewardFromDepositNum(userAddressStr)
356356 let depositNumUser = getIntOrElse(userRewardFromDepositNumKEY, -1)
357357 let depositNumLast = getIntOrElse(keyDepositNumLast(), -1)
358358 func forEachAssetCalcUnclaimedReward (accum,asset) = {
359359 let $t01523615374 = calcReward(userAddressStr, asset, stakedAmountX, depositNumUser, depositNumLast)
360360 let rewardTotal = $t01523615374._1
361361 let cached = $t01523615374._2
362362 let dynamic = $t01523615374._3
363363 let rewardCachedPartKEY = $t01523615374._4
364364 let claimedKEY = keyClaimed(userAddressStr, asset)
365365 let $t01543415471 = accum
366366 let data = $t01543415471._1
367367 let claimedAmtByAsset = $t01543415471._2
368368 let newPart = makeString([asset, toString(rewardTotal)], ":")
369369 let claimedAmtByAssetNew = makeString([claimedAmtByAsset, newPart], "_")
370370 if ((0 >= rewardTotal))
371371 then $Tuple2(data, claimedAmtByAssetNew)
372372 else $Tuple2((((data :+ ScriptTransfer(userAddress, rewardTotal, toAssetVect(asset))) :+ IntegerEntry(claimedKEY, (valueOrElse(getInteger(claimedKEY), 0) + rewardTotal))) :+ IntegerEntry(rewardCachedPartKEY, 0)), claimedAmtByAssetNew)
373373 }
374374
375375 let $t01593116045 = {
376376 let $l = supportedAssetsList
377377 let $s = size($l)
378378 let $acc0 = $Tuple2(nil, "")
379379 func $f0_1 ($a,$i) = if (($i >= $s))
380380 then $a
381381 else forEachAssetCalcUnclaimedReward($a, $l[$i])
382382
383383 func $f0_2 ($a,$i) = if (($i >= $s))
384384 then $a
385385 else throw("List size exceeds 10")
386386
387387 $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)
388388 }
389389 let transfers = $t01593116045._1
390390 let claimedAmtByAssetResult = $t01593116045._2
391391 if ((0 >= size(transfers)))
392392 then nil
393393 else ((transfers :+ IntegerEntry(userRewardFromDepositNumKEY, depositNumLast)) :+ ClaimHistoryEntry(userAddress, i.transactionId, drop(claimedAmtByAssetResult, 1)))
394394 }
395395 }
396396
397397
398398 @Callable(i)
399399 func constructor (minLockAmount,supportedRewardAssets,stakedAssetId) = if ((i.caller != this))
400400 then throw("Permission denied")
401401 else [IntegerEntry(keyMinLockAmount(), minLockAmount), StringEntry(keySupportedRewardAssets(), supportedRewardAssets), StringEntry(keyStakedAssetId(), stakedAssetId)]
402402
403403
404404
405405 @Callable(i)
406406 func stake () = commonStake(i.caller, i)
407407
408408
409409
410410 @Callable(i)
411411 func stakeByOriginCaller () = commonStake(i.originCaller, i)
412412
413413
414414
415415 @Callable(i)
416416 func unstake (amount) = if ((size(i.payments) != 0))
417417 then throw("unstake doesn't require any payment")
418418 else {
419419 let userAddress = i.caller
420420 let userAddressStr = toString(userAddress)
421421 let $t01695417008 = getParamsOrFail()
422422 let stakedAssetId = $t01695417008._1
423423 let minLockAmount = $t01695417008._2
424424 let $t01701117095 = getUserParamsOrFail(userAddress)
425425 let isNewUser = $t01701117095._1
426426 let stakedAmount = $t01701117095._2
427427 let vpEffectiveHeight = $t01701117095._3
428- let swapParamsSTRUCT = asSwapParamsSTRUCT(invoke(neutrinoContract, "swapParamsByUserSYSREADONLY", [userAddressStr, 0], nil))
428+ let swapParamsSTRUCT = asSwapParamsSTRUCT(reentrantInvoke(neutrinoContract, "swapParamsByUserSYSREADONLY", [userAddressStr, 0], nil))
429429 let swapLimitMax = swapParamsSTRUCT._1
430430 let swapLimitSpent = swapParamsSTRUCT._2
431431 let blcks2LmtReset = swapParamsSTRUCT._3
432432 if ((swapLimitSpent > 0))
433433 then throw((("You have already made a swap operation. Wait " + toString((height + blcks2LmtReset))) + " height to unstake"))
434434 else if ((0 >= stakedAmount))
435435 then throw("Nothing to unstake")
436436 else if ((amount > stakedAmount))
437437 then throw(((("Requested " + toString(amount)) + ", but staked only ") + toString(stakedAmount)))
438438 else {
439439 let stakedAmountNEW = (stakedAmount - amount)
440- let $t01775617914 = StatsResult(-(amount), if ((amount == stakedAmount))
440+ let $t01776517923 = StatsResult(-(amount), if ((amount == stakedAmount))
441441 then -1
442442 else 0, if ((amount == stakedAmount))
443443 then -1
444444 else 0)
445- let statsEntries = $t01775617914._1
446- let totalStaked = $t01775617914._2
447- let totalStakedNew = $t01775617914._3
445+ let statsEntries = $t01776517923._1
446+ let totalStaked = $t01776517923._2
447+ let totalStakedNew = $t01776517923._3
448448 ((([ScriptTransfer(userAddress, amount, stakedAssetId), HistoryRecordEntry("unstake", userAddress, i.transactionId, stakedAmount, vpEffectiveHeight, stakedAmountNEW, vpEffectiveHeight)] ++ RewardEntries(false, userAddressStr, stakedAmount)) ++ LockParamsEntry(userAddress, stakedAmountNEW, vpEffectiveHeight)) ++ statsEntries)
449449 }
450450 }
451451
452452
453453
454454 @Callable(i)
455455 func deposit () = if ((size(i.payments) != 1))
456456 then throw("exact 1 payment is allowed only")
457457 else {
458458 let pmt = i.payments[0]
459459 let amount = pmt.amount
460460 let pmtAssetId = valueOrElse(pmt.assetId, WAVESID)
461461 let pmtAssetIdStr = toBase58String(pmtAssetId)
462462 let pmtMultX = if ((pmtAssetId == WAVESID))
463463 then MULTX8
464464 else MULTX6
465465 let amountX = toBigInt(amount)
466466 let totalStaked = getIntOrElse(keyLockParamTotalAmount(), 0)
467467 let totalStakedX = toBigInt(totalStaked)
468468 if ((0 > totalStaked))
469469 then throw("TODO: case is not supported")
470470 else if ((totalStaked == 0))
471471 then IncrementNotDistributedRewardEntry(pmtAssetIdStr, amount)
472472 else {
473473 let rewardPerNsbtX18 = fraction(amountX, MULTX18, totalStakedX)
474474 let depositNumLastKEY = keyDepositNumLast()
475475 let depositNumLast = getIntOrElse(depositNumLastKEY, -1)
476476 let depositNumNew = (depositNumLast + 1)
477477 if (!(contains(supportedAssetsStr, pmtAssetIdStr)))
478478 then throw(((supportedAssetsStr + " doesn't contain ") + pmtAssetIdStr))
479479 else {
480480 func refreshRewardPerNsbtSUM (accum,nextAsset) = {
481481 let rewardPerNsbtSumNewKEY = keyRewardPerNsbtSumAt(depositNumNew, nextAsset)
482482 let sumLastStr = getStrOrElse(keyRewardPerNsbtSumAt(depositNumLast, nextAsset), "0")
483483 (accum :+ (if ((nextAsset == pmtAssetIdStr))
484484 then StringEntry(rewardPerNsbtSumNewKEY, toString((parseBigIntValue(sumLastStr) + rewardPerNsbtX18)))
485485 else StringEntry(rewardPerNsbtSumNewKEY, sumLastStr)))
486486 }
487487
488488 ({
489489 let $l = supportedAssetsList
490490 let $s = size($l)
491491 let $acc0 = nil
492492 func $f0_1 ($a,$i) = if (($i >= $s))
493493 then $a
494494 else refreshRewardPerNsbtSUM($a, $l[$i])
495495
496496 func $f0_2 ($a,$i) = if (($i >= $s))
497497 then $a
498498 else throw("List size exceeds 10")
499499
500500 $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)
501501 } :+ IntegerEntry(depositNumLastKEY, depositNumNew))
502502 }
503503 }
504504 }
505505
506506
507507
508508 @Callable(i)
509509 func claimRewards () = commonClaim(i.caller, i)
510510
511511
512512
513513 @Callable(i)
514514 func claimRewardsByOriginCaller () = commonClaim(i.originCaller, i)
515515
516516
517517
518518 @Callable(i)
519519 func unclaimedRewardsREADONLY (userAddressStr) = {
520520 func forEachAssetZeroReward (accum,asset) = ((accum + makeString([asset, "0", "0"], ":")) + "_")
521521
522522 let unclaimedRewardStr = if ((userAddressStr == ""))
523523 then {
524524 let $l = supportedAssetsList
525525 let $s = size($l)
526526 let $acc0 = ""
527527 func $f0_1 ($a,$i) = if (($i >= $s))
528528 then $a
529529 else forEachAssetZeroReward($a, $l[$i])
530530
531531 func $f0_2 ($a,$i) = if (($i >= $s))
532532 then $a
533533 else throw("List size exceeds 10")
534534
535535 $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)
536536 }
537537 else {
538538 let userAddress = addressFromStringValue(userAddressStr)
539- let $t02048820593 = valueOrElse(getUserParamsOrUnit(userAddress), $Tuple3(true, 0, 0))
540- let isNewUser = $t02048820593._1
541- let stakedAmount = $t02048820593._2
542- let stakingStart = $t02048820593._3
539+ let $t02049720602 = valueOrElse(getUserParamsOrUnit(userAddress), $Tuple3(true, 0, 0))
540+ let isNewUser = $t02049720602._1
541+ let stakedAmount = $t02049720602._2
542+ let stakingStart = $t02049720602._3
543543 let stakedAmountX = toBigInt(stakedAmount)
544544 let userRewardFromDepositNumKEY = keyUserRewardFromDepositNum(userAddressStr)
545545 let depositNumUser = getIntOrElse(userRewardFromDepositNumKEY, -1)
546546 let depositNumLast = getIntOrElse(keyDepositNumLast(), -1)
547547 func forEachAssetCalcUnclaimedReward (accum,asset) = {
548- let $t02093921077 = calcReward(userAddressStr, asset, stakedAmountX, depositNumUser, depositNumLast)
549- let rewardTotal = $t02093921077._1
550- let cached = $t02093921077._2
551- let dynamic = $t02093921077._3
552- let rewardCachedPartKEY = $t02093921077._4
548+ let $t02094821086 = calcReward(userAddressStr, asset, stakedAmountX, depositNumUser, depositNumLast)
549+ let rewardTotal = $t02094821086._1
550+ let cached = $t02094821086._2
551+ let dynamic = $t02094821086._3
552+ let rewardCachedPartKEY = $t02094821086._4
553553 let claimed = valueOrElse(getInteger(keyClaimed(userAddressStr, asset)), 0)
554554 ((accum + makeString([asset, toString(rewardTotal), toString(claimed)], ":")) + "_")
555555 }
556556
557557 let $l = supportedAssetsList
558558 let $s = size($l)
559559 let $acc0 = ""
560560 func $f0_1 ($a,$i) = if (($i >= $s))
561561 then $a
562562 else forEachAssetCalcUnclaimedReward($a, $l[$i])
563563
564564 func $f0_2 ($a,$i) = if (($i >= $s))
565565 then $a
566566 else throw("List size exceeds 10")
567567
568568 $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)
569569 }
570570 $Tuple2(nil, dropRight(unclaimedRewardStr, 1))
571571 }
572572
573573
574574
575575 @Callable(i)
576576 func surfStakingSYSREADONLY (userAddressStrOrEmpty,surfDiff) = {
577577 let surfTotal = getIntOrElse(keyLockParamTotalAmount(), 0)
578578 let gnsbtFromSurfTotal = surfToGnsbt(surfTotal)
579579 if ((userAddressStrOrEmpty == ""))
580580 then $Tuple2(nil, [0, surfTotal, 0, gnsbtFromSurfTotal, 0, height, height])
581581 else {
582582 let userAddress = toAddressOrFail(userAddressStrOrEmpty)
583583 let mergedData = mergeStake(userAddress, surfDiff)
584584 let isNewUser = mergedData._1
585585 let stakedAmount = mergedData._2
586586 let vpEffectiveHeight = mergedData._3
587587 let stakedAmountNEW = mergedData._4
588588 let vpEffectiveHeightNEW = mergedData._5
589589 let surfUser = stakedAmount
590590 let gnsbtFromSurfUser = surfToGnsbt(surfUser)
591591 $Tuple2(nil, [surfUser, surfTotal, gnsbtFromSurfUser, gnsbtFromSurfTotal, vpEffectiveHeight, vpEffectiveHeightNEW])
592592 }
593593 }
594594
595595
596596
597597 @Callable(i)
598598 func gnsbtFromSurfSYSREADONLY (surfAmt) = $Tuple2(nil, surfToGnsbt(surfAmt))
599599
600600
601601
602602 @Callable(i)
603603 func configSYSREADONLY () = {
604604 let minLockAmt = getIntegerValue(keyMinLockAmount())
605605 let surfVotingPowerRestrictivePeriod = (1440 * 14)
606606 let brToStartSurfTransformation = 1150000
607607 $Tuple2(nil, [minLockAmt, (gnsbtFromSurfCoeff * MULT6), surfVotingPowerRestrictivePeriod, brToStartSurfTransformation])
608608 }
609609
610610

github/deemru/w8io/169f3d6 
74.00 ms