tx · AVA7ohQFGuQZWif8dTvBQeCDuixHRGR83REdxJTfSt6n

3M7uzD8rT54fKQcP4s417if1VKU8yromEP2:  -0.05500000 Waves

2025.10.18 01:49 [492657] smart account 3M7uzD8rT54fKQcP4s417if1VKU8yromEP2 > SELF 0.00000000 Waves

{ "type": 13, "id": "AVA7ohQFGuQZWif8dTvBQeCDuixHRGR83REdxJTfSt6n", "fee": 5500000, "feeAssetId": null, "timestamp": 1760741367329, "version": 2, "chainId": 82, "sender": "3M7uzD8rT54fKQcP4s417if1VKU8yromEP2", "senderPublicKey": "2eFPamS5zez1159HEFEJaxs2XPq3AupPbNApNGffRjqf", "proofs": [ "4C36zKPBsn2SUFZo9FFCTffyQTi2YFhgfmo4FvufeV7MUwwo3CG2NEc4paLtfWz3pG5NtpDUw4dYuMRd7tKmgrbS" ], "script": "base64:BgJUCAISBQoDGAEBEgMKAQgSAwoBCBIDCgEIEgMKAQgSCAoGCAgICAgIEgMKAQgSAwoBCBIDCgEIEgkKBwgICAEICAgSAwoBCBIDCgEIEgMKAQESABIAGgAHVkVSU0lPTgIFMS4wLjAAA1NFUAIBXwAGQklHU0VQAgJfXwAIa19BRE1JTlMCC0FETUlOU19QQktTAAprX01BTkFHRVJTAg1NQU5BR0VSU19QQktTAAdrX0xJTUlUAgxWT1RFU19RVU9SVU0ADmtfU0NSSVBUX1RZUEVTAgxTQ1JJUFRfVFlQRVMAC2tfSVRFUkFUSU9OAglJVEVSQVRJT04AEGtfQVBQUk9WRURfQ09VTlQCDkFQUFJPVkVEX0NPVU5UAAxrX3NjcmlwdFR5cGUCCnNjcmlwdFR5cGUADGtfc2NyaXB0SGFzaAIKc2NyaXB0SGFzaAAQa19zdG9yYWdlQWRkcmVzcwIOc3RvcmFnZUFkZHJlc3MABmtfZGVzYwIEZGVzYwAGa19pdGVyAglpdGVyYXRpb24AC2tfaW5pdGlhdG9yAglpbml0aWF0b3IAD2tfY29uZmlybWF0aW9ucwINY29uZmlybWF0aW9ucwANa19yYXdDb2RlSGFzaAILcmF3Q29kZUhhc2gBDF92YWxpZGF0ZVBiawEDcGJrBAdhZGRyZXNzCQCnCAEJANkEAQUDcGJrAwkAAAIFB2FkZHJlc3MFB2FkZHJlc3MECmFkZHJlc3NTdHIJAKUIAQUHYWRkcmVzcwMJAAACBQphZGRyZXNzU3RyBQphZGRyZXNzU3RyBgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgEQX3ZhbGlkYXRlQWRkcmVzcwEHYWRkcmVzcwQEYWRkcgkBEUBleHRyTmF0aXZlKDEwNjIpAQUHYWRkcmVzcwMJAAACBQRhZGRyBQRhZGRyBgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgELX3ZhbGlkYXRlVHMBAnRzBAZjdXJyVHMEByRtYXRjaDAJAO0HAQUGaGVpZ2h0AwkAAQIFByRtYXRjaDACCUJsb2NrSW5mbwQFYmxvY2sFByRtYXRjaDAIBQVibG9jawl0aW1lc3RhbXAJAAIBAhBDYW4ndCBmaW5kIGJsb2NrCQBmAgUCdHMFBmN1cnJUcwEJX2dldExpbWl0AAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEFB2tfTElNSVQCGFF1b3J1bSBsaW1pdCBub3QgZGVmaW5lZAERX2dldEFwcHJvdmVkQ291bnQACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ8IAQUQa19BUFBST1ZFRF9DT1VOVAI1QW1vdW50IG9mIHNpbXVsdGFuZW91c2x5IGFwcHJvdmVkIHNjcmlwdHMgbm90IGRlZmluZWQBCF9pc0FkbWluAQNwYmsECWFkbWluc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBBQhrX0FETUlOUwIABAZhZG1pbnMJALUJAgUJYWRtaW5zU3RyAgEsCQECIT0CCQDPCAIFBmFkbWlucwUDcGJrBQR1bml0AQpfaXNNYW5hZ2VyAQNwYmsEC21hbmFnZXJzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEFCmtfTUFOQUdFUlMCAAQIbWFuYWdlcnMJALUJAgULbWFuYWdlcnNTdHICASwJAQIhPQIJAM8IAgUIbWFuYWdlcnMFA3BiawUEdW5pdAEQX2lzVmFsaWRPcGVyYXRvcgEDcGJrBAttYW5hZ2Vyc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBBQprX01BTkFHRVJTAgAECG1hbmFnZXJzCQC1CQIFC21hbmFnZXJzU3RyAgEsBAdpc0FkbWluCQEIX2lzQWRtaW4BBQNwYmsECWlzTWFuYWdlcgkBCl9pc01hbmFnZXIBBQNwYmsDBQdpc0FkbWluBgUJaXNNYW5hZ2VyAQpzYXZlVHhEYXRhAgZ0YXNrSWQEZGF0YQQHaW5kaWNlcwkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFCQDMCAIABgkAzAgCAAcJAMwIAgAICQDMCAIACQkAzAgCAAoFA25pbAQIZGF0YVNpemUJALECAQUEZGF0YQQKaXNOb3RFbXB0eQMJAQIhPQIFCGRhdGFTaXplAAAGCQACAQIdc2F2ZVR4RGF0YTogRW1wdHkgZGF0YSBwYXNzZWQDCQAAAgUKaXNOb3RFbXB0eQUKaXNOb3RFbXB0eQQJY2h1bmtTaXplALDqAQQLY2h1bmtzQ291bnQDCQAAAgkAagIFCGRhdGFTaXplBQljaHVua1NpemUAAAkAaQIFCGRhdGFTaXplBQljaHVua1NpemUJAGQCCQBpAgUIZGF0YVNpemUFCWNodW5rU2l6ZQABCgEKbWFrZUNodW5rcwIFYWNjdW0FaW5kZXgEBWNodW5rCQCvAgIIBQVhY2N1bQJfMQUJY2h1bmtTaXplAwkAAAIJALECAQUFY2h1bmsAAAUFYWNjdW0ECG5leHREYXRhCQCwAgIIBQVhY2N1bQJfMQUJY2h1bmtTaXplCQCUCgIFCG5leHREYXRhCQDNCAIIBQVhY2N1bQJfMgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFBnRhc2tJZAIJX190eGRhdGFfCQCkAwEFBWluZGV4BQVjaHVuawQHZW50cmllcwoAAiRsBQdpbmRpY2VzCgACJHMJAJADAQUCJGwKAAUkYWNjMAkAlAoCBQRkYXRhCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAg9fX3R4ZGF0YV9jaHVua3MJAKQDAQULY2h1bmtzQ291bnQFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEKbWFrZUNodW5rcwIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACggFB2VudHJpZXMCXzIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4PAWkBBGluaXQDC2FkbWluc19wYmtzBWxpbWl0DWFwcHJvdmVkQ291bnQDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAg5TZWxmIGNhbGwgb25seQQLc2l6ZV9hZG1pbnMDCQECIT0CCQCQAwEFC2FkbWluc19wYmtzAAAGCQACAQInQXQgbGVhc3Qgb25lIGFkbWluIHBiayBzaG91bGQgYmUgcGFzc2VkAwkAAAIFC3NpemVfYWRtaW5zBQtzaXplX2FkbWlucwQMY2hlY2tfYWRtaW5zAwkBASEBCQEJaXNEZWZpbmVkAQkAoggBBQhrX0FETUlOUwYJAAIBAg5BbHJlYWR5IGluaXRlZAMJAAACBQxjaGVja19hZG1pbnMFDGNoZWNrX2FkbWlucwQJemVyb0xpbWl0AwkAZgIFBWxpbWl0AAAGCQACAQITTGltaXQgc2hvdWxkIGJlID4gMAMJAAACBQl6ZXJvTGltaXQFCXplcm9MaW1pdAQIbGltaXRfb2sDCQBmAgkAkAMBBQthZG1pbnNfcGJrcwUFbGltaXQGCQACAQkArAICCQCsAgIJAKwCAgI5TGltaXQgcXVvcnVtIHNob3VsZCBiZSA8PSBzaXplIG9mIGxpc3Qgb2YgYWRtaW5zLCBsaW1pdDogCQCkAwEFBWxpbWl0AhAsIGN1cnJlbnQgc2l6ZTogCQCkAwEJAJADAQULYWRtaW5zX3Bia3MDCQAAAgUIbGltaXRfb2sFCGxpbWl0X29rCgEIdmFsaWRhdGUCBWFjY3VtBG5leHQJAQxfdmFsaWRhdGVQYmsBBQRuZXh0BAh2YWxpZFJlcwoAAiRsBQthZG1pbnNfcGJrcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEIdmFsaWRhdGUCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoDCQAAAgUIdmFsaWRSZXMFCHZhbGlkUmVzBAlhZG1pbnNTdHIJALkJAgULYWRtaW5zX3Bia3MCASwJAMwIAgkBC1N0cmluZ0VudHJ5AgUIa19BRE1JTlMFCWFkbWluc1N0cgkAzAgCCQEMSW50ZWdlckVudHJ5AgUHa19MSU1JVAUFbGltaXQJAMwIAgkBDEludGVnZXJFbnRyeQIFEGtfQVBQUk9WRURfQ09VTlQFDWFwcHJvdmVkQ291bnQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQhhZGRBZG1pbgEDcGJrBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAdpc0FkbWluAwkBCF9pc0FkbWluAQUJY2FsbGVyUGJrBgkAAgECGU9ubHkgYWRtaW5zIGNhbiBhZGQgYWRtaW4DCQAAAgUHaXNBZG1pbgUHaXNBZG1pbgQHaXNWYWxpZAMJAQxfdmFsaWRhdGVQYmsBBQNwYmsGCQACAQIRSW52YWxpZCBhZG1pbiBwYmsDCQAAAgUHaXNWYWxpZAUHaXNWYWxpZAQMbmV3QWRtaW5zU3RyCQCsAgIJAKwCAgkBEUBleHRyTmF0aXZlKDEwNTgpAQUIa19BRE1JTlMCASwFA3BiawkAzAgCCQELU3RyaW5nRW50cnkCBQhrX0FETUlOUwUMbmV3QWRtaW5zU3RyBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQELcmVtb3ZlQWRtaW4BA3BiawQJY2FsbGVyUGJrCQDYBAEIBQFpD2NhbGxlclB1YmxpY0tleQQHaXNBZG1pbgMJAQhfaXNBZG1pbgEFCWNhbGxlclBiawYJAAIBAhlPbmx5IGFkbWlucyBjYW4gYWRkIGFkbWluAwkAAAIFB2lzQWRtaW4FB2lzQWRtaW4EB2lzVmFsaWQDCQEMX3ZhbGlkYXRlUGJrAQUDcGJrBgkAAgECEUludmFsaWQgYWRtaW4gcGJrAwkAAAIFB2lzVmFsaWQFB2lzVmFsaWQEDG9sZEFkbWluc1N0cgkBEUBleHRyTmF0aXZlKDEwNTgpAQUIa19BRE1JTlMECW9sZEFkbWlucwkAtQkCBQxvbGRBZG1pbnNTdHICASwECmNoZWNrQ291bnQDCQBnAgABCQCQAwEFCW9sZEFkbWlucwkAAgECL0Nhbm5vdCByZW1vdmUsIGF0IGxlYXN0IG9uZSBhZG1pbiBzaG91bGQgcmVtYWluBgMJAAACBQpjaGVja0NvdW50BQpjaGVja0NvdW50BAVpbmRleAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQDPCAIFCW9sZEFkbWlucwUDcGJrAhhUaGlzIHBiayBpcyBub3QgYW4gYWRtaW4ECW5ld0FkbWlucwkA0QgCBQlvbGRBZG1pbnMFBWluZGV4BAxuZXdBZG1pbnNTdHIJALkJAgUJbmV3QWRtaW5zAgEsCQDMCAIJAQtTdHJpbmdFbnRyeQIFCGtfQURNSU5TBQxuZXdBZG1pbnNTdHIFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQ1hZGRTY3JpcHRUeXBlAQpzY3JpcHRUeXBlBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5AwkBASEBCQEIX2lzQWRtaW4BBQljYWxsZXJQYmsJAAIBAgtOb3QgYWxsb3dlZAQKY2hlY2tFbXB0eQMJAAACCQCxAgEFCnNjcmlwdFR5cGUAAAkAAgECHFNjcmlwdCB0eXBlIGNhbiBub3QgYmUgZW1wdHkGAwkAAAIFCmNoZWNrRW1wdHkFCmNoZWNrRW1wdHkEDnNjcmlwdFR5cGVzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEFDmtfU0NSSVBUX1RZUEVTAgAEC3NjcmlwdFR5cGVzCQC8CQIFDnNjcmlwdFR5cGVzU3RyAgEsBAtjaGVja0V4aXN0cwMJAQIhPQIJAM8IAgULc2NyaXB0VHlwZXMFCnNjcmlwdFR5cGUFBHVuaXQJAAIBCQCsAgIJAKwCAgINU2NyaXB0IHR5cGUgPAUKc2NyaXB0VHlwZQIPPiBhbHJlYWR5IGFkZGVkBgMJAAACBQtjaGVja0V4aXN0cwULY2hlY2tFeGlzdHMEDm5ld1NjcmlwdFR5cGVzAwkBAiE9AgUOc2NyaXB0VHlwZXNTdHICAAkAzQgCBQtzY3JpcHRUeXBlcwUKc2NyaXB0VHlwZQkAzAgCBQpzY3JpcHRUeXBlBQNuaWwEEW5ld1NjcmlwdFR5cGVzU3RyCQC5CQIFDm5ld1NjcmlwdFR5cGVzAgEsCQDMCAIJAQtTdHJpbmdFbnRyeQIFDmtfU0NSSVBUX1RZUEVTBRFuZXdTY3JpcHRUeXBlc1N0cgUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBEHJlbW92ZVNjcmlwdFR5cGUBCnNjcmlwdFR5cGUECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkDCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECC05vdCBhbGxvd2VkBA5zY3JpcHRUeXBlc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBBQ5rX1NDUklQVF9UWVBFUwIABAtzY3JpcHRUeXBlcwkAvAkCBQ5zY3JpcHRUeXBlc1N0cgIBLAQFaW5kZXgJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAzwgCBQtzY3JpcHRUeXBlcwUKc2NyaXB0VHlwZQkArAICCQCsAgICDVNjcmlwdCB0eXBlIDwFCnNjcmlwdFR5cGUCED4gaXMgbm90IHByZXNlbnQEDm5ld1NjcmlwdFR5cGVzCQDRCAIFC3NjcmlwdFR5cGVzBQVpbmRleAQRbmV3U2NyaXB0VHlwZXNTdHIJALkJAgUObmV3U2NyaXB0VHlwZXMCASwJAMwIAgkBC1N0cmluZ0VudHJ5AgUOa19TQ1JJUFRfVFlQRVMFEW5ld1NjcmlwdFR5cGVzU3RyBQNuaWwBaQEJYWRkU2NyaXB0BgpzY3JpcHRUeXBlDnN0b3JhZ2VBZGRyZXNzC19zY3JpcHRIYXNoBGRlc2MMaW5pdGlhdG9yQWRyC3Jhd0NvZGVIYXNoBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5AwkBASEBCQEQX2lzVmFsaWRPcGVyYXRvcgEFCWNhbGxlclBiawkAAgECC05vdCBhbGxvd2VkBBFzdG9yYWdlU2NyaXB0SGFzaAkA2AQBCQEFdmFsdWUBCQDxBwEJARFAZXh0ck5hdGl2ZSgxMDYyKQEFDnN0b3JhZ2VBZGRyZXNzBA5zY3JpcHRUeXBlc1N0cgkBEUBleHRyTmF0aXZlKDEwNTgpAQUOa19TQ1JJUFRfVFlQRVMEEV9jaGVja1NjcmlwdFR5cGVzAwkAAAIJALMJAgUOc2NyaXB0VHlwZXNTdHIFCnNjcmlwdFR5cGUFBHVuaXQJAAIBCQCsAgICNWludmFsaWQgc2NyaXB0IHR5cGUgb3IgaXQgaXMgbm90IHByZXNlbnQgaW4gc3RvcmFnZTogBQpzY3JpcHRUeXBlBgMJAAACBRFfY2hlY2tTY3JpcHRUeXBlcwURX2NoZWNrU2NyaXB0VHlwZXMEEF9zY3JpcHRUeXBlQ2hlY2sDCQECIT0CBRFzdG9yYWdlU2NyaXB0SGFzaAULX3NjcmlwdEhhc2gJAAIBAjRTdG9yZWQgYW5kIHBhc3NlZCBhcyBhcmd1bWVudCBzY3JpcHRIYXNoZXMgbm90IG1hdGNoBgMJAAACBRBfc2NyaXB0VHlwZUNoZWNrBRBfc2NyaXB0VHlwZUNoZWNrBApfZGVzY0NoZWNrAwkAAAIJALECAQUEZGVzYwAACQACAQIaRGVzY3JpcHRpb24gY2FuJ3QgYmUgZW1wdHkGAwkAAAIFCl9kZXNjQ2hlY2sFCl9kZXNjQ2hlY2sED19pbml0aWF0b3JDaGVjawMJAQIhPQIJAKUIAQkApwgBCAUBaQ9jYWxsZXJQdWJsaWNLZXkFDGluaXRpYXRvckFkcgkAAgECJkluaXRpYXRvciBhZGRyZXNzIGRvZXNuJ3QgbWF0Y2ggY2FsbGVyBgMJAAACBQ9faW5pdGlhdG9yQ2hlY2sFD19pbml0aWF0b3JDaGVjawQJaXRlcmF0aW9uCQBkAgkBC3ZhbHVlT3JFbHNlAgkAnwgBCQCsAgIJAKwCAgULa19JVEVSQVRJT04FA1NFUAUKc2NyaXB0VHlwZQAAAAEECHNjcmlwdElkCQCsAgIJAKwCAgkArAICAghzY3JpcHQlJQUKc2NyaXB0VHlwZQICJSUFC19zY3JpcHRIYXNoCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQxrX3NjcmlwdFR5cGUFCnNjcmlwdFR5cGUJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFEGtfc3RvcmFnZUFkZHJlc3MFDnN0b3JhZ2VBZGRyZXNzCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQxrX3NjcmlwdEhhc2gFEXN0b3JhZ2VTY3JpcHRIYXNoCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQZrX2Rlc2MFBGRlc2MJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQZrX2l0ZXIFCWl0ZXJhdGlvbgkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgUKc2NyaXB0VHlwZQUGQklHU0VQBQZrX2l0ZXIFBkJJR1NFUAkApAMBBQlpdGVyYXRpb24CBl9faGFzaAULX3NjcmlwdEhhc2gJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFC2tfaW5pdGlhdG9yBQxpbml0aWF0b3JBZHIJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDWtfcmF3Q29kZUhhc2gFC3Jhd0NvZGVIYXNoCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQCsAgIJAKwCAgULa19JVEVSQVRJT04FA1NFUAUKc2NyaXB0VHlwZQUJaXRlcmF0aW9uBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEMcmVtb3ZlU2NyaXB0AQhzY3JpcHRJZAQJY2FsbGVyUGJrCQDYBAEIBQFpD2NhbGxlclB1YmxpY0tleQMJAQEhAQkBEF9pc1ZhbGlkT3BlcmF0b3IBBQljYWxsZXJQYmsJAAIBAgtOb3QgYWxsb3dlZAQMaW5pdGlhdG9yQWRyCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFCHNjcmlwdElkAgtfX2luaXRpYXRvcgMJAQIhPQIFDGluaXRpYXRvckFkcgkApQgBCQCnCAEIBQFpD2NhbGxlclB1YmxpY0tleQkAAgECF09ubHkgY3JlYXRvciBjYW4gcmVtb3ZlBApzY3JpcHRUeXBlCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUMa19zY3JpcHRUeXBlBAtfc2NyaXB0SGFzaAkBEUBleHRyTmF0aXZlKDEwNTgpAQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDGtfc2NyaXB0SGFzaAQEaXRlcgkBEUBleHRyTmF0aXZlKDEwNTUpAQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFBmtfaXRlcgQOYXBwcm92ZWRIYXNoZXMJAQt2YWx1ZU9yRWxzZQIJAKIIAQkArAICCQCsAgICB3NjcmlwdF8FCnNjcmlwdFR5cGUCCl9fYXBwcm92ZWQCAAQOX2NoZWNrQXBwcm92ZWQDCQECIT0CCQCzCQIFDmFwcHJvdmVkSGFzaGVzBQtfc2NyaXB0SGFzaAUEdW5pdAkAAgECJ0NhbnQgcmVtb3ZlIGFwcHJvdmVkIHNjcmlwdCwgdXNlIHJldm9rZQYDCQAAAgUOX2NoZWNrQXBwcm92ZWQFDl9jaGVja0FwcHJvdmVkCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQxrX3NjcmlwdFR5cGUJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFEGtfc3RvcmFnZUFkZHJlc3MJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDGtfc2NyaXB0SGFzaAkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUGa19kZXNjCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQZrX2l0ZXIJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIJAKwCAgUKc2NyaXB0VHlwZQINX19pdGVyYXRpb25fXwkApAMBBQRpdGVyAgZfX2hhc2gJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFC2tfaW5pdGlhdG9yCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQ1rX3Jhd0NvZGVIYXNoCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQ9rX2NvbmZpcm1hdGlvbnMFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARVhZGRDb25maXJtYXRpb25TY3JpcHQBCHNjcmlwdElkBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAVsaW1pdAkBCV9nZXRMaW1pdAADCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECF09ubHkgYWRtaW5zIGNhbiBjb25maXJtBAtfc2NyaXB0SGFzaAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCiCAEJAKwCAgUIc2NyaXB0SWQCDF9fc2NyaXB0SGFzaAIPSXZhbGlkIHNjcmlwdElkAwkAAAIFC19zY3JpcHRIYXNoBQtfc2NyaXB0SGFzaAQKc2NyaXB0VHlwZQkBEUBleHRyTmF0aXZlKDEwNTgpAQkArAICBQhzY3JpcHRJZAIMX19zY3JpcHRUeXBlBBFhcHByb3ZlZEhhc2hlc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIJAKwCAgIHc2NyaXB0XwUKc2NyaXB0VHlwZQIKX19hcHByb3ZlZAIABA5fY2hlY2tBcHByb3ZlZAMJAQIhPQIJALMJAgURYXBwcm92ZWRIYXNoZXNTdHIFC19zY3JpcHRIYXNoBQR1bml0CQACAQIfVGhpcyBzY3JpcHQgaXMgYXBwcm92ZWQgYWxyZWFkeQYDCQAAAgUOX2NoZWNrQXBwcm92ZWQFDl9jaGVja0FwcHJvdmVkBA9nbG9iYWxJdGVyYXRpb24JAQt2YWx1ZU9yRWxzZQIJAJ8IAQkArAICCQCsAgIFC2tfSVRFUkFUSU9OBQNTRVAFCnNjcmlwdFR5cGUAAAQNdGhpc0l0ZXJhdGlvbgkBBXZhbHVlAQkAnwgBCQCsAgIFCHNjcmlwdElkAgtfX2l0ZXJhdGlvbgQQY29uZmlybWF0aW9uc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIFCHNjcmlwdElkAg9fX2NvbmZpcm1hdGlvbnMCAAQNY29uZmlybWF0aW9ucwkAtQkCBRBjb25maXJtYXRpb25zU3RyAgEsAwkBAiE9AgkAzwgCBQ1jb25maXJtYXRpb25zBQljYWxsZXJQYmsFBHVuaXQJAAIBAiFZb3UgYWxyZWFkeSBjb25maXJtZWQgdGhpcyBzY3JpcHQEB25ld0xpc3QDCQECIT0CBRBjb25maXJtYXRpb25zU3RyAgAJAM0IAgUNY29uZmlybWF0aW9ucwUJY2FsbGVyUGJrCQDMCAIFCWNhbGxlclBiawUDbmlsBApuZXdMaXN0U3RyCQC5CQIFB25ld0xpc3QCASwECGFwcHJvdmVkCQBnAgkAkAMBBQduZXdMaXN0BQVsaW1pdAQGcmVzdWx0AwUIYXBwcm92ZWQEDWFwcHJvdmVkQ291bnQJARFfZ2V0QXBwcm92ZWRDb3VudAAEDmFwcHJvdmVkSGFzaGVzCQC8CQIFEWFwcHJvdmVkSGFzaGVzU3RyAgEsBBFhcHByb3ZlZEhhc2hlc05ldwMJAQIhPQIFEWFwcHJvdmVkSGFzaGVzU3RyAgAJAM0IAgUOYXBwcm92ZWRIYXNoZXMFC19zY3JpcHRIYXNoCQDMCAIFC19zY3JpcHRIYXNoBQNuaWwDCQBmAgkAkAMBBRFhcHByb3ZlZEhhc2hlc05ldwUNYXBwcm92ZWRDb3VudAQMaGFzaFRvUmVtb3ZlCQCRAwIFEWFwcHJvdmVkSGFzaGVzTmV3AAAEEHNjcmlwdFRvUmVtb3ZlSWQJAKwCAgkArAICCQCsAgICCHNjcmlwdCUlBQpzY3JpcHRUeXBlAgIlJQULX3NjcmlwdEhhc2gEFmFwcHJvdmVkSGFzaGVzU3RyaXBwZWQJANEIAgURYXBwcm92ZWRIYXNoZXNOZXcAAAkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgIHc2NyaXB0XwUKc2NyaXB0VHlwZQIKX19hcHByb3ZlZAkAugkCBRZhcHByb3ZlZEhhc2hlc1N0cmlwcGVkAgEsCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgkArAICBRBzY3JpcHRUb1JlbW92ZUlkBQZCSUdTRVAFD2tfY29uZmlybWF0aW9ucwIABQNuaWwJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgICB3NjcmlwdF8FCnNjcmlwdFR5cGUCCl9fYXBwcm92ZWQJALoJAgURYXBwcm92ZWRIYXNoZXNOZXcCASwFA25pbAUDbmlsCQDOCAIFBnJlc3VsdAkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFCHNjcmlwdElkAg9fX2NvbmZpcm1hdGlvbnMFCm5ld0xpc3RTdHIFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpARhyZXZva2VDb25maXJtYXRpb25TY3JpcHQBCHNjcmlwdElkBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAVsaW1pdAkBCV9nZXRMaW1pdAADCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECFk9ubHkgYWRtaW5zIGNhbiByZXZva2UEC19zY3JpcHRIYXNoCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKIIAQkArAICBQhzY3JpcHRJZAIMX19zY3JpcHRIYXNoAhBJbnZhbGlkIHNjcmlwdElkAwkAAAIFC19zY3JpcHRIYXNoBQtfc2NyaXB0SGFzaAQKc2NyaXB0VHlwZQkBEUBleHRyTmF0aXZlKDEwNTgpAQkArAICBQhzY3JpcHRJZAIMX19zY3JpcHRUeXBlBBFhcHByb3ZlZEhhc2hlc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIJAKwCAgIHc2NyaXB0XwUKc2NyaXB0VHlwZQIKX19hcHByb3ZlZAIABA5hcHByb3ZlZEhhc2hlcwkAvAkCBRFhcHByb3ZlZEhhc2hlc1N0cgIBLAQQY29uZmlybWF0aW9uc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIFCHNjcmlwdElkAg9fX2NvbmZpcm1hdGlvbnMCAAQNY29uZmlybWF0aW9ucwkAtQkCBRBjb25maXJtYXRpb25zU3RyAgEsBAxjb25maXJtSW5kZXgJAM8IAgUNY29uZmlybWF0aW9ucwUJY2FsbGVyUGJrAwkAAAIFDGNvbmZpcm1JbmRleAUEdW5pdAkAAgECMVlvdSBkaWRuJ3QgY29uZmlybSB0aGlzIHNjcmlwdCwgbm90aGluZyB0byByZXZva2UEEG5ld0NvbmZpcm1hdGlvbnMJANEIAgUNY29uZmlybWF0aW9ucwkBBXZhbHVlAQUMY29uZmlybUluZGV4BBNuZXdDb25maXJtYXRpb25zU3RyCQC5CQIFEG5ld0NvbmZpcm1hdGlvbnMCASwEC3dhc0FwcHJvdmVkCQECIT0CCQDPCAIFDmFwcHJvdmVkSGFzaGVzBQtfc2NyaXB0SGFzaAUEdW5pdAQNc3RpbGxBcHByb3ZlZAkAZwIJAJADAQUQbmV3Q29uZmlybWF0aW9ucwUFbGltaXQEDHN0YXRlQ2hhbmdlcwMDBQt3YXNBcHByb3ZlZAkBASEBBQ1zdGlsbEFwcHJvdmVkBwQPc2NyaXB0SGFzaEluZGV4CQEFdmFsdWUBCQDPCAIFDmFwcHJvdmVkSGFzaGVzBQtfc2NyaXB0SGFzaAQRbmV3QXBwcm92ZWRIYXNoZXMJANEIAgUOYXBwcm92ZWRIYXNoZXMFD3NjcmlwdEhhc2hJbmRleAQUbmV3QXBwcm92ZWRIYXNoZXNTdHIJALkJAgURbmV3QXBwcm92ZWRIYXNoZXMCASwJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgICB3NjcmlwdF8FCnNjcmlwdFR5cGUCCl9fYXBwcm92ZWQFFG5ld0FwcHJvdmVkSGFzaGVzU3RyBQNuaWwFA25pbAkAzggCBQxzdGF0ZUNoYW5nZXMJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQhzY3JpcHRJZAIPX19jb25maXJtYXRpb25zBRNuZXdDb25maXJtYXRpb25zU3RyBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEHYWRkVGFzawcEbmFtZQRkYXBwBHR4aWQJZXhlY3V0ZVRzBGRlc2MMaW5pdGlhdG9yQWRyBnR4ZGF0YQQJY2FsbGVyUGJrCQDYBAEIBQFpD2NhbGxlclB1YmxpY0tleQMJAQEhAQkBEF9pc1ZhbGlkT3BlcmF0b3IBBQljYWxsZXJQYmsJAAIBAgtOb3QgYWxsb3dlZAQKX25hbWVDaGVjawMJAAACCQCxAgEFBG5hbWUAAAkAAgECE05hbWUgY2FuJ3QgYmUgZW1wdHkGAwkAAAIFCl9uYW1lQ2hlY2sFCl9uYW1lQ2hlY2sECl9kYXBwQ2hlY2sDCQEBIQEJARBfdmFsaWRhdGVBZGRyZXNzAQUEZGFwcAkAAgECDkluY29ycmVjdCBkYXBwBgMJAAACBQpfZGFwcENoZWNrBQpfZGFwcENoZWNrBApfdHhpZENoZWNrAwkAAAIJALECAQUEdHhpZAAACQACAQITdHhpZCBjYW4ndCBiZSBlbXB0eQYDCQAAAgUKX3R4aWRDaGVjawUKX3R4aWRDaGVjawQPX3RpbWVzdGFtcENoZWNrAwkBASEBCQELX3ZhbGlkYXRlVHMBBQlleGVjdXRlVHMJAAIBAhN0eGlkIGNhbid0IGJlIGVtcHR5BgMJAAACBQ9fdGltZXN0YW1wQ2hlY2sFD190aW1lc3RhbXBDaGVjawQKX2Rlc2NDaGVjawMJAAACCQCxAgEFBGRlc2MAAAkAAgECGkRlc2NyaXB0aW9uIGNhbid0IGJlIGVtcHR5BgMJAAACBQpfZGVzY0NoZWNrBQpfZGVzY0NoZWNrBA9faW5pdGlhdG9yQ2hlY2sDCQECIT0CCQClCAEJAKcIAQgFAWkPY2FsbGVyUHVibGljS2V5BQxpbml0aWF0b3JBZHIJAAIBAiZJbml0aWF0b3IgYWRkcmVzcyBkb2Vzbid0IG1hdGNoIGNhbGxlcgYDCQAAAgUPX2luaXRpYXRvckNoZWNrBQ9faW5pdGlhdG9yQ2hlY2sEDF90eGRhdGFDaGVjawMJAAACCQCxAgEFBnR4ZGF0YQAACQACAQIWVHggZGF0YSBjYW4ndCBiZSBlbXB0eQYDCQAAAgUMX3R4ZGF0YUNoZWNrBQxfdHhkYXRhQ2hlY2sEBnRhc2tJZAkArAICCQCsAgIJAKwCAgIDdHhfBQRkYXBwBQNTRVAFBHR4aWQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQZ0YXNrSWQCBl9fbmFtZQUEbmFtZQkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFBnRhc2tJZAIGX19kYXBwBQRkYXBwCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAgZfX3R4SWQFBHR4aWQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQZ0YXNrSWQCBl9fZGVzYwUEZGVzYwkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFBnRhc2tJZAIIX190eGRhdGEFBnR4ZGF0YQkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFBnRhc2tJZAILX19pbml0aWF0b3IFDGluaXRpYXRvckFkcgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQZ0YXNrSWQCBF9fdHMFCWV4ZWN1dGVUcwUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBCnJlbW92ZVRhc2sBBnRhc2tJZAQJY2FsbGVyUGJrCQDYBAEIBQFpD2NhbGxlclB1YmxpY0tleQMJAQEhAQkBEF9pc1ZhbGlkT3BlcmF0b3IBBQljYWxsZXJQYmsJAAIBAgtOb3QgYWxsb3dlZAQMaW5pdGlhdG9yQWRyCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFBnRhc2tJZAILX19pbml0aWF0b3IDCQECIT0CBQxpbml0aWF0b3JBZHIJAKUIAQkApwgBCAUBaQ9jYWxsZXJQdWJsaWNLZXkJAAIBAhdPbmx5IGNyZWF0b3IgY2FuIHJlbW92ZQkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAIGX19uYW1lCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgUGdGFza0lkAgZfX2RhcHAJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICBQZ0YXNrSWQCBl9fdHhJZAkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAIGX19kZXNjCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgUGdGFza0lkAghfX3R4ZGF0YQkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAILX19pbml0aWF0b3IJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICBQZ0YXNrSWQCBF9fdHMJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICBQZ0YXNrSWQCD19fY29uZmlybWF0aW9ucwkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAIKX19hcHByb3ZlZAUDbmlsAWkBD2FkZENvbmZpcm1hdGlvbgEGdGFza0lkBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAVsaW1pdAkBCV9nZXRMaW1pdAADCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECF09ubHkgYWRtaW5zIGNhbiBjb25maXJtBAR0eElkCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKIIAQkArAICBQZ0YXNrSWQCBl9fdHhJZAILSXZhbGlkIHRhc2sDCQAAAgUEdHhJZAUEdHhJZAQEZGFwcAkBEUBleHRyTmF0aXZlKDEwNTgpAQkArAICBQZ0YXNrSWQCBl9fZGFwcAQQY29uZmlybWF0aW9uc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIFBnRhc2tJZAIPX19jb25maXJtYXRpb25zAgAEDWNvbmZpcm1hdGlvbnMJALUJAgUQY29uZmlybWF0aW9uc1N0cgIBLAMJAQIhPQIJAM8IAgUNY29uZmlybWF0aW9ucwUJY2FsbGVyUGJrBQR1bml0CQACAQIfWW91IGFscmVhZHkgY29uZmlybWVkIHRoaXMgdGFzawQHbmV3TGlzdAMJAQIhPQIFEGNvbmZpcm1hdGlvbnNTdHICAAkAzQgCBQ1jb25maXJtYXRpb25zBQljYWxsZXJQYmsJAMwIAgUJY2FsbGVyUGJrBQNuaWwECm5ld0xpc3RTdHIJALkJAgUHbmV3TGlzdAIBLAQIYXBwcm92ZWQJAGcCCQCQAwEFB25ld0xpc3QFBWxpbWl0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAg9fX2NvbmZpcm1hdGlvbnMFCm5ld0xpc3RTdHIJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAKwCAgkArAICCQCsAgIFBGRhcHAFA1NFUAUEdHhJZAIKX19hcHByb3ZlZAUIYXBwcm92ZWQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQhzZXRMaW1pdAEFbGltaXQECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkDCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECHk9ubHkgd2hpdGVsaXN0ZWQgY2FuIHNldCBsaW1pdAQJemVyb0xpbWl0AwkAZgIFBWxpbWl0AAAGCQACAQITTGltaXQgc2hvdWxkIGJlID4gMAMJAAACBQl6ZXJvTGltaXQFCXplcm9MaW1pdAQGYWRtaW5zCQERQGV4dHJOYXRpdmUoMTA1OCkBBQhrX0FETUlOUwMJAGYCBQVsaW1pdAkAsQIBBQZhZG1pbnMJAAIBCQCsAgIJAKwCAgkArAICAjlMaW1pdCBxdW9ydW0gc2hvdWxkIGJlIDw9IHNpemUgb2YgbGlzdCBvZiBhZG1pbnMsIGxpbWl0OiAJAKQDAQUFbGltaXQCECwgY3VycmVudCBzaXplOiAJAKQDAQkAsQIBBQZhZG1pbnMJAMwIAgkBDEludGVnZXJFbnRyeQIFB2tfTElNSVQFBWxpbWl0BQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEEdGVzdAAEAXgEByRtYXRjaDAJAO0HAQUGaGVpZ2h0AwkAAQIFByRtYXRjaDACCUJsb2NrSW5mbwQFYmxvY2sFByRtYXRjaDAJAAIBCQCkAwEIBQVibG9jawl0aW1lc3RhbXAJAAIBAhBDYW4ndCBmaW5kIGJsb2NrAwkAAAIFAXgFAXgFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQpjbGVhcl90ZXN0AAQHaW5kaWNlcwkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFCQDMCAIABgkAzAgCAAcJAMwIAgAICQDMCAIACQkAzAgCAAoFA25pbAoBDHJlbW92ZUNodW5rcwIFYWNjdW0FaW5kZXgJAM0IAgUFYWNjdW0JAQtEZWxldGVFbnRyeQEJAKwCAgkArAICAgh0ZXN0dGFzawIJX190eGRhdGFfCQCkAwEFBWluZGV4BAdlbnRyaWVzCgACJGwFB2luZGljZXMKAAIkcwkAkAMBBQIkbAoABSRhY2MwCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgIIdGVzdHRhc2sCD19fdHhkYXRhX2NodW5rcwUDbmlsCgEFJGYwXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQxyZW1vdmVDaHVua3MCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoFB2VudHJpZXMAVWnJHA==", "height": 492657, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: AwBdwbz8KPPNz3mDWC6WHGeG6X7mWQcYwek7FKa7dfuB Next: 6XZFEBgNMbtC81GinRzoWxbJhJWzZ4ciJaCdC5XHgpck Diff:
OldNewDifferences
3232 let k_initiator = "initiator"
3333
3434 let k_confirmations = "confirmations"
35+
36+let k_rawCodeHash = "rawCodeHash"
3537
3638 func _validatePbk (pbk) = {
3739 let address = addressFromPublicKey(fromBase58String(pbk))
303305
304306
305307 @Callable(i)
306-func addScript (scriptType,storageAddress,_scriptHash,desc,initiatorAdr) = {
308+func addScript (scriptType,storageAddress,_scriptHash,desc,initiatorAdr,rawCodeHash) = {
307309 let callerPbk = toBase58String(i.callerPublicKey)
308310 if (!(_isValidOperator(callerPbk)))
309311 then throw("Not allowed")
332334 then {
333335 let iteration = (valueOrElse(getInteger(((k_ITERATION + SEP) + scriptType)), 0) + 1)
334336 let scriptId = ((("script%%" + scriptType) + "%%") + _scriptHash)
335-[StringEntry(((scriptId + BIGSEP) + k_scriptType), scriptType), StringEntry(((scriptId + BIGSEP) + k_storageAddress), storageAddress), StringEntry(((scriptId + BIGSEP) + k_scriptHash), storageScriptHash), StringEntry(((scriptId + BIGSEP) + k_desc), desc), IntegerEntry(((scriptId + BIGSEP) + k_iter), iteration), StringEntry((((((scriptType + BIGSEP) + k_iter) + BIGSEP) + toString(iteration)) + "__hash"), _scriptHash), StringEntry(((scriptId + BIGSEP) + k_initiator), initiatorAdr), IntegerEntry(((k_ITERATION + SEP) + scriptType), iteration)]
337+[StringEntry(((scriptId + BIGSEP) + k_scriptType), scriptType), StringEntry(((scriptId + BIGSEP) + k_storageAddress), storageAddress), StringEntry(((scriptId + BIGSEP) + k_scriptHash), storageScriptHash), StringEntry(((scriptId + BIGSEP) + k_desc), desc), IntegerEntry(((scriptId + BIGSEP) + k_iter), iteration), StringEntry((((((scriptType + BIGSEP) + k_iter) + BIGSEP) + toString(iteration)) + "__hash"), _scriptHash), StringEntry(((scriptId + BIGSEP) + k_initiator), initiatorAdr), StringEntry(((scriptId + BIGSEP) + k_rawCodeHash), rawCodeHash), IntegerEntry(((k_ITERATION + SEP) + scriptType), iteration)]
336338 }
337339 else throw("Strict value is not equal to itself.")
338340 }
364366 then throw("Cant remove approved script, use revoke")
365367 else true
366368 if ((_checkApproved == _checkApproved))
367- then [DeleteEntry(((scriptId + BIGSEP) + k_scriptType)), DeleteEntry(((scriptId + BIGSEP) + k_storageAddress)), DeleteEntry(((scriptId + BIGSEP) + k_scriptHash)), DeleteEntry(((scriptId + BIGSEP) + k_desc)), DeleteEntry(((scriptId + BIGSEP) + k_iter)), DeleteEntry((((scriptType + "__iteration__") + toString(iter)) + "__hash")), DeleteEntry(((scriptId + BIGSEP) + k_initiator)), DeleteEntry(((scriptId + BIGSEP) + k_confirmations))]
369+ then [DeleteEntry(((scriptId + BIGSEP) + k_scriptType)), DeleteEntry(((scriptId + BIGSEP) + k_storageAddress)), DeleteEntry(((scriptId + BIGSEP) + k_scriptHash)), DeleteEntry(((scriptId + BIGSEP) + k_desc)), DeleteEntry(((scriptId + BIGSEP) + k_iter)), DeleteEntry((((scriptType + "__iteration__") + toString(iter)) + "__hash")), DeleteEntry(((scriptId + BIGSEP) + k_initiator)), DeleteEntry(((scriptId + BIGSEP) + k_rawCodeHash)), DeleteEntry(((scriptId + BIGSEP) + k_confirmations))]
368370 else throw("Strict value is not equal to itself.")
369371 }
370372 }
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let VERSION = "1.0.0"
55
66 let SEP = "_"
77
88 let BIGSEP = "__"
99
1010 let k_ADMINS = "ADMINS_PBKS"
1111
1212 let k_MANAGERS = "MANAGERS_PBKS"
1313
1414 let k_LIMIT = "VOTES_QUORUM"
1515
1616 let k_SCRIPT_TYPES = "SCRIPT_TYPES"
1717
1818 let k_ITERATION = "ITERATION"
1919
2020 let k_APPROVED_COUNT = "APPROVED_COUNT"
2121
2222 let k_scriptType = "scriptType"
2323
2424 let k_scriptHash = "scriptHash"
2525
2626 let k_storageAddress = "storageAddress"
2727
2828 let k_desc = "desc"
2929
3030 let k_iter = "iteration"
3131
3232 let k_initiator = "initiator"
3333
3434 let k_confirmations = "confirmations"
35+
36+let k_rawCodeHash = "rawCodeHash"
3537
3638 func _validatePbk (pbk) = {
3739 let address = addressFromPublicKey(fromBase58String(pbk))
3840 if ((address == address))
3941 then {
4042 let addressStr = toString(address)
4143 if ((addressStr == addressStr))
4244 then true
4345 else throw("Strict value is not equal to itself.")
4446 }
4547 else throw("Strict value is not equal to itself.")
4648 }
4749
4850
4951 func _validateAddress (address) = {
5052 let addr = addressFromStringValue(address)
5153 if ((addr == addr))
5254 then true
5355 else throw("Strict value is not equal to itself.")
5456 }
5557
5658
5759 func _validateTs (ts) = {
5860 let currTs = match blockInfoByHeight(height) {
5961 case block: BlockInfo =>
6062 block.timestamp
6163 case _ =>
6264 throw("Can't find block")
6365 }
6466 (ts > currTs)
6567 }
6668
6769
6870 func _getLimit () = valueOrErrorMessage(getInteger(k_LIMIT), "Quorum limit not defined")
6971
7072
7173 func _getApprovedCount () = valueOrErrorMessage(getInteger(k_APPROVED_COUNT), "Amount of simultaneously approved scripts not defined")
7274
7375
7476 func _isAdmin (pbk) = {
7577 let adminsStr = valueOrElse(getString(k_ADMINS), "")
7678 let admins = split(adminsStr, ",")
7779 (indexOf(admins, pbk) != unit)
7880 }
7981
8082
8183 func _isManager (pbk) = {
8284 let managersStr = valueOrElse(getString(k_MANAGERS), "")
8385 let managers = split(managersStr, ",")
8486 (indexOf(managers, pbk) != unit)
8587 }
8688
8789
8890 func _isValidOperator (pbk) = {
8991 let managersStr = valueOrElse(getString(k_MANAGERS), "")
9092 let managers = split(managersStr, ",")
9193 let isAdmin = _isAdmin(pbk)
9294 let isManager = _isManager(pbk)
9395 if (isAdmin)
9496 then true
9597 else isManager
9698 }
9799
98100
99101 func saveTxData (taskId,data) = {
100102 let indices = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
101103 let dataSize = size(data)
102104 let isNotEmpty = if ((dataSize != 0))
103105 then true
104106 else throw("saveTxData: Empty data passed")
105107 if ((isNotEmpty == isNotEmpty))
106108 then {
107109 let chunkSize = 30000
108110 let chunksCount = if (((dataSize % chunkSize) == 0))
109111 then (dataSize / chunkSize)
110112 else ((dataSize / chunkSize) + 1)
111113 func makeChunks (accum,index) = {
112114 let chunk = take(accum._1, chunkSize)
113115 if ((size(chunk) == 0))
114116 then accum
115117 else {
116118 let nextData = drop(accum._1, chunkSize)
117119 $Tuple2(nextData, (accum._2 :+ StringEntry(((taskId + "__txdata_") + toString(index)), chunk)))
118120 }
119121 }
120122
121123 let entries = {
122124 let $l = indices
123125 let $s = size($l)
124126 let $acc0 = $Tuple2(data, [StringEntry((taskId + "__txdata_chunks"), toString(chunksCount))])
125127 func $f0_1 ($a,$i) = if (($i >= $s))
126128 then $a
127129 else makeChunks($a, $l[$i])
128130
129131 func $f0_2 ($a,$i) = if (($i >= $s))
130132 then $a
131133 else throw("List size exceeds 10")
132134
133135 $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)
134136 }
135137 entries._2
136138 }
137139 else throw("Strict value is not equal to itself.")
138140 }
139141
140142
141143 @Callable(i)
142144 func init (admins_pbks,limit,approvedCount) = if ((i.caller != this))
143145 then throw("Self call only")
144146 else {
145147 let size_admins = if ((size(admins_pbks) != 0))
146148 then true
147149 else throw("At least one admin pbk should be passed")
148150 if ((size_admins == size_admins))
149151 then {
150152 let check_admins = if (!(isDefined(getString(k_ADMINS))))
151153 then true
152154 else throw("Already inited")
153155 if ((check_admins == check_admins))
154156 then {
155157 let zeroLimit = if ((limit > 0))
156158 then true
157159 else throw("Limit should be > 0")
158160 if ((zeroLimit == zeroLimit))
159161 then {
160162 let limit_ok = if ((size(admins_pbks) > limit))
161163 then true
162164 else throw(((("Limit quorum should be <= size of list of admins, limit: " + toString(limit)) + ", current size: ") + toString(size(admins_pbks))))
163165 if ((limit_ok == limit_ok))
164166 then {
165167 func validate (accum,next) = _validatePbk(next)
166168
167169 let validRes = {
168170 let $l = admins_pbks
169171 let $s = size($l)
170172 let $acc0 = ""
171173 func $f0_1 ($a,$i) = if (($i >= $s))
172174 then $a
173175 else validate($a, $l[$i])
174176
175177 func $f0_2 ($a,$i) = if (($i >= $s))
176178 then $a
177179 else throw("List size exceeds 10")
178180
179181 $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)
180182 }
181183 if ((validRes == validRes))
182184 then {
183185 let adminsStr = makeString(admins_pbks, ",")
184186 [StringEntry(k_ADMINS, adminsStr), IntegerEntry(k_LIMIT, limit), IntegerEntry(k_APPROVED_COUNT, approvedCount)]
185187 }
186188 else throw("Strict value is not equal to itself.")
187189 }
188190 else throw("Strict value is not equal to itself.")
189191 }
190192 else throw("Strict value is not equal to itself.")
191193 }
192194 else throw("Strict value is not equal to itself.")
193195 }
194196 else throw("Strict value is not equal to itself.")
195197 }
196198
197199
198200
199201 @Callable(i)
200202 func addAdmin (pbk) = {
201203 let callerPbk = toBase58String(i.callerPublicKey)
202204 let isAdmin = if (_isAdmin(callerPbk))
203205 then true
204206 else throw("Only admins can add admin")
205207 if ((isAdmin == isAdmin))
206208 then {
207209 let isValid = if (_validatePbk(pbk))
208210 then true
209211 else throw("Invalid admin pbk")
210212 if ((isValid == isValid))
211213 then {
212214 let newAdminsStr = ((getStringValue(k_ADMINS) + ",") + pbk)
213215 [StringEntry(k_ADMINS, newAdminsStr)]
214216 }
215217 else throw("Strict value is not equal to itself.")
216218 }
217219 else throw("Strict value is not equal to itself.")
218220 }
219221
220222
221223
222224 @Callable(i)
223225 func removeAdmin (pbk) = {
224226 let callerPbk = toBase58String(i.callerPublicKey)
225227 let isAdmin = if (_isAdmin(callerPbk))
226228 then true
227229 else throw("Only admins can add admin")
228230 if ((isAdmin == isAdmin))
229231 then {
230232 let isValid = if (_validatePbk(pbk))
231233 then true
232234 else throw("Invalid admin pbk")
233235 if ((isValid == isValid))
234236 then {
235237 let oldAdminsStr = getStringValue(k_ADMINS)
236238 let oldAdmins = split(oldAdminsStr, ",")
237239 let checkCount = if ((1 >= size(oldAdmins)))
238240 then throw("Cannot remove, at least one admin should remain")
239241 else true
240242 if ((checkCount == checkCount))
241243 then {
242244 let index = valueOrErrorMessage(indexOf(oldAdmins, pbk), "This pbk is not an admin")
243245 let newAdmins = removeByIndex(oldAdmins, index)
244246 let newAdminsStr = makeString(newAdmins, ",")
245247 [StringEntry(k_ADMINS, newAdminsStr)]
246248 }
247249 else throw("Strict value is not equal to itself.")
248250 }
249251 else throw("Strict value is not equal to itself.")
250252 }
251253 else throw("Strict value is not equal to itself.")
252254 }
253255
254256
255257
256258 @Callable(i)
257259 func addScriptType (scriptType) = {
258260 let callerPbk = toBase58String(i.callerPublicKey)
259261 if (!(_isAdmin(callerPbk)))
260262 then throw("Not allowed")
261263 else {
262264 let checkEmpty = if ((size(scriptType) == 0))
263265 then throw("Script type can not be empty")
264266 else true
265267 if ((checkEmpty == checkEmpty))
266268 then {
267269 let scriptTypesStr = valueOrElse(getString(k_SCRIPT_TYPES), "")
268270 let scriptTypes = split_4C(scriptTypesStr, ",")
269271 let checkExists = if ((indexOf(scriptTypes, scriptType) != unit))
270272 then throw((("Script type <" + scriptType) + "> already added"))
271273 else true
272274 if ((checkExists == checkExists))
273275 then {
274276 let newScriptTypes = if ((scriptTypesStr != ""))
275277 then (scriptTypes :+ scriptType)
276278 else [scriptType]
277279 let newScriptTypesStr = makeString(newScriptTypes, ",")
278280 [StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)]
279281 }
280282 else throw("Strict value is not equal to itself.")
281283 }
282284 else throw("Strict value is not equal to itself.")
283285 }
284286 }
285287
286288
287289
288290 @Callable(i)
289291 func removeScriptType (scriptType) = {
290292 let callerPbk = toBase58String(i.callerPublicKey)
291293 if (!(_isAdmin(callerPbk)))
292294 then throw("Not allowed")
293295 else {
294296 let scriptTypesStr = valueOrElse(getString(k_SCRIPT_TYPES), "")
295297 let scriptTypes = split_4C(scriptTypesStr, ",")
296298 let index = valueOrErrorMessage(indexOf(scriptTypes, scriptType), (("Script type <" + scriptType) + "> is not present"))
297299 let newScriptTypes = removeByIndex(scriptTypes, index)
298300 let newScriptTypesStr = makeString(newScriptTypes, ",")
299301 [StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)]
300302 }
301303 }
302304
303305
304306
305307 @Callable(i)
306-func addScript (scriptType,storageAddress,_scriptHash,desc,initiatorAdr) = {
308+func addScript (scriptType,storageAddress,_scriptHash,desc,initiatorAdr,rawCodeHash) = {
307309 let callerPbk = toBase58String(i.callerPublicKey)
308310 if (!(_isValidOperator(callerPbk)))
309311 then throw("Not allowed")
310312 else {
311313 let storageScriptHash = toBase58String(value(scriptHash(addressFromStringValue(storageAddress))))
312314 let scriptTypesStr = getStringValue(k_SCRIPT_TYPES)
313315 let _checkScriptTypes = if ((indexOf(scriptTypesStr, scriptType) == unit))
314316 then throw(("invalid script type or it is not present in storage: " + scriptType))
315317 else true
316318 if ((_checkScriptTypes == _checkScriptTypes))
317319 then {
318320 let _scriptTypeCheck = if ((storageScriptHash != _scriptHash))
319321 then throw("Stored and passed as argument scriptHashes not match")
320322 else true
321323 if ((_scriptTypeCheck == _scriptTypeCheck))
322324 then {
323325 let _descCheck = if ((size(desc) == 0))
324326 then throw("Description can't be empty")
325327 else true
326328 if ((_descCheck == _descCheck))
327329 then {
328330 let _initiatorCheck = if ((toString(addressFromPublicKey(i.callerPublicKey)) != initiatorAdr))
329331 then throw("Initiator address doesn't match caller")
330332 else true
331333 if ((_initiatorCheck == _initiatorCheck))
332334 then {
333335 let iteration = (valueOrElse(getInteger(((k_ITERATION + SEP) + scriptType)), 0) + 1)
334336 let scriptId = ((("script%%" + scriptType) + "%%") + _scriptHash)
335-[StringEntry(((scriptId + BIGSEP) + k_scriptType), scriptType), StringEntry(((scriptId + BIGSEP) + k_storageAddress), storageAddress), StringEntry(((scriptId + BIGSEP) + k_scriptHash), storageScriptHash), StringEntry(((scriptId + BIGSEP) + k_desc), desc), IntegerEntry(((scriptId + BIGSEP) + k_iter), iteration), StringEntry((((((scriptType + BIGSEP) + k_iter) + BIGSEP) + toString(iteration)) + "__hash"), _scriptHash), StringEntry(((scriptId + BIGSEP) + k_initiator), initiatorAdr), IntegerEntry(((k_ITERATION + SEP) + scriptType), iteration)]
337+[StringEntry(((scriptId + BIGSEP) + k_scriptType), scriptType), StringEntry(((scriptId + BIGSEP) + k_storageAddress), storageAddress), StringEntry(((scriptId + BIGSEP) + k_scriptHash), storageScriptHash), StringEntry(((scriptId + BIGSEP) + k_desc), desc), IntegerEntry(((scriptId + BIGSEP) + k_iter), iteration), StringEntry((((((scriptType + BIGSEP) + k_iter) + BIGSEP) + toString(iteration)) + "__hash"), _scriptHash), StringEntry(((scriptId + BIGSEP) + k_initiator), initiatorAdr), StringEntry(((scriptId + BIGSEP) + k_rawCodeHash), rawCodeHash), IntegerEntry(((k_ITERATION + SEP) + scriptType), iteration)]
336338 }
337339 else throw("Strict value is not equal to itself.")
338340 }
339341 else throw("Strict value is not equal to itself.")
340342 }
341343 else throw("Strict value is not equal to itself.")
342344 }
343345 else throw("Strict value is not equal to itself.")
344346 }
345347 }
346348
347349
348350
349351 @Callable(i)
350352 func removeScript (scriptId) = {
351353 let callerPbk = toBase58String(i.callerPublicKey)
352354 if (!(_isValidOperator(callerPbk)))
353355 then throw("Not allowed")
354356 else {
355357 let initiatorAdr = getStringValue((scriptId + "__initiator"))
356358 if ((initiatorAdr != toString(addressFromPublicKey(i.callerPublicKey))))
357359 then throw("Only creator can remove")
358360 else {
359361 let scriptType = getStringValue(((scriptId + BIGSEP) + k_scriptType))
360362 let _scriptHash = getStringValue(((scriptId + BIGSEP) + k_scriptHash))
361363 let iter = getIntegerValue(((scriptId + BIGSEP) + k_iter))
362364 let approvedHashes = valueOrElse(getString((("script_" + scriptType) + "__approved")), "")
363365 let _checkApproved = if ((indexOf(approvedHashes, _scriptHash) != unit))
364366 then throw("Cant remove approved script, use revoke")
365367 else true
366368 if ((_checkApproved == _checkApproved))
367- then [DeleteEntry(((scriptId + BIGSEP) + k_scriptType)), DeleteEntry(((scriptId + BIGSEP) + k_storageAddress)), DeleteEntry(((scriptId + BIGSEP) + k_scriptHash)), DeleteEntry(((scriptId + BIGSEP) + k_desc)), DeleteEntry(((scriptId + BIGSEP) + k_iter)), DeleteEntry((((scriptType + "__iteration__") + toString(iter)) + "__hash")), DeleteEntry(((scriptId + BIGSEP) + k_initiator)), DeleteEntry(((scriptId + BIGSEP) + k_confirmations))]
369+ then [DeleteEntry(((scriptId + BIGSEP) + k_scriptType)), DeleteEntry(((scriptId + BIGSEP) + k_storageAddress)), DeleteEntry(((scriptId + BIGSEP) + k_scriptHash)), DeleteEntry(((scriptId + BIGSEP) + k_desc)), DeleteEntry(((scriptId + BIGSEP) + k_iter)), DeleteEntry((((scriptType + "__iteration__") + toString(iter)) + "__hash")), DeleteEntry(((scriptId + BIGSEP) + k_initiator)), DeleteEntry(((scriptId + BIGSEP) + k_rawCodeHash)), DeleteEntry(((scriptId + BIGSEP) + k_confirmations))]
368370 else throw("Strict value is not equal to itself.")
369371 }
370372 }
371373 }
372374
373375
374376
375377 @Callable(i)
376378 func addConfirmationScript (scriptId) = {
377379 let callerPbk = toBase58String(i.callerPublicKey)
378380 let limit = _getLimit()
379381 if (!(_isAdmin(callerPbk)))
380382 then throw("Only admins can confirm")
381383 else {
382384 let _scriptHash = valueOrErrorMessage(getString((scriptId + "__scriptHash")), "Ivalid scriptId")
383385 if ((_scriptHash == _scriptHash))
384386 then {
385387 let scriptType = getStringValue((scriptId + "__scriptType"))
386388 let approvedHashesStr = valueOrElse(getString((("script_" + scriptType) + "__approved")), "")
387389 let _checkApproved = if ((indexOf(approvedHashesStr, _scriptHash) != unit))
388390 then throw("This script is approved already")
389391 else true
390392 if ((_checkApproved == _checkApproved))
391393 then {
392394 let globalIteration = valueOrElse(getInteger(((k_ITERATION + SEP) + scriptType)), 0)
393395 let thisIteration = value(getInteger((scriptId + "__iteration")))
394396 let confirmationsStr = valueOrElse(getString((scriptId + "__confirmations")), "")
395397 let confirmations = split(confirmationsStr, ",")
396398 if ((indexOf(confirmations, callerPbk) != unit))
397399 then throw("You already confirmed this script")
398400 else {
399401 let newList = if ((confirmationsStr != ""))
400402 then (confirmations :+ callerPbk)
401403 else [callerPbk]
402404 let newListStr = makeString(newList, ",")
403405 let approved = (size(newList) >= limit)
404406 let result = if (approved)
405407 then {
406408 let approvedCount = _getApprovedCount()
407409 let approvedHashes = split_4C(approvedHashesStr, ",")
408410 let approvedHashesNew = if ((approvedHashesStr != ""))
409411 then (approvedHashes :+ _scriptHash)
410412 else [_scriptHash]
411413 if ((size(approvedHashesNew) > approvedCount))
412414 then {
413415 let hashToRemove = approvedHashesNew[0]
414416 let scriptToRemoveId = ((("script%%" + scriptType) + "%%") + _scriptHash)
415417 let approvedHashesStripped = removeByIndex(approvedHashesNew, 0)
416418 [StringEntry((("script_" + scriptType) + "__approved"), makeString_2C(approvedHashesStripped, ",")), StringEntry(((scriptToRemoveId + BIGSEP) + k_confirmations), "")]
417419 }
418420 else [StringEntry((("script_" + scriptType) + "__approved"), makeString_2C(approvedHashesNew, ","))]
419421 }
420422 else nil
421423 (result ++ [StringEntry((scriptId + "__confirmations"), newListStr)])
422424 }
423425 }
424426 else throw("Strict value is not equal to itself.")
425427 }
426428 else throw("Strict value is not equal to itself.")
427429 }
428430 }
429431
430432
431433
432434 @Callable(i)
433435 func revokeConfirmationScript (scriptId) = {
434436 let callerPbk = toBase58String(i.callerPublicKey)
435437 let limit = _getLimit()
436438 if (!(_isAdmin(callerPbk)))
437439 then throw("Only admins can revoke")
438440 else {
439441 let _scriptHash = valueOrErrorMessage(getString((scriptId + "__scriptHash")), "Invalid scriptId")
440442 if ((_scriptHash == _scriptHash))
441443 then {
442444 let scriptType = getStringValue((scriptId + "__scriptType"))
443445 let approvedHashesStr = valueOrElse(getString((("script_" + scriptType) + "__approved")), "")
444446 let approvedHashes = split_4C(approvedHashesStr, ",")
445447 let confirmationsStr = valueOrElse(getString((scriptId + "__confirmations")), "")
446448 let confirmations = split(confirmationsStr, ",")
447449 let confirmIndex = indexOf(confirmations, callerPbk)
448450 if ((confirmIndex == unit))
449451 then throw("You didn't confirm this script, nothing to revoke")
450452 else {
451453 let newConfirmations = removeByIndex(confirmations, value(confirmIndex))
452454 let newConfirmationsStr = makeString(newConfirmations, ",")
453455 let wasApproved = (indexOf(approvedHashes, _scriptHash) != unit)
454456 let stillApproved = (size(newConfirmations) >= limit)
455457 let stateChanges = if (if (wasApproved)
456458 then !(stillApproved)
457459 else false)
458460 then {
459461 let scriptHashIndex = value(indexOf(approvedHashes, _scriptHash))
460462 let newApprovedHashes = removeByIndex(approvedHashes, scriptHashIndex)
461463 let newApprovedHashesStr = makeString(newApprovedHashes, ",")
462464 [StringEntry((("script_" + scriptType) + "__approved"), newApprovedHashesStr)]
463465 }
464466 else nil
465467 (stateChanges ++ [StringEntry((scriptId + "__confirmations"), newConfirmationsStr)])
466468 }
467469 }
468470 else throw("Strict value is not equal to itself.")
469471 }
470472 }
471473
472474
473475
474476 @Callable(i)
475477 func addTask (name,dapp,txid,executeTs,desc,initiatorAdr,txdata) = {
476478 let callerPbk = toBase58String(i.callerPublicKey)
477479 if (!(_isValidOperator(callerPbk)))
478480 then throw("Not allowed")
479481 else {
480482 let _nameCheck = if ((size(name) == 0))
481483 then throw("Name can't be empty")
482484 else true
483485 if ((_nameCheck == _nameCheck))
484486 then {
485487 let _dappCheck = if (!(_validateAddress(dapp)))
486488 then throw("Incorrect dapp")
487489 else true
488490 if ((_dappCheck == _dappCheck))
489491 then {
490492 let _txidCheck = if ((size(txid) == 0))
491493 then throw("txid can't be empty")
492494 else true
493495 if ((_txidCheck == _txidCheck))
494496 then {
495497 let _timestampCheck = if (!(_validateTs(executeTs)))
496498 then throw("txid can't be empty")
497499 else true
498500 if ((_timestampCheck == _timestampCheck))
499501 then {
500502 let _descCheck = if ((size(desc) == 0))
501503 then throw("Description can't be empty")
502504 else true
503505 if ((_descCheck == _descCheck))
504506 then {
505507 let _initiatorCheck = if ((toString(addressFromPublicKey(i.callerPublicKey)) != initiatorAdr))
506508 then throw("Initiator address doesn't match caller")
507509 else true
508510 if ((_initiatorCheck == _initiatorCheck))
509511 then {
510512 let _txdataCheck = if ((size(txdata) == 0))
511513 then throw("Tx data can't be empty")
512514 else true
513515 if ((_txdataCheck == _txdataCheck))
514516 then {
515517 let taskId = ((("tx_" + dapp) + SEP) + txid)
516518 [StringEntry((taskId + "__name"), name), StringEntry((taskId + "__dapp"), dapp), StringEntry((taskId + "__txId"), txid), StringEntry((taskId + "__desc"), desc), StringEntry((taskId + "__txdata"), txdata), StringEntry((taskId + "__initiator"), initiatorAdr), IntegerEntry((taskId + "__ts"), executeTs)]
517519 }
518520 else throw("Strict value is not equal to itself.")
519521 }
520522 else throw("Strict value is not equal to itself.")
521523 }
522524 else throw("Strict value is not equal to itself.")
523525 }
524526 else throw("Strict value is not equal to itself.")
525527 }
526528 else throw("Strict value is not equal to itself.")
527529 }
528530 else throw("Strict value is not equal to itself.")
529531 }
530532 else throw("Strict value is not equal to itself.")
531533 }
532534 }
533535
534536
535537
536538 @Callable(i)
537539 func removeTask (taskId) = {
538540 let callerPbk = toBase58String(i.callerPublicKey)
539541 if (!(_isValidOperator(callerPbk)))
540542 then throw("Not allowed")
541543 else {
542544 let initiatorAdr = getStringValue((taskId + "__initiator"))
543545 if ((initiatorAdr != toString(addressFromPublicKey(i.callerPublicKey))))
544546 then throw("Only creator can remove")
545547 else [DeleteEntry((taskId + "__name")), DeleteEntry((taskId + "__dapp")), DeleteEntry((taskId + "__txId")), DeleteEntry((taskId + "__desc")), DeleteEntry((taskId + "__txdata")), DeleteEntry((taskId + "__initiator")), DeleteEntry((taskId + "__ts")), DeleteEntry((taskId + "__confirmations")), DeleteEntry((taskId + "__approved"))]
546548 }
547549 }
548550
549551
550552
551553 @Callable(i)
552554 func addConfirmation (taskId) = {
553555 let callerPbk = toBase58String(i.callerPublicKey)
554556 let limit = _getLimit()
555557 if (!(_isAdmin(callerPbk)))
556558 then throw("Only admins can confirm")
557559 else {
558560 let txId = valueOrErrorMessage(getString((taskId + "__txId")), "Ivalid task")
559561 if ((txId == txId))
560562 then {
561563 let dapp = getStringValue((taskId + "__dapp"))
562564 let confirmationsStr = valueOrElse(getString((taskId + "__confirmations")), "")
563565 let confirmations = split(confirmationsStr, ",")
564566 if ((indexOf(confirmations, callerPbk) != unit))
565567 then throw("You already confirmed this task")
566568 else {
567569 let newList = if ((confirmationsStr != ""))
568570 then (confirmations :+ callerPbk)
569571 else [callerPbk]
570572 let newListStr = makeString(newList, ",")
571573 let approved = (size(newList) >= limit)
572574 [StringEntry((taskId + "__confirmations"), newListStr), BooleanEntry((((dapp + SEP) + txId) + "__approved"), approved)]
573575 }
574576 }
575577 else throw("Strict value is not equal to itself.")
576578 }
577579 }
578580
579581
580582
581583 @Callable(i)
582584 func setLimit (limit) = {
583585 let callerPbk = toBase58String(i.callerPublicKey)
584586 if (!(_isAdmin(callerPbk)))
585587 then throw("Only whitelisted can set limit")
586588 else {
587589 let zeroLimit = if ((limit > 0))
588590 then true
589591 else throw("Limit should be > 0")
590592 if ((zeroLimit == zeroLimit))
591593 then {
592594 let admins = getStringValue(k_ADMINS)
593595 if ((limit > size(admins)))
594596 then throw(((("Limit quorum should be <= size of list of admins, limit: " + toString(limit)) + ", current size: ") + toString(size(admins))))
595597 else [IntegerEntry(k_LIMIT, limit)]
596598 }
597599 else throw("Strict value is not equal to itself.")
598600 }
599601 }
600602
601603
602604
603605 @Callable(i)
604606 func test () = {
605607 let x = match blockInfoByHeight(height) {
606608 case block: BlockInfo =>
607609 throw(toString(block.timestamp))
608610 case _ =>
609611 throw("Can't find block")
610612 }
611613 if ((x == x))
612614 then nil
613615 else throw("Strict value is not equal to itself.")
614616 }
615617
616618
617619
618620 @Callable(i)
619621 func clear_test () = {
620622 let indices = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
621623 func removeChunks (accum,index) = (accum :+ DeleteEntry((("testtask" + "__txdata_") + toString(index))))
622624
623625 let entries = {
624626 let $l = indices
625627 let $s = size($l)
626628 let $acc0 = [DeleteEntry(("testtask" + "__txdata_chunks"))]
627629 func $f0_1 ($a,$i) = if (($i >= $s))
628630 then $a
629631 else removeChunks($a, $l[$i])
630632
631633 func $f0_2 ($a,$i) = if (($i >= $s))
632634 then $a
633635 else throw("List size exceeds 10")
634636
635637 $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)
636638 }
637639 entries
638640 }
639641
640642

github/deemru/w8io
101.10 ms