tx · 86f1wMkBdKnMiHu4udATPPUkn5M37Bhy7qGtJPuoxPDc 3N3irEKCdj77h97aEbfj3fa3uMXCNCXVhKt: -0.01200000 Waves 2023.12.11 10:06 [2881716] smart account 3N3irEKCdj77h97aEbfj3fa3uMXCNCXVhKt > SELF 0.00000000 Waves
{ "type": 13, "id": "86f1wMkBdKnMiHu4udATPPUkn5M37Bhy7qGtJPuoxPDc", "fee": 1200000, "feeAssetId": null, "timestamp": 1702278418882, "version": 1, "sender": "3N3irEKCdj77h97aEbfj3fa3uMXCNCXVhKt", "senderPublicKey": "FACNGaFYnR8yFbQFg9sRBpzGVCyzMyLeVw3JMhGQT1wn", "proofs": [ "2VRNfNLcSsYHbRXoPrkitynsexVuyEwRiPphUqBGb5hddqB8r2oyumuagN8rGg9xgxBZgfYeusLm9nfcXSkqsKoi" ], "script": "base64:BgIWCAISBAoCCAESBQoDGBEBEgMKAQgSADAADGNvbnRyYWN0RmlsZQIRbDJtcF9zdGFraW5nLnJpZGUAA1NFUAICX18ABnNjYWxlOACAwtcvAAdzY2FsZTE4AICAkLu61q3wDQANc2NhbGUxOEJpZ0ludAkAtgIBBQdzY2FsZTE4ABJBRERSRVNTX0JZVEVTX1NJWkUAGgANQkxPQ0tTX0lOX0RBWQCgCwAMYWRtaW5BZGRyZXNzCQERQGV4dHJOYXRpdmUoMTA2MikBAiMzTXBzN0NacUI5blViRWlyWXlDTU1vQTdWYnFyeEx2SkZTQgEIdGhyb3dFcnIBA21zZwkAAgEJAKwCAgkArAICBQxjb250cmFjdEZpbGUCAjogBQNtc2cACmtleUFzc2V0SWQJALkJAgkAzAgCAgIlcwkAzAgCAgdhc3NldElkBQNuaWwFA1NFUAATa2V5RW1pc3Npb25QZXJCbG9jawkAuQkCCQDMCAICAiVzCQDMCAICEGVtaXNzaW9uUGVyQmxvY2sFA25pbAUDU0VQABlrZXlFbWlzc2lvblBlcmlvZEluQmxvY2tzCQC5CQIJAMwIAgICJXMJAMwIAgIWZW1pc3Npb25QZXJpb2RJbkJsb2NrcwUDbmlsBQNTRVAADWtleVN0YXJ0QmxvY2sJALkJAgkAzAgCAgIlcwkAzAgCAgpzdGFydEJsb2NrBQNuaWwFA1NFUAAQa2V5VG90YWxMcEFtb3VudAkAuQkCCQDMCAICAiVzCQDMCAICDXRvdGFsTHBBbW91bnQFA25pbAUDU0VQABNrZXlUb3RhbEFzc2V0QW1vdW50CQC5CQIJAMwIAgICJXMJAMwIAgIQdG90YWxBc3NldEFtb3VudAUDbmlsBQNTRVAAFmtleVRvdGFsTG9ja2VkTHBBbW91bnQJALkJAgkAzAgCAgIlcwkAzAgCAhN0b3RhbExvY2tlZExwQW1vdW50BQNuaWwFA1NFUAEPa2V5VXNlckxwQW1vdW50AQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgIMdXNlckxwQW1vdW50CQDMCAIFC3VzZXJBZGRyZXNzBQNuaWwFA1NFUAEVa2V5VXNlckxvY2tlZExwQW1vdW50AQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgISdXNlckxvY2tlZExwQW1vdW50CQDMCAIFC3VzZXJBZGRyZXNzBQNuaWwFA1NFUAETa2V5VXNlclN0YWtpbmdOb2RlcwELdXNlckFkZHJlc3MJALkJAgkAzAgCAgQlcyVzCQDMCAICEHVzZXJTdGFraW5nTm9kZXMJAMwIAgULdXNlckFkZHJlc3MFA25pbAUDU0VQARlrZXlVc2VyU3Rha2luZ05vZGVzU2hhcmVzAQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgIWdXNlclN0YWtpbmdOb2Rlc1NoYXJlcwkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQNTRVABGmtleVVzZXJUb3RhbEFzc2V0V2l0aGRyYXduAQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgITdG90YWxBc3NldFdpdGhkcmF3bgkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQNTRVABF2tleVVzZXJUb3RhbEFzc2V0U3Rha2VkAQt1c2VyQWRkcmVzcwkAuQkCCQDMCAICBCVzJXMJAMwIAgIQdG90YWxBc3NldFN0YWtlZAkAzAgCBQt1c2VyQWRkcmVzcwUDbmlsBQNTRVABCmtleUhpc3RvcnkDBHR5cGULdXNlckFkZHJlc3MEdHhJZAkAuQkCCQDMCAICBiVzJXMlcwkAzAgCBQR0eXBlCQDMCAIFC3VzZXJBZGRyZXNzCQDMCAIJANgEAQUEdHhJZAUDbmlsBQNTRVABDWZvcm1hdEhpc3RvcnkEC3RvdGFsUHJvZml0BXByaWNlEHRvdGFsQXNzZXRBbW91bnQNdG90YWxMcEFtb3VudAkAuQkCCQDMCAICCCVkJWQlZCVkCQDMCAIJAKQDAQULdG90YWxQcm9maXQJAMwIAgkApgMBBQVwcmljZQkAzAgCCQCkAwEFEHRvdGFsQXNzZXRBbW91bnQJAMwIAgkApAMBBQ10b3RhbExwQW1vdW50BQNuaWwFA1NFUAANdG90YWxMcEFtb3VudAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBRBrZXlUb3RhbExwQW1vdW50AAAAEHRvdGFsQXNzZXRBbW91bnQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUTa2V5VG90YWxBc3NldEFtb3VudAAAABN0b3RhbExvY2tlZExwQW1vdW50CQELdmFsdWVPckVsc2UCCQCaCAIFBHRoaXMFFmtleVRvdGFsTG9ja2VkTHBBbW91bnQAAAANYXNzZXRJZFN0cmluZwkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzBQprZXlBc3NldElkAgVXQVZFUwAMYXNzZXRJZEJ5dGVzAwkAAAIFDWFzc2V0SWRTdHJpbmcCBVdBVkVTBQR1bml0CQDZBAEFDWFzc2V0SWRTdHJpbmcAFmVtaXNzaW9uUGVyaW9kSW5CbG9ja3MJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUZa2V5RW1pc3Npb25QZXJpb2RJbkJsb2NrcwUNQkxPQ0tTX0lOX0RBWQAQZW1pc3Npb25QZXJCbG9jawkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBRNrZXlFbWlzc2lvblBlckJsb2NrAAAAEWVtaXNzaW9uUGVyUGVyaW9kCQBoAgUQZW1pc3Npb25QZXJCbG9jawUWZW1pc3Npb25QZXJpb2RJbkJsb2NrcwAKc3RhcnRCbG9jawkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBQ1rZXlTdGFydEJsb2NrBQZoZWlnaHQBGXN0cmluZ0xpc3RUb0ludExpc3RIZWxwZXICA2FjYwV2YWx1ZQkAzQgCBQNhY2MJAQ1wYXJzZUludFZhbHVlAQUFdmFsdWUBGGNhbGNUb3RhbFByb2ZpdEZvckhlaWdodAEBaAQLc3RhcnRQZXJpb2QJAGsDBQpzdGFydEJsb2NrAAEFFmVtaXNzaW9uUGVyaW9kSW5CbG9ja3MEDmVsYXBzZWRQZXJpb2RzCQBlAgkAaQIFAWgFFmVtaXNzaW9uUGVyaW9kSW5CbG9ja3MFC3N0YXJ0UGVyaW9kCQCWAwEJAMwIAgAACQDMCAIJAGgCBRFlbWlzc2lvblBlclBlcmlvZAUOZWxhcHNlZFBlcmlvZHMFA25pbAEPY2FsY1RvdGFsUHJvZml0AAkBGGNhbGNUb3RhbFByb2ZpdEZvckhlaWdodAEFBmhlaWdodAEUZ2V0TWF4QXNzZXRBdmFpbGFibGUABAckbWF0Y2gwBQxhc3NldElkQnl0ZXMDCQABAgUHJG1hdGNoMAIEVW5pdAQBdQUHJG1hdGNoMAgJAO8HAQUEdGhpcwlhdmFpbGFibGUDCQABAgUHJG1hdGNoMAIKQnl0ZVZlY3RvcgQBYgUHJG1hdGNoMAkA8AcCBQR0aGlzBQFiCQACAQILTWF0Y2ggZXJyb3IBK2dldFRvdGFsQXNzZXRBbW91bnRXaXRoUHJvZml0T3JNYXhBdmFpbGFibGUBCGF0SGVpZ2h0BBp0b3RhbEFzc2V0QW1vdW50V2l0aFByb2ZpdAkAZAIFEHRvdGFsQXNzZXRBbW91bnQJARhjYWxjVG90YWxQcm9maXRGb3JIZWlnaHQBBQhhdEhlaWdodAQLdG90YWxBbW91bnQJAJcDAQkAzAgCBRp0b3RhbEFzc2V0QW1vdW50V2l0aFByb2ZpdAkAzAgCCQEUZ2V0TWF4QXNzZXRBdmFpbGFibGUABQNuaWwDCQAAAgUNdG90YWxMcEFtb3VudAAAAAAFC3RvdGFsQW1vdW50ARBnZXRQcmljZUF0SGVpZ2h0AQFoAwkBAiE9AgUNdG90YWxMcEFtb3VudAAACQC8AgMJALYCAQkBK2dldFRvdGFsQXNzZXRBbW91bnRXaXRoUHJvZml0T3JNYXhBdmFpbGFibGUBBQFoBQ1zY2FsZTE4QmlnSW50CQC2AgEFDXRvdGFsTHBBbW91bnQFDXNjYWxlMThCaWdJbnQBD2dldEN1cnJlbnRQcmljZQAJARBnZXRQcmljZUF0SGVpZ2h0AQUGaGVpZ2h0ARJnZXRSZW1haW5pbmdCbG9ja3MAAwkAAAIFEGVtaXNzaW9uUGVyQmxvY2sAAAAACQBrAwkAZQIJARRnZXRNYXhBc3NldEF2YWlsYWJsZQAJAStnZXRUb3RhbEFzc2V0QW1vdW50V2l0aFByb2ZpdE9yTWF4QXZhaWxhYmxlAQUGaGVpZ2h0AAEFEGVtaXNzaW9uUGVyQmxvY2sBF2dldFVzZXJTdGFraW5nTm9kZXNEYXRhAQt1c2VyQWRkcmVzcwQIbm9kZXNSYXcJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwkBE2tleVVzZXJTdGFraW5nTm9kZXMBBQt1c2VyQWRkcmVzcwIABAlzaGFyZXNSYXcJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwkBGWtleVVzZXJTdGFraW5nTm9kZXNTaGFyZXMBBQt1c2VyQWRkcmVzcwIABAlub2Rlc0xpc3QDCQAAAgUIbm9kZXNSYXcCAAUDbmlsCQC1CQIFCG5vZGVzUmF3BQNTRVAEEHNoYXJlc1N0cmluZ0xpc3QDCQAAAgUJc2hhcmVzUmF3AgAFA25pbAkAtQkCBQlzaGFyZXNSYXcFA1NFUAQKc2hhcmVzTGlzdAoAAiRsBRBzaGFyZXNTdHJpbmdMaXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJARlzdHJpbmdMaXN0VG9JbnRMaXN0SGVscGVyAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAyMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQJAJQKAgUJbm9kZXNMaXN0BQpzaGFyZXNMaXN0AQ9jYWxjQXNzZXRGcm9tTHACCGxwQW1vdW50CGF0SGVpZ2h0CQCWAwEJAMwIAgAACQDMCAIJAKADAQkAvAIDCQC2AgEFCGxwQW1vdW50CQEQZ2V0UHJpY2VBdEhlaWdodAEFCGF0SGVpZ2h0BQ1zY2FsZTE4QmlnSW50BQNuaWwBD2NhbGNMcEZyb21Bc3NldAILYXNzZXRBbW91bnQIYXRIZWlnaHQJAJYDAQkAzAgCAAAJAMwIAgkAoAMBCQC8AgMJALYCAQULYXNzZXRBbW91bnQFDXNjYWxlMThCaWdJbnQJARBnZXRQcmljZUF0SGVpZ2h0AQUIYXRIZWlnaHQFA25pbAEPZ2V0VXNlckxwQW1vdW50AQt1c2VyQWRkcmVzcwkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEPa2V5VXNlckxwQW1vdW50AQULdXNlckFkZHJlc3MAAAEVZ2V0VXNlckxvY2tlZExwQW1vdW50AQt1c2VyQWRkcmVzcwkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEVa2V5VXNlckxvY2tlZExwQW1vdW50AQULdXNlckFkZHJlc3MAAAEgZ2V0VXNlckF2YWlsYWJsZUFzc2V0c1RvV2l0aGRyYXcBC3VzZXJBZGRyZXNzBAx1c2VyTHBBbW91bnQJAQ9nZXRVc2VyTHBBbW91bnQBBQt1c2VyQWRkcmVzcwkBD2NhbGNBc3NldEZyb21McAIFDHVzZXJMcEFtb3VudAUGaGVpZ2h0AQ9nZXRTdGFrZUFjdGlvbnMDAWkLdXNlckFkZHJlc3MLc3Rha2VIZWlnaHQEBmNoZWNrcwkAzAgCAwkAZwIFC3N0YWtlSGVpZ2h0BQpzdGFydEJsb2NrBgkBCHRocm93RXJyAQImc3RhcnRCbG9jayBpcyBncmVhdGVyIHRoYW4gc3Rha2VIZWlnaHQJAMwIAgMJAAACCQCQAwEIBQFpCHBheW1lbnRzAAEGCQEIdGhyb3dFcnIBAhhzaG91bGQgaW5jbHVkZSAxIHBheW1lbnQJAMwIAgMJAAACCAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQFDGFzc2V0SWRCeXRlcwYJAQh0aHJvd0VycgEJAKwCAgIVcGF5bWVudCBzaG91bGQgYmUgaW4gBQ1hc3NldElkU3RyaW5nCQDMCAIDCQBmAggJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQAAAYCJ3BheW1lbnQgYW1vdW50IHNob3VsZCBiZSBncmVhdGVyIHRoYW4gMAkAzAgCAwkAAAIJAMgBAQkA2QQBBQt1c2VyQWRkcmVzcwUSQUREUkVTU19CWVRFU19TSVpFBgkBCHRocm93RXJyAQIZdXNlciBhZGRyZXNzIGlzIG5vdCB2YWxpZAUDbmlsAwkAAAIFBmNoZWNrcwUGY2hlY2tzBA1wYXltZW50QW1vdW50CAkAkQMCCAUBaQhwYXltZW50cwAABmFtb3VudAQPcGF5bWVudExwQW1vdW50CQEPY2FsY0xwRnJvbUFzc2V0AgUNcGF5bWVudEFtb3VudAULc3Rha2VIZWlnaHQEDHVzZXJMcEFtb3VudAkBD2dldFVzZXJMcEFtb3VudAEFC3VzZXJBZGRyZXNzBBV1c2VyVG90YWxTdGFrZWRBbW91bnQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwkBF2tleVVzZXJUb3RhbEFzc2V0U3Rha2VkAQULdXNlckFkZHJlc3MAAAQQbmV3VG90YWxMcEFtb3VudAkAZAIFDXRvdGFsTHBBbW91bnQFD3BheW1lbnRMcEFtb3VudAQTbmV3VG90YWxBc3NldEFtb3VudAkBD2NhbGNBc3NldEZyb21McAIFEG5ld1RvdGFsTHBBbW91bnQFC3N0YWtlSGVpZ2h0BA9uZXdVc2VyTHBBbW91bnQJAGQCBQx1c2VyTHBBbW91bnQFD3BheW1lbnRMcEFtb3VudAQYbmV3VXNlclRvdGFsU3Rha2VkQW1vdW50CQBkAgUVdXNlclRvdGFsU3Rha2VkQW1vdW50BQ1wYXltZW50QW1vdW50CQDMCAIJAQtTdHJpbmdFbnRyeQIJAQprZXlIaXN0b3J5AwIFc3Rha2UFC3VzZXJBZGRyZXNzCAUBaQ10cmFuc2FjdGlvbklkCQENZm9ybWF0SGlzdG9yeQQJAQ9jYWxjVG90YWxQcm9maXQACQEPZ2V0Q3VycmVudFByaWNlAAUNdG90YWxMcEFtb3VudAUQdG90YWxBc3NldEFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgUQa2V5VG90YWxMcEFtb3VudAUQbmV3VG90YWxMcEFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgUTa2V5VG90YWxBc3NldEFtb3VudAUTbmV3VG90YWxBc3NldEFtb3VudAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBD2tleVVzZXJMcEFtb3VudAEFC3VzZXJBZGRyZXNzBQ9uZXdVc2VyTHBBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIJARdrZXlVc2VyVG90YWxBc3NldFN0YWtlZAEFC3VzZXJBZGRyZXNzBRhuZXdVc2VyVG90YWxTdGFrZWRBbW91bnQJAMwIAgkBDEludGVnZXJFbnRyeQIFDWtleVN0YXJ0QmxvY2sFC3N0YWtlSGVpZ2h0BQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4EAWkBDXN0YWtlRm9yQWRtaW4CC3VzZXJBZGRyZXNzC3N0YWtlSGVpZ2h0BAVjaGVjawkAzAgCAwkAAAIIBQFpBmNhbGxlcgUMYWRtaW5BZGRyZXNzBgkBCHRocm93RXJyAQIRcGVybWlzc2lvbiBkZW5pZWQFA25pbAMJAAACBQVjaGVjawUFY2hlY2sJAQ9nZXRTdGFrZUFjdGlvbnMDBQFpBQt1c2VyQWRkcmVzcwULc3Rha2VIZWlnaHQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEHYWlyZHJvcAMLYWRkcmVzc0xpc3QKYW1vdW50TGlzdA1haXJkcm9wSGVpZ2h0CgEDc3VtAgVhY2N1bQRuZXh0AwkAZgIAAAUEbmV4dAkBCHRocm93RXJyAQIjbmVnYXRpdmUgYW1vdW50IHZhbHVlIGluIGFtb3VudExpc3QJAGQCBQVhY2N1bQUEbmV4dAQNYW1vdW50TGlzdFN1bQoAAiRsBQphbW91bnRMaXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAAACgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQNzdW0CBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDkwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAAVABYAFwAYABkAGgAbABwAHQAeAB8AIAAhACIAIwAkACUAJgAnACgAKQAqACsALAAtAC4ALwAwADEAMgAzADQANQA2ADcAOAA5ADoAOwA8AD0APgA/AEAAQQBCAEMARABFAEYARwBIAEkASgBLAEwATQBOAE8AUABRAFIAUwBUAFUAVgBXAFgAWQBaBAVjaGVjawkAzAgCAwkAAAIIBQFpBmNhbGxlcgUMYWRtaW5BZGRyZXNzBgkBCHRocm93RXJyAQIRcGVybWlzc2lvbiBkZW5pZWQJAMwIAgMJAGcCBQ1haXJkcm9wSGVpZ2h0BQpzdGFydEJsb2NrBgkBCHRocm93RXJyAQIpYWlyZHJvcEhlaWdodCBpcyBncmVhdGVyIHRoYW4gc3Rha2VIZWlnaHQJAMwIAgMJAAACCQCQAwEIBQFpCHBheW1lbnRzAAEGCQEIdGhyb3dFcnIBAhhzaG91bGQgaW5jbHVkZSAxIHBheW1lbnQJAMwIAgMJAAACCAkAkQMCCAUBaQhwYXltZW50cwAAB2Fzc2V0SWQFDGFzc2V0SWRCeXRlcwYJAQh0aHJvd0VycgEJAKwCAgIVcGF5bWVudCBzaG91bGQgYmUgaW4gBQ1hc3NldElkU3RyaW5nCQDMCAIDCQBmAggJAJEDAggFAWkIcGF5bWVudHMAAAZhbW91bnQAAAYCJ3BheW1lbnQgYW1vdW50IHNob3VsZCBiZSBncmVhdGVyIHRoYW4gMAkAzAgCAwkAAAIJAJADAQULYWRkcmVzc0xpc3QJAJADAQUKYW1vdW50TGlzdAYJAQh0aHJvd0VycgECLWFkZHJlc3NMaXN0IHNob3VsZCBiZSBzYW1lIHNpemUgYXMgYW1vdW50TGlzdAkAzAgCAwkAZwIICQCRAwIIBQFpCHBheW1lbnRzAAAGYW1vdW50BQ1hbW91bnRMaXN0U3VtBgkBCHRocm93RXJyAQItcGF5bWVudCBhbW91bnQgaXMgbGVzcyB0aGFuIHN1bSBvZiBhbW91bnRMaXN0BQNuaWwDCQAAAgUFY2hlY2sFBWNoZWNrCgEWZ2V0QWlyZHJvcFN0YXRlQ2hhbmdlcwIFYWNjdW0LYXNzZXRBbW91bnQECyR0MDgzNjQ4NDE1BQVhY2N1bQQGcmVzdWx0CAULJHQwODM2NDg0MTUCXzEEBWluZGV4CAULJHQwODM2NDg0MTUCXzIEB3RvdGFsTHAIBQskdDA4MzY0ODQxNQJfMwQNcHJvY2Vzc2VkTGlzdAgFCyR0MDgzNjQ4NDE1Al80BA1hZGRyZXNzU3RyaW5nCQCRAwIFC2FkZHJlc3NMaXN0BQVpbmRleAQHYWRkcmVzcwQHJG1hdGNoMAkApggBBQ1hZGRyZXNzU3RyaW5nAwkAAQIFByRtYXRjaDACB0FkZHJlc3MEA2FkcgUHJG1hdGNoMAUDYWRyCQEIdGhyb3dFcnIBAh5pbnZhbGlkIGFkZHJlc3MgaW4gYWRkcmVzc0xpc3QEAmNoCQDMCAIDCQEBIQEJAQ9jb250YWluc0VsZW1lbnQCBQ1wcm9jZXNzZWRMaXN0BQdhZGRyZXNzBgkBCHRocm93RXJyAQIgZHVwbGljYXRlIGFkZHJlc3MgaXMgYWRkcmVzc0xpc3QFA25pbAMJAAACBQJjaAUCY2gEDWFkZGVkTHBBbW91bnQJAQ9jYWxjTHBGcm9tQXNzZXQCBQthc3NldEFtb3VudAUNYWlyZHJvcEhlaWdodAQPdXNlckxvY2tlZExwS2V5CQEVa2V5VXNlckxvY2tlZExwQW1vdW50AQUNYWRkcmVzc1N0cmluZwQLb2xkTHBBbW91bnQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUEdGhpcwUPdXNlckxvY2tlZExwS2V5AAAJAJYKBAkAzQgCBQZyZXN1bHQJAQxJbnRlZ2VyRW50cnkCBQ91c2VyTG9ja2VkTHBLZXkJAGQCBQtvbGRMcEFtb3VudAUNYWRkZWRMcEFtb3VudAkAZAIFBWluZGV4AAEJAGQCBQd0b3RhbExwBQ1hZGRlZExwQW1vdW50CQDNCAIFDXByb2Nlc3NlZExpc3QFB2FkZHJlc3MJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4ECyR0MDkxOTQ5MzExCgACJGwFCmFtb3VudExpc3QKAAIkcwkAkAMBBQIkbAoABSRhY2MwCQCWCgQFA25pbAAAAAAFA25pbAoBBSRmMV8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEWZ2V0QWlyZHJvcFN0YXRlQ2hhbmdlcwIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMV8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgOTAJAQUkZjFfMgIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIJAQUkZjFfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgALAAwADQAOAA8AEAARABIAEwAUABUAFgAXABgAGQAaABsAHAAdAB4AHwAgACEAIgAjACQAJQAmACcAKAApACoAKwAsAC0ALgAvADAAMQAyADMANAA1ADYANwA4ADkAOgA7ADwAPQA+AD8AQABBAEIAQwBEAEUARgBHAEgASQBKAEsATABNAE4ATwBQAFEAUgBTAFQAVQBWAFcAWABZAFoEDmFpcmRyb3BFbnRyaWVzCAULJHQwOTE5NDkzMTECXzEEAl9hCAULJHQwOTE5NDkzMTECXzIEGGFkZGVkVG90YWxMb2NrZWRMcEFtb3VudAgFCyR0MDkxOTQ5MzExAl8zBAJfYggFCyR0MDkxOTQ5MzExAl80BA1uZXdUb3RhbEFzc2V0CQEPY2FsY0Fzc2V0RnJvbUxwAgkAZAIFDXRvdGFsTHBBbW91bnQFGGFkZGVkVG90YWxMb2NrZWRMcEFtb3VudAUNYWlyZHJvcEhlaWdodAkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCBRZrZXlUb3RhbExvY2tlZExwQW1vdW50CQBkAgUTdG90YWxMb2NrZWRMcEFtb3VudAUYYWRkZWRUb3RhbExvY2tlZExwQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCBRBrZXlUb3RhbExwQW1vdW50CQBkAgUNdG90YWxMcEFtb3VudAUYYWRkZWRUb3RhbExvY2tlZExwQW1vdW50CQDMCAIJAQxJbnRlZ2VyRW50cnkCBRNrZXlUb3RhbEFzc2V0QW1vdW50BQ1uZXdUb3RhbEFzc2V0CQDMCAIJAQxJbnRlZ2VyRW50cnkCBQ1rZXlTdGFydEJsb2NrBQZoZWlnaHQFA25pbAUOYWlyZHJvcEVudHJpZXMJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEVZ2V0VXNlckFzc2V0c1JFQURPTkxZAQt1c2VyQWRkcmVzcwQMdXNlckxwQW1vdW50CQEPZ2V0VXNlckxwQW1vdW50AQULdXNlckFkZHJlc3MEEnVzZXJMb2NrZWRMcEFtb3VudAkBFWdldFVzZXJMb2NrZWRMcEFtb3VudAEFC3VzZXJBZGRyZXNzBBV1c2VyTG9ja2VkQXNzZXRBbW91bnQJAQ9jYWxjQXNzZXRGcm9tTHACBRJ1c2VyTG9ja2VkTHBBbW91bnQFBmhlaWdodAQcdXNlckF2YWlsYWJsZUFzc2V0VG9XaXRoZHJhdwkBIGdldFVzZXJBdmFpbGFibGVBc3NldHNUb1dpdGhkcmF3AQULdXNlckFkZHJlc3MEFXVzZXJUb3RhbFN0YWtlZEFtb3VudAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEXa2V5VXNlclRvdGFsQXNzZXRTdGFrZWQBBQt1c2VyQWRkcmVzcwAABBd1c2VyVG90YWxBc3NldFdpdGhkcmF3bgkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzCQEaa2V5VXNlclRvdGFsQXNzZXRXaXRoZHJhd24BBQt1c2VyQWRkcmVzcwAABA0kdDAxMDY4NDEwNzc2CQEXZ2V0VXNlclN0YWtpbmdOb2Rlc0RhdGEBBQt1c2VyQWRkcmVzcwQUdXNlclN0YWtpbmdOb2Rlc0xpc3QIBQ0kdDAxMDY4NDEwNzc2Al8xBBl1c2VyU3Rha2luZ05vZGVTaGFyZXNMaXN0CAUNJHQwMTA2ODQxMDc3NgJfMgkAlAoCBQNuaWwJAJwKCgUMdXNlckxwQW1vdW50BRx1c2VyQXZhaWxhYmxlQXNzZXRUb1dpdGhkcmF3CQEPZ2V0Q3VycmVudFByaWNlAAUVdXNlclRvdGFsU3Rha2VkQW1vdW50BRd1c2VyVG90YWxBc3NldFdpdGhkcmF3bgUSdXNlckxvY2tlZExwQW1vdW50BRV1c2VyTG9ja2VkQXNzZXRBbW91bnQFFHVzZXJTdGFraW5nTm9kZXNMaXN0BRl1c2VyU3Rha2luZ05vZGVTaGFyZXNMaXN0CQESZ2V0UmVtYWluaW5nQmxvY2tzAAFpARZnZXRUb3RhbEFzc2V0c1JFQURPTkxZAAkAlAoCBQNuaWwJAJgKBgUNdG90YWxMcEFtb3VudAkBK2dldFRvdGFsQXNzZXRBbW91bnRXaXRoUHJvZml0T3JNYXhBdmFpbGFibGUBBQZoZWlnaHQJAQ9nZXRDdXJyZW50UHJpY2UABRN0b3RhbExvY2tlZExwQW1vdW50CQEPY2FsY0Fzc2V0RnJvbUxwAgUTdG90YWxMb2NrZWRMcEFtb3VudAUGaGVpZ2h0CQESZ2V0UmVtYWluaW5nQmxvY2tzAACvHrl+", "chainId": 84, "height": 2881716, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: Hq2jw7mQaM9AKWTJ8tdbXiGETFKmrbGH7GJNMr3cuhiX Next: E7zhNJLDwJLYAR8UJ4QesJGw2e3qc4rFZ2W7USKCxYnt Diff:
Old | New | Differences | |
---|---|---|---|
76 | 76 | ||
77 | 77 | let emissionPerPeriod = (emissionPerBlock * emissionPeriodInBlocks) | |
78 | 78 | ||
79 | + | let startBlock = valueOrElse(getInteger(this, keyStartBlock), height) | |
80 | + | ||
79 | 81 | func stringListToIntListHelper (acc,value) = (acc :+ parseIntValue(value)) | |
80 | 82 | ||
81 | 83 | ||
82 | 84 | func calcTotalProfitForHeight (h) = { | |
83 | - | let startBlock = valueOrElse(getInteger(this, keyStartBlock), height) | |
84 | 85 | let startPeriod = fraction(startBlock, 1, emissionPeriodInBlocks) | |
85 | 86 | let elapsedPeriods = ((h / emissionPeriodInBlocks) - startPeriod) | |
86 | 87 | max([0, (emissionPerPeriod * elapsedPeriods)]) | |
167 | 168 | } | |
168 | 169 | ||
169 | 170 | ||
170 | - | func getClearStakingNodesActions (userAddress) = [DeleteEntry(keyUserStakingNodes(userAddress)), DeleteEntry(keyUserStakingNodesShares(userAddress))] | |
171 | - | ||
172 | - | ||
173 | 171 | func getStakeActions (i,userAddress,stakeHeight) = { | |
174 | - | let checks = [if ((size(i.payments) == 1)) | |
172 | + | let checks = [if ((stakeHeight >= startBlock)) | |
173 | + | then true | |
174 | + | else throwErr("startBlock is greater than stakeHeight"), if ((size(i.payments) == 1)) | |
175 | 175 | then true | |
176 | 176 | else throwErr("should include 1 payment"), if ((i.payments[0].assetId == assetIdBytes)) | |
177 | 177 | then true | |
190 | 190 | let newTotalAssetAmount = calcAssetFromLp(newTotalLpAmount, stakeHeight) | |
191 | 191 | let newUserLpAmount = (userLpAmount + paymentLpAmount) | |
192 | 192 | let newUserTotalStakedAmount = (userTotalStakedAmount + paymentAmount) | |
193 | - | [StringEntry(keyHistory("stake", userAddress, i.transactionId), formatHistory(calcTotalProfit(), getCurrentPrice(), totalLpAmount, totalAssetAmount)), IntegerEntry(keyTotalLpAmount, newTotalLpAmount), IntegerEntry(keyTotalAssetAmount, newTotalAssetAmount), IntegerEntry(keyUserLpAmount(userAddress), newUserLpAmount), IntegerEntry(keyUserTotalAssetStaked(userAddress), newUserTotalStakedAmount), IntegerEntry(keyStartBlock, | |
193 | + | [StringEntry(keyHistory("stake", userAddress, i.transactionId), formatHistory(calcTotalProfit(), getCurrentPrice(), totalLpAmount, totalAssetAmount)), IntegerEntry(keyTotalLpAmount, newTotalLpAmount), IntegerEntry(keyTotalAssetAmount, newTotalAssetAmount), IntegerEntry(keyUserLpAmount(userAddress), newUserLpAmount), IntegerEntry(keyUserTotalAssetStaked(userAddress), newUserTotalStakedAmount), IntegerEntry(keyStartBlock, stakeHeight)] | |
194 | 194 | } | |
195 | 195 | else throw("Strict value is not equal to itself.") | |
196 | 196 | } | |
230 | 230 | } | |
231 | 231 | let check = [if ((i.caller == adminAddress)) | |
232 | 232 | then true | |
233 | - | else throwErr("permission denied"), if ((size(i.payments) == 1)) | |
233 | + | else throwErr("permission denied"), if ((airdropHeight >= startBlock)) | |
234 | + | then true | |
235 | + | else throwErr("airdropHeight is greater than stakeHeight"), if ((size(i.payments) == 1)) | |
234 | 236 | then true | |
235 | 237 | else throwErr("should include 1 payment"), if ((i.payments[0].assetId == assetIdBytes)) | |
236 | 238 | then true | |
244 | 246 | if ((check == check)) | |
245 | 247 | then { | |
246 | 248 | func getAirdropStateChanges (accum,assetAmount) = { | |
247 | - | let $ | |
248 | - | let result = $ | |
249 | - | let index = $ | |
250 | - | let totalLp = $ | |
251 | - | let processedList = $ | |
249 | + | let $t083648415 = accum | |
250 | + | let result = $t083648415._1 | |
251 | + | let index = $t083648415._2 | |
252 | + | let totalLp = $t083648415._3 | |
253 | + | let processedList = $t083648415._4 | |
252 | 254 | let addressString = addressList[index] | |
253 | 255 | let address = match addressFromString(addressString) { | |
254 | 256 | case adr: Address => | |
269 | 271 | else throw("Strict value is not equal to itself.") | |
270 | 272 | } | |
271 | 273 | ||
272 | - | let $ | |
274 | + | let $t091949311 = { | |
273 | 275 | let $l = amountList | |
274 | 276 | let $s = size($l) | |
275 | 277 | let $acc0 = $Tuple4(nil, 0, 0, nil) | |
283 | 285 | ||
284 | 286 | $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90) | |
285 | 287 | } | |
286 | - | let airdropEntries = $ | |
287 | - | let _a = $ | |
288 | - | let addedTotalLockedLpAmount = $ | |
289 | - | let _b = $ | |
288 | + | let airdropEntries = $t091949311._1 | |
289 | + | let _a = $t091949311._2 | |
290 | + | let addedTotalLockedLpAmount = $t091949311._3 | |
291 | + | let _b = $t091949311._4 | |
290 | 292 | let newTotalAsset = calcAssetFromLp((totalLpAmount + addedTotalLockedLpAmount), airdropHeight) | |
291 | 293 | ([IntegerEntry(keyTotalLockedLpAmount, (totalLockedLpAmount + addedTotalLockedLpAmount)), IntegerEntry(keyTotalLpAmount, (totalLpAmount + addedTotalLockedLpAmount)), IntegerEntry(keyTotalAssetAmount, newTotalAsset), IntegerEntry(keyStartBlock, height)] ++ airdropEntries) | |
292 | 294 | } | |
303 | 305 | let userAvailableAssetToWithdraw = getUserAvailableAssetsToWithdraw(userAddress) | |
304 | 306 | let userTotalStakedAmount = valueOrElse(getInteger(this, keyUserTotalAssetStaked(userAddress)), 0) | |
305 | 307 | let userTotalAssetWithdrawn = valueOrElse(getInteger(this, keyUserTotalAssetWithdrawn(userAddress)), 0) | |
306 | - | let $ | |
307 | - | let userStakingNodesList = $ | |
308 | - | let userStakingNodeSharesList = $ | |
308 | + | let $t01068410776 = getUserStakingNodesData(userAddress) | |
309 | + | let userStakingNodesList = $t01068410776._1 | |
310 | + | let userStakingNodeSharesList = $t01068410776._2 | |
309 | 311 | $Tuple2(nil, $Tuple10(userLpAmount, userAvailableAssetToWithdraw, getCurrentPrice(), userTotalStakedAmount, userTotalAssetWithdrawn, userLockedLpAmount, userLockedAssetAmount, userStakingNodesList, userStakingNodeSharesList, getRemainingBlocks())) | |
310 | 312 | } | |
311 | 313 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let contractFile = "l2mp_staking.ride" | |
5 | 5 | ||
6 | 6 | let SEP = "__" | |
7 | 7 | ||
8 | 8 | let scale8 = 100000000 | |
9 | 9 | ||
10 | 10 | let scale18 = 1000000000000000000 | |
11 | 11 | ||
12 | 12 | let scale18BigInt = toBigInt(scale18) | |
13 | 13 | ||
14 | 14 | let ADDRESS_BYTES_SIZE = 26 | |
15 | 15 | ||
16 | 16 | let BLOCKS_IN_DAY = 1440 | |
17 | 17 | ||
18 | 18 | let adminAddress = addressFromStringValue("3Mps7CZqB9nUbEirYyCMMoA7VbqrxLvJFSB") | |
19 | 19 | ||
20 | 20 | func throwErr (msg) = throw(((contractFile + ": ") + msg)) | |
21 | 21 | ||
22 | 22 | ||
23 | 23 | let keyAssetId = makeString(["%s", "assetId"], SEP) | |
24 | 24 | ||
25 | 25 | let keyEmissionPerBlock = makeString(["%s", "emissionPerBlock"], SEP) | |
26 | 26 | ||
27 | 27 | let keyEmissionPeriodInBlocks = makeString(["%s", "emissionPeriodInBlocks"], SEP) | |
28 | 28 | ||
29 | 29 | let keyStartBlock = makeString(["%s", "startBlock"], SEP) | |
30 | 30 | ||
31 | 31 | let keyTotalLpAmount = makeString(["%s", "totalLpAmount"], SEP) | |
32 | 32 | ||
33 | 33 | let keyTotalAssetAmount = makeString(["%s", "totalAssetAmount"], SEP) | |
34 | 34 | ||
35 | 35 | let keyTotalLockedLpAmount = makeString(["%s", "totalLockedLpAmount"], SEP) | |
36 | 36 | ||
37 | 37 | func keyUserLpAmount (userAddress) = makeString(["%s%s", "userLpAmount", userAddress], SEP) | |
38 | 38 | ||
39 | 39 | ||
40 | 40 | func keyUserLockedLpAmount (userAddress) = makeString(["%s%s", "userLockedLpAmount", userAddress], SEP) | |
41 | 41 | ||
42 | 42 | ||
43 | 43 | func keyUserStakingNodes (userAddress) = makeString(["%s%s", "userStakingNodes", userAddress], SEP) | |
44 | 44 | ||
45 | 45 | ||
46 | 46 | func keyUserStakingNodesShares (userAddress) = makeString(["%s%s", "userStakingNodesShares", userAddress], SEP) | |
47 | 47 | ||
48 | 48 | ||
49 | 49 | func keyUserTotalAssetWithdrawn (userAddress) = makeString(["%s%s", "totalAssetWithdrawn", userAddress], SEP) | |
50 | 50 | ||
51 | 51 | ||
52 | 52 | func keyUserTotalAssetStaked (userAddress) = makeString(["%s%s", "totalAssetStaked", userAddress], SEP) | |
53 | 53 | ||
54 | 54 | ||
55 | 55 | func keyHistory (type,userAddress,txId) = makeString(["%s%s%s", type, userAddress, toBase58String(txId)], SEP) | |
56 | 56 | ||
57 | 57 | ||
58 | 58 | func formatHistory (totalProfit,price,totalAssetAmount,totalLpAmount) = makeString(["%d%d%d%d", toString(totalProfit), toString(price), toString(totalAssetAmount), toString(totalLpAmount)], SEP) | |
59 | 59 | ||
60 | 60 | ||
61 | 61 | let totalLpAmount = valueOrElse(getInteger(this, keyTotalLpAmount), 0) | |
62 | 62 | ||
63 | 63 | let totalAssetAmount = valueOrElse(getInteger(this, keyTotalAssetAmount), 0) | |
64 | 64 | ||
65 | 65 | let totalLockedLpAmount = valueOrElse(getInteger(this, keyTotalLockedLpAmount), 0) | |
66 | 66 | ||
67 | 67 | let assetIdString = valueOrElse(getString(this, keyAssetId), "WAVES") | |
68 | 68 | ||
69 | 69 | let assetIdBytes = if ((assetIdString == "WAVES")) | |
70 | 70 | then unit | |
71 | 71 | else fromBase58String(assetIdString) | |
72 | 72 | ||
73 | 73 | let emissionPeriodInBlocks = valueOrElse(getInteger(this, keyEmissionPeriodInBlocks), BLOCKS_IN_DAY) | |
74 | 74 | ||
75 | 75 | let emissionPerBlock = valueOrElse(getInteger(this, keyEmissionPerBlock), 0) | |
76 | 76 | ||
77 | 77 | let emissionPerPeriod = (emissionPerBlock * emissionPeriodInBlocks) | |
78 | 78 | ||
79 | + | let startBlock = valueOrElse(getInteger(this, keyStartBlock), height) | |
80 | + | ||
79 | 81 | func stringListToIntListHelper (acc,value) = (acc :+ parseIntValue(value)) | |
80 | 82 | ||
81 | 83 | ||
82 | 84 | func calcTotalProfitForHeight (h) = { | |
83 | - | let startBlock = valueOrElse(getInteger(this, keyStartBlock), height) | |
84 | 85 | let startPeriod = fraction(startBlock, 1, emissionPeriodInBlocks) | |
85 | 86 | let elapsedPeriods = ((h / emissionPeriodInBlocks) - startPeriod) | |
86 | 87 | max([0, (emissionPerPeriod * elapsedPeriods)]) | |
87 | 88 | } | |
88 | 89 | ||
89 | 90 | ||
90 | 91 | func calcTotalProfit () = calcTotalProfitForHeight(height) | |
91 | 92 | ||
92 | 93 | ||
93 | 94 | func getMaxAssetAvailable () = match assetIdBytes { | |
94 | 95 | case u: Unit => | |
95 | 96 | wavesBalance(this).available | |
96 | 97 | case b: ByteVector => | |
97 | 98 | assetBalance(this, b) | |
98 | 99 | case _ => | |
99 | 100 | throw("Match error") | |
100 | 101 | } | |
101 | 102 | ||
102 | 103 | ||
103 | 104 | func getTotalAssetAmountWithProfitOrMaxAvailable (atHeight) = { | |
104 | 105 | let totalAssetAmountWithProfit = (totalAssetAmount + calcTotalProfitForHeight(atHeight)) | |
105 | 106 | let totalAmount = min([totalAssetAmountWithProfit, getMaxAssetAvailable()]) | |
106 | 107 | if ((totalLpAmount == 0)) | |
107 | 108 | then 0 | |
108 | 109 | else totalAmount | |
109 | 110 | } | |
110 | 111 | ||
111 | 112 | ||
112 | 113 | func getPriceAtHeight (h) = if ((totalLpAmount != 0)) | |
113 | 114 | then fraction(toBigInt(getTotalAssetAmountWithProfitOrMaxAvailable(h)), scale18BigInt, toBigInt(totalLpAmount)) | |
114 | 115 | else scale18BigInt | |
115 | 116 | ||
116 | 117 | ||
117 | 118 | func getCurrentPrice () = getPriceAtHeight(height) | |
118 | 119 | ||
119 | 120 | ||
120 | 121 | func getRemainingBlocks () = if ((emissionPerBlock == 0)) | |
121 | 122 | then 0 | |
122 | 123 | else fraction((getMaxAssetAvailable() - getTotalAssetAmountWithProfitOrMaxAvailable(height)), 1, emissionPerBlock) | |
123 | 124 | ||
124 | 125 | ||
125 | 126 | func getUserStakingNodesData (userAddress) = { | |
126 | 127 | let nodesRaw = valueOrElse(getString(this, keyUserStakingNodes(userAddress)), "") | |
127 | 128 | let sharesRaw = valueOrElse(getString(this, keyUserStakingNodesShares(userAddress)), "") | |
128 | 129 | let nodesList = if ((nodesRaw == "")) | |
129 | 130 | then nil | |
130 | 131 | else split(nodesRaw, SEP) | |
131 | 132 | let sharesStringList = if ((sharesRaw == "")) | |
132 | 133 | then nil | |
133 | 134 | else split(sharesRaw, SEP) | |
134 | 135 | let sharesList = { | |
135 | 136 | let $l = sharesStringList | |
136 | 137 | let $s = size($l) | |
137 | 138 | let $acc0 = nil | |
138 | 139 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
139 | 140 | then $a | |
140 | 141 | else stringListToIntListHelper($a, $l[$i]) | |
141 | 142 | ||
142 | 143 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
143 | 144 | then $a | |
144 | 145 | else throw("List size exceeds 20") | |
145 | 146 | ||
146 | 147 | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20) | |
147 | 148 | } | |
148 | 149 | $Tuple2(nodesList, sharesList) | |
149 | 150 | } | |
150 | 151 | ||
151 | 152 | ||
152 | 153 | func calcAssetFromLp (lpAmount,atHeight) = max([0, toInt(fraction(toBigInt(lpAmount), getPriceAtHeight(atHeight), scale18BigInt))]) | |
153 | 154 | ||
154 | 155 | ||
155 | 156 | func calcLpFromAsset (assetAmount,atHeight) = max([0, toInt(fraction(toBigInt(assetAmount), scale18BigInt, getPriceAtHeight(atHeight)))]) | |
156 | 157 | ||
157 | 158 | ||
158 | 159 | func getUserLpAmount (userAddress) = valueOrElse(getInteger(this, keyUserLpAmount(userAddress)), 0) | |
159 | 160 | ||
160 | 161 | ||
161 | 162 | func getUserLockedLpAmount (userAddress) = valueOrElse(getInteger(this, keyUserLockedLpAmount(userAddress)), 0) | |
162 | 163 | ||
163 | 164 | ||
164 | 165 | func getUserAvailableAssetsToWithdraw (userAddress) = { | |
165 | 166 | let userLpAmount = getUserLpAmount(userAddress) | |
166 | 167 | calcAssetFromLp(userLpAmount, height) | |
167 | 168 | } | |
168 | 169 | ||
169 | 170 | ||
170 | - | func getClearStakingNodesActions (userAddress) = [DeleteEntry(keyUserStakingNodes(userAddress)), DeleteEntry(keyUserStakingNodesShares(userAddress))] | |
171 | - | ||
172 | - | ||
173 | 171 | func getStakeActions (i,userAddress,stakeHeight) = { | |
174 | - | let checks = [if ((size(i.payments) == 1)) | |
172 | + | let checks = [if ((stakeHeight >= startBlock)) | |
173 | + | then true | |
174 | + | else throwErr("startBlock is greater than stakeHeight"), if ((size(i.payments) == 1)) | |
175 | 175 | then true | |
176 | 176 | else throwErr("should include 1 payment"), if ((i.payments[0].assetId == assetIdBytes)) | |
177 | 177 | then true | |
178 | 178 | else throwErr(("payment should be in " + assetIdString)), if ((i.payments[0].amount > 0)) | |
179 | 179 | then true | |
180 | 180 | else "payment amount should be greater than 0", if ((size(fromBase58String(userAddress)) == ADDRESS_BYTES_SIZE)) | |
181 | 181 | then true | |
182 | 182 | else throwErr("user address is not valid")] | |
183 | 183 | if ((checks == checks)) | |
184 | 184 | then { | |
185 | 185 | let paymentAmount = i.payments[0].amount | |
186 | 186 | let paymentLpAmount = calcLpFromAsset(paymentAmount, stakeHeight) | |
187 | 187 | let userLpAmount = getUserLpAmount(userAddress) | |
188 | 188 | let userTotalStakedAmount = valueOrElse(getInteger(this, keyUserTotalAssetStaked(userAddress)), 0) | |
189 | 189 | let newTotalLpAmount = (totalLpAmount + paymentLpAmount) | |
190 | 190 | let newTotalAssetAmount = calcAssetFromLp(newTotalLpAmount, stakeHeight) | |
191 | 191 | let newUserLpAmount = (userLpAmount + paymentLpAmount) | |
192 | 192 | let newUserTotalStakedAmount = (userTotalStakedAmount + paymentAmount) | |
193 | - | [StringEntry(keyHistory("stake", userAddress, i.transactionId), formatHistory(calcTotalProfit(), getCurrentPrice(), totalLpAmount, totalAssetAmount)), IntegerEntry(keyTotalLpAmount, newTotalLpAmount), IntegerEntry(keyTotalAssetAmount, newTotalAssetAmount), IntegerEntry(keyUserLpAmount(userAddress), newUserLpAmount), IntegerEntry(keyUserTotalAssetStaked(userAddress), newUserTotalStakedAmount), IntegerEntry(keyStartBlock, | |
193 | + | [StringEntry(keyHistory("stake", userAddress, i.transactionId), formatHistory(calcTotalProfit(), getCurrentPrice(), totalLpAmount, totalAssetAmount)), IntegerEntry(keyTotalLpAmount, newTotalLpAmount), IntegerEntry(keyTotalAssetAmount, newTotalAssetAmount), IntegerEntry(keyUserLpAmount(userAddress), newUserLpAmount), IntegerEntry(keyUserTotalAssetStaked(userAddress), newUserTotalStakedAmount), IntegerEntry(keyStartBlock, stakeHeight)] | |
194 | 194 | } | |
195 | 195 | else throw("Strict value is not equal to itself.") | |
196 | 196 | } | |
197 | 197 | ||
198 | 198 | ||
199 | 199 | @Callable(i) | |
200 | 200 | func stakeForAdmin (userAddress,stakeHeight) = { | |
201 | 201 | let check = [if ((i.caller == adminAddress)) | |
202 | 202 | then true | |
203 | 203 | else throwErr("permission denied")] | |
204 | 204 | if ((check == check)) | |
205 | 205 | then getStakeActions(i, userAddress, stakeHeight) | |
206 | 206 | else throw("Strict value is not equal to itself.") | |
207 | 207 | } | |
208 | 208 | ||
209 | 209 | ||
210 | 210 | ||
211 | 211 | @Callable(i) | |
212 | 212 | func airdrop (addressList,amountList,airdropHeight) = { | |
213 | 213 | func sum (accum,next) = if ((0 > next)) | |
214 | 214 | then throwErr("negative amount value in amountList") | |
215 | 215 | else (accum + next) | |
216 | 216 | ||
217 | 217 | let amountListSum = { | |
218 | 218 | let $l = amountList | |
219 | 219 | let $s = size($l) | |
220 | 220 | let $acc0 = 0 | |
221 | 221 | func $f0_1 ($a,$i) = if (($i >= $s)) | |
222 | 222 | then $a | |
223 | 223 | else sum($a, $l[$i]) | |
224 | 224 | ||
225 | 225 | func $f0_2 ($a,$i) = if (($i >= $s)) | |
226 | 226 | then $a | |
227 | 227 | else throw("List size exceeds 90") | |
228 | 228 | ||
229 | 229 | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($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), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90) | |
230 | 230 | } | |
231 | 231 | let check = [if ((i.caller == adminAddress)) | |
232 | 232 | then true | |
233 | - | else throwErr("permission denied"), if ((size(i.payments) == 1)) | |
233 | + | else throwErr("permission denied"), if ((airdropHeight >= startBlock)) | |
234 | + | then true | |
235 | + | else throwErr("airdropHeight is greater than stakeHeight"), if ((size(i.payments) == 1)) | |
234 | 236 | then true | |
235 | 237 | else throwErr("should include 1 payment"), if ((i.payments[0].assetId == assetIdBytes)) | |
236 | 238 | then true | |
237 | 239 | else throwErr(("payment should be in " + assetIdString)), if ((i.payments[0].amount > 0)) | |
238 | 240 | then true | |
239 | 241 | else "payment amount should be greater than 0", if ((size(addressList) == size(amountList))) | |
240 | 242 | then true | |
241 | 243 | else throwErr("addressList should be same size as amountList"), if ((i.payments[0].amount >= amountListSum)) | |
242 | 244 | then true | |
243 | 245 | else throwErr("payment amount is less than sum of amountList")] | |
244 | 246 | if ((check == check)) | |
245 | 247 | then { | |
246 | 248 | func getAirdropStateChanges (accum,assetAmount) = { | |
247 | - | let $ | |
248 | - | let result = $ | |
249 | - | let index = $ | |
250 | - | let totalLp = $ | |
251 | - | let processedList = $ | |
249 | + | let $t083648415 = accum | |
250 | + | let result = $t083648415._1 | |
251 | + | let index = $t083648415._2 | |
252 | + | let totalLp = $t083648415._3 | |
253 | + | let processedList = $t083648415._4 | |
252 | 254 | let addressString = addressList[index] | |
253 | 255 | let address = match addressFromString(addressString) { | |
254 | 256 | case adr: Address => | |
255 | 257 | adr | |
256 | 258 | case _ => | |
257 | 259 | throwErr("invalid address in addressList") | |
258 | 260 | } | |
259 | 261 | let ch = [if (!(containsElement(processedList, address))) | |
260 | 262 | then true | |
261 | 263 | else throwErr("duplicate address is addressList")] | |
262 | 264 | if ((ch == ch)) | |
263 | 265 | then { | |
264 | 266 | let addedLpAmount = calcLpFromAsset(assetAmount, airdropHeight) | |
265 | 267 | let userLockedLpKey = keyUserLockedLpAmount(addressString) | |
266 | 268 | let oldLpAmount = valueOrElse(getInteger(this, userLockedLpKey), 0) | |
267 | 269 | $Tuple4((result :+ IntegerEntry(userLockedLpKey, (oldLpAmount + addedLpAmount))), (index + 1), (totalLp + addedLpAmount), (processedList :+ address)) | |
268 | 270 | } | |
269 | 271 | else throw("Strict value is not equal to itself.") | |
270 | 272 | } | |
271 | 273 | ||
272 | - | let $ | |
274 | + | let $t091949311 = { | |
273 | 275 | let $l = amountList | |
274 | 276 | let $s = size($l) | |
275 | 277 | let $acc0 = $Tuple4(nil, 0, 0, nil) | |
276 | 278 | func $f1_1 ($a,$i) = if (($i >= $s)) | |
277 | 279 | then $a | |
278 | 280 | else getAirdropStateChanges($a, $l[$i]) | |
279 | 281 | ||
280 | 282 | func $f1_2 ($a,$i) = if (($i >= $s)) | |
281 | 283 | then $a | |
282 | 284 | else throw("List size exceeds 90") | |
283 | 285 | ||
284 | 286 | $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20), 21), 22), 23), 24), 25), 26), 27), 28), 29), 30), 31), 32), 33), 34), 35), 36), 37), 38), 39), 40), 41), 42), 43), 44), 45), 46), 47), 48), 49), 50), 51), 52), 53), 54), 55), 56), 57), 58), 59), 60), 61), 62), 63), 64), 65), 66), 67), 68), 69), 70), 71), 72), 73), 74), 75), 76), 77), 78), 79), 80), 81), 82), 83), 84), 85), 86), 87), 88), 89), 90) | |
285 | 287 | } | |
286 | - | let airdropEntries = $ | |
287 | - | let _a = $ | |
288 | - | let addedTotalLockedLpAmount = $ | |
289 | - | let _b = $ | |
288 | + | let airdropEntries = $t091949311._1 | |
289 | + | let _a = $t091949311._2 | |
290 | + | let addedTotalLockedLpAmount = $t091949311._3 | |
291 | + | let _b = $t091949311._4 | |
290 | 292 | let newTotalAsset = calcAssetFromLp((totalLpAmount + addedTotalLockedLpAmount), airdropHeight) | |
291 | 293 | ([IntegerEntry(keyTotalLockedLpAmount, (totalLockedLpAmount + addedTotalLockedLpAmount)), IntegerEntry(keyTotalLpAmount, (totalLpAmount + addedTotalLockedLpAmount)), IntegerEntry(keyTotalAssetAmount, newTotalAsset), IntegerEntry(keyStartBlock, height)] ++ airdropEntries) | |
292 | 294 | } | |
293 | 295 | else throw("Strict value is not equal to itself.") | |
294 | 296 | } | |
295 | 297 | ||
296 | 298 | ||
297 | 299 | ||
298 | 300 | @Callable(i) | |
299 | 301 | func getUserAssetsREADONLY (userAddress) = { | |
300 | 302 | let userLpAmount = getUserLpAmount(userAddress) | |
301 | 303 | let userLockedLpAmount = getUserLockedLpAmount(userAddress) | |
302 | 304 | let userLockedAssetAmount = calcAssetFromLp(userLockedLpAmount, height) | |
303 | 305 | let userAvailableAssetToWithdraw = getUserAvailableAssetsToWithdraw(userAddress) | |
304 | 306 | let userTotalStakedAmount = valueOrElse(getInteger(this, keyUserTotalAssetStaked(userAddress)), 0) | |
305 | 307 | let userTotalAssetWithdrawn = valueOrElse(getInteger(this, keyUserTotalAssetWithdrawn(userAddress)), 0) | |
306 | - | let $ | |
307 | - | let userStakingNodesList = $ | |
308 | - | let userStakingNodeSharesList = $ | |
308 | + | let $t01068410776 = getUserStakingNodesData(userAddress) | |
309 | + | let userStakingNodesList = $t01068410776._1 | |
310 | + | let userStakingNodeSharesList = $t01068410776._2 | |
309 | 311 | $Tuple2(nil, $Tuple10(userLpAmount, userAvailableAssetToWithdraw, getCurrentPrice(), userTotalStakedAmount, userTotalAssetWithdrawn, userLockedLpAmount, userLockedAssetAmount, userStakingNodesList, userStakingNodeSharesList, getRemainingBlocks())) | |
310 | 312 | } | |
311 | 313 | ||
312 | 314 | ||
313 | 315 | ||
314 | 316 | @Callable(i) | |
315 | 317 | func getTotalAssetsREADONLY () = $Tuple2(nil, $Tuple6(totalLpAmount, getTotalAssetAmountWithProfitOrMaxAvailable(height), getCurrentPrice(), totalLockedLpAmount, calcAssetFromLp(totalLockedLpAmount, height), getRemainingBlocks())) | |
316 | 318 | ||
317 | 319 |
github/deemru/w8io/169f3d6 70.92 ms ◑