tx · Gw6do8G8y6KZTyEGjefg8wDrMofW8BjUxqcQqGTDsFR8

3M5Ts2aPKpcGeKZ23N1ZhosqRQoHvxGQv8d:  -0.02850000 Waves

2025.10.25 00:24 [551551] smart account 3M5Ts2aPKpcGeKZ23N1ZhosqRQoHvxGQv8d > SELF 0.00000000 Waves

{ "type": 13, "id": "Gw6do8G8y6KZTyEGjefg8wDrMofW8BjUxqcQqGTDsFR8", "fee": 2850000, "feeAssetId": null, "timestamp": 1761341071344, "version": 2, "chainId": 82, "sender": "3M5Ts2aPKpcGeKZ23N1ZhosqRQoHvxGQv8d", "senderPublicKey": "DLyBoqv63q4kZWbFVE64DNbweTRqJNcTSN5D4mif888p", "proofs": [ "5uS7DEyRLWYfz9maUbDm7tpipANNEgkM5yViizL723Aicfxe9RhNDKVuLsHmwjjVVuimN3TmECmAzQNuinp1ENZ" ], "script": "base64:BgJZCAISBQoDGAEBEgMKAQgSAwoBCBIDCgEIEgMKAQgSAwoBARIICgYICAgICAgSAwoBCBIDCgEIEgMKAQgSCQoHCAgIAQgICBIDCgEIEgMKAQgSAwoBCBIAEgAbAAdWRVJTSU9OAgUxLjAuMAADU0VQAgFfAAZCSUdTRVACAl9fAAhrX0FETUlOUwILQURNSU5TX1BCS1MACmtfTUFOQUdFUlMCDU1BTkFHRVJTX1BCS1MAB2tfTElNSVQCDFZPVEVTX1FVT1JVTQAOa19TQ1JJUFRfVFlQRVMCDFNDUklQVF9UWVBFUwALa19JVEVSQVRJT04CCUlURVJBVElPTgAQa19BUFBST1ZFRF9DT1VOVAIOQVBQUk9WRURfQ09VTlQADGtfc2NyaXB0VHlwZQIKc2NyaXB0VHlwZQAMa19zY3JpcHRIYXNoAgpzY3JpcHRIYXNoABBrX3N0b3JhZ2VBZGRyZXNzAg5zdG9yYWdlQWRkcmVzcwAGa19kZXNjAgRkZXNjAAZrX2l0ZXICCWl0ZXJhdGlvbgALa19pbml0aWF0b3ICCWluaXRpYXRvcgAPa19jb25maXJtYXRpb25zAg1jb25maXJtYXRpb25zAA1rX3Jhd0NvZGVIYXNoAgtyYXdDb2RlSGFzaAEMX3ZhbGlkYXRlUGJrAQNwYmsEB2FkZHJlc3MJAKcIAQkA2QQBBQNwYmsDCQAAAgUHYWRkcmVzcwUHYWRkcmVzcwQKYWRkcmVzc1N0cgkApQgBBQdhZGRyZXNzAwkAAAIFCmFkZHJlc3NTdHIFCmFkZHJlc3NTdHIGCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuARBfdmFsaWRhdGVBZGRyZXNzAQdhZGRyZXNzBARhZGRyCQERQGV4dHJOYXRpdmUoMTA2MikBBQdhZGRyZXNzAwkAAAIFBGFkZHIFBGFkZHIGCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAQtfdmFsaWRhdGVUcwECdHMEBmN1cnJUcwQHJG1hdGNoMAkA7QcBBQZoZWlnaHQDCQABAgUHJG1hdGNoMAIJQmxvY2tJbmZvBAVibG9jawUHJG1hdGNoMAgFBWJsb2NrCXRpbWVzdGFtcAkAAgECEENhbid0IGZpbmQgYmxvY2sEB29uZUhvdXIJAGgCCQBoAgA8ADwA6AcEBW1pblRzCQBlAgUGY3VyclRzBQdvbmVIb3VyCQBnAgUCdHMFBW1pblRzAQxpc1R4QXBwcm92ZWQBA2ludgkBC3ZhbHVlT3JFbHNlAgkAmwgCCQEFdmFsdWUBBQR0aGlzCQCsAgIJAKwCAgkArAICCQClCAEFBHRoaXMCAV8JANgEAQgFA2ludg10cmFuc2FjdGlvbklkAgpfX2FwcHJvdmVkBwEJX2dldExpbWl0AAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEFB2tfTElNSVQCGFF1b3J1bSBsaW1pdCBub3QgZGVmaW5lZAERX2dldEFwcHJvdmVkQ291bnQACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ8IAQUQa19BUFBST1ZFRF9DT1VOVAI1QW1vdW50IG9mIHNpbXVsdGFuZW91c2x5IGFwcHJvdmVkIHNjcmlwdHMgbm90IGRlZmluZWQBCF9pc0FkbWluAQNwYmsECWFkbWluc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBBQhrX0FETUlOUwIABAZhZG1pbnMJALUJAgUJYWRtaW5zU3RyAgEsCQECIT0CCQDPCAIFBmFkbWlucwUDcGJrBQR1bml0AQpfaXNNYW5hZ2VyAQNwYmsEC21hbmFnZXJzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEFCmtfTUFOQUdFUlMCAAQIbWFuYWdlcnMJALUJAgULbWFuYWdlcnNTdHICASwJAQIhPQIJAM8IAgUIbWFuYWdlcnMFA3BiawUEdW5pdAEQX2lzVmFsaWRPcGVyYXRvcgEDcGJrBAttYW5hZ2Vyc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBBQprX01BTkFHRVJTAgAECG1hbmFnZXJzCQC1CQIFC21hbmFnZXJzU3RyAgEsBAdpc0FkbWluCQEIX2lzQWRtaW4BBQNwYmsECWlzTWFuYWdlcgkBCl9pc01hbmFnZXIBBQNwYmsDBQdpc0FkbWluBgUJaXNNYW5hZ2VyAQpzYXZlVHhEYXRhAgZ0YXNrSWQEZGF0YQQHaW5kaWNlcwkAzAgCAAEJAMwIAgACCQDMCAIAAwkAzAgCAAQJAMwIAgAFCQDMCAIABgkAzAgCAAcJAMwIAgAICQDMCAIACQkAzAgCAAoFA25pbAQIZGF0YVNpemUJALECAQUEZGF0YQQKaXNOb3RFbXB0eQMJAQIhPQIFCGRhdGFTaXplAAAGCQACAQIdc2F2ZVR4RGF0YTogRW1wdHkgZGF0YSBwYXNzZWQDCQAAAgUKaXNOb3RFbXB0eQUKaXNOb3RFbXB0eQQJY2h1bmtTaXplALDqAQQLY2h1bmtzQ291bnQDCQAAAgkAagIFCGRhdGFTaXplBQljaHVua1NpemUAAAkAaQIFCGRhdGFTaXplBQljaHVua1NpemUJAGQCCQBpAgUIZGF0YVNpemUFCWNodW5rU2l6ZQABCgEKbWFrZUNodW5rcwIFYWNjdW0FaW5kZXgEBWNodW5rCQCvAgIIBQVhY2N1bQJfMQUJY2h1bmtTaXplAwkAAAIJALECAQUFY2h1bmsAAAUFYWNjdW0ECG5leHREYXRhCQCwAgIIBQVhY2N1bQJfMQUJY2h1bmtTaXplCQCUCgIFCG5leHREYXRhCQDNCAIIBQVhY2N1bQJfMgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFBnRhc2tJZAIJX190eGRhdGFfCQCkAwEFBWluZGV4BQVjaHVuawQHZW50cmllcwoAAiRsBQdpbmRpY2VzCgACJHMJAJADAQUCJGwKAAUkYWNjMAkAlAoCBQRkYXRhCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAg9fX3R4ZGF0YV9jaHVua3MJAKQDAQULY2h1bmtzQ291bnQFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEKbWFrZUNodW5rcwIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACggFB2VudHJpZXMCXzIJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4QAWkBBGluaXQDC2FkbWluc19wYmtzBWxpbWl0DWFwcHJvdmVkQ291bnQDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAg5TZWxmIGNhbGwgb25seQQLc2l6ZV9hZG1pbnMDCQECIT0CCQCQAwEFC2FkbWluc19wYmtzAAAGCQACAQInQXQgbGVhc3Qgb25lIGFkbWluIHBiayBzaG91bGQgYmUgcGFzc2VkAwkAAAIFC3NpemVfYWRtaW5zBQtzaXplX2FkbWlucwQMY2hlY2tfYWRtaW5zAwkBASEBCQEJaXNEZWZpbmVkAQkAoggBBQhrX0FETUlOUwYJAAIBAg5BbHJlYWR5IGluaXRlZAMJAAACBQxjaGVja19hZG1pbnMFDGNoZWNrX2FkbWlucwQJemVyb0xpbWl0AwkAZgIFBWxpbWl0AAAGCQACAQITTGltaXQgc2hvdWxkIGJlID4gMAMJAAACBQl6ZXJvTGltaXQFCXplcm9MaW1pdAQIbGltaXRfb2sDCQBmAgkAkAMBBQthZG1pbnNfcGJrcwUFbGltaXQGCQACAQkArAICCQCsAgIJAKwCAgI5TGltaXQgcXVvcnVtIHNob3VsZCBiZSA8PSBzaXplIG9mIGxpc3Qgb2YgYWRtaW5zLCBsaW1pdDogCQCkAwEFBWxpbWl0AhAsIGN1cnJlbnQgc2l6ZTogCQCkAwEJAJADAQULYWRtaW5zX3Bia3MDCQAAAgUIbGltaXRfb2sFCGxpbWl0X29rCgEIdmFsaWRhdGUCBWFjY3VtBG5leHQJAQxfdmFsaWRhdGVQYmsBBQRuZXh0BAh2YWxpZFJlcwoAAiRsBQthZG1pbnNfcGJrcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzACAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEIdmFsaWRhdGUCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoDCQAAAgUIdmFsaWRSZXMFCHZhbGlkUmVzBAlhZG1pbnNTdHIJALkJAgULYWRtaW5zX3Bia3MCASwJAMwIAgkBC1N0cmluZ0VudHJ5AgUIa19BRE1JTlMFCWFkbWluc1N0cgkAzAgCCQEMSW50ZWdlckVudHJ5AgUHa19MSU1JVAUFbGltaXQJAMwIAgkBDEludGVnZXJFbnRyeQIFEGtfQVBQUk9WRURfQ09VTlQFDWFwcHJvdmVkQ291bnQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQhhZGRBZG1pbgEDcGJrBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAdpc0FkbWluAwkBCF9pc0FkbWluAQUJY2FsbGVyUGJrBgkAAgECGU9ubHkgYWRtaW5zIGNhbiBhZGQgYWRtaW4DCQAAAgUHaXNBZG1pbgUHaXNBZG1pbgQKaXNBcHByb3ZlZAMJAQxpc1R4QXBwcm92ZWQBBQFpBgkAAgECHkFkbWluIGNhbGxzIHNob3VsZCBiZSBhcHByb3ZlZAMJAAACBQppc0FwcHJvdmVkBQppc0FwcHJvdmVkBAdpc1ZhbGlkAwkBDF92YWxpZGF0ZVBiawEFA3BiawYJAAIBAhFJbnZhbGlkIGFkbWluIHBiawMJAAACBQdpc1ZhbGlkBQdpc1ZhbGlkBAxuZXdBZG1pbnNTdHIJAKwCAgkArAICCQERQGV4dHJOYXRpdmUoMTA1OCkBBQhrX0FETUlOUwIBLAUDcGJrCQDMCAIJAQtTdHJpbmdFbnRyeQIFCGtfQURNSU5TBQxuZXdBZG1pbnNTdHIFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQtyZW1vdmVBZG1pbgEDcGJrBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAdpc0FkbWluAwkBCF9pc0FkbWluAQUJY2FsbGVyUGJrBgkAAgECGU9ubHkgYWRtaW5zIGNhbiBhZGQgYWRtaW4DCQAAAgUHaXNBZG1pbgUHaXNBZG1pbgQKaXNBcHByb3ZlZAMJAQxpc1R4QXBwcm92ZWQBBQFpBgkAAgECHkFkbWluIGNhbGxzIHNob3VsZCBiZSBhcHByb3ZlZAMJAAACBQppc0FwcHJvdmVkBQppc0FwcHJvdmVkBAdpc1ZhbGlkAwkBDF92YWxpZGF0ZVBiawEFA3BiawYJAAIBAhFJbnZhbGlkIGFkbWluIHBiawMJAAACBQdpc1ZhbGlkBQdpc1ZhbGlkBAxvbGRBZG1pbnNTdHIJARFAZXh0ck5hdGl2ZSgxMDU4KQEFCGtfQURNSU5TBAlvbGRBZG1pbnMJALUJAgUMb2xkQWRtaW5zU3RyAgEsBApjaGVja0NvdW50AwkAZwIAAQkAkAMBBQlvbGRBZG1pbnMJAAIBAi9DYW5ub3QgcmVtb3ZlLCBhdCBsZWFzdCBvbmUgYWRtaW4gc2hvdWxkIHJlbWFpbgYDCQAAAgUKY2hlY2tDb3VudAUKY2hlY2tDb3VudAQFaW5kZXgJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAzwgCBQlvbGRBZG1pbnMFA3BiawIYVGhpcyBwYmsgaXMgbm90IGFuIGFkbWluBAluZXdBZG1pbnMJANEIAgUJb2xkQWRtaW5zBQVpbmRleAQMbmV3QWRtaW5zU3RyCQC5CQIFCW5ld0FkbWlucwIBLAkAzAgCCQELU3RyaW5nRW50cnkCBQhrX0FETUlOUwUMbmV3QWRtaW5zU3RyBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQENYWRkU2NyaXB0VHlwZQEKc2NyaXB0VHlwZQQJY2FsbGVyUGJrCQDYBAEIBQFpD2NhbGxlclB1YmxpY0tleQMJAQEhAQkBCF9pc0FkbWluAQUJY2FsbGVyUGJrCQACAQILTm90IGFsbG93ZWQECmlzQXBwcm92ZWQDCQEMaXNUeEFwcHJvdmVkAQUBaQYJAAIBAh5BZG1pbiBjYWxscyBzaG91bGQgYmUgYXBwcm92ZWQDCQAAAgUKaXNBcHByb3ZlZAUKaXNBcHByb3ZlZAQKY2hlY2tFbXB0eQMJAAACCQCxAgEFCnNjcmlwdFR5cGUAAAkAAgECHFNjcmlwdCB0eXBlIGNhbiBub3QgYmUgZW1wdHkGAwkAAAIFCmNoZWNrRW1wdHkFCmNoZWNrRW1wdHkEDnNjcmlwdFR5cGVzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEFDmtfU0NSSVBUX1RZUEVTAgAEC3NjcmlwdFR5cGVzCQC8CQIFDnNjcmlwdFR5cGVzU3RyAgEsBAtjaGVja0V4aXN0cwMJAQIhPQIJAM8IAgULc2NyaXB0VHlwZXMFCnNjcmlwdFR5cGUFBHVuaXQJAAIBCQCsAgIJAKwCAgINU2NyaXB0IHR5cGUgPAUKc2NyaXB0VHlwZQIPPiBhbHJlYWR5IGFkZGVkBgMJAAACBQtjaGVja0V4aXN0cwULY2hlY2tFeGlzdHMEDm5ld1NjcmlwdFR5cGVzAwkBAiE9AgUOc2NyaXB0VHlwZXNTdHICAAkAzQgCBQtzY3JpcHRUeXBlcwUKc2NyaXB0VHlwZQkAzAgCBQpzY3JpcHRUeXBlBQNuaWwEEW5ld1NjcmlwdFR5cGVzU3RyCQC5CQIFDm5ld1NjcmlwdFR5cGVzAgEsCQDMCAIJAQtTdHJpbmdFbnRyeQIFDmtfU0NSSVBUX1RZUEVTBRFuZXdTY3JpcHRUeXBlc1N0cgUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBEHJlbW92ZVNjcmlwdFR5cGUBCnNjcmlwdFR5cGUECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkDCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECC05vdCBhbGxvd2VkBAppc0FwcHJvdmVkAwkBDGlzVHhBcHByb3ZlZAEFAWkGCQACAQIeQWRtaW4gY2FsbHMgc2hvdWxkIGJlIGFwcHJvdmVkAwkAAAIFCmlzQXBwcm92ZWQFCmlzQXBwcm92ZWQEDnNjcmlwdFR5cGVzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEFDmtfU0NSSVBUX1RZUEVTAgAEC3NjcmlwdFR5cGVzCQC8CQIFDnNjcmlwdFR5cGVzU3RyAgEsBAVpbmRleAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQDPCAIFC3NjcmlwdFR5cGVzBQpzY3JpcHRUeXBlCQCsAgIJAKwCAgINU2NyaXB0IHR5cGUgPAUKc2NyaXB0VHlwZQIQPiBpcyBub3QgcHJlc2VudAQObmV3U2NyaXB0VHlwZXMJANEIAgULc2NyaXB0VHlwZXMFBWluZGV4BBFuZXdTY3JpcHRUeXBlc1N0cgkAuQkCBQ5uZXdTY3JpcHRUeXBlcwIBLAkAzAgCCQELU3RyaW5nRW50cnkCBQ5rX1NDUklQVF9UWVBFUwURbmV3U2NyaXB0VHlwZXNTdHIFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQhzZXRMaW1pdAEFbGltaXQECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkDCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECHk9ubHkgd2hpdGVsaXN0ZWQgY2FuIHNldCBsaW1pdAQKaXNBcHByb3ZlZAMJAQxpc1R4QXBwcm92ZWQBBQFpBgkAAgECHkFkbWluIGNhbGxzIHNob3VsZCBiZSBhcHByb3ZlZAMJAAACBQppc0FwcHJvdmVkBQppc0FwcHJvdmVkBAl6ZXJvTGltaXQDCQBmAgUFbGltaXQAAAYJAAIBAhNMaW1pdCBzaG91bGQgYmUgPiAwAwkAAAIFCXplcm9MaW1pdAUJemVyb0xpbWl0BAZhZG1pbnMJARFAZXh0ck5hdGl2ZSgxMDU4KQEFCGtfQURNSU5TAwkAZgIFBWxpbWl0CQCxAgEFBmFkbWlucwkAAgEJAKwCAgkArAICCQCsAgICOUxpbWl0IHF1b3J1bSBzaG91bGQgYmUgPD0gc2l6ZSBvZiBsaXN0IG9mIGFkbWlucywgbGltaXQ6IAkApAMBBQVsaW1pdAIQLCBjdXJyZW50IHNpemU6IAkApAMBCQCxAgEFBmFkbWlucwkAzAgCCQEMSW50ZWdlckVudHJ5AgUHa19MSU1JVAUFbGltaXQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQlhZGRTY3JpcHQGCnNjcmlwdFR5cGUOc3RvcmFnZUFkZHJlc3MLX3NjcmlwdEhhc2gEZGVzYwxpbml0aWF0b3JBZHILcmF3Q29kZUhhc2gECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkDCQEBIQEJARBfaXNWYWxpZE9wZXJhdG9yAQUJY2FsbGVyUGJrCQACAQILTm90IGFsbG93ZWQEEXN0b3JhZ2VTY3JpcHRIYXNoCQDYBAEJAQV2YWx1ZQEJAPEHAQkBEUBleHRyTmF0aXZlKDEwNjIpAQUOc3RvcmFnZUFkZHJlc3MEDnNjcmlwdFR5cGVzU3RyCQERQGV4dHJOYXRpdmUoMTA1OCkBBQ5rX1NDUklQVF9UWVBFUwQRX2NoZWNrU2NyaXB0VHlwZXMDCQAAAgkAswkCBQ5zY3JpcHRUeXBlc1N0cgUKc2NyaXB0VHlwZQUEdW5pdAkAAgEJAKwCAgI1aW52YWxpZCBzY3JpcHQgdHlwZSBvciBpdCBpcyBub3QgcHJlc2VudCBpbiBzdG9yYWdlOiAFCnNjcmlwdFR5cGUGAwkAAAIFEV9jaGVja1NjcmlwdFR5cGVzBRFfY2hlY2tTY3JpcHRUeXBlcwQQX3NjcmlwdFR5cGVDaGVjawMJAQIhPQIFEXN0b3JhZ2VTY3JpcHRIYXNoBQtfc2NyaXB0SGFzaAkAAgECNFN0b3JlZCBhbmQgcGFzc2VkIGFzIGFyZ3VtZW50IHNjcmlwdEhhc2hlcyBub3QgbWF0Y2gGAwkAAAIFEF9zY3JpcHRUeXBlQ2hlY2sFEF9zY3JpcHRUeXBlQ2hlY2sECl9kZXNjQ2hlY2sDCQAAAgkAsQIBBQRkZXNjAAAJAAIBAhpEZXNjcmlwdGlvbiBjYW4ndCBiZSBlbXB0eQYDCQAAAgUKX2Rlc2NDaGVjawUKX2Rlc2NDaGVjawQPX2luaXRpYXRvckNoZWNrAwkBAiE9AgkApQgBCQCnCAEIBQFpD2NhbGxlclB1YmxpY0tleQUMaW5pdGlhdG9yQWRyCQACAQImSW5pdGlhdG9yIGFkZHJlc3MgZG9lc24ndCBtYXRjaCBjYWxsZXIGAwkAAAIFD19pbml0aWF0b3JDaGVjawUPX2luaXRpYXRvckNoZWNrBAlpdGVyYXRpb24JAGQCCQELdmFsdWVPckVsc2UCCQCfCAEJAKwCAgkArAICBQtrX0lURVJBVElPTgUDU0VQBQpzY3JpcHRUeXBlAAAAAQQIc2NyaXB0SWQJAKwCAgkArAICCQCsAgICCHNjcmlwdCUlBQpzY3JpcHRUeXBlAgIlJQULX3NjcmlwdEhhc2gJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDGtfc2NyaXB0VHlwZQUKc2NyaXB0VHlwZQkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUQa19zdG9yYWdlQWRkcmVzcwUOc3RvcmFnZUFkZHJlc3MJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDGtfc2NyaXB0SGFzaAURc3RvcmFnZVNjcmlwdEhhc2gJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFBmtfZGVzYwUEZGVzYwkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFBmtfaXRlcgUJaXRlcmF0aW9uCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICBQpzY3JpcHRUeXBlBQZCSUdTRVAFBmtfaXRlcgUGQklHU0VQCQCkAwEFCWl0ZXJhdGlvbgIGX19oYXNoBQtfc2NyaXB0SGFzaAkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAULa19pbml0aWF0b3IFDGluaXRpYXRvckFkcgkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUNa19yYXdDb2RlSGFzaAULcmF3Q29kZUhhc2gJAMwIAgkBDEludGVnZXJFbnRyeQIJAKwCAgkArAICBQtrX0lURVJBVElPTgUDU0VQBQpzY3JpcHRUeXBlBQlpdGVyYXRpb24FA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQxyZW1vdmVTY3JpcHQBCHNjcmlwdElkBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5AwkBASEBCQEQX2lzVmFsaWRPcGVyYXRvcgEFCWNhbGxlclBiawkAAgECC05vdCBhbGxvd2VkBAxpbml0aWF0b3JBZHIJARFAZXh0ck5hdGl2ZSgxMDU4KQEJAKwCAgUIc2NyaXB0SWQCC19faW5pdGlhdG9yAwkBAiE9AgUMaW5pdGlhdG9yQWRyCQClCAEJAKcIAQgFAWkPY2FsbGVyUHVibGljS2V5CQACAQIXT25seSBjcmVhdG9yIGNhbiByZW1vdmUECnNjcmlwdFR5cGUJARFAZXh0ck5hdGl2ZSgxMDU4KQEJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQxrX3NjcmlwdFR5cGUEC19zY3JpcHRIYXNoCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUMa19zY3JpcHRIYXNoBARpdGVyCQERQGV4dHJOYXRpdmUoMTA1NSkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUGa19pdGVyBA5hcHByb3ZlZEhhc2hlcwkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIJAKwCAgIHc2NyaXB0XwUKc2NyaXB0VHlwZQIKX19hcHByb3ZlZAIABA5fY2hlY2tBcHByb3ZlZAMJAQIhPQIJALMJAgUOYXBwcm92ZWRIYXNoZXMFC19zY3JpcHRIYXNoBQR1bml0CQACAQInQ2FudCByZW1vdmUgYXBwcm92ZWQgc2NyaXB0LCB1c2UgcmV2b2tlBgMJAAACBQ5fY2hlY2tBcHByb3ZlZAUOX2NoZWNrQXBwcm92ZWQJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDGtfc2NyaXB0VHlwZQkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUQa19zdG9yYWdlQWRkcmVzcwkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAUMa19zY3JpcHRIYXNoCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgkArAICBQhzY3JpcHRJZAUGQklHU0VQBQZrX2Rlc2MJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFBmtfaXRlcgkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIJAKwCAgkArAICBQpzY3JpcHRUeXBlAg1fX2l0ZXJhdGlvbl9fCQCkAwEFBGl0ZXICBl9faGFzaAkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIJAKwCAgUIc2NyaXB0SWQFBkJJR1NFUAULa19pbml0aWF0b3IJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFDWtfcmF3Q29kZUhhc2gJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgIFCHNjcmlwdElkBQZCSUdTRVAFD2tfY29uZmlybWF0aW9ucwUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBFWFkZENvbmZpcm1hdGlvblNjcmlwdAEIc2NyaXB0SWQECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkEBWxpbWl0CQEJX2dldExpbWl0AAMJAQEhAQkBCF9pc0FkbWluAQUJY2FsbGVyUGJrCQACAQIXT25seSBhZG1pbnMgY2FuIGNvbmZpcm0EC19zY3JpcHRIYXNoCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKIIAQkArAICBQhzY3JpcHRJZAIMX19zY3JpcHRIYXNoAg9JdmFsaWQgc2NyaXB0SWQDCQAAAgULX3NjcmlwdEhhc2gFC19zY3JpcHRIYXNoBApzY3JpcHRUeXBlCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFCHNjcmlwdElkAgxfX3NjcmlwdFR5cGUEEWFwcHJvdmVkSGFzaGVzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEJAKwCAgkArAICAgdzY3JpcHRfBQpzY3JpcHRUeXBlAgpfX2FwcHJvdmVkAgAEDl9jaGVja0FwcHJvdmVkAwkBAiE9AgkAswkCBRFhcHByb3ZlZEhhc2hlc1N0cgULX3NjcmlwdEhhc2gFBHVuaXQJAAIBAh9UaGlzIHNjcmlwdCBpcyBhcHByb3ZlZCBhbHJlYWR5BgMJAAACBQ5fY2hlY2tBcHByb3ZlZAUOX2NoZWNrQXBwcm92ZWQED2dsb2JhbEl0ZXJhdGlvbgkBC3ZhbHVlT3JFbHNlAgkAnwgBCQCsAgIJAKwCAgULa19JVEVSQVRJT04FA1NFUAUKc2NyaXB0VHlwZQAABA10aGlzSXRlcmF0aW9uCQEFdmFsdWUBCQCfCAEJAKwCAgUIc2NyaXB0SWQCC19faXRlcmF0aW9uBBBjb25maXJtYXRpb25zU3RyCQELdmFsdWVPckVsc2UCCQCiCAEJAKwCAgUIc2NyaXB0SWQCD19fY29uZmlybWF0aW9ucwIABA1jb25maXJtYXRpb25zCQC1CQIFEGNvbmZpcm1hdGlvbnNTdHICASwDCQECIT0CCQDPCAIFDWNvbmZpcm1hdGlvbnMFCWNhbGxlclBiawUEdW5pdAkAAgECIVlvdSBhbHJlYWR5IGNvbmZpcm1lZCB0aGlzIHNjcmlwdAQHbmV3TGlzdAMJAQIhPQIFEGNvbmZpcm1hdGlvbnNTdHICAAkAzQgCBQ1jb25maXJtYXRpb25zBQljYWxsZXJQYmsJAMwIAgUJY2FsbGVyUGJrBQNuaWwECm5ld0xpc3RTdHIJALkJAgUHbmV3TGlzdAIBLAQIYXBwcm92ZWQJAGcCCQCQAwEFB25ld0xpc3QFBWxpbWl0BAZyZXN1bHQDBQhhcHByb3ZlZAQNYXBwcm92ZWRDb3VudAkBEV9nZXRBcHByb3ZlZENvdW50AAQOYXBwcm92ZWRIYXNoZXMJALwJAgURYXBwcm92ZWRIYXNoZXNTdHICASwEEWFwcHJvdmVkSGFzaGVzTmV3AwkBAiE9AgURYXBwcm92ZWRIYXNoZXNTdHICAAkAzQgCBQ5hcHByb3ZlZEhhc2hlcwULX3NjcmlwdEhhc2gJAMwIAgULX3NjcmlwdEhhc2gFA25pbAMJAGYCCQCQAwEFEWFwcHJvdmVkSGFzaGVzTmV3BQ1hcHByb3ZlZENvdW50BAxoYXNoVG9SZW1vdmUJAJEDAgURYXBwcm92ZWRIYXNoZXNOZXcAAAQQc2NyaXB0VG9SZW1vdmVJZAkArAICCQCsAgIJAKwCAgIIc2NyaXB0JSUFCnNjcmlwdFR5cGUCAiUlBQtfc2NyaXB0SGFzaAQWYXBwcm92ZWRIYXNoZXNTdHJpcHBlZAkA0QgCBRFhcHByb3ZlZEhhc2hlc05ldwAACQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgkArAICAgdzY3JpcHRfBQpzY3JpcHRUeXBlAgpfX2FwcHJvdmVkCQC6CQIFFmFwcHJvdmVkSGFzaGVzU3RyaXBwZWQCASwJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICCQCsAgIFEHNjcmlwdFRvUmVtb3ZlSWQFBkJJR1NFUAUPa19jb25maXJtYXRpb25zAgAFA25pbAkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgIHc2NyaXB0XwUKc2NyaXB0VHlwZQIKX19hcHByb3ZlZAkAugkCBRFhcHByb3ZlZEhhc2hlc05ldwIBLAUDbmlsBQNuaWwJAM4IAgUGcmVzdWx0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUIc2NyaXB0SWQCD19fY29uZmlybWF0aW9ucwUKbmV3TGlzdFN0cgUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBGHJldm9rZUNvbmZpcm1hdGlvblNjcmlwdAEIc2NyaXB0SWQECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkEBWxpbWl0CQEJX2dldExpbWl0AAMJAQEhAQkBCF9pc0FkbWluAQUJY2FsbGVyUGJrCQACAQIWT25seSBhZG1pbnMgY2FuIHJldm9rZQQLX3NjcmlwdEhhc2gJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAoggBCQCsAgIFCHNjcmlwdElkAgxfX3NjcmlwdEhhc2gCEEludmFsaWQgc2NyaXB0SWQDCQAAAgULX3NjcmlwdEhhc2gFC19zY3JpcHRIYXNoBApzY3JpcHRUeXBlCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFCHNjcmlwdElkAgxfX3NjcmlwdFR5cGUEEWFwcHJvdmVkSGFzaGVzU3RyCQELdmFsdWVPckVsc2UCCQCiCAEJAKwCAgkArAICAgdzY3JpcHRfBQpzY3JpcHRUeXBlAgpfX2FwcHJvdmVkAgAEDmFwcHJvdmVkSGFzaGVzCQC8CQIFEWFwcHJvdmVkSGFzaGVzU3RyAgEsBBBjb25maXJtYXRpb25zU3RyCQELdmFsdWVPckVsc2UCCQCiCAEJAKwCAgUIc2NyaXB0SWQCD19fY29uZmlybWF0aW9ucwIABA1jb25maXJtYXRpb25zCQC1CQIFEGNvbmZpcm1hdGlvbnNTdHICASwEDGNvbmZpcm1JbmRleAkAzwgCBQ1jb25maXJtYXRpb25zBQljYWxsZXJQYmsDCQAAAgUMY29uZmlybUluZGV4BQR1bml0CQACAQIxWW91IGRpZG4ndCBjb25maXJtIHRoaXMgc2NyaXB0LCBub3RoaW5nIHRvIHJldm9rZQQQbmV3Q29uZmlybWF0aW9ucwkA0QgCBQ1jb25maXJtYXRpb25zCQEFdmFsdWUBBQxjb25maXJtSW5kZXgEE25ld0NvbmZpcm1hdGlvbnNTdHIJALkJAgUQbmV3Q29uZmlybWF0aW9ucwIBLAQLd2FzQXBwcm92ZWQJAQIhPQIJAM8IAgUOYXBwcm92ZWRIYXNoZXMFC19zY3JpcHRIYXNoBQR1bml0BA1zdGlsbEFwcHJvdmVkCQBnAgkAkAMBBRBuZXdDb25maXJtYXRpb25zBQVsaW1pdAQMc3RhdGVDaGFuZ2VzAwMFC3dhc0FwcHJvdmVkCQEBIQEFDXN0aWxsQXBwcm92ZWQHBA9zY3JpcHRIYXNoSW5kZXgJAQV2YWx1ZQEJAM8IAgUOYXBwcm92ZWRIYXNoZXMFC19zY3JpcHRIYXNoBBFuZXdBcHByb3ZlZEhhc2hlcwkA0QgCBQ5hcHByb3ZlZEhhc2hlcwUPc2NyaXB0SGFzaEluZGV4BBRuZXdBcHByb3ZlZEhhc2hlc1N0cgkAuQkCBRFuZXdBcHByb3ZlZEhhc2hlcwIBLAkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIJAKwCAgIHc2NyaXB0XwUKc2NyaXB0VHlwZQIKX19hcHByb3ZlZAUUbmV3QXBwcm92ZWRIYXNoZXNTdHIFA25pbAUDbmlsCQDOCAIFDHN0YXRlQ2hhbmdlcwkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFCHNjcmlwdElkAg9fX2NvbmZpcm1hdGlvbnMFE25ld0NvbmZpcm1hdGlvbnNTdHIFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQdhZGRUYXNrBwRuYW1lBGRhcHAEdHhpZAlleGVjdXRlVHMEZGVzYwxpbml0aWF0b3JBZHIGdHhkYXRhBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5AwkBASEBCQEQX2lzVmFsaWRPcGVyYXRvcgEFCWNhbGxlclBiawkAAgECC05vdCBhbGxvd2VkBApfbmFtZUNoZWNrAwkAAAIJALECAQUEbmFtZQAACQACAQITTmFtZSBjYW4ndCBiZSBlbXB0eQYDCQAAAgUKX25hbWVDaGVjawUKX25hbWVDaGVjawQKX2RhcHBDaGVjawMJAQEhAQkBEF92YWxpZGF0ZUFkZHJlc3MBBQRkYXBwCQACAQIOSW5jb3JyZWN0IGRhcHAGAwkAAAIFCl9kYXBwQ2hlY2sFCl9kYXBwQ2hlY2sECl90eGlkQ2hlY2sDCQAAAgkAsQIBBQR0eGlkAAAJAAIBAhN0eGlkIGNhbid0IGJlIGVtcHR5BgMJAAACBQpfdHhpZENoZWNrBQpfdHhpZENoZWNrBA9fdGltZXN0YW1wQ2hlY2sDCQEBIQEJAQtfdmFsaWRhdGVUcwEFCWV4ZWN1dGVUcwkAAgECREludmFsaWQgZXhlY3V0aW9uIHRpbWVzdGFtcCAtIGNhbm5vdCBiZSBtb3JlIHRoYW4gMSBob3VyIGluIHRoZSBwYXN0BgMJAAACBQ9fdGltZXN0YW1wQ2hlY2sFD190aW1lc3RhbXBDaGVjawQKX2Rlc2NDaGVjawMJAAACCQCxAgEFBGRlc2MAAAkAAgECGkRlc2NyaXB0aW9uIGNhbid0IGJlIGVtcHR5BgMJAAACBQpfZGVzY0NoZWNrBQpfZGVzY0NoZWNrBA9faW5pdGlhdG9yQ2hlY2sDCQECIT0CCQClCAEJAKcIAQgFAWkPY2FsbGVyUHVibGljS2V5BQxpbml0aWF0b3JBZHIJAAIBAiZJbml0aWF0b3IgYWRkcmVzcyBkb2Vzbid0IG1hdGNoIGNhbGxlcgYDCQAAAgUPX2luaXRpYXRvckNoZWNrBQ9faW5pdGlhdG9yQ2hlY2sEDF90eGRhdGFDaGVjawMJAAACCQCxAgEFBnR4ZGF0YQAACQACAQIWVHggZGF0YSBjYW4ndCBiZSBlbXB0eQYDCQAAAgUMX3R4ZGF0YUNoZWNrBQxfdHhkYXRhQ2hlY2sEBnRhc2tJZAkArAICCQCsAgIJAKwCAgIDdHhfBQRkYXBwBQNTRVAFBHR4aWQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQZ0YXNrSWQCBl9fbmFtZQUEbmFtZQkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFBnRhc2tJZAIGX19kYXBwBQRkYXBwCQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAgZfX3R4SWQFBHR4aWQJAMwIAgkBC1N0cmluZ0VudHJ5AgkArAICBQZ0YXNrSWQCBl9fZGVzYwUEZGVzYwkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFBnRhc2tJZAIIX190eGRhdGEFBnR4ZGF0YQkAzAgCCQELU3RyaW5nRW50cnkCCQCsAgIFBnRhc2tJZAILX19pbml0aWF0b3IFDGluaXRpYXRvckFkcgkAzAgCCQEMSW50ZWdlckVudHJ5AgkArAICBQZ0YXNrSWQCBF9fdHMFCWV4ZWN1dGVUcwUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBCnJlbW92ZVRhc2sBBnRhc2tJZAQJY2FsbGVyUGJrCQDYBAEIBQFpD2NhbGxlclB1YmxpY0tleQMJAQEhAQkBEF9pc1ZhbGlkT3BlcmF0b3IBBQljYWxsZXJQYmsJAAIBAgtOb3QgYWxsb3dlZAQMaW5pdGlhdG9yQWRyCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFBnRhc2tJZAILX19pbml0aWF0b3IDCQECIT0CBQxpbml0aWF0b3JBZHIJAKUIAQkApwgBCAUBaQ9jYWxsZXJQdWJsaWNLZXkJAAIBAhdPbmx5IGNyZWF0b3IgY2FuIHJlbW92ZQkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAIGX19uYW1lCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgUGdGFza0lkAgZfX2RhcHAJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICBQZ0YXNrSWQCBl9fdHhJZAkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAIGX19kZXNjCQDMCAIJAQtEZWxldGVFbnRyeQEJAKwCAgUGdGFza0lkAghfX3R4ZGF0YQkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAILX19pbml0aWF0b3IJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICBQZ0YXNrSWQCBF9fdHMJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICBQZ0YXNrSWQCD19fY29uZmlybWF0aW9ucwkAzAgCCQELRGVsZXRlRW50cnkBCQCsAgIFBnRhc2tJZAIKX19hcHByb3ZlZAUDbmlsAWkBD2FkZENvbmZpcm1hdGlvbgEGdGFza0lkBAljYWxsZXJQYmsJANgEAQgFAWkPY2FsbGVyUHVibGljS2V5BAVsaW1pdAkBCV9nZXRMaW1pdAADCQEBIQEJAQhfaXNBZG1pbgEFCWNhbGxlclBiawkAAgECF09ubHkgYWRtaW5zIGNhbiBjb25maXJtBAR0eElkCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKIIAQkArAICBQZ0YXNrSWQCBl9fdHhJZAILSXZhbGlkIHRhc2sDCQAAAgUEdHhJZAUEdHhJZAQEZGFwcAkBEUBleHRyTmF0aXZlKDEwNTgpAQkArAICBQZ0YXNrSWQCBl9fZGFwcAQQY29uZmlybWF0aW9uc1N0cgkBC3ZhbHVlT3JFbHNlAgkAoggBCQCsAgIFBnRhc2tJZAIPX19jb25maXJtYXRpb25zAgAEDWNvbmZpcm1hdGlvbnMJALUJAgUQY29uZmlybWF0aW9uc1N0cgIBLAMJAQIhPQIJAM8IAgUNY29uZmlybWF0aW9ucwUJY2FsbGVyUGJrBQR1bml0CQACAQIfWW91IGFscmVhZHkgY29uZmlybWVkIHRoaXMgdGFzawQHbmV3TGlzdAMJAQIhPQIFEGNvbmZpcm1hdGlvbnNTdHICAAkAzQgCBQ1jb25maXJtYXRpb25zBQljYWxsZXJQYmsJAMwIAgUJY2FsbGVyUGJrBQNuaWwECm5ld0xpc3RTdHIJALkJAgUHbmV3TGlzdAIBLAQIYXBwcm92ZWQJAGcCCQCQAwEFB25ld0xpc3QFBWxpbWl0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAg9fX2NvbmZpcm1hdGlvbnMFCm5ld0xpc3RTdHIJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAKwCAgkArAICCQCsAgIFBGRhcHAFA1NFUAUEdHhJZAIKX19hcHByb3ZlZAUIYXBwcm92ZWQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQpyZXZva2VUYXNrAQZ0YXNrSWQECWNhbGxlclBiawkA2AQBCAUBaQ9jYWxsZXJQdWJsaWNLZXkEBWxpbWl0CQEJX2dldExpbWl0AAMJAQEhAQkBCF9pc0FkbWluAQUJY2FsbGVyUGJrCQACAQIUT25seSBhZG1pbnMgY2FuIGNhbGwEBHR4SWQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAoggBCQCsAgIFBnRhc2tJZAIGX190eElkAgtJdmFsaWQgdGFzawMJAAACBQR0eElkBQR0eElkBBBjb25maXJtYXRpb25zU3RyCQELdmFsdWVPckVsc2UCCQCiCAEJAKwCAgUGdGFza0lkAg9fX2NvbmZpcm1hdGlvbnMCAAQNY29uZmlybWF0aW9ucwkAtQkCBRBjb25maXJtYXRpb25zU3RyAgEsBAdpbmRleE15CQDPCAIFDWNvbmZpcm1hdGlvbnMFCWNhbGxlclBiawMJAAACBQdpbmRleE15BQR1bml0CQACAQIwWW91IGRpZCBub3QgY29uZmlybSB0aGlzIHRhc2ssIG5vdGhpbmcgdG8gcmV2b2tlBARkYXBwCQERQGV4dHJOYXRpdmUoMTA1OCkBCQCsAgIFBnRhc2tJZAIGX19kYXBwBAduZXdMaXN0CQDRCAIFDWNvbmZpcm1hdGlvbnMJAQV2YWx1ZQEFB2luZGV4TXkECm5ld0xpc3RTdHIJALkJAgUHbmV3TGlzdAIBLAQIYXBwcm92ZWQJAGcCCQCQAwEFB25ld0xpc3QFBWxpbWl0CQDMCAIJAQtTdHJpbmdFbnRyeQIJAKwCAgUGdGFza0lkAg9fX2NvbmZpcm1hdGlvbnMFCm5ld0xpc3RTdHIJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAKwCAgkArAICCQCsAgIFBGRhcHAFA1NFUAUEdHhJZAIKX19hcHByb3ZlZAUIYXBwcm92ZWQFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQR0ZXN0AAQBeAQHJG1hdGNoMAkA7QcBBQZoZWlnaHQDCQABAgUHJG1hdGNoMAIJQmxvY2tJbmZvBAVibG9jawUHJG1hdGNoMAkAAgEJAKQDAQgFBWJsb2NrCXRpbWVzdGFtcAkAAgECEENhbid0IGZpbmQgYmxvY2sDCQAAAgUBeAUBeAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBCmNsZWFyX3Rlc3QABAdpbmRpY2VzCQDMCAIAAQkAzAgCAAIJAMwIAgADCQDMCAIABAkAzAgCAAUJAMwIAgAGCQDMCAIABwkAzAgCAAgJAMwIAgAJCQDMCAIACgUDbmlsCgEMcmVtb3ZlQ2h1bmtzAgVhY2N1bQVpbmRleAkAzQgCBQVhY2N1bQkBC0RlbGV0ZUVudHJ5AQkArAICCQCsAgICCHRlc3R0YXNrAglfX3R4ZGF0YV8JAKQDAQUFaW5kZXgEB2VudHJpZXMKAAIkbAUHaW5kaWNlcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAJAMwIAgkBC0RlbGV0ZUVudHJ5AQkArAICAgh0ZXN0dGFzawIPX190eGRhdGFfY2h1bmtzBQNuaWwKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBDHJlbW92ZUNodW5rcwIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQIUTGlzdCBzaXplIGV4Y2VlZHMgMTAJAQUkZjBfMgIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIJAQUkZjBfMQIFBSRhY2MwAAAAAQACAAMABAAFAAYABwAIAAkACgUHZW50cmllcwECdHgBBnZlcmlmeQAEEE1VTFRJU0lHX0FERFJFU1MJAKYIAQIjM003dXpEOHJUNTRmS1FjUDRzNDE3aWYxVktVOHlyb21FUDIEDUNPTlRSQUNUX1RZUEUCEUFQUFJPVkVSX0NPTlRSQUNUAwkBAiE9AgUQTVVMVElTSUdfQUREUkVTUwUEdW5pdAQHJG1hdGNoMAUCdHgDCQABAgUHJG1hdGNoMAIUU2V0U2NyaXB0VHJhbnNhY3Rpb24EA3R0eAUHJG1hdGNoMAQLc2NyaXB0SGFzaGEJANgEAQkBBXZhbHVlAQkA9gMBCQEFdmFsdWUBCAUDdHR4BnNjcmlwdAQOYXBwcm92ZWRIYXNoZXMJAQV2YWx1ZQEJAJ0IAgkBBXZhbHVlAQUQTVVMVElTSUdfQUREUkVTUwkArAICCQCsAgICB3NjcmlwdF8FDUNPTlRSQUNUX1RZUEUCCl9fYXBwcm92ZWQJAQIhPQIJALMJAgUOYXBwcm92ZWRIYXNoZXMFC3NjcmlwdEhhc2hhBQR1bml0BANvdHgFByRtYXRjaDAJAQt2YWx1ZU9yRWxzZQIJAJsIAgkBBXZhbHVlAQUEdGhpcwkArAICCQCsAgIJAKwCAgkApQgBBQR0aGlzAgFfCQDYBAEIBQNvdHgCaWQCCl9fYXBwcm92ZWQHCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAACAUCdHgPc2VuZGVyUHVibGljS2V58dgszQ==", "height": 551551, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: none Next: none Full:
OldNewDifferences
1-# no script
1+{-# STDLIB_VERSION 6 #-}
2+{-# SCRIPT_TYPE ACCOUNT #-}
3+{-# CONTENT_TYPE DAPP #-}
4+let VERSION = "1.0.0"
5+
6+let SEP = "_"
7+
8+let BIGSEP = "__"
9+
10+let k_ADMINS = "ADMINS_PBKS"
11+
12+let k_MANAGERS = "MANAGERS_PBKS"
13+
14+let k_LIMIT = "VOTES_QUORUM"
15+
16+let k_SCRIPT_TYPES = "SCRIPT_TYPES"
17+
18+let k_ITERATION = "ITERATION"
19+
20+let k_APPROVED_COUNT = "APPROVED_COUNT"
21+
22+let k_scriptType = "scriptType"
23+
24+let k_scriptHash = "scriptHash"
25+
26+let k_storageAddress = "storageAddress"
27+
28+let k_desc = "desc"
29+
30+let k_iter = "iteration"
31+
32+let k_initiator = "initiator"
33+
34+let k_confirmations = "confirmations"
35+
36+let k_rawCodeHash = "rawCodeHash"
37+
38+func _validatePbk (pbk) = {
39+ let address = addressFromPublicKey(fromBase58String(pbk))
40+ if ((address == address))
41+ then {
42+ let addressStr = toString(address)
43+ if ((addressStr == addressStr))
44+ then true
45+ else throw("Strict value is not equal to itself.")
46+ }
47+ else throw("Strict value is not equal to itself.")
48+ }
49+
50+
51+func _validateAddress (address) = {
52+ let addr = addressFromStringValue(address)
53+ if ((addr == addr))
54+ then true
55+ else throw("Strict value is not equal to itself.")
56+ }
57+
58+
59+func _validateTs (ts) = {
60+ let currTs = match blockInfoByHeight(height) {
61+ case block: BlockInfo =>
62+ block.timestamp
63+ case _ =>
64+ throw("Can't find block")
65+ }
66+ let oneHour = ((60 * 60) * 1000)
67+ let minTs = (currTs - oneHour)
68+ (ts >= minTs)
69+ }
70+
71+
72+func isTxApproved (inv) = valueOrElse(getBoolean(value(this), (((toString(this) + "_") + toBase58String(inv.transactionId)) + "__approved")), false)
73+
74+
75+func _getLimit () = valueOrErrorMessage(getInteger(k_LIMIT), "Quorum limit not defined")
76+
77+
78+func _getApprovedCount () = valueOrErrorMessage(getInteger(k_APPROVED_COUNT), "Amount of simultaneously approved scripts not defined")
79+
80+
81+func _isAdmin (pbk) = {
82+ let adminsStr = valueOrElse(getString(k_ADMINS), "")
83+ let admins = split(adminsStr, ",")
84+ (indexOf(admins, pbk) != unit)
85+ }
86+
87+
88+func _isManager (pbk) = {
89+ let managersStr = valueOrElse(getString(k_MANAGERS), "")
90+ let managers = split(managersStr, ",")
91+ (indexOf(managers, pbk) != unit)
92+ }
93+
94+
95+func _isValidOperator (pbk) = {
96+ let managersStr = valueOrElse(getString(k_MANAGERS), "")
97+ let managers = split(managersStr, ",")
98+ let isAdmin = _isAdmin(pbk)
99+ let isManager = _isManager(pbk)
100+ if (isAdmin)
101+ then true
102+ else isManager
103+ }
104+
105+
106+func saveTxData (taskId,data) = {
107+ let indices = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
108+ let dataSize = size(data)
109+ let isNotEmpty = if ((dataSize != 0))
110+ then true
111+ else throw("saveTxData: Empty data passed")
112+ if ((isNotEmpty == isNotEmpty))
113+ then {
114+ let chunkSize = 30000
115+ let chunksCount = if (((dataSize % chunkSize) == 0))
116+ then (dataSize / chunkSize)
117+ else ((dataSize / chunkSize) + 1)
118+ func makeChunks (accum,index) = {
119+ let chunk = take(accum._1, chunkSize)
120+ if ((size(chunk) == 0))
121+ then accum
122+ else {
123+ let nextData = drop(accum._1, chunkSize)
124+ $Tuple2(nextData, (accum._2 :+ StringEntry(((taskId + "__txdata_") + toString(index)), chunk)))
125+ }
126+ }
127+
128+ let entries = {
129+ let $l = indices
130+ let $s = size($l)
131+ let $acc0 = $Tuple2(data, [StringEntry((taskId + "__txdata_chunks"), toString(chunksCount))])
132+ func $f0_1 ($a,$i) = if (($i >= $s))
133+ then $a
134+ else makeChunks($a, $l[$i])
135+
136+ func $f0_2 ($a,$i) = if (($i >= $s))
137+ then $a
138+ else throw("List size exceeds 10")
139+
140+ $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)
141+ }
142+ entries._2
143+ }
144+ else throw("Strict value is not equal to itself.")
145+ }
146+
147+
148+@Callable(i)
149+func init (admins_pbks,limit,approvedCount) = if ((i.caller != this))
150+ then throw("Self call only")
151+ else {
152+ let size_admins = if ((size(admins_pbks) != 0))
153+ then true
154+ else throw("At least one admin pbk should be passed")
155+ if ((size_admins == size_admins))
156+ then {
157+ let check_admins = if (!(isDefined(getString(k_ADMINS))))
158+ then true
159+ else throw("Already inited")
160+ if ((check_admins == check_admins))
161+ then {
162+ let zeroLimit = if ((limit > 0))
163+ then true
164+ else throw("Limit should be > 0")
165+ if ((zeroLimit == zeroLimit))
166+ then {
167+ let limit_ok = if ((size(admins_pbks) > limit))
168+ then true
169+ else throw(((("Limit quorum should be <= size of list of admins, limit: " + toString(limit)) + ", current size: ") + toString(size(admins_pbks))))
170+ if ((limit_ok == limit_ok))
171+ then {
172+ func validate (accum,next) = _validatePbk(next)
173+
174+ let validRes = {
175+ let $l = admins_pbks
176+ let $s = size($l)
177+ let $acc0 = ""
178+ func $f0_1 ($a,$i) = if (($i >= $s))
179+ then $a
180+ else validate($a, $l[$i])
181+
182+ func $f0_2 ($a,$i) = if (($i >= $s))
183+ then $a
184+ else throw("List size exceeds 10")
185+
186+ $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)
187+ }
188+ if ((validRes == validRes))
189+ then {
190+ let adminsStr = makeString(admins_pbks, ",")
191+[StringEntry(k_ADMINS, adminsStr), IntegerEntry(k_LIMIT, limit), IntegerEntry(k_APPROVED_COUNT, approvedCount)]
192+ }
193+ else throw("Strict value is not equal to itself.")
194+ }
195+ else throw("Strict value is not equal to itself.")
196+ }
197+ else throw("Strict value is not equal to itself.")
198+ }
199+ else throw("Strict value is not equal to itself.")
200+ }
201+ else throw("Strict value is not equal to itself.")
202+ }
203+
204+
205+
206+@Callable(i)
207+func addAdmin (pbk) = {
208+ let callerPbk = toBase58String(i.callerPublicKey)
209+ let isAdmin = if (_isAdmin(callerPbk))
210+ then true
211+ else throw("Only admins can add admin")
212+ if ((isAdmin == isAdmin))
213+ then {
214+ let isApproved = if (isTxApproved(i))
215+ then true
216+ else throw("Admin calls should be approved")
217+ if ((isApproved == isApproved))
218+ then {
219+ let isValid = if (_validatePbk(pbk))
220+ then true
221+ else throw("Invalid admin pbk")
222+ if ((isValid == isValid))
223+ then {
224+ let newAdminsStr = ((getStringValue(k_ADMINS) + ",") + pbk)
225+[StringEntry(k_ADMINS, newAdminsStr)]
226+ }
227+ else throw("Strict value is not equal to itself.")
228+ }
229+ else throw("Strict value is not equal to itself.")
230+ }
231+ else throw("Strict value is not equal to itself.")
232+ }
233+
234+
235+
236+@Callable(i)
237+func removeAdmin (pbk) = {
238+ let callerPbk = toBase58String(i.callerPublicKey)
239+ let isAdmin = if (_isAdmin(callerPbk))
240+ then true
241+ else throw("Only admins can add admin")
242+ if ((isAdmin == isAdmin))
243+ then {
244+ let isApproved = if (isTxApproved(i))
245+ then true
246+ else throw("Admin calls should be approved")
247+ if ((isApproved == isApproved))
248+ then {
249+ let isValid = if (_validatePbk(pbk))
250+ then true
251+ else throw("Invalid admin pbk")
252+ if ((isValid == isValid))
253+ then {
254+ let oldAdminsStr = getStringValue(k_ADMINS)
255+ let oldAdmins = split(oldAdminsStr, ",")
256+ let checkCount = if ((1 >= size(oldAdmins)))
257+ then throw("Cannot remove, at least one admin should remain")
258+ else true
259+ if ((checkCount == checkCount))
260+ then {
261+ let index = valueOrErrorMessage(indexOf(oldAdmins, pbk), "This pbk is not an admin")
262+ let newAdmins = removeByIndex(oldAdmins, index)
263+ let newAdminsStr = makeString(newAdmins, ",")
264+[StringEntry(k_ADMINS, newAdminsStr)]
265+ }
266+ else throw("Strict value is not equal to itself.")
267+ }
268+ else throw("Strict value is not equal to itself.")
269+ }
270+ else throw("Strict value is not equal to itself.")
271+ }
272+ else throw("Strict value is not equal to itself.")
273+ }
274+
275+
276+
277+@Callable(i)
278+func addScriptType (scriptType) = {
279+ let callerPbk = toBase58String(i.callerPublicKey)
280+ if (!(_isAdmin(callerPbk)))
281+ then throw("Not allowed")
282+ else {
283+ let isApproved = if (isTxApproved(i))
284+ then true
285+ else throw("Admin calls should be approved")
286+ if ((isApproved == isApproved))
287+ then {
288+ let checkEmpty = if ((size(scriptType) == 0))
289+ then throw("Script type can not be empty")
290+ else true
291+ if ((checkEmpty == checkEmpty))
292+ then {
293+ let scriptTypesStr = valueOrElse(getString(k_SCRIPT_TYPES), "")
294+ let scriptTypes = split_4C(scriptTypesStr, ",")
295+ let checkExists = if ((indexOf(scriptTypes, scriptType) != unit))
296+ then throw((("Script type <" + scriptType) + "> already added"))
297+ else true
298+ if ((checkExists == checkExists))
299+ then {
300+ let newScriptTypes = if ((scriptTypesStr != ""))
301+ then (scriptTypes :+ scriptType)
302+ else [scriptType]
303+ let newScriptTypesStr = makeString(newScriptTypes, ",")
304+[StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)]
305+ }
306+ else throw("Strict value is not equal to itself.")
307+ }
308+ else throw("Strict value is not equal to itself.")
309+ }
310+ else throw("Strict value is not equal to itself.")
311+ }
312+ }
313+
314+
315+
316+@Callable(i)
317+func removeScriptType (scriptType) = {
318+ let callerPbk = toBase58String(i.callerPublicKey)
319+ if (!(_isAdmin(callerPbk)))
320+ then throw("Not allowed")
321+ else {
322+ let isApproved = if (isTxApproved(i))
323+ then true
324+ else throw("Admin calls should be approved")
325+ if ((isApproved == isApproved))
326+ then {
327+ let scriptTypesStr = valueOrElse(getString(k_SCRIPT_TYPES), "")
328+ let scriptTypes = split_4C(scriptTypesStr, ",")
329+ let index = valueOrErrorMessage(indexOf(scriptTypes, scriptType), (("Script type <" + scriptType) + "> is not present"))
330+ let newScriptTypes = removeByIndex(scriptTypes, index)
331+ let newScriptTypesStr = makeString(newScriptTypes, ",")
332+[StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)]
333+ }
334+ else throw("Strict value is not equal to itself.")
335+ }
336+ }
337+
338+
339+
340+@Callable(i)
341+func setLimit (limit) = {
342+ let callerPbk = toBase58String(i.callerPublicKey)
343+ if (!(_isAdmin(callerPbk)))
344+ then throw("Only whitelisted can set limit")
345+ else {
346+ let isApproved = if (isTxApproved(i))
347+ then true
348+ else throw("Admin calls should be approved")
349+ if ((isApproved == isApproved))
350+ then {
351+ let zeroLimit = if ((limit > 0))
352+ then true
353+ else throw("Limit should be > 0")
354+ if ((zeroLimit == zeroLimit))
355+ then {
356+ let admins = getStringValue(k_ADMINS)
357+ if ((limit > size(admins)))
358+ then throw(((("Limit quorum should be <= size of list of admins, limit: " + toString(limit)) + ", current size: ") + toString(size(admins))))
359+ else [IntegerEntry(k_LIMIT, limit)]
360+ }
361+ else throw("Strict value is not equal to itself.")
362+ }
363+ else throw("Strict value is not equal to itself.")
364+ }
365+ }
366+
367+
368+
369+@Callable(i)
370+func addScript (scriptType,storageAddress,_scriptHash,desc,initiatorAdr,rawCodeHash) = {
371+ let callerPbk = toBase58String(i.callerPublicKey)
372+ if (!(_isValidOperator(callerPbk)))
373+ then throw("Not allowed")
374+ else {
375+ let storageScriptHash = toBase58String(value(scriptHash(addressFromStringValue(storageAddress))))
376+ let scriptTypesStr = getStringValue(k_SCRIPT_TYPES)
377+ let _checkScriptTypes = if ((indexOf(scriptTypesStr, scriptType) == unit))
378+ then throw(("invalid script type or it is not present in storage: " + scriptType))
379+ else true
380+ if ((_checkScriptTypes == _checkScriptTypes))
381+ then {
382+ let _scriptTypeCheck = if ((storageScriptHash != _scriptHash))
383+ then throw("Stored and passed as argument scriptHashes not match")
384+ else true
385+ if ((_scriptTypeCheck == _scriptTypeCheck))
386+ then {
387+ let _descCheck = if ((size(desc) == 0))
388+ then throw("Description can't be empty")
389+ else true
390+ if ((_descCheck == _descCheck))
391+ then {
392+ let _initiatorCheck = if ((toString(addressFromPublicKey(i.callerPublicKey)) != initiatorAdr))
393+ then throw("Initiator address doesn't match caller")
394+ else true
395+ if ((_initiatorCheck == _initiatorCheck))
396+ then {
397+ let iteration = (valueOrElse(getInteger(((k_ITERATION + SEP) + scriptType)), 0) + 1)
398+ let scriptId = ((("script%%" + scriptType) + "%%") + _scriptHash)
399+[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)]
400+ }
401+ else throw("Strict value is not equal to itself.")
402+ }
403+ else throw("Strict value is not equal to itself.")
404+ }
405+ else throw("Strict value is not equal to itself.")
406+ }
407+ else throw("Strict value is not equal to itself.")
408+ }
409+ }
410+
411+
412+
413+@Callable(i)
414+func removeScript (scriptId) = {
415+ let callerPbk = toBase58String(i.callerPublicKey)
416+ if (!(_isValidOperator(callerPbk)))
417+ then throw("Not allowed")
418+ else {
419+ let initiatorAdr = getStringValue((scriptId + "__initiator"))
420+ if ((initiatorAdr != toString(addressFromPublicKey(i.callerPublicKey))))
421+ then throw("Only creator can remove")
422+ else {
423+ let scriptType = getStringValue(((scriptId + BIGSEP) + k_scriptType))
424+ let _scriptHash = getStringValue(((scriptId + BIGSEP) + k_scriptHash))
425+ let iter = getIntegerValue(((scriptId + BIGSEP) + k_iter))
426+ let approvedHashes = valueOrElse(getString((("script_" + scriptType) + "__approved")), "")
427+ let _checkApproved = if ((indexOf(approvedHashes, _scriptHash) != unit))
428+ then throw("Cant remove approved script, use revoke")
429+ else true
430+ if ((_checkApproved == _checkApproved))
431+ 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))]
432+ else throw("Strict value is not equal to itself.")
433+ }
434+ }
435+ }
436+
437+
438+
439+@Callable(i)
440+func addConfirmationScript (scriptId) = {
441+ let callerPbk = toBase58String(i.callerPublicKey)
442+ let limit = _getLimit()
443+ if (!(_isAdmin(callerPbk)))
444+ then throw("Only admins can confirm")
445+ else {
446+ let _scriptHash = valueOrErrorMessage(getString((scriptId + "__scriptHash")), "Ivalid scriptId")
447+ if ((_scriptHash == _scriptHash))
448+ then {
449+ let scriptType = getStringValue((scriptId + "__scriptType"))
450+ let approvedHashesStr = valueOrElse(getString((("script_" + scriptType) + "__approved")), "")
451+ let _checkApproved = if ((indexOf(approvedHashesStr, _scriptHash) != unit))
452+ then throw("This script is approved already")
453+ else true
454+ if ((_checkApproved == _checkApproved))
455+ then {
456+ let globalIteration = valueOrElse(getInteger(((k_ITERATION + SEP) + scriptType)), 0)
457+ let thisIteration = value(getInteger((scriptId + "__iteration")))
458+ let confirmationsStr = valueOrElse(getString((scriptId + "__confirmations")), "")
459+ let confirmations = split(confirmationsStr, ",")
460+ if ((indexOf(confirmations, callerPbk) != unit))
461+ then throw("You already confirmed this script")
462+ else {
463+ let newList = if ((confirmationsStr != ""))
464+ then (confirmations :+ callerPbk)
465+ else [callerPbk]
466+ let newListStr = makeString(newList, ",")
467+ let approved = (size(newList) >= limit)
468+ let result = if (approved)
469+ then {
470+ let approvedCount = _getApprovedCount()
471+ let approvedHashes = split_4C(approvedHashesStr, ",")
472+ let approvedHashesNew = if ((approvedHashesStr != ""))
473+ then (approvedHashes :+ _scriptHash)
474+ else [_scriptHash]
475+ if ((size(approvedHashesNew) > approvedCount))
476+ then {
477+ let hashToRemove = approvedHashesNew[0]
478+ let scriptToRemoveId = ((("script%%" + scriptType) + "%%") + _scriptHash)
479+ let approvedHashesStripped = removeByIndex(approvedHashesNew, 0)
480+[StringEntry((("script_" + scriptType) + "__approved"), makeString_2C(approvedHashesStripped, ",")), StringEntry(((scriptToRemoveId + BIGSEP) + k_confirmations), "")]
481+ }
482+ else [StringEntry((("script_" + scriptType) + "__approved"), makeString_2C(approvedHashesNew, ","))]
483+ }
484+ else nil
485+ (result ++ [StringEntry((scriptId + "__confirmations"), newListStr)])
486+ }
487+ }
488+ else throw("Strict value is not equal to itself.")
489+ }
490+ else throw("Strict value is not equal to itself.")
491+ }
492+ }
493+
494+
495+
496+@Callable(i)
497+func revokeConfirmationScript (scriptId) = {
498+ let callerPbk = toBase58String(i.callerPublicKey)
499+ let limit = _getLimit()
500+ if (!(_isAdmin(callerPbk)))
501+ then throw("Only admins can revoke")
502+ else {
503+ let _scriptHash = valueOrErrorMessage(getString((scriptId + "__scriptHash")), "Invalid scriptId")
504+ if ((_scriptHash == _scriptHash))
505+ then {
506+ let scriptType = getStringValue((scriptId + "__scriptType"))
507+ let approvedHashesStr = valueOrElse(getString((("script_" + scriptType) + "__approved")), "")
508+ let approvedHashes = split_4C(approvedHashesStr, ",")
509+ let confirmationsStr = valueOrElse(getString((scriptId + "__confirmations")), "")
510+ let confirmations = split(confirmationsStr, ",")
511+ let confirmIndex = indexOf(confirmations, callerPbk)
512+ if ((confirmIndex == unit))
513+ then throw("You didn't confirm this script, nothing to revoke")
514+ else {
515+ let newConfirmations = removeByIndex(confirmations, value(confirmIndex))
516+ let newConfirmationsStr = makeString(newConfirmations, ",")
517+ let wasApproved = (indexOf(approvedHashes, _scriptHash) != unit)
518+ let stillApproved = (size(newConfirmations) >= limit)
519+ let stateChanges = if (if (wasApproved)
520+ then !(stillApproved)
521+ else false)
522+ then {
523+ let scriptHashIndex = value(indexOf(approvedHashes, _scriptHash))
524+ let newApprovedHashes = removeByIndex(approvedHashes, scriptHashIndex)
525+ let newApprovedHashesStr = makeString(newApprovedHashes, ",")
526+[StringEntry((("script_" + scriptType) + "__approved"), newApprovedHashesStr)]
527+ }
528+ else nil
529+ (stateChanges ++ [StringEntry((scriptId + "__confirmations"), newConfirmationsStr)])
530+ }
531+ }
532+ else throw("Strict value is not equal to itself.")
533+ }
534+ }
535+
536+
537+
538+@Callable(i)
539+func addTask (name,dapp,txid,executeTs,desc,initiatorAdr,txdata) = {
540+ let callerPbk = toBase58String(i.callerPublicKey)
541+ if (!(_isValidOperator(callerPbk)))
542+ then throw("Not allowed")
543+ else {
544+ let _nameCheck = if ((size(name) == 0))
545+ then throw("Name can't be empty")
546+ else true
547+ if ((_nameCheck == _nameCheck))
548+ then {
549+ let _dappCheck = if (!(_validateAddress(dapp)))
550+ then throw("Incorrect dapp")
551+ else true
552+ if ((_dappCheck == _dappCheck))
553+ then {
554+ let _txidCheck = if ((size(txid) == 0))
555+ then throw("txid can't be empty")
556+ else true
557+ if ((_txidCheck == _txidCheck))
558+ then {
559+ let _timestampCheck = if (!(_validateTs(executeTs)))
560+ then throw("Invalid execution timestamp - cannot be more than 1 hour in the past")
561+ else true
562+ if ((_timestampCheck == _timestampCheck))
563+ then {
564+ let _descCheck = if ((size(desc) == 0))
565+ then throw("Description can't be empty")
566+ else true
567+ if ((_descCheck == _descCheck))
568+ then {
569+ let _initiatorCheck = if ((toString(addressFromPublicKey(i.callerPublicKey)) != initiatorAdr))
570+ then throw("Initiator address doesn't match caller")
571+ else true
572+ if ((_initiatorCheck == _initiatorCheck))
573+ then {
574+ let _txdataCheck = if ((size(txdata) == 0))
575+ then throw("Tx data can't be empty")
576+ else true
577+ if ((_txdataCheck == _txdataCheck))
578+ then {
579+ let taskId = ((("tx_" + dapp) + SEP) + txid)
580+[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)]
581+ }
582+ else throw("Strict value is not equal to itself.")
583+ }
584+ else throw("Strict value is not equal to itself.")
585+ }
586+ else throw("Strict value is not equal to itself.")
587+ }
588+ else throw("Strict value is not equal to itself.")
589+ }
590+ else throw("Strict value is not equal to itself.")
591+ }
592+ else throw("Strict value is not equal to itself.")
593+ }
594+ else throw("Strict value is not equal to itself.")
595+ }
596+ }
597+
598+
599+
600+@Callable(i)
601+func removeTask (taskId) = {
602+ let callerPbk = toBase58String(i.callerPublicKey)
603+ if (!(_isValidOperator(callerPbk)))
604+ then throw("Not allowed")
605+ else {
606+ let initiatorAdr = getStringValue((taskId + "__initiator"))
607+ if ((initiatorAdr != toString(addressFromPublicKey(i.callerPublicKey))))
608+ then throw("Only creator can remove")
609+ 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"))]
610+ }
611+ }
612+
613+
614+
615+@Callable(i)
616+func addConfirmation (taskId) = {
617+ let callerPbk = toBase58String(i.callerPublicKey)
618+ let limit = _getLimit()
619+ if (!(_isAdmin(callerPbk)))
620+ then throw("Only admins can confirm")
621+ else {
622+ let txId = valueOrErrorMessage(getString((taskId + "__txId")), "Ivalid task")
623+ if ((txId == txId))
624+ then {
625+ let dapp = getStringValue((taskId + "__dapp"))
626+ let confirmationsStr = valueOrElse(getString((taskId + "__confirmations")), "")
627+ let confirmations = split(confirmationsStr, ",")
628+ if ((indexOf(confirmations, callerPbk) != unit))
629+ then throw("You already confirmed this task")
630+ else {
631+ let newList = if ((confirmationsStr != ""))
632+ then (confirmations :+ callerPbk)
633+ else [callerPbk]
634+ let newListStr = makeString(newList, ",")
635+ let approved = (size(newList) >= limit)
636+[StringEntry((taskId + "__confirmations"), newListStr), BooleanEntry((((dapp + SEP) + txId) + "__approved"), approved)]
637+ }
638+ }
639+ else throw("Strict value is not equal to itself.")
640+ }
641+ }
642+
643+
644+
645+@Callable(i)
646+func revokeTask (taskId) = {
647+ let callerPbk = toBase58String(i.callerPublicKey)
648+ let limit = _getLimit()
649+ if (!(_isAdmin(callerPbk)))
650+ then throw("Only admins can call")
651+ else {
652+ let txId = valueOrErrorMessage(getString((taskId + "__txId")), "Ivalid task")
653+ if ((txId == txId))
654+ then {
655+ let confirmationsStr = valueOrElse(getString((taskId + "__confirmations")), "")
656+ let confirmations = split(confirmationsStr, ",")
657+ let indexMy = indexOf(confirmations, callerPbk)
658+ if ((indexMy == unit))
659+ then throw("You did not confirm this task, nothing to revoke")
660+ else {
661+ let dapp = getStringValue((taskId + "__dapp"))
662+ let newList = removeByIndex(confirmations, value(indexMy))
663+ let newListStr = makeString(newList, ",")
664+ let approved = (size(newList) >= limit)
665+[StringEntry((taskId + "__confirmations"), newListStr), BooleanEntry((((dapp + SEP) + txId) + "__approved"), approved)]
666+ }
667+ }
668+ else throw("Strict value is not equal to itself.")
669+ }
670+ }
671+
672+
673+
674+@Callable(i)
675+func test () = {
676+ let x = match blockInfoByHeight(height) {
677+ case block: BlockInfo =>
678+ throw(toString(block.timestamp))
679+ case _ =>
680+ throw("Can't find block")
681+ }
682+ if ((x == x))
683+ then nil
684+ else throw("Strict value is not equal to itself.")
685+ }
686+
687+
688+
689+@Callable(i)
690+func clear_test () = {
691+ let indices = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
692+ func removeChunks (accum,index) = (accum :+ DeleteEntry((("testtask" + "__txdata_") + toString(index))))
693+
694+ let entries = {
695+ let $l = indices
696+ let $s = size($l)
697+ let $acc0 = [DeleteEntry(("testtask" + "__txdata_chunks"))]
698+ func $f0_1 ($a,$i) = if (($i >= $s))
699+ then $a
700+ else removeChunks($a, $l[$i])
701+
702+ func $f0_2 ($a,$i) = if (($i >= $s))
703+ then $a
704+ else throw("List size exceeds 10")
705+
706+ $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)
707+ }
708+ entries
709+ }
710+
711+
712+@Verifier(tx)
713+func verify () = {
714+ let MULTISIG_ADDRESS = addressFromString("3M7uzD8rT54fKQcP4s417if1VKU8yromEP2")
715+ let CONTRACT_TYPE = "APPROVER_CONTRACT"
716+ if ((MULTISIG_ADDRESS != unit))
717+ then match tx {
718+ case ttx: SetScriptTransaction =>
719+ let scriptHasha = toBase58String(value(blake2b256(value(ttx.script))))
720+ let approvedHashes = value(getString(value(MULTISIG_ADDRESS), (("script_" + CONTRACT_TYPE) + "__approved")))
721+ (indexOf(approvedHashes, scriptHasha) != unit)
722+ case otx =>
723+ valueOrElse(getBoolean(value(this), (((toString(this) + "_") + toBase58String(otx.id)) + "__approved")), false)
724+ }
725+ else sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
726+ }
727+

github/deemru/w8io
29.13 ms