tx · CMCQseThWrf5ma1kFxZLiGmdmSeJAtf9FvsshJbegvwe

3N3gSFrTwgiLAFZHMuJrEttht1nPhnkpvjs:  -0.01700000 Waves

2022.11.24 14:43 [2331129] smart account 3N3gSFrTwgiLAFZHMuJrEttht1nPhnkpvjs > SELF 0.00000000 Waves

{ "type": 13, "id": "CMCQseThWrf5ma1kFxZLiGmdmSeJAtf9FvsshJbegvwe", "fee": 1700000, "feeAssetId": null, "timestamp": 1669290207509, "version": 2, "chainId": 84, "sender": "3N3gSFrTwgiLAFZHMuJrEttht1nPhnkpvjs", "senderPublicKey": "5mttAQ8x5ig98rgbv6rZmwXQ2pAYSwgCkQPSVBcr7BVp", "proofs": [ "24AeVgjjidZ1sEd5w9Qxjd8dg65MeUpzW8gZRAv2TDd4rTEipctAvmjtuPXtbpCLnhinK7EaJ4Suptszj5TfMHJn" ], "script": "base64:BgIuCAISBgoECAEBARIECgIBARIHCgUICAEBGBIHCgUICAEBGBIDCgEBEgMKAQESAFUAC3JldmlzaW9uTnVtAgAAA1NFUAICX18AB0xJU1RTRVACAToADURFRkFVTFRRVU9SVU0AoMIeAApVUkxQQVRURVJOAhpodHRwczovL2ZvcnVtLm5ldXRyaW5vLmF0LwAITUFYVElUTEUAoAEABk1BWFVSTAD6AQANTUFYVk9USU5HVElNRQCAkOTABAAFTVVMVDYAwIQ9AA5ERUZBVUxUUEFZTUVOVACAlOvcAwAUREVGQVVMVENSRUFUSU9OR05TQlQAgJTr3AMAClBBU1RNQVJHSU4AgLq3AwAMRlVUVVJFTUFSR0lOAMDLyQIAEmdvdklkeFByb3Bvc2FsVHhJZAABAApnb3ZJZHhUeXBlAAIADGdvdklkeEF1dGhvcgADAAlnb3ZJZHhVcmwABAALZ292SWR4VGl0bGUABQASZ292SWR4Q3JlYXRpb25UaW1lAAYAC2dvdklkeFN0YXJ0AAcACWdvdklkeEVuZAAIAAtnb3ZJZHhUeElkcwAJAAxnb3ZJZHhRdW9ydW0ACgANZ292SWR4T3B0aW9ucwALABNnb3ZTdGF0dXNJZHhJc1ZhbGlkAAEAEmdvdlN0YXR1c0lkeFdpbk9wdAACABRnb3ZTdGF0dXNJZHhXaW5Wb3RlcwADABZnb3ZTdGF0dXNJZHhUb3RhbFZvdGVzAAQAFWdvdlN0YXR1c0lkeFNjQXBwbGllZAAFABJnb3ZTdGF0dXNJZHhTY1RpbWUABgAWZ292U3RhdHVzSWR4SXNDYW5jZWxlZAAHAQ9nZXRTdHJpbmdPckZhaWwCB2FkZHJlc3MDa2V5CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJ0IAgUHYWRkcmVzcwUDa2V5CQC5CQIJAMwIAgIKbWFuZGF0b3J5IAkAzAgCCQClCAEFB2FkZHJlc3MJAMwIAgIBLgkAzAgCBQNrZXkJAMwIAgIPIGlzIG5vdCBkZWZpbmVkBQNuaWwCAAEMZ2V0SW50T3JFbHNlAgNrZXkKZGVmYXVsdFZhbAkBC3ZhbHVlT3JFbHNlAgkAmggCBQR0aGlzBQNrZXkFCmRlZmF1bHRWYWwAGUlkeENvbnRyb2xDZmdOZXV0cmlub0RhcHAAAQAYSWR4Q29udHJvbENmZ0F1Y3Rpb25EYXBwAAIAFElkeENvbnRyb2xDZmdScGREYXBwAAMAFUlkeENvbnRyb2xDZmdNYXRoRGFwcAAEABxJZHhDb250cm9sQ2ZnTGlxdWlkYXRpb25EYXBwAAUAFUlkeENvbnRyb2xDZmdSZXN0RGFwcAAGAB1JZHhDb250cm9sQ2ZnTm9kZVJlZ2lzdHJ5RGFwcAAHABxJZHhDb250cm9sQ2ZnTnNidFN0YWtpbmdEYXBwAAgAGUlkeENvbnRyb2xDZmdNZWRpYXRvckRhcHAACQAcSWR4Q29udHJvbENmZ1N1cmZTdGFraW5nRGFwcAAKACBJZHhDb250cm9sQ2ZnR25zYnRDb250cm9sbGVyRGFwcAALABdJZHhDb250cm9sQ2ZnUmVzdFYyRGFwcAAMABtJZHhDb250cm9sQ2ZnR292ZXJuYW5jZURhcHAADQERa2V5Q29udHJvbEFkZHJlc3MAAhwlcyVzX19jb25maWdfX2NvbnRyb2xBZGRyZXNzAQ1rZXlDb250cm9sQ2ZnAAIRJXNfX2NvbnRyb2xDb25maWcBFHJlYWRDb250cm9sQ2ZnT3JGYWlsAQdjb250cm9sCQC8CQIJAQ9nZXRTdHJpbmdPckZhaWwCBQdjb250cm9sCQENa2V5Q29udHJvbENmZwAFA1NFUAEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgpjb250cm9sQ2ZnA2lkeAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJAJEDAgUKY29udHJvbENmZwUDaWR4CQCsAgICLUNvbnRyb2wgY2ZnIGRvZXNuJ3QgY29udGFpbiBhZGRyZXNzIGF0IGluZGV4IAkApAMBBQNpZHgAD2NvbnRyb2xDb250cmFjdAkBEUBleHRyTmF0aXZlKDEwNjIpAQkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzCQERa2V5Q29udHJvbEFkZHJlc3MAAiMzUDVCZmQ1OFBQZk52Qk0ySHk4UWZiY0RxTWVOdHpnN0tmUAAKY29udHJvbENmZwkBFHJlYWRDb250cm9sQ2ZnT3JGYWlsAQUPY29udHJvbENvbnRyYWN0ABBuZXV0cmlub0NvbnRyYWN0CQEYZ2V0Q29udHJhY3RBZGRyZXNzT3JGYWlsAgUKY29udHJvbENmZwUZSWR4Q29udHJvbENmZ05ldXRyaW5vRGFwcAAXZ25zYnRDb250cm9sbGVyQ29udHJhY3QJARhnZXRDb250cmFjdEFkZHJlc3NPckZhaWwCBQpjb250cm9sQ2ZnBSBJZHhDb250cm9sQ2ZnR25zYnRDb250cm9sbGVyRGFwcAEYa2V5UXVvcnVtUmVxdWlyZWRQZXJjZW50AAISJXNfX3F1b3J1bVJlcXVpcmVkARJrZXlQYXltZW50UmVxdWlyZWQAAhMlc19fcGF5bWVudFJlcXVpcmVkARBrZXlHbnNidFJlcXVpcmVkAAIRJXNfX2dOc2J0UmVxdWlyZWQBEWtleUxhc3RQcm9wb3NhbElkAAIOJXNfX3Byb3Bvc2FsSWQBFGtleUxhc3RVcGRhdGVWZXJzaW9uAAIRJXNfX3VwZGF0ZVZlcnNpb24BGWtleVByb3Bvc2FsU3RhdHVzRGF0YUJ5SWQBCnByb3Bvc2FsSWQJAKwCAgIaJXMlZF9fcHJvcG9zYWxTdGF0dXNEYXRhX18JAKQDAQUKcHJvcG9zYWxJZAETa2V5UHJvcG9zYWxEYXRhQnlJZAEKcHJvcG9zYWxJZAkArAICAhQlcyVkX19wcm9wb3NhbERhdGFfXwkApAMBBQpwcm9wb3NhbElkAR1rZXlQcm9wb3NhbFZvdGVzQnlJZEFuZE9wdGlvbgIKcHJvcG9zYWxJZANvcHQJALkJAgkAzAgCAgYlcyVkJWQJAMwIAgIKdm90ZXNCeU9wdAkAzAgCCQCkAwEFCnByb3Bvc2FsSWQJAMwIAgkApAMBBQNvcHQFA25pbAUDU0VQARtrZXlQcm9wb3NhbFZvdGVzQnlJZEFuZFVzZXICCnByb3Bvc2FsSWQIdXNlckFkZHIJALkJAgkAzAgCAgYlcyVkJXMJAMwIAgILdm90ZXNCeVVzZXIJAMwIAgkApAMBBQpwcm9wb3NhbElkCQDMCAIFCHVzZXJBZGRyBQNuaWwFA1NFUAEca2V5UHJvcG9zYWxDaG9pY2VCeUlkQW5kVXNlcgIKcHJvcG9zYWxJZAh1c2VyQWRkcgkAuQkCCQDMCAICBiVzJWQlcwkAzAgCAgxvcHRpb25CeVVzZXIJAMwIAgkApAMBBQpwcm9wb3NhbElkCQDMCAIFCHVzZXJBZGRyBQNuaWwFA1NFUAESa2V5QXBwbHlJblByb2dyZXNzAAITJXNfX2FwcGx5SW5Qcm9ncmVzcwESa2V5UHJvcG9zYWxJZEJ5VXJsAQlmb3J1bUxpbmsJAKwCAgIXJXMlc19fcHJvcG9zYWxJZEJ5VXJsX18JANoEAQkAmwMBBQlmb3J1bUxpbmsBHmtleU51bVVuaXF1ZVZvdGVyc0J5UHJvcG9zYWxJZAEKcHJvcG9zYWxJZAkArAICAhElcyVkX19udW1Wb3RlcnNfXwkApAMBBQpwcm9wb3NhbElkARhrZXlTdGF0c0F2ZXJVbmlxdWVWb3RlcnMAAiAlcyVzJXNfX3N0YXRzX19hdmdfX3VuaXF1ZVZvdGVycwEWa2V5U3RhdHNBdmVyR25zYnRWb3RlZAACHiVzJXMlc19fc3RhdHNfX2F2Z19fZ25zYnRWb3RlZAEVa2V5U3RhdHNVbmlxdWVBdXRob3JzAAIaJXMlc19fc3RhdHNfX3VuaXF1ZUF1dGhvcnMBF2tleU51bVByb3Bvc2Fsc0J5QXV0aG9yAQphZGRyZXNzU3RyCQCsAgICHCVzJXNfX251bVByb3Bvc2Fsc0J5QXV0aG9yX18FCmFkZHJlc3NTdHIBD2tleUFwcGx5SGlzdG9yeQEJdGltZXN0YW1wCQCsAgICFCVzJWRfX2FwcGx5SGlzdG9yeV9fCQCkAwEFCXRpbWVzdGFtcAEJYXNBbnlMaXN0AQF2BAckbWF0Y2gwBQF2AwkAAQIFByRtYXRjaDACCUxpc3RbQW55XQQBbAUHJG1hdGNoMAUBbAkAAgECG2ZhaWwgdG8gY2FzdCBpbnRvIExpc3RbQW55XQEFYXNJbnQBAXYEByRtYXRjaDAFAXYDCQABAgUHJG1hdGNoMAIDSW50BAFpBQckbWF0Y2gwBQFpCQACAQIVZmFpbCB0byBjYXN0IGludG8gSW50AQpzdGF0dXNEYXRhBw1pc1ZvdGluZ1ZhbGlkCXdpbk9wdGlvbg53aW5PcHRpb25Wb3Rlcwp0b3RhbFZvdGVzEWFyZVNjcmlwdHNBcHBsaWVkEHNjcmlwdHNUaW1lc3RhbXAOY2FuY2VsZWRCeVRlYW0JALkJAgkAzAgCAg4lYiVkJWQlZCViJWQlYgkAzAgCCQClAwEFDWlzVm90aW5nVmFsaWQJAMwIAgkApAMBBQl3aW5PcHRpb24JAMwIAgkApAMBBQ53aW5PcHRpb25Wb3RlcwkAzAgCCQCkAwEFCnRvdGFsVm90ZXMJAMwIAgkApQMBBRFhcmVTY3JpcHRzQXBwbGllZAkAzAgCCQCkAwEFEHNjcmlwdHNUaW1lc3RhbXAJAMwIAgkApQMBBQ5jYW5jZWxlZEJ5VGVhbQUDbmlsBQNTRVABDHByb3Bvc2FsRGF0YQsMcHJvcG9zYWxUeElkBHR5cGUGYXV0aG9yCWZvcnVtTGluawV0aXRsZQxwcm9wb3NhbFRpbWUPdm90aW5nU3RhcnRUaW1lDXZvdGluZ0VuZFRpbWUFdHhJZHMNcXVvcnVtSW5HbnNidAdvcHRpb25zCQC5CQIJAMwIAgIWJXMlcyVzJXMlcyVkJWQlZCVzJWQlcwkAzAgCBQxwcm9wb3NhbFR4SWQJAMwIAgUEdHlwZQkAzAgCBQZhdXRob3IJAMwIAgUJZm9ydW1MaW5rCQDMCAIFBXRpdGxlCQDMCAIJAKQDAQUMcHJvcG9zYWxUaW1lCQDMCAIJAKQDAQUPdm90aW5nU3RhcnRUaW1lCQDMCAIJAKQDAQUNdm90aW5nRW5kVGltZQkAzAgCBQV0eElkcwkAzAgCCQCkAwEFDXF1b3J1bUluR25zYnQJAMwIAgUHb3B0aW9ucwUDbmlsBQNTRVABC2NoZWNrVHhMaXN0AQZ0eExpc3QDCQBmAgkAkAMBBQZ0eExpc3QAFAkAAgEJAKwCAgIXVG9vIG1hbnkgdHJhbnNhY3Rpb25zOiAJAKQDAQkAkAMBBQZ0eExpc3QKAQhjb21iaW5lcgIDYWNjAnR4AwkBAiE9AgkAyAEBCQDZBAEFAnR4ACAJAAIBCQCsAgICDFdyb25nIHR4SWQ6IAUCdHgDCQAAAgUDYWNjAgAFAnR4CQCsAgIJAKwCAgUDYWNjBQdMSVNUU0VQBQJ0eAoAAiRsBQZ0eExpc3QKAAIkcwkAkAMBBQIkbAoABSRhY2MwAgAKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBCGNvbWJpbmVyAgUCJGEJAJEDAgUCJGwFAiRpCgEFJGYwXzICAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAAIBAhRMaXN0IHNpemUgZXhjZWVkcyAyMAkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUABgAHAAgACQAKAAsADAANAA4ADwAQABEAEgATABQAC0lkeEVmZlRvdGFsAAAACklkeEVmZlVzZXIAAQERZ2V0RWZmZWN0aXZlR25zYnQBEnVzZXJBZGRyU3RyT3JFbXB0eQQJZ25zYnREYXRhCQEJYXNBbnlMaXN0AQkA/AcEBRdnbnNidENvbnRyb2xsZXJDb250cmFjdAIUZ25zYnRJbmZvU1lTUkVBRE9OTFkJAMwIAgUSdXNlckFkZHJTdHJPckVtcHR5CQDMCAIAAAkAzAgCAAAFA25pbAUDbmlsBAhuc2J0RGF0YQkBCWFzQW55TGlzdAEJAJEDAgUJZ25zYnREYXRhAAIEDHVzZXJGcm9tTnNidAkBBWFzSW50AQkAkQMCBQhuc2J0RGF0YQACBA10b3RhbEZyb21Oc2J0CQEFYXNJbnQBCQCRAwIFCG5zYnREYXRhAAMEEnVzZXJNYXR1cmVGcm9tU3VyZgkBBWFzSW50AQkAkQMCBQlnbnNidERhdGEACQQTdG90YWxNYXR1cmVGcm9tU3VyZgkBBWFzSW50AQkAkQMCBQlnbnNidERhdGEABgkAzAgCCQBkAgUNdG90YWxGcm9tTnNidAUTdG90YWxNYXR1cmVGcm9tU3VyZgkAzAgCCQBkAgUMdXNlckZyb21Oc2J0BRJ1c2VyTWF0dXJlRnJvbVN1cmYFA25pbAEOaW5pdGlhdGVWb3RpbmcLB3BheW1lbnQMcHJvcG9zYWxUeElkBHR5cGUGYXV0aG9yCWZvcnVtTGluawV0aXRsZQ92b3RpbmdTdGFydFRpbWUNdm90aW5nRW5kVGltZQZzdGF0dXMGdHhMaXN0C29wdGlvbnNMaXN0AwkBAiE9AggFB3BheW1lbnQHYXNzZXRJZAUEdW5pdAkAAgECG0FsbG93ZWQgV0FWRVMgcGF5bWVudCBvbmx5IQQGcG10UmVxCQEMZ2V0SW50T3JFbHNlAgkBEmtleVBheW1lbnRSZXF1aXJlZAAFDkRFRkFVTFRQQVlNRU5UAwkAZgIFBnBtdFJlcQgFB3BheW1lbnQGYW1vdW50CQACAQkArAICAiRQYXltZW50IGF0dGFjaGVkIHNob3VsZCBiZSBhdCBsZWFzdCAJAKQDAQUGcG10UmVxAwkBAiE9AgkBBXZhbHVlAQkAswkCBQlmb3J1bUxpbmsFClVSTFBBVFRFUk4AAAkAAgECC0ludmFsaWQgdXJsAwkAZgIJALECAQUJZm9ydW1MaW5rBQZNQVhVUkwJAAIBAg1VcmwgdG9vIGxvbmchBAxyZWdpc3RlcmVkSWQJAJ8IAQkBEmtleVByb3Bvc2FsSWRCeVVybAEFCWZvcnVtTGluawMJAQlpc0RlZmluZWQBBQxyZWdpc3RlcmVkSWQJAAIBCQCsAgICOFZvdGluZyB3aXRoIHN1Y2ggZm9ydW0gbGluayBpcyBhbHJlYWR5IHJlZ2lzdGVyZWQgYnkgaWQ9CQCkAwEJAQV2YWx1ZQEFDHJlZ2lzdGVyZWRJZAMJAAACBQV0aXRsZQIACQACAQIOVGl0bGUgaXMgZW1wdHkDCQBmAgkAsQIBBQV0aXRsZQUITUFYVElUTEUJAAIBAg5Ub28gbG9uZyB0aXRsZQQMcHJvcG9zYWxUaW1lCAUJbGFzdEJsb2NrCXRpbWVzdGFtcAMJAGYCBQxwcm9wb3NhbFRpbWUFD3ZvdGluZ1N0YXJ0VGltZQkAAgEJAKwCAgkArAICCQCsAgICEHZvdGluZ1N0YXJ0VGltZT0JAKQDAQUPdm90aW5nU3RhcnRUaW1lAhAgPCBwcm9wb3NhbFRpbWU9CQCkAwEFDHByb3Bvc2FsVGltZQMJAGYCBQ92b3RpbmdTdGFydFRpbWUFDXZvdGluZ0VuZFRpbWUJAAIBCQCsAgIJAKwCAgkArAICAg52b3RpbmdFbmRUaW1lPQkApAMBBQ12b3RpbmdFbmRUaW1lAhMgPCB2b3RpbmdTdGFydFRpbWU9CQCkAwEFD3ZvdGluZ1N0YXJ0VGltZQMJAGYCCQBlAgUNdm90aW5nRW5kVGltZQUPdm90aW5nU3RhcnRUaW1lBQ1NQVhWT1RJTkdUSU1FCQACAQkArAICCQCsAgIJAKwCAgIbVm90aW5nIHBlcmlvZCBleGNlZWRzIG1heDogCQCkAwEJAGUCBQ12b3RpbmdFbmRUaW1lBQ92b3RpbmdTdGFydFRpbWUCAyA+IAkApAMBBQ1NQVhWT1RJTkdUSU1FBAV0eElkcwMJAAACBQR0eXBlAgRJREVBAgAJAQtjaGVja1R4TGlzdAEFBnR4TGlzdAMJAGcCAAEJAJADAQULb3B0aW9uc0xpc3QJAAIBAhdUb28gZmV3IGNob2ljZXMgdG8gdm90ZQQDZWZmCQERZ2V0RWZmZWN0aXZlR25zYnQBBQZhdXRob3IECmduc2J0VG90YWwJAJEDAgUDZWZmBQtJZHhFZmZUb3RhbAQJZ05zYnRVc2VyCQCRAwIFA2VmZgUKSWR4RWZmVXNlcgQIZ25zYnRSZXEJAQxnZXRJbnRPckVsc2UCCQEQa2V5R25zYnRSZXF1aXJlZAAFFERFRkFVTFRDUkVBVElPTkdOU0JUAwkAZgIFCGduc2J0UmVxBQlnTnNidFVzZXIJAAIBCQCsAgIJAKwCAgISWW91IG5lZWQgYXQgbGVhc3QgCQCkAwEFCGduc2J0UmVxAhcgZ05zYnQgdG8gY3JlYXRlIHZvdGluZwQMYW1vdW50TGVhc2VkCQD8BwQFEG5ldXRyaW5vQ29udHJhY3QCC2FjY2VwdFdhdmVzBQNuaWwJAMwIAgUHcGF5bWVudAUDbmlsAwkAAAIFDGFtb3VudExlYXNlZAUMYW1vdW50TGVhc2VkBAZxdW9ydW0JAQxnZXRJbnRPckVsc2UCCQEYa2V5UXVvcnVtUmVxdWlyZWRQZXJjZW50AAUNREVGQVVMVFFVT1JVTQQNcXVvcnVtSW5HbnNidAkAawMFBnF1b3J1bQUKZ25zYnRUb3RhbAUFTVVMVDYECnByb3Bvc2FsSWQJAGQCCQEMZ2V0SW50T3JFbHNlAgkBEWtleUxhc3RQcm9wb3NhbElkAAAAAAEEFG51bVByb3Bvc2Fsc0J5QXV0aG9yCQBkAgkBDGdldEludE9yRWxzZQIJARdrZXlOdW1Qcm9wb3NhbHNCeUF1dGhvcgEFBmF1dGhvcgAAAAEEC3VuaXFBdXRob3JzCQBkAgkBDGdldEludE9yRWxzZQIJARVrZXlTdGF0c1VuaXF1ZUF1dGhvcnMAAAADCQAAAgUUbnVtUHJvcG9zYWxzQnlBdXRob3IAAQABAAAECm9wdGlvbnNTdHIJALkJAgULb3B0aW9uc0xpc3QFB0xJU1RTRVAJAJQKAgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBEWtleUxhc3RQcm9wb3NhbElkAAUKcHJvcG9zYWxJZAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBEmtleVByb3Bvc2FsSWRCeVVybAEFCWZvcnVtTGluawUKcHJvcG9zYWxJZAkAzAgCCQELU3RyaW5nRW50cnkCCQEZa2V5UHJvcG9zYWxTdGF0dXNEYXRhQnlJZAEFCnByb3Bvc2FsSWQJAQpzdGF0dXNEYXRhBwcAAAAAAAAHAAAHCQDMCAIJAQtTdHJpbmdFbnRyeQIJARNrZXlQcm9wb3NhbERhdGFCeUlkAQUKcHJvcG9zYWxJZAkBDHByb3Bvc2FsRGF0YQsFDHByb3Bvc2FsVHhJZAUEdHlwZQUGYXV0aG9yBQlmb3J1bUxpbmsFBXRpdGxlBQxwcm9wb3NhbFRpbWUFD3ZvdGluZ1N0YXJ0VGltZQUNdm90aW5nRW5kVGltZQUFdHhJZHMFDXF1b3J1bUluR25zYnQFCm9wdGlvbnNTdHIJAMwIAgkBDEludGVnZXJFbnRyeQIJARdrZXlOdW1Qcm9wb3NhbHNCeUF1dGhvcgEFBmF1dGhvcgUUbnVtUHJvcG9zYWxzQnlBdXRob3IJAMwIAgkBDEludGVnZXJFbnRyeQIJARVrZXlTdGF0c1VuaXF1ZUF1dGhvcnMABQt1bmlxQXV0aG9ycwUDbmlsBQxwcm9wb3NhbFR4SWQJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BDWNhbGNXaW5PcHRpb24HCnByb3Bvc2FsSWQLb3B0aW9uc0xpc3QOaXNQcmV2T3B0aW9uYWwJb2xkQ2hvaWNlEG9wdGlvbmFsVG90YWxPbGQJbmV3Q2hvaWNlE25ld1RvdGFsQnlOZXdDaG9pY2UKAQhmaW5kQmVzdAIDYWNjBGVsZW0EA2lkeAkBBXZhbHVlAQkAzwgCBQtvcHRpb25zTGlzdAUEZWxlbQQDdmFsAwUOaXNQcmV2T3B0aW9uYWwDCQAAAgUDaWR4BQluZXdDaG9pY2UFE25ld1RvdGFsQnlOZXdDaG9pY2UJAQxnZXRJbnRPckVsc2UCCQEda2V5UHJvcG9zYWxWb3Rlc0J5SWRBbmRPcHRpb24CBQpwcm9wb3NhbElkBQNpZHgAAAMJAAACBQNpZHgJAQV2YWx1ZQEFCW9sZENob2ljZQUQb3B0aW9uYWxUb3RhbE9sZAMJAAACBQNpZHgFCW5ld0Nob2ljZQUTbmV3VG90YWxCeU5ld0Nob2ljZQkBDGdldEludE9yRWxzZQIJAR1rZXlQcm9wb3NhbFZvdGVzQnlJZEFuZE9wdGlvbgIFCnByb3Bvc2FsSWQFA2lkeAAAAwkAZgIIBQNhY2MCXzIFA3ZhbAUDYWNjCQCUCgIFA2lkeAUDdmFsCgACJGwFC29wdGlvbnNMaXN0CgACJHMJAJADAQUCJGwKAAUkYWNjMAkAlAoCAAAAAAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEIZmluZEJlc3QCBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDEwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoBEHVwZGF0ZVN0YXR1c0RhdGEEB29sZERhdGEHaXNWYWxpZAluZXdXaW5PcHQNbmV3VG90YWxWb3RlcwkAuQkCCQDMCAICDiViJWQlZCVkJWIlZCViCQDMCAIJAKUDAQUHaXNWYWxpZAkAzAgCCQCkAwEIBQluZXdXaW5PcHQCXzEJAMwIAgkApAMBCAUJbmV3V2luT3B0Al8yCQDMCAIJAKQDAQUNbmV3VG90YWxWb3RlcwkAzAgCCQCRAwIFB29sZERhdGEFFWdvdlN0YXR1c0lkeFNjQXBwbGllZAkAzAgCCQCRAwIFB29sZERhdGEFEmdvdlN0YXR1c0lkeFNjVGltZQkAzAgCCQCRAwIFB29sZERhdGEFFmdvdlN0YXR1c0lkeElzQ2FuY2VsZWQFA25pbAUDU0VQARFzdGF0dXNBcHBseVNjcmlwdAEHb2xkRGF0YQkAuQkCCQDMCAICDiViJWQlZCVkJWIlZCViCQDMCAIJAJEDAgUHb2xkRGF0YQUTZ292U3RhdHVzSWR4SXNWYWxpZAkAzAgCCQCRAwIFB29sZERhdGEFEmdvdlN0YXR1c0lkeFdpbk9wdAkAzAgCCQCRAwIFB29sZERhdGEFFGdvdlN0YXR1c0lkeFdpblZvdGVzCQDMCAIJAJEDAgUHb2xkRGF0YQUWZ292U3RhdHVzSWR4VG90YWxWb3RlcwkAzAgCAgR0cnVlCQDMCAIJAJEDAgUHb2xkRGF0YQUSZ292U3RhdHVzSWR4U2NUaW1lCQDMCAIJAJEDAgUHb2xkRGF0YQUWZ292U3RhdHVzSWR4SXNDYW5jZWxlZAUDbmlsBQNTRVABEEV4ZWN1dGlvbkhpc3RvcnkDCnByb3Bvc2FsSWQFdGl0bGUDdXJsBApnbnNidFRvdGFsCQCRAwIJARFnZXRFZmZlY3RpdmVHbnNidAECAAULSWR4RWZmVG90YWwEB3R1cm5vdXQAoMIeCQELU3RyaW5nRW50cnkCCQEPa2V5QXBwbHlIaXN0b3J5AQgFCWxhc3RCbG9jawl0aW1lc3RhbXAJALkJAgkAzAgCAgolZCVkJWQlcyVzCQDMCAIJAKQDAQUKcHJvcG9zYWxJZAkAzAgCCQCkAwEFCmduc2J0VG90YWwJAMwIAgkApAMBBQd0dXJub3V0CQDMCAIFBXRpdGxlCQDMCAIFA3VybAUDbmlsBQNTRVAHAWkBC2NvbnN0cnVjdG9yBAtjb250cm9sQWRkcg5nTnNidFJlcVRvSW5pdA53YXZlc1JlcVRvSW5pdA1xdW9ydW1SZXFQZXJjAwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIRUGVybWlzc2lvbiBkZW5pZWQJAMwIAgkBC1N0cmluZ0VudHJ5AgkBEWtleUNvbnRyb2xBZGRyZXNzAAULY29udHJvbEFkZHIJAMwIAgkBDEludGVnZXJFbnRyeQIJARBrZXlHbnNidFJlcXVpcmVkAAUOZ05zYnRSZXFUb0luaXQJAMwIAgkBDEludGVnZXJFbnRyeQIJARJrZXlQYXltZW50UmVxdWlyZWQABQ53YXZlc1JlcVRvSW5pdAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBGGtleVF1b3J1bVJlcXVpcmVkUGVyY2VudAAFDXF1b3J1bVJlcVBlcmMFA25pbAFpAQhjYXN0Vm90ZQIKcHJvcG9zYWxJZAZjaG9pY2UEDnVzZXJBZGRyZXNzU3RyCQClCAEIBQFpBmNhbGxlcgQLZHluYW1pY0RhdGEJALUJAgkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMJARlrZXlQcm9wb3NhbFN0YXR1c0RhdGFCeUlkAQUKcHJvcG9zYWxJZAUDU0VQAwkAAAIJAJEDAgULZHluYW1pY0RhdGEFFmdvdlN0YXR1c0lkeElzQ2FuY2VsZWQCBHRydWUJAAIBAhpWb3RpbmcgaXMgY2FuY2VsZWQgYnkgdGVhbQQIcHJvcERhdGEJALUJAgkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMJARNrZXlQcm9wb3NhbERhdGFCeUlkAQUKcHJvcG9zYWxJZAUDU0VQBAVzdGFydAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFCHByb3BEYXRhBQtnb3ZJZHhTdGFydAQDZW5kCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUIcHJvcERhdGEFCWdvdklkeEVuZAQDbm93CAUJbGFzdEJsb2NrCXRpbWVzdGFtcAMJAGYCBQVzdGFydAUDbm93CQACAQIWVm90aW5nIG5vdCBzdGFydGVkIHlldAMJAGcCBQNub3cFA2VuZAkAAgECF1ZvdGluZyBhbHJlYWR5IGZpbmlzaGVkBBBhdmFpbGFibGVPcHRpb25zCQC1CQIJAJEDAgUIcHJvcERhdGEFDWdvdklkeE9wdGlvbnMFB0xJU1RTRVAECm51bU9wdGlvbnMJAJADAQUQYXZhaWxhYmxlT3B0aW9ucwMJAGcCAAEFCm51bU9wdGlvbnMJAAIBAhdUb28gZmV3IGNob2ljZXMgdG8gdm90ZQMJAGcCBQZjaG9pY2UFCm51bU9wdGlvbnMJAAIBCQCsAgICG1Vua25vd24gY2hvaWNlISBNdXN0IGJlIDAuLgkApAMBCQBlAgUKbnVtT3B0aW9ucwABBANlZmYJARFnZXRFZmZlY3RpdmVHbnNidAEFDnVzZXJBZGRyZXNzU3RyBAhnbnNidEFtdAkAkQMCBQNlZmYFCklkeEVmZlVzZXIDCQBnAgAABQhnbnNidEFtdAkAAgECEG5vIGduc2J0IHRvIHZvdGUECmduc2J0VG90YWwJAJEDAgUDZWZmBQtJZHhFZmZUb3RhbAQJb2xkQ2hvaWNlCQCfCAEJARxrZXlQcm9wb3NhbENob2ljZUJ5SWRBbmRVc2VyAgUKcHJvcG9zYWxJZAUOdXNlckFkZHJlc3NTdHIEDG9sZFVzZXJWb3RlcwMJAQEhAQkBCWlzRGVmaW5lZAEFCW9sZENob2ljZQAACQEMZ2V0SW50T3JFbHNlAgkBG2tleVByb3Bvc2FsVm90ZXNCeUlkQW5kVXNlcgIFCnByb3Bvc2FsSWQFDnVzZXJBZGRyZXNzU3RyAAAEE29sZFRvdGFsQnlPbGRDaG9pY2UDCQEJaXNEZWZpbmVkAQUJb2xkQ2hvaWNlCQEMZ2V0SW50T3JFbHNlAgkBHWtleVByb3Bvc2FsVm90ZXNCeUlkQW5kT3B0aW9uAgUKcHJvcG9zYWxJZAkBBXZhbHVlAQUJb2xkQ2hvaWNlAAAAAAQTb2xkVG90YWxCeU5ld0Nob2ljZQkBDGdldEludE9yRWxzZQIJAR1rZXlQcm9wb3NhbFZvdGVzQnlJZEFuZE9wdGlvbgIFCnByb3Bvc2FsSWQFBmNob2ljZQAABAhvbGRUb3RhbAkBDXBhcnNlSW50VmFsdWUBCQCRAwIFC2R5bmFtaWNEYXRhBRZnb3ZTdGF0dXNJZHhUb3RhbFZvdGVzBBNuZXdUb3RhbEJ5T2xkQ2hvaWNlAwkBASEBCQEJaXNEZWZpbmVkAQUJb2xkQ2hvaWNlAAAJAGQCCQBlAgUTb2xkVG90YWxCeU9sZENob2ljZQUMb2xkVXNlclZvdGVzAwkAAAIJAQV2YWx1ZQEFCW9sZENob2ljZQUGY2hvaWNlBQhnbnNidEFtdAAABBNuZXdUb3RhbEJ5TmV3Q2hvaWNlAwMJAQlpc0RlZmluZWQBBQlvbGRDaG9pY2UJAAACCQEFdmFsdWUBBQlvbGRDaG9pY2UFBmNob2ljZQcFE25ld1RvdGFsQnlPbGRDaG9pY2UJAGQCBRNvbGRUb3RhbEJ5TmV3Q2hvaWNlBQhnbnNidEFtdAQIbmV3VG90YWwJAGQCCQBlAgUIb2xkVG90YWwFDG9sZFVzZXJWb3RlcwUIZ25zYnRBbXQED2lzUXVvcnVtUmVhY2hlZAkAZwIFCG5ld1RvdGFsCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUIcHJvcERhdGEFDGdvdklkeFF1b3J1bQQVbnVtVm90ZXJzQnlQcm9wb3NhbElkCQEMZ2V0SW50T3JFbHNlAgkBHmtleU51bVVuaXF1ZVZvdGVyc0J5UHJvcG9zYWxJZAEFCnByb3Bvc2FsSWQAAAQUb2xkQXZlclVuaXF1ZVZvdGVyczYJAQxnZXRJbnRPckVsc2UCCQEYa2V5U3RhdHNBdmVyVW5pcXVlVm90ZXJzAAAABAxudW1Qcm9wb3NhbHMJARFAZXh0ck5hdGl2ZSgxMDU1KQEJARFrZXlMYXN0UHJvcG9zYWxJZAAECnVuaXF1ZURpZmYDCQAAAgUMb2xkVXNlclZvdGVzAAAAAQAABBRuZXdBdmVyVW5pcXVlVm90ZXJzNgkAZAIFFG9sZEF2ZXJVbmlxdWVWb3RlcnM2CQBrAwUKdW5pcXVlRGlmZgUFTVVMVDYFDG51bVByb3Bvc2FscwQMb2xkQXZlckduc2J0CQEMZ2V0SW50T3JFbHNlAgkBFmtleVN0YXRzQXZlckduc2J0Vm90ZWQAAAAEDG5ld0F2ZXJHbnNidAkAZAIFDG9sZEF2ZXJHbnNidAkAaQIJAGUCBQhnbnNidEFtdAUMb2xkVXNlclZvdGVzBQxudW1Qcm9wb3NhbHMEDmlzUHJldk9wdGlvbmFsAwkBASEBCQEJaXNEZWZpbmVkAQUJb2xkQ2hvaWNlBgkAAAIJAQV2YWx1ZQEFCW9sZENob2ljZQUGY2hvaWNlBBBvcHRpb25hbFRvdGFsT2xkAwUOaXNQcmV2T3B0aW9uYWwFA25pbAkAzAgCCQEMSW50ZWdlckVudHJ5AgkBHWtleVByb3Bvc2FsVm90ZXNCeUlkQW5kT3B0aW9uAgUKcHJvcG9zYWxJZAkBBXZhbHVlAQUJb2xkQ2hvaWNlBRNuZXdUb3RhbEJ5T2xkQ2hvaWNlBQNuaWwEBndpbk9wdAkBDWNhbGNXaW5PcHRpb24HBQpwcm9wb3NhbElkBRBhdmFpbGFibGVPcHRpb25zBQ5pc1ByZXZPcHRpb25hbAUJb2xkQ2hvaWNlBRNuZXdUb3RhbEJ5T2xkQ2hvaWNlBQZjaG9pY2UFE25ld1RvdGFsQnlOZXdDaG9pY2UJAJQKAgkAzggCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEca2V5UHJvcG9zYWxDaG9pY2VCeUlkQW5kVXNlcgIFCnByb3Bvc2FsSWQFDnVzZXJBZGRyZXNzU3RyBQZjaG9pY2UJAMwIAgkBDEludGVnZXJFbnRyeQIJARtrZXlQcm9wb3NhbFZvdGVzQnlJZEFuZFVzZXICBQpwcm9wb3NhbElkBQ51c2VyQWRkcmVzc1N0cgUIZ25zYnRBbXQJAMwIAgkBDEludGVnZXJFbnRyeQIJAR1rZXlQcm9wb3NhbFZvdGVzQnlJZEFuZE9wdGlvbgIFCnByb3Bvc2FsSWQFBmNob2ljZQUTbmV3VG90YWxCeU5ld0Nob2ljZQkAzAgCCQEMSW50ZWdlckVudHJ5AgkBHmtleU51bVVuaXF1ZVZvdGVyc0J5UHJvcG9zYWxJZAEFCnByb3Bvc2FsSWQJAGQCBRVudW1Wb3RlcnNCeVByb3Bvc2FsSWQFCnVuaXF1ZURpZmYJAMwIAgkBDEludGVnZXJFbnRyeQIJARhrZXlTdGF0c0F2ZXJVbmlxdWVWb3RlcnMABRRuZXdBdmVyVW5pcXVlVm90ZXJzNgkAzAgCCQEMSW50ZWdlckVudHJ5AgkBFmtleVN0YXRzQXZlckduc2J0Vm90ZWQABQxuZXdBdmVyR25zYnQJAMwIAgkBC1N0cmluZ0VudHJ5AgkBGWtleVByb3Bvc2FsU3RhdHVzRGF0YUJ5SWQBBQpwcm9wb3NhbElkCQEQdXBkYXRlU3RhdHVzRGF0YQQFC2R5bmFtaWNEYXRhBQ9pc1F1b3J1bVJlYWNoZWQFBndpbk9wdAUIbmV3VG90YWwFA25pbAUQb3B0aW9uYWxUb3RhbE9sZAUEdW5pdAFpARJpbml0aWF0ZUlkZWFWb3RpbmcFCWZvcnVtTGluawV0aXRsZQ92b3RpbmdTdGFydFRpbWUNdm90aW5nRW5kVGltZQtvcHRpb25zTGlzdAMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQkAAgECHEV4YWN0bHkgb25lIHBheW1lbnQgcmVxdWlyZWQDCQECIT0CCQCQAwEFC29wdGlvbnNMaXN0AAIJAAIBAitFeGFjdGx5IDIgb3B0aW9uIFsnTk8nLCAnWUVTJ10gYXJlIGV4cGVjdGVkAwkBAiE9AgkAkQMCBQtvcHRpb25zTGlzdAAAAgJOTwkAAgECHU9wdGlvbiBOTyBzaG91bGQgYmUgdGhlIGZpcnN0AwkBAiE9AgkAkQMCBQtvcHRpb25zTGlzdAABAgNZRVMJAAIBAh9PcHRpb24gWUVTIHNob3VsZCBiZSB0aGUgc2Vjb25kCQEOaW5pdGlhdGVWb3RpbmcLCQEFdmFsdWUBCQCRAwIIBQFpCHBheW1lbnRzAAAJANgEAQgFAWkNdHJhbnNhY3Rpb25JZAIESURFQQkApQgBCAUBaQZjYWxsZXIFCWZvcnVtTGluawUFdGl0bGUFD3ZvdGluZ1N0YXJ0VGltZQUNdm90aW5nRW5kVGltZQIHUEVORElORwUDbmlsBQtvcHRpb25zTGlzdAFpARRpbml0aWF0ZVVwZGF0ZVZvdGluZwUJZm9ydW1MaW5rBXRpdGxlD3ZvdGluZ1N0YXJ0VGltZQ12b3RpbmdFbmRUaW1lBnR4TGlzdAMJAQIhPQIJAJADAQgFAWkIcGF5bWVudHMAAQkAAgECHEV4YWN0bHkgb25lIHBheW1lbnQgcmVxdWlyZWQDCQBmAgABCQCQAwEFBnR4TGlzdAkAAgECGlRyYW5zYWN0aW9ucyBsaXN0IGlzIGVtcHR5AwkBAiE9AggFAWkGY2FsbGVyBQR0aGlzCQACAQIObm90IGF1dGhvcml6ZWQJAQ5pbml0aWF0ZVZvdGluZwsJAQV2YWx1ZQEJAJEDAggFAWkIcGF5bWVudHMAAAkA2AQBCAUBaQ10cmFuc2FjdGlvbklkAgZVUERBVEUJAKUIAQgFAWkGY2FsbGVyBQlmb3J1bUxpbmsFBXRpdGxlBQ92b3RpbmdTdGFydFRpbWUFDXZvdGluZ0VuZFRpbWUCB1BFTkRJTkcFBnR4TGlzdAkAzAgCAgJOTwkAzAgCAgNZRVMFA25pbAFpAQxjYW5jZWxWb3RpbmcBCnByb3Bvc2FsSWQDCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAg5ub3QgYXV0aG9yaXplZAQLY3VycmVudERhdGEJAQ9nZXRTdHJpbmdPckZhaWwCBQR0aGlzCQEZa2V5UHJvcG9zYWxTdGF0dXNEYXRhQnlJZAEFCnByb3Bvc2FsSWQEC3VwZGF0ZWREYXRhCQCsAgIJAKwCAgkArwICBQtjdXJyZW50RGF0YQkBBXZhbHVlAQkAtwkCBQtjdXJyZW50RGF0YQUDU0VQBQNTRVACBHRydWUJAJQKAgkAzAgCCQELU3RyaW5nRW50cnkCCQEZa2V5UHJvcG9zYWxTdGF0dXNEYXRhQnlJZAEFCnByb3Bvc2FsSWQFC3VwZGF0ZWREYXRhBQNuaWwFBHVuaXQBaQELYXBwbHlVcGRhdGUBCnByb3Bvc2FsSWQECHByb3BEYXRhCQC1CQIJAQ9nZXRTdHJpbmdPckZhaWwCBQR0aGlzCQETa2V5UHJvcG9zYWxEYXRhQnlJZAEFCnByb3Bvc2FsSWQFA1NFUAQDZW5kCQENcGFyc2VJbnRWYWx1ZQEJAJEDAgUIcHJvcERhdGEFCWdvdklkeEVuZAQDbm93CAUJbGFzdEJsb2NrCXRpbWVzdGFtcAMJAGYCBQNlbmQFA25vdwkAAgECGlZvdGluZyBpcyBub3QgZmluaXNoZWQgeWV0AwkBAiE9AgIGVVBEQVRFCQCRAwIFCHByb3BEYXRhBQpnb3ZJZHhUeXBlCQACAQIfT25seSBVUERBVEUgdHlwZSBjYW4gYmUgYXBwbGllZAQLZHluYW1pY0RhdGEJALUJAgkBD2dldFN0cmluZ09yRmFpbAIFBHRoaXMJARlrZXlQcm9wb3NhbFN0YXR1c0RhdGFCeUlkAQUKcHJvcG9zYWxJZAUDU0VQAwkAAAIJAJEDAgULZHluYW1pY0RhdGEFFmdvdlN0YXR1c0lkeElzQ2FuY2VsZWQCBHRydWUJAAIBAhJWb3RpbmcgaXMgY2FuY2VsZWQDCQECIT0CCQCRAwIFC2R5bmFtaWNEYXRhBRNnb3ZTdGF0dXNJZHhJc1ZhbGlkAgR0cnVlCQACAQIVVm90aW5nIHN0YXR1cyBpbnZhbGlkAwkBAiE9AgkAkQMCBQtkeW5hbWljRGF0YQUSZ292U3RhdHVzSWR4V2luT3B0AgExCQACAQIhV2lubmVyIGlzICdOTycgLSBub3RoaW5nIHRvIGFwcGx5AwkAAAIJAJEDAgULZHluYW1pY0RhdGEFFWdvdlN0YXR1c0lkeFNjQXBwbGllZAIEdHJ1ZQkAAgECG1NjcmlwdHMgYXJlIGFscmVhZHkgYXBwbGllZAQKc2NyaXB0VGltZQkBDXBhcnNlSW50VmFsdWUBCQCRAwIFC2R5bmFtaWNEYXRhBRJnb3ZTdGF0dXNJZHhTY1RpbWUDCQBmAgkAZQIFA25vdwUKUEFTVE1BUkdJTgUKc2NyaXB0VGltZQkAAgEJAKwCAgkArAICAhJTY3JpcHRzIHRpbWVzdGFtcD0JAKQDAQUKc2NyaXB0VGltZQIqIGlzIHRvbyBmYXIgaW4gdGhlIHBhc3QsIG1heCAyIGhycyBhbGxvd2VkAwkAZgIFCnNjcmlwdFRpbWUJAGQCBQNub3cFDEZVVFVSRU1BUkdJTgkAAgEJAKwCAgkArAICAhJTY3JpcHRzIHRpbWVzdGFtcD0JAKQDAQUKc2NyaXB0VGltZQIuIGlzIHRvbyBmYXIgaW4gdGhlIGZ1dHVyZSwgbWF4IDEuNSBocnMgYWxsb3dlZAQMaW5Qcm9ncmVzc0lkCQEMZ2V0SW50T3JFbHNlAgkBEmtleUFwcGx5SW5Qcm9ncmVzcwAA////////////AQMJAQIhPQIFDGluUHJvZ3Jlc3NJZAD///////////8BCQACAQkArAICCQCsAgICC3Byb3Bvc2FsSWQ9CQCkAwEFDGluUHJvZ3Jlc3NJZAIrIGlzIGFscmVhZHkgYmVpbmcgYXBwbGllZC4gRmluaXNoIGl0IGZpcnN0IQQIc2h1dGRvd24JAPwHBAUPY29udHJvbENvbnRyYWN0AhVjYWxsRW1lcmdlbmN5U2h1dGRvd24JAMwIAgIaQXBwbHlpbmcgR292ZXJuYW5jZSBVUERBVEUFA25pbAUDbmlsAwkAAAIFCHNodXRkb3duBQhzaHV0ZG93bgkAlAoCCQDMCAIJAQxJbnRlZ2VyRW50cnkCCQESa2V5QXBwbHlJblByb2dyZXNzAAUKcHJvcG9zYWxJZAUDbmlsBQR1bml0CQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBC2ZpbmlzaEFwcGx5AAQKcHJvcG9zYWxJZAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCfCAEJARJrZXlBcHBseUluUHJvZ3Jlc3MAAidObyBhcHBseSBpbiBwcm9ncmVzcywgbm90aGluZyB0byBmaW5pc2gECHByb3BEYXRhCQC1CQIJAQ9nZXRTdHJpbmdPckZhaWwCBQR0aGlzCQETa2V5UHJvcG9zYWxEYXRhQnlJZAEFCnByb3Bvc2FsSWQFA1NFUAQGdHhMaXN0CQC1CQIJAJEDAgUIcHJvcERhdGEFC2dvdklkeFR4SWRzBQdMSVNUU0VQBAtkeW5hbWljRGF0YQkAtQkCCQEPZ2V0U3RyaW5nT3JGYWlsAgUEdGhpcwkBGWtleVByb3Bvc2FsU3RhdHVzRGF0YUJ5SWQBBQpwcm9wb3NhbElkBQNTRVADCQAAAgkAkQMCBQtkeW5hbWljRGF0YQUVZ292U3RhdHVzSWR4U2NBcHBsaWVkAgR0cnVlCQACAQIbU2NyaXB0cyBhcmUgYWxyZWFkeSBhcHBsaWVkCgEHY2hlY2tlcgIDYWNjAnR4AwkBASEBCQEJaXNEZWZpbmVkAQkA6QcBCQDZBAEFAnR4CQACAQkArAICAhJOT1QgYXBwbGllZCB0eElkOiAFAnR4BQR1bml0BAdpZ25vcmVkCgACJGwFBnR4TGlzdAoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFBHVuaXQKAQUkZjBfMQICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkBB2NoZWNrZXICBQIkYQkAkQMCBQIkbAUCJGkKAQUkZjBfMgICJGECJGkDCQBnAgUCJGkFAiRzBQIkYQkAAgECFExpc3Qgc2l6ZSBleGNlZWRzIDIwCQEFJGYwXzICCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECCQEFJGYwXzECBQUkYWNjMAAAAAEAAgADAAQABQAGAAcACAAJAAoACwAMAA0ADgAPABAAEQASABMAFAQHdmVyc2lvbgkAZAIJAQxnZXRJbnRPckVsc2UCCQEUa2V5TGFzdFVwZGF0ZVZlcnNpb24AAAAAAQkAlAoCCQDMCAIJAQtEZWxldGVFbnRyeQEJARJrZXlBcHBseUluUHJvZ3Jlc3MACQDMCAIJAQxJbnRlZ2VyRW50cnkCCQEUa2V5TGFzdFVwZGF0ZVZlcnNpb24ABQd2ZXJzaW9uCQDMCAIJAQtTdHJpbmdFbnRyeQIJARlrZXlQcm9wb3NhbFN0YXR1c0RhdGFCeUlkAQUKcHJvcG9zYWxJZAkBEXN0YXR1c0FwcGx5U2NyaXB0AQULZHluYW1pY0RhdGEJAMwIAgkBEEV4ZWN1dGlvbkhpc3RvcnkDBQd2ZXJzaW9uCQCRAwIFCHByb3BEYXRhBQtnb3ZJZHhUaXRsZQkAkQMCBQhwcm9wRGF0YQUJZ292SWR4VXJsBQNuaWwFB2lnbm9yZWQAFx3mRg==", "height": 2331129, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 4143eFgP3jkCfZ5wMJFHanTcR1H5ymyoRibdGtpXcTJs Next: 8xrWaRC9k2KAeGEBtUmfT1AhnrpSR59BDpXhkXVf5Zzf Diff:
OldNewDifferences
338338
339339
340340 @Callable(i)
341+func constructor (controlAddr,gNsbtReqToInit,wavesReqToInit,quorumReqPerc) = if ((i.caller != this))
342+ then throw("Permission denied")
343+ else [StringEntry(keyControlAddress(), controlAddr), IntegerEntry(keyGnsbtRequired(), gNsbtReqToInit), IntegerEntry(keyPaymentRequired(), wavesReqToInit), IntegerEntry(keyQuorumRequiredPercent(), quorumReqPerc)]
344+
345+
346+
347+@Callable(i)
341348 func castVote (proposalId,choice) = {
342349 let userAddressStr = toString(i.caller)
343350 let dynamicData = split(getStringOrFail(this, keyProposalStatusDataById(proposalId)), SEP)
Full:
OldNewDifferences
11 {-# STDLIB_VERSION 6 #-}
22 {-# SCRIPT_TYPE ACCOUNT #-}
33 {-# CONTENT_TYPE DAPP #-}
44 let revisionNum = ""
55
66 let SEP = "__"
77
88 let LISTSEP = ":"
99
1010 let DEFAULTQUORUM = 500000
1111
1212 let URLPATTERN = "https://forum.neutrino.at/"
1313
1414 let MAXTITLE = 160
1515
1616 let MAXURL = 250
1717
1818 let MAXVOTINGTIME = 1209600000
1919
2020 let MULT6 = 1000000
2121
2222 let DEFAULTPAYMENT = 1000000000
2323
2424 let DEFAULTCREATIONGNSBT = 1000000000
2525
2626 let PASTMARGIN = 7200000
2727
2828 let FUTUREMARGIN = 5400000
2929
3030 let govIdxProposalTxId = 1
3131
3232 let govIdxType = 2
3333
3434 let govIdxAuthor = 3
3535
3636 let govIdxUrl = 4
3737
3838 let govIdxTitle = 5
3939
4040 let govIdxCreationTime = 6
4141
4242 let govIdxStart = 7
4343
4444 let govIdxEnd = 8
4545
4646 let govIdxTxIds = 9
4747
4848 let govIdxQuorum = 10
4949
5050 let govIdxOptions = 11
5151
5252 let govStatusIdxIsValid = 1
5353
5454 let govStatusIdxWinOpt = 2
5555
5656 let govStatusIdxWinVotes = 3
5757
5858 let govStatusIdxTotalVotes = 4
5959
6060 let govStatusIdxScApplied = 5
6161
6262 let govStatusIdxScTime = 6
6363
6464 let govStatusIdxIsCanceled = 7
6565
6666 func getStringOrFail (address,key) = valueOrErrorMessage(getString(address, key), makeString(["mandatory ", toString(address), ".", key, " is not defined"], ""))
6767
6868
6969 func getIntOrElse (key,defaultVal) = valueOrElse(getInteger(this, key), defaultVal)
7070
7171
7272 let IdxControlCfgNeutrinoDapp = 1
7373
7474 let IdxControlCfgAuctionDapp = 2
7575
7676 let IdxControlCfgRpdDapp = 3
7777
7878 let IdxControlCfgMathDapp = 4
7979
8080 let IdxControlCfgLiquidationDapp = 5
8181
8282 let IdxControlCfgRestDapp = 6
8383
8484 let IdxControlCfgNodeRegistryDapp = 7
8585
8686 let IdxControlCfgNsbtStakingDapp = 8
8787
8888 let IdxControlCfgMediatorDapp = 9
8989
9090 let IdxControlCfgSurfStakingDapp = 10
9191
9292 let IdxControlCfgGnsbtControllerDapp = 11
9393
9494 let IdxControlCfgRestV2Dapp = 12
9595
9696 let IdxControlCfgGovernanceDapp = 13
9797
9898 func keyControlAddress () = "%s%s__config__controlAddress"
9999
100100
101101 func keyControlCfg () = "%s__controlConfig"
102102
103103
104104 func readControlCfgOrFail (control) = split_4C(getStringOrFail(control, keyControlCfg()), SEP)
105105
106106
107107 func getContractAddressOrFail (controlCfg,idx) = valueOrErrorMessage(addressFromString(controlCfg[idx]), ("Control cfg doesn't contain address at index " + toString(idx)))
108108
109109
110110 let controlContract = addressFromStringValue(valueOrElse(getString(this, keyControlAddress()), "3P5Bfd58PPfNvBM2Hy8QfbcDqMeNtzg7KfP"))
111111
112112 let controlCfg = readControlCfgOrFail(controlContract)
113113
114114 let neutrinoContract = getContractAddressOrFail(controlCfg, IdxControlCfgNeutrinoDapp)
115115
116116 let gnsbtControllerContract = getContractAddressOrFail(controlCfg, IdxControlCfgGnsbtControllerDapp)
117117
118118 func keyQuorumRequiredPercent () = "%s__quorumRequired"
119119
120120
121121 func keyPaymentRequired () = "%s__paymentRequired"
122122
123123
124124 func keyGnsbtRequired () = "%s__gNsbtRequired"
125125
126126
127127 func keyLastProposalId () = "%s__proposalId"
128128
129129
130130 func keyLastUpdateVersion () = "%s__updateVersion"
131131
132132
133133 func keyProposalStatusDataById (proposalId) = ("%s%d__proposalStatusData__" + toString(proposalId))
134134
135135
136136 func keyProposalDataById (proposalId) = ("%s%d__proposalData__" + toString(proposalId))
137137
138138
139139 func keyProposalVotesByIdAndOption (proposalId,opt) = makeString(["%s%d%d", "votesByOpt", toString(proposalId), toString(opt)], SEP)
140140
141141
142142 func keyProposalVotesByIdAndUser (proposalId,userAddr) = makeString(["%s%d%s", "votesByUser", toString(proposalId), userAddr], SEP)
143143
144144
145145 func keyProposalChoiceByIdAndUser (proposalId,userAddr) = makeString(["%s%d%s", "optionByUser", toString(proposalId), userAddr], SEP)
146146
147147
148148 func keyApplyInProgress () = "%s__applyInProgress"
149149
150150
151151 func keyProposalIdByUrl (forumLink) = ("%s%s__proposalIdByUrl__" + toBase64String(toBytes(forumLink)))
152152
153153
154154 func keyNumUniqueVotersByProposalId (proposalId) = ("%s%d__numVoters__" + toString(proposalId))
155155
156156
157157 func keyStatsAverUniqueVoters () = "%s%s%s__stats__avg__uniqueVoters"
158158
159159
160160 func keyStatsAverGnsbtVoted () = "%s%s%s__stats__avg__gnsbtVoted"
161161
162162
163163 func keyStatsUniqueAuthors () = "%s%s__stats__uniqueAuthors"
164164
165165
166166 func keyNumProposalsByAuthor (addressStr) = ("%s%s__numProposalsByAuthor__" + addressStr)
167167
168168
169169 func keyApplyHistory (timestamp) = ("%s%d__applyHistory__" + toString(timestamp))
170170
171171
172172 func asAnyList (v) = match v {
173173 case l: List[Any] =>
174174 l
175175 case _ =>
176176 throw("fail to cast into List[Any]")
177177 }
178178
179179
180180 func asInt (v) = match v {
181181 case i: Int =>
182182 i
183183 case _ =>
184184 throw("fail to cast into Int")
185185 }
186186
187187
188188 func statusData (isVotingValid,winOption,winOptionVotes,totalVotes,areScriptsApplied,scriptsTimestamp,canceledByTeam) = makeString(["%b%d%d%d%b%d%b", toString(isVotingValid), toString(winOption), toString(winOptionVotes), toString(totalVotes), toString(areScriptsApplied), toString(scriptsTimestamp), toString(canceledByTeam)], SEP)
189189
190190
191191 func proposalData (proposalTxId,type,author,forumLink,title,proposalTime,votingStartTime,votingEndTime,txIds,quorumInGnsbt,options) = makeString(["%s%s%s%s%s%d%d%d%s%d%s", proposalTxId, type, author, forumLink, title, toString(proposalTime), toString(votingStartTime), toString(votingEndTime), txIds, toString(quorumInGnsbt), options], SEP)
192192
193193
194194 func checkTxList (txList) = if ((size(txList) > 20))
195195 then throw(("Too many transactions: " + toString(size(txList))))
196196 else {
197197 func combiner (acc,tx) = if ((size(fromBase58String(tx)) != 32))
198198 then throw(("Wrong txId: " + tx))
199199 else if ((acc == ""))
200200 then tx
201201 else ((acc + LISTSEP) + tx)
202202
203203 let $l = txList
204204 let $s = size($l)
205205 let $acc0 = ""
206206 func $f0_1 ($a,$i) = if (($i >= $s))
207207 then $a
208208 else combiner($a, $l[$i])
209209
210210 func $f0_2 ($a,$i) = if (($i >= $s))
211211 then $a
212212 else throw("List size exceeds 20")
213213
214214 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20)
215215 }
216216
217217
218218 let IdxEffTotal = 0
219219
220220 let IdxEffUser = 1
221221
222222 func getEffectiveGnsbt (userAddrStrOrEmpty) = {
223223 let gnsbtData = asAnyList(invoke(gnsbtControllerContract, "gnsbtInfoSYSREADONLY", [userAddrStrOrEmpty, 0, 0], nil))
224224 let nsbtData = asAnyList(gnsbtData[2])
225225 let userFromNsbt = asInt(nsbtData[2])
226226 let totalFromNsbt = asInt(nsbtData[3])
227227 let userMatureFromSurf = asInt(gnsbtData[9])
228228 let totalMatureFromSurf = asInt(gnsbtData[6])
229229 [(totalFromNsbt + totalMatureFromSurf), (userFromNsbt + userMatureFromSurf)]
230230 }
231231
232232
233233 func initiateVoting (payment,proposalTxId,type,author,forumLink,title,votingStartTime,votingEndTime,status,txList,optionsList) = if ((payment.assetId != unit))
234234 then throw("Allowed WAVES payment only!")
235235 else {
236236 let pmtReq = getIntOrElse(keyPaymentRequired(), DEFAULTPAYMENT)
237237 if ((pmtReq > payment.amount))
238238 then throw(("Payment attached should be at least " + toString(pmtReq)))
239239 else if ((value(indexOf(forumLink, URLPATTERN)) != 0))
240240 then throw("Invalid url")
241241 else if ((size(forumLink) > MAXURL))
242242 then throw("Url too long!")
243243 else {
244244 let registeredId = getInteger(keyProposalIdByUrl(forumLink))
245245 if (isDefined(registeredId))
246246 then throw(("Voting with such forum link is already registered by id=" + toString(value(registeredId))))
247247 else if ((title == ""))
248248 then throw("Title is empty")
249249 else if ((size(title) > MAXTITLE))
250250 then throw("Too long title")
251251 else {
252252 let proposalTime = lastBlock.timestamp
253253 if ((proposalTime > votingStartTime))
254254 then throw(((("votingStartTime=" + toString(votingStartTime)) + " < proposalTime=") + toString(proposalTime)))
255255 else if ((votingStartTime > votingEndTime))
256256 then throw(((("votingEndTime=" + toString(votingEndTime)) + " < votingStartTime=") + toString(votingStartTime)))
257257 else if (((votingEndTime - votingStartTime) > MAXVOTINGTIME))
258258 then throw(((("Voting period exceeds max: " + toString((votingEndTime - votingStartTime))) + " > ") + toString(MAXVOTINGTIME)))
259259 else {
260260 let txIds = if ((type == "IDEA"))
261261 then ""
262262 else checkTxList(txList)
263263 if ((1 >= size(optionsList)))
264264 then throw("Too few choices to vote")
265265 else {
266266 let eff = getEffectiveGnsbt(author)
267267 let gnsbtTotal = eff[IdxEffTotal]
268268 let gNsbtUser = eff[IdxEffUser]
269269 let gnsbtReq = getIntOrElse(keyGnsbtRequired(), DEFAULTCREATIONGNSBT)
270270 if ((gnsbtReq > gNsbtUser))
271271 then throw((("You need at least " + toString(gnsbtReq)) + " gNsbt to create voting"))
272272 else {
273273 let amountLeased = invoke(neutrinoContract, "acceptWaves", nil, [payment])
274274 if ((amountLeased == amountLeased))
275275 then {
276276 let quorum = getIntOrElse(keyQuorumRequiredPercent(), DEFAULTQUORUM)
277277 let quorumInGnsbt = fraction(quorum, gnsbtTotal, MULT6)
278278 let proposalId = (getIntOrElse(keyLastProposalId(), 0) + 1)
279279 let numProposalsByAuthor = (getIntOrElse(keyNumProposalsByAuthor(author), 0) + 1)
280280 let uniqAuthors = (getIntOrElse(keyStatsUniqueAuthors(), 0) + (if ((numProposalsByAuthor == 1))
281281 then 1
282282 else 0))
283283 let optionsStr = makeString(optionsList, LISTSEP)
284284 $Tuple2([IntegerEntry(keyLastProposalId(), proposalId), IntegerEntry(keyProposalIdByUrl(forumLink), proposalId), StringEntry(keyProposalStatusDataById(proposalId), statusData(false, 0, 0, 0, false, 0, false)), StringEntry(keyProposalDataById(proposalId), proposalData(proposalTxId, type, author, forumLink, title, proposalTime, votingStartTime, votingEndTime, txIds, quorumInGnsbt, optionsStr)), IntegerEntry(keyNumProposalsByAuthor(author), numProposalsByAuthor), IntegerEntry(keyStatsUniqueAuthors(), uniqAuthors)], proposalTxId)
285285 }
286286 else throw("Strict value is not equal to itself.")
287287 }
288288 }
289289 }
290290 }
291291 }
292292 }
293293
294294
295295 func calcWinOption (proposalId,optionsList,isPrevOptional,oldChoice,optionalTotalOld,newChoice,newTotalByNewChoice) = {
296296 func findBest (acc,elem) = {
297297 let idx = value(indexOf(optionsList, elem))
298298 let val = if (isPrevOptional)
299299 then if ((idx == newChoice))
300300 then newTotalByNewChoice
301301 else getIntOrElse(keyProposalVotesByIdAndOption(proposalId, idx), 0)
302302 else if ((idx == value(oldChoice)))
303303 then optionalTotalOld
304304 else if ((idx == newChoice))
305305 then newTotalByNewChoice
306306 else getIntOrElse(keyProposalVotesByIdAndOption(proposalId, idx), 0)
307307 if ((acc._2 > val))
308308 then acc
309309 else $Tuple2(idx, val)
310310 }
311311
312312 let $l = optionsList
313313 let $s = size($l)
314314 let $acc0 = $Tuple2(0, 0)
315315 func $f0_1 ($a,$i) = if (($i >= $s))
316316 then $a
317317 else findBest($a, $l[$i])
318318
319319 func $f0_2 ($a,$i) = if (($i >= $s))
320320 then $a
321321 else throw("List size exceeds 10")
322322
323323 $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)
324324 }
325325
326326
327327 func updateStatusData (oldData,isValid,newWinOpt,newTotalVotes) = makeString(["%b%d%d%d%b%d%b", toString(isValid), toString(newWinOpt._1), toString(newWinOpt._2), toString(newTotalVotes), oldData[govStatusIdxScApplied], oldData[govStatusIdxScTime], oldData[govStatusIdxIsCanceled]], SEP)
328328
329329
330330 func statusApplyScript (oldData) = makeString(["%b%d%d%d%b%d%b", oldData[govStatusIdxIsValid], oldData[govStatusIdxWinOpt], oldData[govStatusIdxWinVotes], oldData[govStatusIdxTotalVotes], "true", oldData[govStatusIdxScTime], oldData[govStatusIdxIsCanceled]], SEP)
331331
332332
333333 func ExecutionHistory (proposalId,title,url) = {
334334 let gnsbtTotal = getEffectiveGnsbt("")[IdxEffTotal]
335335 let turnout = 500000
336336 StringEntry(keyApplyHistory(lastBlock.timestamp), makeString(["%d%d%d%s%s", toString(proposalId), toString(gnsbtTotal), toString(turnout), title, url], SEP))
337337 }
338338
339339
340340 @Callable(i)
341+func constructor (controlAddr,gNsbtReqToInit,wavesReqToInit,quorumReqPerc) = if ((i.caller != this))
342+ then throw("Permission denied")
343+ else [StringEntry(keyControlAddress(), controlAddr), IntegerEntry(keyGnsbtRequired(), gNsbtReqToInit), IntegerEntry(keyPaymentRequired(), wavesReqToInit), IntegerEntry(keyQuorumRequiredPercent(), quorumReqPerc)]
344+
345+
346+
347+@Callable(i)
341348 func castVote (proposalId,choice) = {
342349 let userAddressStr = toString(i.caller)
343350 let dynamicData = split(getStringOrFail(this, keyProposalStatusDataById(proposalId)), SEP)
344351 if ((dynamicData[govStatusIdxIsCanceled] == "true"))
345352 then throw("Voting is canceled by team")
346353 else {
347354 let propData = split(getStringOrFail(this, keyProposalDataById(proposalId)), SEP)
348355 let start = parseIntValue(propData[govIdxStart])
349356 let end = parseIntValue(propData[govIdxEnd])
350357 let now = lastBlock.timestamp
351358 if ((start > now))
352359 then throw("Voting not started yet")
353360 else if ((now >= end))
354361 then throw("Voting already finished")
355362 else {
356363 let availableOptions = split(propData[govIdxOptions], LISTSEP)
357364 let numOptions = size(availableOptions)
358365 if ((1 >= numOptions))
359366 then throw("Too few choices to vote")
360367 else if ((choice >= numOptions))
361368 then throw(("Unknown choice! Must be 0.." + toString((numOptions - 1))))
362369 else {
363370 let eff = getEffectiveGnsbt(userAddressStr)
364371 let gnsbtAmt = eff[IdxEffUser]
365372 if ((0 >= gnsbtAmt))
366373 then throw("no gnsbt to vote")
367374 else {
368375 let gnsbtTotal = eff[IdxEffTotal]
369376 let oldChoice = getInteger(keyProposalChoiceByIdAndUser(proposalId, userAddressStr))
370377 let oldUserVotes = if (!(isDefined(oldChoice)))
371378 then 0
372379 else getIntOrElse(keyProposalVotesByIdAndUser(proposalId, userAddressStr), 0)
373380 let oldTotalByOldChoice = if (isDefined(oldChoice))
374381 then getIntOrElse(keyProposalVotesByIdAndOption(proposalId, value(oldChoice)), 0)
375382 else 0
376383 let oldTotalByNewChoice = getIntOrElse(keyProposalVotesByIdAndOption(proposalId, choice), 0)
377384 let oldTotal = parseIntValue(dynamicData[govStatusIdxTotalVotes])
378385 let newTotalByOldChoice = if (!(isDefined(oldChoice)))
379386 then 0
380387 else ((oldTotalByOldChoice - oldUserVotes) + (if ((value(oldChoice) == choice))
381388 then gnsbtAmt
382389 else 0))
383390 let newTotalByNewChoice = if (if (isDefined(oldChoice))
384391 then (value(oldChoice) == choice)
385392 else false)
386393 then newTotalByOldChoice
387394 else (oldTotalByNewChoice + gnsbtAmt)
388395 let newTotal = ((oldTotal - oldUserVotes) + gnsbtAmt)
389396 let isQuorumReached = (newTotal >= parseIntValue(propData[govIdxQuorum]))
390397 let numVotersByProposalId = getIntOrElse(keyNumUniqueVotersByProposalId(proposalId), 0)
391398 let oldAverUniqueVoters6 = getIntOrElse(keyStatsAverUniqueVoters(), 0)
392399 let numProposals = getIntegerValue(keyLastProposalId())
393400 let uniqueDiff = if ((oldUserVotes == 0))
394401 then 1
395402 else 0
396403 let newAverUniqueVoters6 = (oldAverUniqueVoters6 + fraction(uniqueDiff, MULT6, numProposals))
397404 let oldAverGnsbt = getIntOrElse(keyStatsAverGnsbtVoted(), 0)
398405 let newAverGnsbt = (oldAverGnsbt + ((gnsbtAmt - oldUserVotes) / numProposals))
399406 let isPrevOptional = if (!(isDefined(oldChoice)))
400407 then true
401408 else (value(oldChoice) == choice)
402409 let optionalTotalOld = if (isPrevOptional)
403410 then nil
404411 else [IntegerEntry(keyProposalVotesByIdAndOption(proposalId, value(oldChoice)), newTotalByOldChoice)]
405412 let winOpt = calcWinOption(proposalId, availableOptions, isPrevOptional, oldChoice, newTotalByOldChoice, choice, newTotalByNewChoice)
406413 $Tuple2(([IntegerEntry(keyProposalChoiceByIdAndUser(proposalId, userAddressStr), choice), IntegerEntry(keyProposalVotesByIdAndUser(proposalId, userAddressStr), gnsbtAmt), IntegerEntry(keyProposalVotesByIdAndOption(proposalId, choice), newTotalByNewChoice), IntegerEntry(keyNumUniqueVotersByProposalId(proposalId), (numVotersByProposalId + uniqueDiff)), IntegerEntry(keyStatsAverUniqueVoters(), newAverUniqueVoters6), IntegerEntry(keyStatsAverGnsbtVoted(), newAverGnsbt), StringEntry(keyProposalStatusDataById(proposalId), updateStatusData(dynamicData, isQuorumReached, winOpt, newTotal))] ++ optionalTotalOld), unit)
407414 }
408415 }
409416 }
410417 }
411418 }
412419
413420
414421
415422 @Callable(i)
416423 func initiateIdeaVoting (forumLink,title,votingStartTime,votingEndTime,optionsList) = if ((size(i.payments) != 1))
417424 then throw("Exactly one payment required")
418425 else if ((size(optionsList) != 2))
419426 then throw("Exactly 2 option ['NO', 'YES'] are expected")
420427 else if ((optionsList[0] != "NO"))
421428 then throw("Option NO should be the first")
422429 else if ((optionsList[1] != "YES"))
423430 then throw("Option YES should be the second")
424431 else initiateVoting(value(i.payments[0]), toBase58String(i.transactionId), "IDEA", toString(i.caller), forumLink, title, votingStartTime, votingEndTime, "PENDING", nil, optionsList)
425432
426433
427434
428435 @Callable(i)
429436 func initiateUpdateVoting (forumLink,title,votingStartTime,votingEndTime,txList) = if ((size(i.payments) != 1))
430437 then throw("Exactly one payment required")
431438 else if ((1 > size(txList)))
432439 then throw("Transactions list is empty")
433440 else if ((i.caller != this))
434441 then throw("not authorized")
435442 else initiateVoting(value(i.payments[0]), toBase58String(i.transactionId), "UPDATE", toString(i.caller), forumLink, title, votingStartTime, votingEndTime, "PENDING", txList, ["NO", "YES"])
436443
437444
438445
439446 @Callable(i)
440447 func cancelVoting (proposalId) = if ((i.caller != this))
441448 then throw("not authorized")
442449 else {
443450 let currentData = getStringOrFail(this, keyProposalStatusDataById(proposalId))
444451 let updatedData = ((take(currentData, value(lastIndexOf(currentData, SEP))) + SEP) + "true")
445452 $Tuple2([StringEntry(keyProposalStatusDataById(proposalId), updatedData)], unit)
446453 }
447454
448455
449456
450457 @Callable(i)
451458 func applyUpdate (proposalId) = {
452459 let propData = split(getStringOrFail(this, keyProposalDataById(proposalId)), SEP)
453460 let end = parseIntValue(propData[govIdxEnd])
454461 let now = lastBlock.timestamp
455462 if ((end > now))
456463 then throw("Voting is not finished yet")
457464 else if (("UPDATE" != propData[govIdxType]))
458465 then throw("Only UPDATE type can be applied")
459466 else {
460467 let dynamicData = split(getStringOrFail(this, keyProposalStatusDataById(proposalId)), SEP)
461468 if ((dynamicData[govStatusIdxIsCanceled] == "true"))
462469 then throw("Voting is canceled")
463470 else if ((dynamicData[govStatusIdxIsValid] != "true"))
464471 then throw("Voting status invalid")
465472 else if ((dynamicData[govStatusIdxWinOpt] != "1"))
466473 then throw("Winner is 'NO' - nothing to apply")
467474 else if ((dynamicData[govStatusIdxScApplied] == "true"))
468475 then throw("Scripts are already applied")
469476 else {
470477 let scriptTime = parseIntValue(dynamicData[govStatusIdxScTime])
471478 if (((now - PASTMARGIN) > scriptTime))
472479 then throw((("Scripts timestamp=" + toString(scriptTime)) + " is too far in the past, max 2 hrs allowed"))
473480 else if ((scriptTime > (now + FUTUREMARGIN)))
474481 then throw((("Scripts timestamp=" + toString(scriptTime)) + " is too far in the future, max 1.5 hrs allowed"))
475482 else {
476483 let inProgressId = getIntOrElse(keyApplyInProgress(), -1)
477484 if ((inProgressId != -1))
478485 then throw((("proposalId=" + toString(inProgressId)) + " is already being applied. Finish it first!"))
479486 else {
480487 let shutdown = invoke(controlContract, "callEmergencyShutdown", ["Applying Governance UPDATE"], nil)
481488 if ((shutdown == shutdown))
482489 then $Tuple2([IntegerEntry(keyApplyInProgress(), proposalId)], unit)
483490 else throw("Strict value is not equal to itself.")
484491 }
485492 }
486493 }
487494 }
488495 }
489496
490497
491498
492499 @Callable(i)
493500 func finishApply () = {
494501 let proposalId = valueOrErrorMessage(getInteger(keyApplyInProgress()), "No apply in progress, nothing to finish")
495502 let propData = split(getStringOrFail(this, keyProposalDataById(proposalId)), SEP)
496503 let txList = split(propData[govIdxTxIds], LISTSEP)
497504 let dynamicData = split(getStringOrFail(this, keyProposalStatusDataById(proposalId)), SEP)
498505 if ((dynamicData[govStatusIdxScApplied] == "true"))
499506 then throw("Scripts are already applied")
500507 else {
501508 func checker (acc,tx) = if (!(isDefined(transactionHeightById(fromBase58String(tx)))))
502509 then throw(("NOT applied txId: " + tx))
503510 else unit
504511
505512 let ignored = {
506513 let $l = txList
507514 let $s = size($l)
508515 let $acc0 = unit
509516 func $f0_1 ($a,$i) = if (($i >= $s))
510517 then $a
511518 else checker($a, $l[$i])
512519
513520 func $f0_2 ($a,$i) = if (($i >= $s))
514521 then $a
515522 else throw("List size exceeds 20")
516523
517524 $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20)
518525 }
519526 let version = (getIntOrElse(keyLastUpdateVersion(), 0) + 1)
520527 $Tuple2([DeleteEntry(keyApplyInProgress()), IntegerEntry(keyLastUpdateVersion(), version), StringEntry(keyProposalStatusDataById(proposalId), statusApplyScript(dynamicData)), ExecutionHistory(version, propData[govIdxTitle], propData[govIdxUrl])], ignored)
521528 }
522529 }
523530
524531

github/deemru/w8io/169f3d6 
86.62 ms