tx · B7Wf966y9YKVJPyzFPmxaWQ7GA2uqrLCF5G7fmpicnMY 3Mu5kasZ85VY5xUCpPYoWr6fBzh6eGZwcnt: -0.01000000 Waves 2023.05.22 10:45 [2588892] smart account 3Mu5kasZ85VY5xUCpPYoWr6fBzh6eGZwcnt > SELF 0.00000000 Waves
{ "type": 13, "id": "B7Wf966y9YKVJPyzFPmxaWQ7GA2uqrLCF5G7fmpicnMY", "fee": 1000000, "feeAssetId": null, "timestamp": 1684741565920, "version": 2, "chainId": 84, "sender": "3Mu5kasZ85VY5xUCpPYoWr6fBzh6eGZwcnt", "senderPublicKey": "6VehQ4nhpSn9EW76xuyThbHcUFYhjEbb4sJDrys5ZEx5", "proofs": [ "3yAbeUjocupCa7Ve7fnBPJVdw5jbPmpYjvxFS9G4P2JnMgUGYfBJvPLhf1ynnhpxJdksBdbxLX4wDvcA8FGgBVax" ], "script": "base64:BgInCAISCQoHCAEBAQEBCBIDCgEIEgMKAQESABIAEgASAwoBARIDCgEBMwADU0VQAgJfXwEHd3JhcEVycgEDbXNnCQC5CQIJAMwIAgIOZW1pc3Npb24ucmlkZToJAMwIAgkApQgBBQR0aGlzCQDMCAIFA21zZwUDbmlsAgEgAQh0aHJvd0VycgEDbXNnCQACAQkBB3dyYXBFcnIBBQNtc2cBD2dldFN0cmluZ09yRmFpbAIHYWRkcmVzcwNrZXkJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQdhZGRyZXNzBQNrZXkJAQd3cmFwRXJyAQkArAICCQCsAgICD21hbmRhdG9yeSB0aGlzLgUDa2V5Ag8gaXMgbm90IGRlZmluZWQBEWtleUZhY3RvcnlBZGRyZXNzAAIcJXMlc19fY29uZmlnX19mYWN0b3J5QWRkcmVzcwAZa2V5Vm90aW5nVmVyaWZpZWRDb250cmFjdAIkJXMlc19fY29uZmlnX192b3RpbmdWZXJpZmllZENvbnRyYWN0ACJrZXlWb3RpbmdFbWlzc2lvbkNhbmRpZGF0ZUNvbnRyYWN0Ai0lcyVzX19jb25maWdfX3ZvdGluZ0VtaXNzaW9uQ2FuZGlkYXRlQ29udHJhY3QAFGtleVVzZXJQb29sc0NvbnRyYWN0Ah8lcyVzX19jb25maWdfX3VzZXJQb29sc0NvbnRyYWN0AB1rZXlWb3RpbmdFbWlzc2lvblJhdGVDb250cmFjdAIoJXMlc19fY29uZmlnX192b3RpbmdFbWlzc2lvblJhdGVDb250cmFjdAAWdm90aW5nVmVyaWZpZWRDb250cmFjdAkBEUBleHRyTmF0aXZlKDEwNjIpAQkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMFGWtleVZvdGluZ1ZlcmlmaWVkQ29udHJhY3QAH3ZvdGluZ0VtaXNzaW9uQ2FuZGlkYXRlQ29udHJhY3QJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQ9nZXRTdHJpbmdPckZhaWwCBQR0aGlzBSJrZXlWb3RpbmdFbWlzc2lvbkNhbmRpZGF0ZUNvbnRyYWN0ABF1c2VyUG9vbHNDb250cmFjdAkBEUBleHRyTmF0aXZlKDEwNjIpAQkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMFFGtleVVzZXJQb29sc0NvbnRyYWN0ABp2b3RpbmdFbWlzc2lvblJhdGVDb250cmFjdAkBEUBleHRyTmF0aXZlKDEwNjIpAQkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMFHWtleVZvdGluZ0VtaXNzaW9uUmF0ZUNvbnRyYWN0ABhJZHhGYWN0b3J5Q2ZnU3Rha2luZ0RhcHAAAQAZSWR4RmFjdG9yeUNmZ0Jvb3N0aW5nRGFwcAACABRJZHhGYWN0b3J5Q2ZnSWRvRGFwcAADABVJZHhGYWN0b3J5Q2ZnVGVhbURhcHAABAAZSWR4RmFjdG9yeUNmZ0VtaXNzaW9uRGFwcAAFABVJZHhGYWN0b3J5Q2ZnUmVzdERhcHAABgAZSWR4RmFjdG9yeUNmZ1NsaXBwYWdlRGFwcAAHABpJZHhGYWN0b3J5Q2ZnR3d4UmV3YXJkRGFwcAAKAQ1rZXlGYWN0b3J5Q2ZnAAIRJXNfX2ZhY3RvcnlDb25maWcBGmtleUZhY3RvcnlMcDJBc3NldHNNYXBwaW5nAQpscEFzc2V0U3RyCQC5CQIJAMwIAgIGJXMlcyVzCQDMCAIFCmxwQXNzZXRTdHIJAMwIAgIebWFwcGluZ3NfX2xwQXNzZXQyUG9vbENvbnRyYWN0BQNuaWwFA1NFUAEQa2V5RmFjdG9yeUxwTGlzdAACECVzX19scFRva2Vuc0xpc3QBJmtleUZhY3RvcnlMcEFzc2V0VG9Qb29sQ29udHJhY3RBZGRyZXNzAQpscEFzc2V0U3RyCQC5CQIJAMwIAgIGJXMlcyVzCQDMCAIFCmxwQXNzZXRTdHIJAMwIAgIebWFwcGluZ3NfX2xwQXNzZXQyUG9vbENvbnRyYWN0BQNuaWwFA1NFUAEUa2V5RmFjdG9yeVBvb2xXZWlnaHQBD2NvbnRyYWN0QWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgIKcG9vbFdlaWdodAkAzAgCBQ9jb250cmFjdEFkZHJlc3MFA25pbAUDU0VQARhyZWFkRmFjdG9yeUFkZHJlc3NPckZhaWwACQERQGV4dHJOYXRpdmUoMTA2MikBCQEPZ2V0U3RyaW5nT3JGYWlsAgUEdGhpcwkBEWtleUZhY3RvcnlBZGRyZXNzAAEKcmVhZExwTGlzdAAJALUJAgkBC3ZhbHVlT3JFbHNlAgkAnQgCCQEYcmVhZEZhY3RvcnlBZGRyZXNzT3JGYWlsAAkBEGtleUZhY3RvcnlMcExpc3QAAgAFA1NFUAEUcmVhZEZhY3RvcnlDZmdPckZhaWwBB2ZhY3RvcnkJALUJAgkBD2dldFN0cmluZ09yRmFpbAIFB2ZhY3RvcnkJAQ1rZXlGYWN0b3J5Q2ZnAAUDU0VQARhnZXRCb29zdGluZ0FkZHJlc3NPckZhaWwBCmZhY3RvcnlDZmcJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAJEDAgUKZmFjdG9yeUNmZwUZSWR4RmFjdG9yeUNmZ0Jvb3N0aW5nRGFwcAEYZ2V0RW1pc3Npb25BZGRyZXNzT3JGYWlsAQpmYWN0b3J5Q2ZnCQERQGV4dHJOYXRpdmUoMTA2MikBCQCRAwIFCmZhY3RvcnlDZmcFGUlkeEZhY3RvcnlDZmdFbWlzc2lvbkRhcHABF2dldFN0YWtpbmdBZGRyZXNzT3JGYWlsAQpmYWN0b3J5Q2ZnCQERQGV4dHJOYXRpdmUoMTA2MikBCQCRAwIFCmZhY3RvcnlDZmcFGElkeEZhY3RvcnlDZmdTdGFraW5nRGFwcAEZZ2V0R3d4UmV3YXJkQWRkcmVzc09yRmFpbAEKZmFjdG9yeUNmZwkBEUBleHRyTmF0aXZlKDEwNjIpAQkAkQMCBQpmYWN0b3J5Q2ZnBRpJZHhGYWN0b3J5Q2ZnR3d4UmV3YXJkRGFwcAEJa2V5Q29uZmlnAAIKJXNfX2NvbmZpZwEZa2V5UmF0ZVBlckJsb2NrTWF4Q3VycmVudAACHiVzJXNfX3JhdGVQZXJCbG9ja01heF9fY3VycmVudAEba2V5UmF0ZVBlckJsb2NrTWF4U3RhcnRGcm9tAgl0aW1lc3RhbXAFYmxvY2sJALkJAgkAzAgCAiAlcyVzJWQlZF9fcmF0ZVBlckJsb2NrTWF4X19zdGFydAkAzAgCCQCkAwEFCXRpbWVzdGFtcAkAzAgCCQCkAwEFBWJsb2NrBQNuaWwFA1NFUAEWa2V5UmF0ZVBlckJsb2NrQ3VycmVudAACGyVzJXNfX3JhdGVQZXJCbG9ja19fY3VycmVudAEYa2V5UmF0ZVBlckJsb2NrU3RhcnRGcm9tAgl0aW1lc3RhbXAFYmxvY2sJALkJAgkAzAgCAh0lcyVzJWQlZF9fcmF0ZVBlckJsb2NrX19zdGFydAkAzAgCCQCkAwEFCXRpbWVzdGFtcAkAzAgCCQCkAwEFBWJsb2NrBQNuaWwFA1NFUAEVa2V5RW1pc3Npb25TdGFydEJsb2NrAAIaJXMlc19fZW1pc3Npb25fX3N0YXJ0QmxvY2sBG2tleUVtaXNzaW9uRHVyYXRpb25JbkJsb2NrcwACGCVzJXNfX2VtaXNzaW9uX19kdXJhdGlvbgETa2V5RW1pc3Npb25FbmRCbG9jawACGCVzJXNfX2VtaXNzaW9uX19lbmRCbG9jawENa2V5Qm9vc3RDb2VmZgACDiVzX19ib29zdENvZWZmABFib29zdENvZWZmRGVmYXVsdAADAApib29zdENvZWZmCQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMJAQ1rZXlCb29zdENvZWZmAAURYm9vc3RDb2VmZkRlZmF1bHQBGmtleUd3eEhvbGRlcnNSZXdhcmRDdXJyZW50AAIfJXMlc19fZ3d4SG9sZGVyc1Jld2FyZF9fY3VycmVudAEXa2V5R3d4SG9sZGVyc1Jld2FyZE5leHQAAhwlcyVzX19nd3hIb2xkZXJzUmV3YXJkX19uZXh0ARNrZXlNYW5hZ2VyUHVibGljS2V5AAIUJXNfX21hbmFnZXJQdWJsaWNLZXkBFmtleU1hbmFnZXJWYXVsdEFkZHJlc3MAAhclc19fbWFuYWdlclZhdWx0QWRkcmVzcwEcZ2V0TWFuYWdlclZhdWx0QWRkcmVzc09yVGhpcwAEByRtYXRjaDAJAKIIAQkBFmtleU1hbmFnZXJWYXVsdEFkZHJlc3MAAwkAAQIFByRtYXRjaDACBlN0cmluZwQBcwUHJG1hdGNoMAkBEUBleHRyTmF0aXZlKDEwNjIpAQUBcwUEdGhpcwEWbWFuYWdlclB1YmxpY0tleU9yVW5pdAAEE21hbmFnZXJWYXVsdEFkZHJlc3MJARxnZXRNYW5hZ2VyVmF1bHRBZGRyZXNzT3JUaGlzAAQHJG1hdGNoMAkAnQgCBRNtYW5hZ2VyVmF1bHRBZGRyZXNzCQETa2V5TWFuYWdlclB1YmxpY0tleQADCQABAgUHJG1hdGNoMAIGU3RyaW5nBAFzBQckbWF0Y2gwCQDZBAEFAXMDCQABAgUHJG1hdGNoMAIEVW5pdAUEdW5pdAkAAgECC01hdGNoIGVycm9yAQttdXN0TWFuYWdlcgEBaQQCcGQJAQh0aHJvd0VycgECEVBlcm1pc3Npb24gZGVuaWVkBAckbWF0Y2gwCQEWbWFuYWdlclB1YmxpY0tleU9yVW5pdAADCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQCcGsFByRtYXRjaDADCQAAAggFAWkPY2FsbGVyUHVibGljS2V5BQJwawYFAnBkAwkAAQIFByRtYXRjaDACBFVuaXQDCQAAAggFAWkGY2FsbGVyBQR0aGlzBgUCcGQJAAIBAgtNYXRjaCBlcnJvcggBaQELY29uc3RydWN0b3IHDmZhY3RvcnlBZGRyZXNzD3JhdGVQZXJCbG9ja01heAxyYXRlUGVyQmxvY2sSZW1pc3Npb25TdGFydEJsb2NrEGVtaXNzaW9uRHVyYXRpb24WZW1pc3Npb25TdGFydFRpbWVzdGFtcAx3eEFzc2V0SWRTdHIEC2NoZWNrQ2FsbGVyCQELbXVzdE1hbmFnZXIBBQFpAwkAAAIFC2NoZWNrQ2FsbGVyBQtjaGVja0NhbGxlcgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBG2tleVJhdGVQZXJCbG9ja01heFN0YXJ0RnJvbQIFFmVtaXNzaW9uU3RhcnRUaW1lc3RhbXAFEmVtaXNzaW9uU3RhcnRCbG9jawUPcmF0ZVBlckJsb2NrTWF4CQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEZa2V5UmF0ZVBlckJsb2NrTWF4Q3VycmVudAAFD3JhdGVQZXJCbG9ja01heAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBGGtleVJhdGVQZXJCbG9ja1N0YXJ0RnJvbQIFFmVtaXNzaW9uU3RhcnRUaW1lc3RhbXAFEmVtaXNzaW9uU3RhcnRCbG9jawUMcmF0ZVBlckJsb2NrCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEWa2V5UmF0ZVBlckJsb2NrQ3VycmVudAAFDHJhdGVQZXJCbG9jawkAzAgCCQEMSW50ZWdlckVudHJ5AgkBFWtleUVtaXNzaW9uU3RhcnRCbG9jawAFEmVtaXNzaW9uU3RhcnRCbG9jawkAzAgCCQEMSW50ZWdlckVudHJ5AgkBG2tleUVtaXNzaW9uRHVyYXRpb25JbkJsb2NrcwAFEGVtaXNzaW9uRHVyYXRpb24JAMwIAgkBDEludGVnZXJFbnRyeQIJARNrZXlFbWlzc2lvbkVuZEJsb2NrAAkAZAIFEmVtaXNzaW9uU3RhcnRCbG9jawUQZW1pc3Npb25EdXJhdGlvbgkAzAgCCQELU3RyaW5nRW50cnkCCQERa2V5RmFjdG9yeUFkZHJlc3MABQ5mYWN0b3J5QWRkcmVzcwkAzAgCCQELU3RyaW5nRW50cnkCCQEJa2V5Q29uZmlnAAkArAICAgQlc19fBQx3eEFzc2V0SWRTdHIFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQ1jb25zdHJ1Y3RvclYyARl2b3RpbmdWZXJpZmllZENvbnRyYWN0UHJtBAtjaGVja0NhbGxlcgkBC211c3RNYW5hZ2VyAQUBaQMJAAACBQtjaGVja0NhbGxlcgULY2hlY2tDYWxsZXIJAJQKAgkAzAgCCQELU3RyaW5nRW50cnkCBRlrZXlWb3RpbmdWZXJpZmllZENvbnRyYWN0BRl2b3RpbmdWZXJpZmllZENvbnRyYWN0UHJtBQNuaWwFBHVuaXQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEEZW1pdAEGYW1vdW50AwkAZwIAAAUGYW1vdW50CQCUCgIFA25pbAUDbmlsBA9mYWN0b3J5Q29udHJhY3QJARhyZWFkRmFjdG9yeUFkZHJlc3NPckZhaWwABApmYWN0b3J5Q2ZnCQEUcmVhZEZhY3RvcnlDZmdPckZhaWwBBQ9mYWN0b3J5Q29udHJhY3QED3N0YWtpbmdDb250cmFjdAkBF2dldFN0YWtpbmdBZGRyZXNzT3JGYWlsAQUKZmFjdG9yeUNmZwQSZ3d4UmV3YXJkc0NvbnRyYWN0CQEZZ2V0R3d4UmV3YXJkQWRkcmVzc09yRmFpbAEFCmZhY3RvcnlDZmcEBmNhbGxlcggFAWkGY2FsbGVyAwMJAQIhPQIFBmNhbGxlcgUPc3Rha2luZ0NvbnRyYWN0CQECIT0CBQZjYWxsZXIFEmd3eFJld2FyZHNDb250cmFjdAcJAQh0aHJvd0VycgECEnBlcm1pc3Npb25zIGRlbmllZAQMd3hBc3NldElkU3RyCQCRAwIJALUJAgkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMJAQlrZXlDb25maWcABQNTRVAAAQQJd3hBc3NldElkCQDZBAEFDHd4QXNzZXRJZFN0cgkAlAoCCQDMCAIJAQ5TY3JpcHRUcmFuc2ZlcgMFBmNhbGxlcgUGYW1vdW50BQl3eEFzc2V0SWQFA25pbAkAzAgCBQl3eEFzc2V0SWQFA25pbAFpAQRidXJuAAQPZmFjdG9yeUNvbnRyYWN0CQEYcmVhZEZhY3RvcnlBZGRyZXNzT3JGYWlsAAQKZmFjdG9yeUNmZwkBFHJlYWRGYWN0b3J5Q2ZnT3JGYWlsAQUPZmFjdG9yeUNvbnRyYWN0BBBib29zdGluZ0NvbnRyYWN0CQEYZ2V0Qm9vc3RpbmdBZGRyZXNzT3JGYWlsAQUKZmFjdG9yeUNmZwMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQkBCHRocm93RXJyAQIcZXhhY3Qgb25lIHBheW1lbnQgaXMgYWxsb3dlZAQDcG10CQCRAwIIBQFpCHBheW1lbnRzAAAEB2Fzc2V0SWQJAQV2YWx1ZQEIBQNwbXQHYXNzZXRJZAQJcG10QW1vdW50CAUDcG10BmFtb3VudAQJd3hBc3NldElkCQDZBAEJAJEDAgkAtQkCCQEPZ2V0U3RyaW5nT3JGYWlsAgUEdGhpcwkBCWtleUNvbmZpZwAFA1NFUAABAwkBAiE9AgUHYXNzZXRJZAUJd3hBc3NldElkCQEIdGhyb3dFcnIBAhtpbnZhbGlkIHd4QXNzZXRJZCBpcyBwYXNzZWQDCQEBIQEJAQ9jb250YWluc0VsZW1lbnQCCQDMCAIFEGJvb3N0aW5nQ29udHJhY3QJAMwIAgUWdm90aW5nVmVyaWZpZWRDb250cmFjdAkAzAgCBR92b3RpbmdFbWlzc2lvbkNhbmRpZGF0ZUNvbnRyYWN0CQDMCAIFEXVzZXJQb29sc0NvbnRyYWN0BQNuaWwIBQFpBmNhbGxlcgkBCHRocm93RXJyAQIScGVybWlzc2lvbnMgZGVuaWVkCQDMCAIJAQRCdXJuAgUHYXNzZXRJZAUJcG10QW1vdW50CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgIXJXMlcyVzX19oaXN0b3J5X19idXJuX18JANgEAQgFAWkNdHJhbnNhY3Rpb25JZAkAuQkCCQDMCAICBiVkJWQlZAkAzAgCCQCkAwEFBmhlaWdodAkAzAgCCQCkAwEIBQlsYXN0QmxvY2sJdGltZXN0YW1wCQDMCAIJAKQDAQUJcG10QW1vdW50BQNuaWwFA1NFUAUDbmlsAWkBFmd3eEhvbGRlcnNSZXdhcmRVcGRhdGUABA9mYWN0b3J5Q29udHJhY3QJARhyZWFkRmFjdG9yeUFkZHJlc3NPckZhaWwABApmYWN0b3J5Q2ZnCQEUcmVhZEZhY3RvcnlDZmdPckZhaWwBBQ9mYWN0b3J5Q29udHJhY3QEEmd3eFJld2FyZHNDb250cmFjdAkBGWdldEd3eFJld2FyZEFkZHJlc3NPckZhaWwBBQpmYWN0b3J5Q2ZnBAtjaGVja0NhbGxlcgMJAAACCAUBaQZjYWxsZXIFEmd3eFJld2FyZHNDb250cmFjdAYJAQttdXN0TWFuYWdlcgEFAWkDCQAAAgULY2hlY2tDYWxsZXIFC2NoZWNrQ2FsbGVyBBZnd3hIb2xkZXJSZXdhcmRDdXJyZW50CQELdmFsdWVPckVsc2UCCQCfCAEJARprZXlHd3hIb2xkZXJzUmV3YXJkQ3VycmVudAAAAAQTZ3d4SG9sZGVyUmV3YXJkTmV4dAkBC3ZhbHVlT3JFbHNlAgkAnwgBCQEXa2V5R3d4SG9sZGVyc1Jld2FyZE5leHQABRZnd3hIb2xkZXJSZXdhcmRDdXJyZW50BA9zaG91bGRCZVVwZGF0ZWQJAQIhPQIFFmd3eEhvbGRlclJld2FyZEN1cnJlbnQFE2d3eEhvbGRlclJld2FyZE5leHQEB2FjdGlvbnMDBQ9zaG91bGRCZVVwZGF0ZWQJAMwIAgkBDEludGVnZXJFbnRyeQIJARprZXlHd3hIb2xkZXJzUmV3YXJkQ3VycmVudAAFE2d3eEhvbGRlclJld2FyZE5leHQJAMwIAgkBC0RlbGV0ZUVudHJ5AQkBF2tleUd3eEhvbGRlcnNSZXdhcmROZXh0AAUDbmlsBQNuaWwJAJQKAgUHYWN0aW9ucwUPc2hvdWxkQmVVcGRhdGVkCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBFWdldEJvb3N0Q29lZmZSRUFET05MWQAJAJQKAgUDbmlsBQpib29zdENvZWZmAWkBGmNoYW5nZVJhdGVQZXJCbG9ja0lOVEVSTkFMAQ9uZXdSYXRlUGVyQmxvY2sEBWNoZWNrAwkAAAIIBQFpBmNhbGxlcgUEdGhpcwYJAQh0aHJvd0VycgECHHNob3VsZCBiZSBpbnZva2VkIGludGVybmFsbHkDCQAAAgUFY2hlY2sFBWNoZWNrBA5ibG9ja1RpbWVzdGFtcAQHJG1hdGNoMAkA7QcBBQZoZWlnaHQDCQABAgUHJG1hdGNoMAIJQmxvY2tJbmZvBAJiaQUHJG1hdGNoMAgFAmJpCXRpbWVzdGFtcAkBCHRocm93RXJyAQIZQmxvY2tJbmZvLnRpbWVzdGFtcCBlcnJvcgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBGGtleVJhdGVQZXJCbG9ja1N0YXJ0RnJvbQIFDmJsb2NrVGltZXN0YW1wBQZoZWlnaHQFD25ld1JhdGVQZXJCbG9jawkAzAgCCQEMSW50ZWdlckVudHJ5AgkBFmtleVJhdGVQZXJCbG9ja0N1cnJlbnQABQ9uZXdSYXRlUGVyQmxvY2sFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARJjaGFuZ2VSYXRlUGVyQmxvY2sBD25ld1JhdGVQZXJCbG9jawQFY2hlY2sDCQAAAggFAWkGY2FsbGVyBRp2b3RpbmdFbWlzc2lvblJhdGVDb250cmFjdAYJAQh0aHJvd0VycgECL3Nob3VsZCBiZSBpbnZva2VkIGJ5IHZvdGluZ0VtaXNzaW9uUmF0ZUNvbnRyYWN0AwkAAAIFBWNoZWNrBQVjaGVjawQPZmFjdG9yeUNvbnRyYWN0CQEYcmVhZEZhY3RvcnlBZGRyZXNzT3JGYWlsAAQKZmFjdG9yeUNmZwkBFHJlYWRGYWN0b3J5Q2ZnT3JGYWlsAQUPZmFjdG9yeUNvbnRyYWN0BBBib29zdGluZ0NvbnRyYWN0CQEYZ2V0Qm9vc3RpbmdBZGRyZXNzT3JGYWlsAQUKZmFjdG9yeUNmZwQLaW50ZXJuYWxJbnYJAPwHBAUEdGhpcwIaY2hhbmdlUmF0ZVBlckJsb2NrSU5URVJOQUwJAMwIAgUPbmV3UmF0ZVBlckJsb2NrBQNuaWwFA25pbAMJAAACBQtpbnRlcm5hbEludgULaW50ZXJuYWxJbnYEDmJvb3N0aW5nSW52b2tlCQD9BwQFEGJvb3N0aW5nQ29udHJhY3QCFW9uQm9vc3RFbWlzc2lvblVwZGF0ZQUDbmlsBQNuaWwJAJQKAgUDbmlsBQ5ib29zdGluZ0ludm9rZQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgECdHgBBnZlcmlmeQAED3RhcmdldFB1YmxpY0tleQQHJG1hdGNoMAkBFm1hbmFnZXJQdWJsaWNLZXlPclVuaXQAAwkAAQIFByRtYXRjaDACCkJ5dGVWZWN0b3IEAnBrBQckbWF0Y2gwBQJwawMJAAECBQckbWF0Y2gwAgRVbml0CAUCdHgPc2VuZGVyUHVibGljS2V5CQACAQILTWF0Y2ggZXJyb3IJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAAFD3RhcmdldFB1YmxpY0tleaOvxFs=", "height": 2588892, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 9NDyJKWYnWDKMHQFP2czF5Z7CZPRE9oTZVCDLmQsP48b Next: 8cTTKCu9mNNEmPMnZGfWW83a2WK5TKZ8N7RrBh3rjAcS Diff:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let SEP = "__" | |
5 | + | ||
6 | + | func wrapErr (msg) = makeString(["emission.ride:", toString(this), msg], " ") | |
7 | + | ||
8 | + | ||
9 | + | func throwErr (msg) = throw(wrapErr(msg)) | |
10 | + | ||
11 | + | ||
12 | + | func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), wrapErr((("mandatory this." + key) + " is not defined"))) | |
13 | + | ||
14 | + | ||
15 | + | func keyFactoryAddress () = "%s%s__config__factoryAddress" | |
16 | + | ||
17 | + | ||
18 | + | let keyVotingVerifiedContract = "%s%s__config__votingVerifiedContract" | |
19 | + | ||
20 | + | let keyVotingEmissionCandidateContract = "%s%s__config__votingEmissionCandidateContract" | |
21 | + | ||
22 | + | let keyUserPoolsContract = "%s%s__config__userPoolsContract" | |
23 | + | ||
24 | + | let keyVotingEmissionRateContract = "%s%s__config__votingEmissionRateContract" | |
25 | + | ||
26 | + | let votingVerifiedContract = addressFromStringValue(getStringOrFail(this, keyVotingVerifiedContract)) | |
27 | + | ||
28 | + | let votingEmissionCandidateContract = addressFromStringValue(getStringOrFail(this, keyVotingEmissionCandidateContract)) | |
29 | + | ||
30 | + | let userPoolsContract = addressFromStringValue(getStringOrFail(this, keyUserPoolsContract)) | |
31 | + | ||
32 | + | let votingEmissionRateContract = addressFromStringValue(getStringOrFail(this, keyVotingEmissionRateContract)) | |
33 | + | ||
34 | + | let IdxFactoryCfgStakingDapp = 1 | |
35 | + | ||
36 | + | let IdxFactoryCfgBoostingDapp = 2 | |
37 | + | ||
38 | + | let IdxFactoryCfgIdoDapp = 3 | |
39 | + | ||
40 | + | let IdxFactoryCfgTeamDapp = 4 | |
41 | + | ||
42 | + | let IdxFactoryCfgEmissionDapp = 5 | |
43 | + | ||
44 | + | let IdxFactoryCfgRestDapp = 6 | |
45 | + | ||
46 | + | let IdxFactoryCfgSlippageDapp = 7 | |
47 | + | ||
48 | + | let IdxFactoryCfgGwxRewardDapp = 10 | |
49 | + | ||
50 | + | func keyFactoryCfg () = "%s__factoryConfig" | |
51 | + | ||
52 | + | ||
53 | + | func keyFactoryLp2AssetsMapping (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
54 | + | ||
55 | + | ||
56 | + | func keyFactoryLpList () = "%s__lpTokensList" | |
57 | + | ||
58 | + | ||
59 | + | func keyFactoryLpAssetToPoolContractAddress (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
60 | + | ||
61 | + | ||
62 | + | func keyFactoryPoolWeight (contractAddress) = makeString(["%s%s", "poolWeight", contractAddress], SEP) | |
63 | + | ||
64 | + | ||
65 | + | func readFactoryAddressOrFail () = addressFromStringValue(getStringOrFail(this, keyFactoryAddress())) | |
66 | + | ||
67 | + | ||
68 | + | func readLpList () = split(valueOrElse(getString(readFactoryAddressOrFail(), keyFactoryLpList()), ""), SEP) | |
69 | + | ||
70 | + | ||
71 | + | func readFactoryCfgOrFail (factory) = split(getStringOrFail(factory, keyFactoryCfg()), SEP) | |
72 | + | ||
73 | + | ||
74 | + | func getBoostingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgBoostingDapp]) | |
75 | + | ||
76 | + | ||
77 | + | func getEmissionAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgEmissionDapp]) | |
78 | + | ||
79 | + | ||
80 | + | func getStakingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgStakingDapp]) | |
81 | + | ||
82 | + | ||
83 | + | func getGwxRewardAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgGwxRewardDapp]) | |
84 | + | ||
85 | + | ||
86 | + | func keyConfig () = "%s__config" | |
87 | + | ||
88 | + | ||
89 | + | func keyRatePerBlockMaxCurrent () = "%s%s__ratePerBlockMax__current" | |
90 | + | ||
91 | + | ||
92 | + | func keyRatePerBlockMaxStartFrom (timestamp,block) = makeString(["%s%s%d%d__ratePerBlockMax__start", toString(timestamp), toString(block)], SEP) | |
93 | + | ||
94 | + | ||
95 | + | func keyRatePerBlockCurrent () = "%s%s__ratePerBlock__current" | |
96 | + | ||
97 | + | ||
98 | + | func keyRatePerBlockStartFrom (timestamp,block) = makeString(["%s%s%d%d__ratePerBlock__start", toString(timestamp), toString(block)], SEP) | |
99 | + | ||
100 | + | ||
101 | + | func keyEmissionStartBlock () = "%s%s__emission__startBlock" | |
102 | + | ||
103 | + | ||
104 | + | func keyEmissionDurationInBlocks () = "%s%s__emission__duration" | |
105 | + | ||
106 | + | ||
107 | + | func keyEmissionEndBlock () = "%s%s__emission__endBlock" | |
108 | + | ||
109 | + | ||
110 | + | func keyBoostCoeff () = "%s__boostCoeff" | |
111 | + | ||
112 | + | ||
113 | + | let boostCoeffDefault = 3 | |
114 | + | ||
115 | + | let boostCoeff = valueOrElse(getInteger(this, keyBoostCoeff()), boostCoeffDefault) | |
116 | + | ||
117 | + | func keyGwxHoldersRewardCurrent () = "%s%s__gwxHoldersReward__current" | |
118 | + | ||
119 | + | ||
120 | + | func keyGwxHoldersRewardNext () = "%s%s__gwxHoldersReward__next" | |
121 | + | ||
122 | + | ||
123 | + | func keyManagerPublicKey () = "%s__managerPublicKey" | |
124 | + | ||
125 | + | ||
126 | + | func keyManagerVaultAddress () = "%s__managerVaultAddress" | |
127 | + | ||
128 | + | ||
129 | + | func getManagerVaultAddressOrThis () = match getString(keyManagerVaultAddress()) { | |
130 | + | case s: String => | |
131 | + | addressFromStringValue(s) | |
132 | + | case _ => | |
133 | + | this | |
134 | + | } | |
135 | + | ||
136 | + | ||
137 | + | func managerPublicKeyOrUnit () = { | |
138 | + | let managerVaultAddress = getManagerVaultAddressOrThis() | |
139 | + | match getString(managerVaultAddress, keyManagerPublicKey()) { | |
140 | + | case s: String => | |
141 | + | fromBase58String(s) | |
142 | + | case _: Unit => | |
143 | + | unit | |
144 | + | case _ => | |
145 | + | throw("Match error") | |
146 | + | } | |
147 | + | } | |
148 | + | ||
149 | + | ||
150 | + | func mustManager (i) = { | |
151 | + | let pd = throwErr("Permission denied") | |
152 | + | match managerPublicKeyOrUnit() { | |
153 | + | case pk: ByteVector => | |
154 | + | if ((i.callerPublicKey == pk)) | |
155 | + | then true | |
156 | + | else pd | |
157 | + | case _: Unit => | |
158 | + | if ((i.caller == this)) | |
159 | + | then true | |
160 | + | else pd | |
161 | + | case _ => | |
162 | + | throw("Match error") | |
163 | + | } | |
164 | + | } | |
4 | 165 | ||
5 | 166 | ||
6 | 167 | @Callable(i) | |
7 | - | func getNum () = $Tuple2(nil, 7) | |
168 | + | func constructor (factoryAddress,ratePerBlockMax,ratePerBlock,emissionStartBlock,emissionDuration,emissionStartTimestamp,wxAssetIdStr) = { | |
169 | + | let checkCaller = mustManager(i) | |
170 | + | if ((checkCaller == checkCaller)) | |
171 | + | then [IntegerEntry(keyRatePerBlockMaxStartFrom(emissionStartTimestamp, emissionStartBlock), ratePerBlockMax), IntegerEntry(keyRatePerBlockMaxCurrent(), ratePerBlockMax), IntegerEntry(keyRatePerBlockStartFrom(emissionStartTimestamp, emissionStartBlock), ratePerBlock), IntegerEntry(keyRatePerBlockCurrent(), ratePerBlock), IntegerEntry(keyEmissionStartBlock(), emissionStartBlock), IntegerEntry(keyEmissionDurationInBlocks(), emissionDuration), IntegerEntry(keyEmissionEndBlock(), (emissionStartBlock + emissionDuration)), StringEntry(keyFactoryAddress(), factoryAddress), StringEntry(keyConfig(), ("%s__" + wxAssetIdStr))] | |
172 | + | else throw("Strict value is not equal to itself.") | |
173 | + | } | |
8 | 174 | ||
9 | 175 | ||
10 | 176 | ||
11 | 177 | @Callable(i) | |
12 | - | func setterINTERNAL (n) = { | |
13 | - | let check = if ((i.caller == this)) | |
178 | + | func constructorV2 (votingVerifiedContractPrm) = { | |
179 | + | let checkCaller = mustManager(i) | |
180 | + | if ((checkCaller == checkCaller)) | |
181 | + | then $Tuple2([StringEntry(keyVotingVerifiedContract, votingVerifiedContractPrm)], unit) | |
182 | + | else throw("Strict value is not equal to itself.") | |
183 | + | } | |
184 | + | ||
185 | + | ||
186 | + | ||
187 | + | @Callable(i) | |
188 | + | func emit (amount) = if ((0 >= amount)) | |
189 | + | then $Tuple2(nil, nil) | |
190 | + | else { | |
191 | + | let factoryContract = readFactoryAddressOrFail() | |
192 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
193 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
194 | + | let gwxRewardsContract = getGwxRewardAddressOrFail(factoryCfg) | |
195 | + | let caller = i.caller | |
196 | + | if (if ((caller != stakingContract)) | |
197 | + | then (caller != gwxRewardsContract) | |
198 | + | else false) | |
199 | + | then throwErr("permissions denied") | |
200 | + | else { | |
201 | + | let wxAssetIdStr = split(getStringOrFail(this, keyConfig()), SEP)[1] | |
202 | + | let wxAssetId = fromBase58String(wxAssetIdStr) | |
203 | + | $Tuple2([ScriptTransfer(caller, amount, wxAssetId)], [wxAssetId]) | |
204 | + | } | |
205 | + | } | |
206 | + | ||
207 | + | ||
208 | + | ||
209 | + | @Callable(i) | |
210 | + | func burn () = { | |
211 | + | let factoryContract = readFactoryAddressOrFail() | |
212 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
213 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
214 | + | if ((size(i.payments) != 1)) | |
215 | + | then throwErr("exact one payment is allowed") | |
216 | + | else { | |
217 | + | let pmt = i.payments[0] | |
218 | + | let assetId = value(pmt.assetId) | |
219 | + | let pmtAmount = pmt.amount | |
220 | + | let wxAssetId = fromBase58String(split(getStringOrFail(this, keyConfig()), SEP)[1]) | |
221 | + | if ((assetId != wxAssetId)) | |
222 | + | then throwErr("invalid wxAssetId is passed") | |
223 | + | else if (!(containsElement([boostingContract, votingVerifiedContract, votingEmissionCandidateContract, userPoolsContract], i.caller))) | |
224 | + | then throwErr("permissions denied") | |
225 | + | else [Burn(assetId, pmtAmount), StringEntry(("%s%s%s__history__burn__" + toBase58String(i.transactionId)), makeString(["%d%d%d", toString(height), toString(lastBlock.timestamp), toString(pmtAmount)], SEP))] | |
226 | + | } | |
227 | + | } | |
228 | + | ||
229 | + | ||
230 | + | ||
231 | + | @Callable(i) | |
232 | + | func gwxHoldersRewardUpdate () = { | |
233 | + | let factoryContract = readFactoryAddressOrFail() | |
234 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
235 | + | let gwxRewardsContract = getGwxRewardAddressOrFail(factoryCfg) | |
236 | + | let checkCaller = if ((i.caller == gwxRewardsContract)) | |
14 | 237 | then true | |
15 | - | else | |
16 | - | if (( | |
238 | + | else mustManager(i) | |
239 | + | if ((checkCaller == checkCaller)) | |
17 | 240 | then { | |
18 | - | let | |
19 | - | | |
20 | - | | |
21 | - | | |
22 | - | | |
23 | - | | |
24 | - | ||
241 | + | let gwxHolderRewardCurrent = valueOrElse(getInteger(keyGwxHoldersRewardCurrent()), 0) | |
242 | + | let gwxHolderRewardNext = valueOrElse(getInteger(keyGwxHoldersRewardNext()), gwxHolderRewardCurrent) | |
243 | + | let shouldBeUpdated = (gwxHolderRewardCurrent != gwxHolderRewardNext) | |
244 | + | let actions = if (shouldBeUpdated) | |
245 | + | then [IntegerEntry(keyGwxHoldersRewardCurrent(), gwxHolderRewardNext), DeleteEntry(keyGwxHoldersRewardNext())] | |
246 | + | else nil | |
247 | + | $Tuple2(actions, shouldBeUpdated) | |
25 | 248 | } | |
26 | 249 | else throw("Strict value is not equal to itself.") | |
27 | 250 | } | |
29 | 252 | ||
30 | 253 | ||
31 | 254 | @Callable(i) | |
32 | - | func setter (n) = { | |
33 | - | let otherContract = addressFromStringValue("3MpMuGN8rW9EpPwzETAQz88Pe9TQhSwiriq") | |
34 | - | let tc = invoke(this, "setterINTERNAL", [n], nil) | |
35 | - | if ((tc == tc)) | |
255 | + | func getBoostCoeffREADONLY () = $Tuple2(nil, boostCoeff) | |
256 | + | ||
257 | + | ||
258 | + | ||
259 | + | @Callable(i) | |
260 | + | func changeRatePerBlockINTERNAL (newRatePerBlock) = { | |
261 | + | let check = if ((i.caller == this)) | |
262 | + | then true | |
263 | + | else throwErr("should be invoked internally") | |
264 | + | if ((check == check)) | |
36 | 265 | then { | |
37 | - | let oc = invoke(otherContract, "update", nil, nil) | |
38 | - | $Tuple2(nil, oc) | |
266 | + | let blockTimestamp = match blockInfoByHeight(height) { | |
267 | + | case bi: BlockInfo => | |
268 | + | bi.timestamp | |
269 | + | case _ => | |
270 | + | throwErr("BlockInfo.timestamp error") | |
271 | + | } | |
272 | + | [IntegerEntry(keyRatePerBlockStartFrom(blockTimestamp, height), newRatePerBlock), IntegerEntry(keyRatePerBlockCurrent(), newRatePerBlock)] | |
39 | 273 | } | |
40 | 274 | else throw("Strict value is not equal to itself.") | |
41 | 275 | } | |
42 | 276 | ||
43 | 277 | ||
278 | + | ||
279 | + | @Callable(i) | |
280 | + | func changeRatePerBlock (newRatePerBlock) = { | |
281 | + | let check = if ((i.caller == votingEmissionRateContract)) | |
282 | + | then true | |
283 | + | else throwErr("should be invoked by votingEmissionRateContract") | |
284 | + | if ((check == check)) | |
285 | + | then { | |
286 | + | let factoryContract = readFactoryAddressOrFail() | |
287 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
288 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
289 | + | let internalInv = invoke(this, "changeRatePerBlockINTERNAL", [newRatePerBlock], nil) | |
290 | + | if ((internalInv == internalInv)) | |
291 | + | then { | |
292 | + | let boostingInvoke = reentrantInvoke(boostingContract, "onBoostEmissionUpdate", nil, nil) | |
293 | + | $Tuple2(nil, boostingInvoke) | |
294 | + | } | |
295 | + | else throw("Strict value is not equal to itself.") | |
296 | + | } | |
297 | + | else throw("Strict value is not equal to itself.") | |
298 | + | } | |
299 | + | ||
300 | + | ||
301 | + | @Verifier(tx) | |
302 | + | func verify () = { | |
303 | + | let targetPublicKey = match managerPublicKeyOrUnit() { | |
304 | + | case pk: ByteVector => | |
305 | + | pk | |
306 | + | case _: Unit => | |
307 | + | tx.senderPublicKey | |
308 | + | case _ => | |
309 | + | throw("Match error") | |
310 | + | } | |
311 | + | sigVerify(tx.bodyBytes, tx.proofs[0], targetPublicKey) | |
312 | + | } | |
313 | + |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | + | let SEP = "__" | |
5 | + | ||
6 | + | func wrapErr (msg) = makeString(["emission.ride:", toString(this), msg], " ") | |
7 | + | ||
8 | + | ||
9 | + | func throwErr (msg) = throw(wrapErr(msg)) | |
10 | + | ||
11 | + | ||
12 | + | func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), wrapErr((("mandatory this." + key) + " is not defined"))) | |
13 | + | ||
14 | + | ||
15 | + | func keyFactoryAddress () = "%s%s__config__factoryAddress" | |
16 | + | ||
17 | + | ||
18 | + | let keyVotingVerifiedContract = "%s%s__config__votingVerifiedContract" | |
19 | + | ||
20 | + | let keyVotingEmissionCandidateContract = "%s%s__config__votingEmissionCandidateContract" | |
21 | + | ||
22 | + | let keyUserPoolsContract = "%s%s__config__userPoolsContract" | |
23 | + | ||
24 | + | let keyVotingEmissionRateContract = "%s%s__config__votingEmissionRateContract" | |
25 | + | ||
26 | + | let votingVerifiedContract = addressFromStringValue(getStringOrFail(this, keyVotingVerifiedContract)) | |
27 | + | ||
28 | + | let votingEmissionCandidateContract = addressFromStringValue(getStringOrFail(this, keyVotingEmissionCandidateContract)) | |
29 | + | ||
30 | + | let userPoolsContract = addressFromStringValue(getStringOrFail(this, keyUserPoolsContract)) | |
31 | + | ||
32 | + | let votingEmissionRateContract = addressFromStringValue(getStringOrFail(this, keyVotingEmissionRateContract)) | |
33 | + | ||
34 | + | let IdxFactoryCfgStakingDapp = 1 | |
35 | + | ||
36 | + | let IdxFactoryCfgBoostingDapp = 2 | |
37 | + | ||
38 | + | let IdxFactoryCfgIdoDapp = 3 | |
39 | + | ||
40 | + | let IdxFactoryCfgTeamDapp = 4 | |
41 | + | ||
42 | + | let IdxFactoryCfgEmissionDapp = 5 | |
43 | + | ||
44 | + | let IdxFactoryCfgRestDapp = 6 | |
45 | + | ||
46 | + | let IdxFactoryCfgSlippageDapp = 7 | |
47 | + | ||
48 | + | let IdxFactoryCfgGwxRewardDapp = 10 | |
49 | + | ||
50 | + | func keyFactoryCfg () = "%s__factoryConfig" | |
51 | + | ||
52 | + | ||
53 | + | func keyFactoryLp2AssetsMapping (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
54 | + | ||
55 | + | ||
56 | + | func keyFactoryLpList () = "%s__lpTokensList" | |
57 | + | ||
58 | + | ||
59 | + | func keyFactoryLpAssetToPoolContractAddress (lpAssetStr) = makeString(["%s%s%s", lpAssetStr, "mappings__lpAsset2PoolContract"], SEP) | |
60 | + | ||
61 | + | ||
62 | + | func keyFactoryPoolWeight (contractAddress) = makeString(["%s%s", "poolWeight", contractAddress], SEP) | |
63 | + | ||
64 | + | ||
65 | + | func readFactoryAddressOrFail () = addressFromStringValue(getStringOrFail(this, keyFactoryAddress())) | |
66 | + | ||
67 | + | ||
68 | + | func readLpList () = split(valueOrElse(getString(readFactoryAddressOrFail(), keyFactoryLpList()), ""), SEP) | |
69 | + | ||
70 | + | ||
71 | + | func readFactoryCfgOrFail (factory) = split(getStringOrFail(factory, keyFactoryCfg()), SEP) | |
72 | + | ||
73 | + | ||
74 | + | func getBoostingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgBoostingDapp]) | |
75 | + | ||
76 | + | ||
77 | + | func getEmissionAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgEmissionDapp]) | |
78 | + | ||
79 | + | ||
80 | + | func getStakingAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgStakingDapp]) | |
81 | + | ||
82 | + | ||
83 | + | func getGwxRewardAddressOrFail (factoryCfg) = addressFromStringValue(factoryCfg[IdxFactoryCfgGwxRewardDapp]) | |
84 | + | ||
85 | + | ||
86 | + | func keyConfig () = "%s__config" | |
87 | + | ||
88 | + | ||
89 | + | func keyRatePerBlockMaxCurrent () = "%s%s__ratePerBlockMax__current" | |
90 | + | ||
91 | + | ||
92 | + | func keyRatePerBlockMaxStartFrom (timestamp,block) = makeString(["%s%s%d%d__ratePerBlockMax__start", toString(timestamp), toString(block)], SEP) | |
93 | + | ||
94 | + | ||
95 | + | func keyRatePerBlockCurrent () = "%s%s__ratePerBlock__current" | |
96 | + | ||
97 | + | ||
98 | + | func keyRatePerBlockStartFrom (timestamp,block) = makeString(["%s%s%d%d__ratePerBlock__start", toString(timestamp), toString(block)], SEP) | |
99 | + | ||
100 | + | ||
101 | + | func keyEmissionStartBlock () = "%s%s__emission__startBlock" | |
102 | + | ||
103 | + | ||
104 | + | func keyEmissionDurationInBlocks () = "%s%s__emission__duration" | |
105 | + | ||
106 | + | ||
107 | + | func keyEmissionEndBlock () = "%s%s__emission__endBlock" | |
108 | + | ||
109 | + | ||
110 | + | func keyBoostCoeff () = "%s__boostCoeff" | |
111 | + | ||
112 | + | ||
113 | + | let boostCoeffDefault = 3 | |
114 | + | ||
115 | + | let boostCoeff = valueOrElse(getInteger(this, keyBoostCoeff()), boostCoeffDefault) | |
116 | + | ||
117 | + | func keyGwxHoldersRewardCurrent () = "%s%s__gwxHoldersReward__current" | |
118 | + | ||
119 | + | ||
120 | + | func keyGwxHoldersRewardNext () = "%s%s__gwxHoldersReward__next" | |
121 | + | ||
122 | + | ||
123 | + | func keyManagerPublicKey () = "%s__managerPublicKey" | |
124 | + | ||
125 | + | ||
126 | + | func keyManagerVaultAddress () = "%s__managerVaultAddress" | |
127 | + | ||
128 | + | ||
129 | + | func getManagerVaultAddressOrThis () = match getString(keyManagerVaultAddress()) { | |
130 | + | case s: String => | |
131 | + | addressFromStringValue(s) | |
132 | + | case _ => | |
133 | + | this | |
134 | + | } | |
135 | + | ||
136 | + | ||
137 | + | func managerPublicKeyOrUnit () = { | |
138 | + | let managerVaultAddress = getManagerVaultAddressOrThis() | |
139 | + | match getString(managerVaultAddress, keyManagerPublicKey()) { | |
140 | + | case s: String => | |
141 | + | fromBase58String(s) | |
142 | + | case _: Unit => | |
143 | + | unit | |
144 | + | case _ => | |
145 | + | throw("Match error") | |
146 | + | } | |
147 | + | } | |
148 | + | ||
149 | + | ||
150 | + | func mustManager (i) = { | |
151 | + | let pd = throwErr("Permission denied") | |
152 | + | match managerPublicKeyOrUnit() { | |
153 | + | case pk: ByteVector => | |
154 | + | if ((i.callerPublicKey == pk)) | |
155 | + | then true | |
156 | + | else pd | |
157 | + | case _: Unit => | |
158 | + | if ((i.caller == this)) | |
159 | + | then true | |
160 | + | else pd | |
161 | + | case _ => | |
162 | + | throw("Match error") | |
163 | + | } | |
164 | + | } | |
4 | 165 | ||
5 | 166 | ||
6 | 167 | @Callable(i) | |
7 | - | func getNum () = $Tuple2(nil, 7) | |
168 | + | func constructor (factoryAddress,ratePerBlockMax,ratePerBlock,emissionStartBlock,emissionDuration,emissionStartTimestamp,wxAssetIdStr) = { | |
169 | + | let checkCaller = mustManager(i) | |
170 | + | if ((checkCaller == checkCaller)) | |
171 | + | then [IntegerEntry(keyRatePerBlockMaxStartFrom(emissionStartTimestamp, emissionStartBlock), ratePerBlockMax), IntegerEntry(keyRatePerBlockMaxCurrent(), ratePerBlockMax), IntegerEntry(keyRatePerBlockStartFrom(emissionStartTimestamp, emissionStartBlock), ratePerBlock), IntegerEntry(keyRatePerBlockCurrent(), ratePerBlock), IntegerEntry(keyEmissionStartBlock(), emissionStartBlock), IntegerEntry(keyEmissionDurationInBlocks(), emissionDuration), IntegerEntry(keyEmissionEndBlock(), (emissionStartBlock + emissionDuration)), StringEntry(keyFactoryAddress(), factoryAddress), StringEntry(keyConfig(), ("%s__" + wxAssetIdStr))] | |
172 | + | else throw("Strict value is not equal to itself.") | |
173 | + | } | |
8 | 174 | ||
9 | 175 | ||
10 | 176 | ||
11 | 177 | @Callable(i) | |
12 | - | func setterINTERNAL (n) = { | |
13 | - | let check = if ((i.caller == this)) | |
178 | + | func constructorV2 (votingVerifiedContractPrm) = { | |
179 | + | let checkCaller = mustManager(i) | |
180 | + | if ((checkCaller == checkCaller)) | |
181 | + | then $Tuple2([StringEntry(keyVotingVerifiedContract, votingVerifiedContractPrm)], unit) | |
182 | + | else throw("Strict value is not equal to itself.") | |
183 | + | } | |
184 | + | ||
185 | + | ||
186 | + | ||
187 | + | @Callable(i) | |
188 | + | func emit (amount) = if ((0 >= amount)) | |
189 | + | then $Tuple2(nil, nil) | |
190 | + | else { | |
191 | + | let factoryContract = readFactoryAddressOrFail() | |
192 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
193 | + | let stakingContract = getStakingAddressOrFail(factoryCfg) | |
194 | + | let gwxRewardsContract = getGwxRewardAddressOrFail(factoryCfg) | |
195 | + | let caller = i.caller | |
196 | + | if (if ((caller != stakingContract)) | |
197 | + | then (caller != gwxRewardsContract) | |
198 | + | else false) | |
199 | + | then throwErr("permissions denied") | |
200 | + | else { | |
201 | + | let wxAssetIdStr = split(getStringOrFail(this, keyConfig()), SEP)[1] | |
202 | + | let wxAssetId = fromBase58String(wxAssetIdStr) | |
203 | + | $Tuple2([ScriptTransfer(caller, amount, wxAssetId)], [wxAssetId]) | |
204 | + | } | |
205 | + | } | |
206 | + | ||
207 | + | ||
208 | + | ||
209 | + | @Callable(i) | |
210 | + | func burn () = { | |
211 | + | let factoryContract = readFactoryAddressOrFail() | |
212 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
213 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
214 | + | if ((size(i.payments) != 1)) | |
215 | + | then throwErr("exact one payment is allowed") | |
216 | + | else { | |
217 | + | let pmt = i.payments[0] | |
218 | + | let assetId = value(pmt.assetId) | |
219 | + | let pmtAmount = pmt.amount | |
220 | + | let wxAssetId = fromBase58String(split(getStringOrFail(this, keyConfig()), SEP)[1]) | |
221 | + | if ((assetId != wxAssetId)) | |
222 | + | then throwErr("invalid wxAssetId is passed") | |
223 | + | else if (!(containsElement([boostingContract, votingVerifiedContract, votingEmissionCandidateContract, userPoolsContract], i.caller))) | |
224 | + | then throwErr("permissions denied") | |
225 | + | else [Burn(assetId, pmtAmount), StringEntry(("%s%s%s__history__burn__" + toBase58String(i.transactionId)), makeString(["%d%d%d", toString(height), toString(lastBlock.timestamp), toString(pmtAmount)], SEP))] | |
226 | + | } | |
227 | + | } | |
228 | + | ||
229 | + | ||
230 | + | ||
231 | + | @Callable(i) | |
232 | + | func gwxHoldersRewardUpdate () = { | |
233 | + | let factoryContract = readFactoryAddressOrFail() | |
234 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
235 | + | let gwxRewardsContract = getGwxRewardAddressOrFail(factoryCfg) | |
236 | + | let checkCaller = if ((i.caller == gwxRewardsContract)) | |
14 | 237 | then true | |
15 | - | else | |
16 | - | if (( | |
238 | + | else mustManager(i) | |
239 | + | if ((checkCaller == checkCaller)) | |
17 | 240 | then { | |
18 | - | let | |
19 | - | | |
20 | - | | |
21 | - | | |
22 | - | | |
23 | - | | |
24 | - | ||
241 | + | let gwxHolderRewardCurrent = valueOrElse(getInteger(keyGwxHoldersRewardCurrent()), 0) | |
242 | + | let gwxHolderRewardNext = valueOrElse(getInteger(keyGwxHoldersRewardNext()), gwxHolderRewardCurrent) | |
243 | + | let shouldBeUpdated = (gwxHolderRewardCurrent != gwxHolderRewardNext) | |
244 | + | let actions = if (shouldBeUpdated) | |
245 | + | then [IntegerEntry(keyGwxHoldersRewardCurrent(), gwxHolderRewardNext), DeleteEntry(keyGwxHoldersRewardNext())] | |
246 | + | else nil | |
247 | + | $Tuple2(actions, shouldBeUpdated) | |
25 | 248 | } | |
26 | 249 | else throw("Strict value is not equal to itself.") | |
27 | 250 | } | |
28 | 251 | ||
29 | 252 | ||
30 | 253 | ||
31 | 254 | @Callable(i) | |
32 | - | func setter (n) = { | |
33 | - | let otherContract = addressFromStringValue("3MpMuGN8rW9EpPwzETAQz88Pe9TQhSwiriq") | |
34 | - | let tc = invoke(this, "setterINTERNAL", [n], nil) | |
35 | - | if ((tc == tc)) | |
255 | + | func getBoostCoeffREADONLY () = $Tuple2(nil, boostCoeff) | |
256 | + | ||
257 | + | ||
258 | + | ||
259 | + | @Callable(i) | |
260 | + | func changeRatePerBlockINTERNAL (newRatePerBlock) = { | |
261 | + | let check = if ((i.caller == this)) | |
262 | + | then true | |
263 | + | else throwErr("should be invoked internally") | |
264 | + | if ((check == check)) | |
36 | 265 | then { | |
37 | - | let oc = invoke(otherContract, "update", nil, nil) | |
38 | - | $Tuple2(nil, oc) | |
266 | + | let blockTimestamp = match blockInfoByHeight(height) { | |
267 | + | case bi: BlockInfo => | |
268 | + | bi.timestamp | |
269 | + | case _ => | |
270 | + | throwErr("BlockInfo.timestamp error") | |
271 | + | } | |
272 | + | [IntegerEntry(keyRatePerBlockStartFrom(blockTimestamp, height), newRatePerBlock), IntegerEntry(keyRatePerBlockCurrent(), newRatePerBlock)] | |
39 | 273 | } | |
40 | 274 | else throw("Strict value is not equal to itself.") | |
41 | 275 | } | |
42 | 276 | ||
43 | 277 | ||
278 | + | ||
279 | + | @Callable(i) | |
280 | + | func changeRatePerBlock (newRatePerBlock) = { | |
281 | + | let check = if ((i.caller == votingEmissionRateContract)) | |
282 | + | then true | |
283 | + | else throwErr("should be invoked by votingEmissionRateContract") | |
284 | + | if ((check == check)) | |
285 | + | then { | |
286 | + | let factoryContract = readFactoryAddressOrFail() | |
287 | + | let factoryCfg = readFactoryCfgOrFail(factoryContract) | |
288 | + | let boostingContract = getBoostingAddressOrFail(factoryCfg) | |
289 | + | let internalInv = invoke(this, "changeRatePerBlockINTERNAL", [newRatePerBlock], nil) | |
290 | + | if ((internalInv == internalInv)) | |
291 | + | then { | |
292 | + | let boostingInvoke = reentrantInvoke(boostingContract, "onBoostEmissionUpdate", nil, nil) | |
293 | + | $Tuple2(nil, boostingInvoke) | |
294 | + | } | |
295 | + | else throw("Strict value is not equal to itself.") | |
296 | + | } | |
297 | + | else throw("Strict value is not equal to itself.") | |
298 | + | } | |
299 | + | ||
300 | + | ||
301 | + | @Verifier(tx) | |
302 | + | func verify () = { | |
303 | + | let targetPublicKey = match managerPublicKeyOrUnit() { | |
304 | + | case pk: ByteVector => | |
305 | + | pk | |
306 | + | case _: Unit => | |
307 | + | tx.senderPublicKey | |
308 | + | case _ => | |
309 | + | throw("Match error") | |
310 | + | } | |
311 | + | sigVerify(tx.bodyBytes, tx.proofs[0], targetPublicKey) | |
312 | + | } | |
313 | + |
github/deemru/w8io/169f3d6 38.07 ms ◑![]()