tx · Czh1pFyoDSAcCkRqS1g6RJWqrmt7KHeRhAUA7SW6MBqg 3PGWhtvCSz1TRJhe3BJSrvQigrtKuyXsgma: -0.01000000 Waves 2023.03.30 17:42 [3578397] smart account 3PGWhtvCSz1TRJhe3BJSrvQigrtKuyXsgma > SELF 0.00000000 Waves
{ "type": 13, "id": "Czh1pFyoDSAcCkRqS1g6RJWqrmt7KHeRhAUA7SW6MBqg", "fee": 1000000, "feeAssetId": null, "timestamp": 1680187433859, "version": 1, "sender": "3PGWhtvCSz1TRJhe3BJSrvQigrtKuyXsgma", "senderPublicKey": "4MpZMd8gCopeSW5sZNQx6DnKTcckE7ckhX1QArMdUiyT", "proofs": [ "4tUa81tK4ZzK8uhDzSHFo5MDQZjnowsj29ZMM9xJjvXuhxymo1PdvPxE8hW6EoFwHaLN69BJj7Xb5QWmSHaJxCFA" ], "script": "base64:BgIWCAISAwoBCBIDCgEIEgMKAQgSAwoBCA4BC3dyaXRlU3RyaW5nAgNrZXkFdmFsdWUJAQtTdHJpbmdFbnRyeQIFA2tleQUFdmFsdWUBEHdyaXRlQ29uc3RTdHJpbmcCA2tleQV2YWx1ZQMJAQEhAQkBCWlzRGVmaW5lZAEJAJ0IAgUEdGhpcwUDa2V5CQELU3RyaW5nRW50cnkCBQNrZXkFBXZhbHVlCQACAQkArAICAhVhbHJlYWR5IGluaXRpYWxpemVkOiAFA2tleQANY29uZmlnQWRkcmVzcwUEdGhpcwAGSEVJR0hUBQZoZWlnaHQBBmFkdmlzZQAJAKwCAgIIaGVpZ2h0OiAJAKQDAQUGSEVJR0hUAANvcHMJAMwIAgITZ292ZXJuYW5jZV9hcHBseV90eAkAzAgCAgRtaW50CQDMCAICBXN0YWtlCQDMCAICCHN0YWtlRm9yCQDMCAICB3Vuc3Rha2UJAMwIAgILdW5zdGFrZUZyb20JAMwIAgIVZ292ZXJuYW5jZV9wcm9wb3NlX3R4CQDMCAICEmdvdmVybmFuY2Vfdm90ZV90eAkAzAgCAhdnb3Zlcm5hbmNlX3JldHJhY3Rfdm90ZQkAzAgCAh5nb3Zlcm5hbmNlX3JldHJhY3RfcHJvcG9zYWxfdHgJAMwIAgIHcHJvamVjdAkAzAgCAgZjb21taXQJAMwIAgIKdW5kb0NvbW1pdAkAzAgCAgpwcm9qZWN0Rm9yCQDMCAICDWluY3JlcmFzZVNraW4JAMwIAgIJY29tbWl0Rm9yCQDMCAICBmludmVzdAkAzAgCAgVzdGFydAkAzAgCAgVzbGFzaAkAzAgCAghjb21wbGV0ZQkAzAgCAgZjYW5jZWwJAMwIAgIKdW5kb0ludmVzdAkAzAgCAg11bmRvQ29tbWl0Rm9yCQDMCAICBXN0YXJ0CQDMCAICBXNsYXNoBQNuaWwAEnN5c3RlbUNvbnRyb2xTdG9yZQIOc3lzdGVtX2FsbG93ZWQBDm9wQWxsb3dlZFN0b3JlAQJvcAMJAQlpc0RlZmluZWQBCQDPCAIFA29wcwUCb3AJAKwCAgkArAICAgNvcF8FAm9wAghfYWxsb3dlZAkAAgEJAKwCAgkArAICAhZJbnZhbGlkIGFyZ3VtZW50OiBvcC4gBQJvcAIcIG9wZXJhdGlvbiBpcyBub3QgcmVnaXN0ZXJlZAEOb3BVbnBhdXNlU3RvcmUBAm9wCQCsAgIJAKwCAgIDb3BfBQJvcAIMX3VucGF1c2VkX2J5AA5zZW50aW5lbHNTdG9yZQIJc2VudGluZWxzAAxzZW50aW5lbHNTdHIJAJ0IAgUEdGhpcwUOc2VudGluZWxzU3RvcmUACXNlbnRpbmVscwMJAQlpc0RlZmluZWQBBQxzZW50aW5lbHNTdHIJALUJAgkBBXZhbHVlAQUMc2VudGluZWxzU3RyAgEsBQNuaWwBDmFzc2VydFNlbnRpbmVsAQFzCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAM8IAgUJc2VudGluZWxzCQClCAEFAXMCDnBhdXNlT3AgZGVuaWVkAQZxdW9ydW0ABANhbGwJAJADAQUJc2VudGluZWxzAwkAZgIAAwUDYWxsAAEJAGQCCQBpAgUDYWxsAAIAAQQBaQEJb3BBbGxvd2VkAQJvcAQFc3RvcmUJAQ5vcEFsbG93ZWRTdG9yZQEFAm9wAwkBCGNvbnRhaW5zAgkBC3ZhbHVlT3JFbHNlAgkAnQgCBQR0aGlzAghzdG9wbGlzdAIACQClCAEIBQFpDG9yaWdpbkNhbGxlcgkAAgECHG9wZXJhdGlvbiBwYXVzZWQgYnkgc3RvcGxpc3QDCQEIY29udGFpbnMCCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoaXMCA2V4YwIACQClCAEIBQFpDG9yaWdpbkNhbGxlcgkAlAoCBQNuaWwGBAhlbmFibGVkMAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzBRJzeXN0ZW1Db250cm9sU3RvcmUGBAhlbmFibGVkMQkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzBQVzdG9yZQYDCQEBIQEFCGVuYWJsZWQwCQACAQkArAICCQCsAgICFG9wZXJhdGlvbiBwYXVzZWQgYnkgBRJzeXN0ZW1Db250cm9sU3RvcmUCBj1mYWxzZQMJAQEhAQUIZW5hYmxlZDEJAAIBCQCsAgIJAKwCAgIUb3BlcmF0aW9uIHBhdXNlZCBieSAFBXN0b3JlAgY9ZmFsc2UJAJQKAgUDbmlsBgFpAQdwYXVzZU9wAQJvcAQBcwkBDmFzc2VydFNlbnRpbmVsAQgFAWkGY2FsbGVyAwkAAAIFAXMFAXMJAMwIAgkBDEJvb2xlYW5FbnRyeQIJAQ5vcEFsbG93ZWRTdG9yZQEFAm9wBwUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBCXVucGF1c2VPcAECb3AEAXMJAQ5hc3NlcnRTZW50aW5lbAEIBQFpBmNhbGxlcgMJAAACBQFzBQFzBAVzdG9yZQkBDm9wQWxsb3dlZFN0b3JlAQUCb3ADCQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMFBXN0b3JlBgkAAgECHHVucGF1c2VPcDogaW52YWxpZCBvcGVyYXRpb24ECHNlbnRpbmVsCQClCAEIBQFpBmNhbGxlcgQPdW5wYXVzZWRCeVN0b3JlCQEOb3BVbnBhdXNlU3RvcmUBBQJvcAQNdW5wYXVzZWRCeVN0cgkAnQgCBQR0aGlzBQ91bnBhdXNlZEJ5U3RvcmUEAXEJAQZxdW9ydW0AAwkBCWlzRGVmaW5lZAEFDXVucGF1c2VkQnlTdHIED3VucGF1c2VkQnlWYWx1ZQkBBXZhbHVlAQUNdW5wYXVzZWRCeVN0cgQKdW5wYXVzZWRCeQkAtQkCBQ91bnBhdXNlZEJ5VmFsdWUCASwDCQEJaXNEZWZpbmVkAQkAzwgCBQp1bnBhdXNlZEJ5BQhzZW50aW5lbAkAAgECEWR1cGxpY2F0ZSB1bnBhdXNlAwkAAAIJAGQCCQCQAwEFCnVucGF1c2VkQnkAAQUBcQkAzAgCCQELRGVsZXRlRW50cnkBBQVzdG9yZQkAzAgCCQELRGVsZXRlRW50cnkBBQ91bnBhdXNlZEJ5U3RvcmUFA25pbAkAzAgCCQELU3RyaW5nRW50cnkCBQ91bnBhdXNlZEJ5U3RvcmUJAKwCAgkArAICBQ91bnBhdXNlZEJ5VmFsdWUCASwFCHNlbnRpbmVsBQNuaWwDCQAAAgUBcQABCQDMCAIJAQtEZWxldGVFbnRyeQEFBXN0b3JlBQNuaWwJAMwIAgkBC1N0cmluZ0VudHJ5AgUPdW5wYXVzZWRCeVN0b3JlBQhzZW50aW5lbAUDbmlsCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAWkBC2FkZFNlbnRpbmVsAQhzZW50aW5lbAMJAQIhPQIIBQFpBmNhbGxlcgUEdGhpcwkAAgECGm9ubHkgc2VsZiBjYW4gYWRkIHNlbnRpbmVsAwkBASEBCQEJaXNEZWZpbmVkAQUMc2VudGluZWxzU3RyCQDMCAIJAQtTdHJpbmdFbnRyeQIFDnNlbnRpbmVsc1N0b3JlBQhzZW50aW5lbAUDbmlsAwkBCWlzRGVmaW5lZAEJAM8IAgUJc2VudGluZWxzBQhzZW50aW5lbAUDbmlsCQDMCAIJAQtTdHJpbmdFbnRyeQIFDnNlbnRpbmVsc1N0b3JlCQCsAgIJAKwCAgkBBXZhbHVlAQUMc2VudGluZWxzU3RyAgEsBQhzZW50aW5lbAUDbmlsAQJ0eAEGdmVyaWZ5AAQLdGhpc0FkZHJlc3MJAKUIAQUEdGhpcwQEQkFTRQDoBwQQdm90aW5nQWRkcmVzc1N0cgkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCdCAIFDWNvbmZpZ0FkZHJlc3MCD2NvbnRyYWN0X3ZvdGluZwIYdm90aW5nIGFkZHJlc3Mgbm90IGZvdW5kBA52b3RpbmdDb250cmFjdAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQ1jb25maWdBZGRyZXNzAg9jb250cmFjdF92b3RpbmcCEm5vIGNvbnRyYWN0X3ZvdGluZwIXaW52YWxpZCBjb250cmFjdF92b3RpbmcEC3F1b3J1bVJhdGlvCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNY29uZmlnQWRkcmVzcwkArAICBRB2b3RpbmdBZGRyZXNzU3RyAhZfcHJvcG9zYWxfcXVvcnVtX3JhdGlvAiJwcm9wb3NhbFF1b3J1bVJhdGlvIGlzIG5vdCBkZWZpbmVkBAtwYXNzZWRSYXRpbwkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCaCAIFDWNvbmZpZ0FkZHJlc3MJAKwCAgUQdm90aW5nQWRkcmVzc1N0cgIWX3Byb3Bvc2FsX3Bhc3NlZF9yYXRpbwIlcHJvcG9zYWxUaHJlc2hvbGRSYXRpbyBpcyBub3QgZGVmaW5lZAQCaWQJANgEAQgFAnR4AmlkBAh2b3Rlc1llcwkBC3ZhbHVlT3JFbHNlAgkAmggCBQ52b3RpbmdDb250cmFjdAkArAICAg1wcm9wb3NhbF95ZXNfBQJpZAAABAd2b3Rlc05vCQELdmFsdWVPckVsc2UCCQCaCAIFDnZvdGluZ0NvbnRyYWN0CQCsAgICDHByb3Bvc2FsX25vXwUCaWQAAAQOcHJvcG9zYWxIZWlnaHQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQ52b3RpbmdDb250cmFjdAkArAICAhBwcm9wb3NhbF9oZWlnaHRfBQJpZAIXcHJvcG9zYWwgbm90IHJlZ2lzdGVyZWQECmFwcGx5U3RhcnQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUOdm90aW5nQ29udHJhY3QJAKwCAgIUcHJvcG9zYWxfYXBwbHlzdGFydF8FAmlkAAAECGFwcGx5RW5kCQELdmFsdWVPckVsc2UCCQCaCAIFDnZvdGluZ0NvbnRyYWN0CQCsAgICEnByb3Bvc2FsX2FwcGx5ZW5kXwUCaWQAAAQKdG90YWxQb3dlcgkBC3ZhbHVlT3JFbHNlAgkAmggCBQ52b3RpbmdDb250cmFjdAkArAICAg9wcm9wb3NhbF9wb3dlcl8FAmlkAAAEB2VuYWJsZWQJAQt2YWx1ZU9yRWxzZQIJAJsIAgUNY29uZmlnQWRkcmVzcwkBDm9wQWxsb3dlZFN0b3JlAQITZ292ZXJuYW5jZV9hcHBseV90eAYEB3ZvdGVZZXMFCHZvdGVzWWVzBAZ2b3RlTm8FB3ZvdGVzTm8ECnRvdGFsVm90ZXMJAGQCBQd2b3RlWWVzBQZ2b3RlTm8ECWhhc1F1b3J1bQkAZwIJAGkCCQBoAgUKdG90YWxWb3RlcwUEQkFTRQUKdG90YWxQb3dlcgULcXVvcnVtUmF0aW8ECWhhc1Bhc3NlZAkAZwIJAGkCCQBoAgUHdm90ZVllcwUEQkFTRQUKdG90YWxWb3RlcwULcGFzc2VkUmF0aW8ECHRvb0Vhcmx5CQBnAgUKYXBwbHlTdGFydAUGSEVJR0hUBAd0b29MYXRlCQBnAgUGSEVJR0hUBQhhcHBseUVuZAQJdGltZURlYnVnCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICEXByb3Bvc2FsSGVpZ2h0ID0gCQCkAwEFDnByb3Bvc2FsSGVpZ2h0Ag8sIGFwcGx5U3RhcnQgPSAJAKQDAQUKYXBwbHlTdGFydAINLCBhcHBseUVuZCA9IAkApAMBBQhhcHBseUVuZAILLCBIRUlHSFQgPSAJAKQDAQUGSEVJR0hUBAhieVZvdGluZwMJAQEhAQUHZW5hYmxlZAkAAgECK3R4IGFwcGxpY2F0aW9uIHRocm93IGdvdmVybmFuY2Ugbm90IGVuYWJsZWQDBQh0b29FYXJseQkAAgEJAKwCAgIucHJvcG9zYWwgY2FuJ3QgYmUgZXhlY3V0ZWQgYXMgaXQncyB0b28gZWFybHk6IAUJdGltZURlYnVnAwUHdG9vTGF0ZQkAAgEJAKwCAgIscHJvcG9zYWwgY2FuJ3QgYmUgZXhlY3V0ZWQgYXMgaXQncyB0b28gbGF0ZToFCXRpbWVEZWJ1ZwMJAQEhAQUJaGFzUXVvcnVtCQACAQkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgILbm8gcXVvcnVtOiACDHRvdGFsVm90ZXM6IAkApAMBBQp0b3RhbFZvdGVzAg4sIHRvdGFsUG93ZXI6IAkApAMBBQp0b3RhbFBvd2VyAg8sIHF1b3J1bVJhdGlvOiAJAKQDAQULcXVvcnVtUmF0aW8DCQEBIQEFCWhhc1Bhc3NlZAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICF25vIHRocmVzaG9sZCBhY2hpZXZlZDogAgl2b3RlWWVzOiAJAKQDAQUHdm90ZVllcwIKLCB2b3RlTm86IAkApAMBBQZ2b3RlTm8CDywgcGFzc2VkUmF0aW86IAkApAMBBQtwYXNzZWRSYXRpbwYDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAACAUCdHgPc2VuZGVyUHVibGljS2V5BgUIYnlWb3RpbmegK8pD", "chainId": 87, "height": 3578397, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 8FJ3LdnK2ye2TZC4JFWiDT3Zvfk5dy84o7tde5zV3JYo Next: 4tAa5vcTppjzsLaGqrHWCzyo39w1C17tEwK5N6LtBMvq Diff:
Old | New | Differences | |
---|---|---|---|
16 | 16 | func advise () = ("height: " + toString(HEIGHT)) | |
17 | 17 | ||
18 | 18 | ||
19 | - | let opsStr = "stake_power|mint" | |
20 | - | ||
21 | - | let ops = split_4C(opsStr, "|") | |
19 | + | let ops = ["governance_apply_tx", "mint", "stake", "stakeFor", "unstake", "unstakeFrom", "governance_propose_tx", "governance_vote_tx", "governance_retract_vote", "governance_retract_proposal_tx", "project", "commit", "undoCommit", "projectFor", "increraseSkin", "commitFor", "invest", "start", "slash", "complete", "cancel", "undoInvest", "undoCommitFor", "start", "slash"] | |
22 | 20 | ||
23 | 21 | let systemControlStore = "system_allowed" | |
24 | 22 | ||
25 | - | func opAllowedStore (op) = (("op_" + op) + "_allowed") | |
23 | + | func opAllowedStore (op) = if (isDefined(indexOf(ops, op))) | |
24 | + | then (("op_" + op) + "_allowed") | |
25 | + | else throw((("Invalid argument: op. " + op) + " operation is not registered")) | |
26 | 26 | ||
27 | 27 | ||
28 | 28 | func opUnpauseStore (op) = (("op_" + op) + "_unpaused_by") | |
48 | 48 | ||
49 | 49 | ||
50 | 50 | @Callable(i) | |
51 | - | func opAllowed (op) = if (contains(valueOrElse(getString(this, "stoplist"), ""), toString(i.originCaller))) | |
52 | - | then throw("operation paused by stoplist") | |
53 | - | else if (contains(valueOrElse(getString(this, "exc"), ""), toString(i.originCaller))) | |
54 | - | then $Tuple2(nil, true) | |
55 | - | else { | |
56 | - | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
57 | - | let enabled1 = valueOrElse(getBoolean(this, opAllowedStore(op)), true) | |
58 | - | if (!(enabled0)) | |
59 | - | then throw((("operation paused by " + systemControlStore) + "=false")) | |
60 | - | else if (!(enabled1)) | |
61 | - | then throw((("operation paused by " + opAllowedStore(op)) + "=false")) | |
62 | - | else $Tuple2(nil, true) | |
63 | - | } | |
51 | + | func opAllowed (op) = { | |
52 | + | let store = opAllowedStore(op) | |
53 | + | if (contains(valueOrElse(getString(this, "stoplist"), ""), toString(i.originCaller))) | |
54 | + | then throw("operation paused by stoplist") | |
55 | + | else if (contains(valueOrElse(getString(this, "exc"), ""), toString(i.originCaller))) | |
56 | + | then $Tuple2(nil, true) | |
57 | + | else { | |
58 | + | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
59 | + | let enabled1 = valueOrElse(getBoolean(this, store), true) | |
60 | + | if (!(enabled0)) | |
61 | + | then throw((("operation paused by " + systemControlStore) + "=false")) | |
62 | + | else if (!(enabled1)) | |
63 | + | then throw((("operation paused by " + store) + "=false")) | |
64 | + | else $Tuple2(nil, true) | |
65 | + | } | |
66 | + | } | |
64 | 67 | ||
65 | 68 | ||
66 | 69 | ||
121 | 124 | func verify () = { | |
122 | 125 | let thisAddress = toString(this) | |
123 | 126 | let BASE = 1000 | |
124 | - | let stakingContract = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(configAddress, "contact_staking"), "no contact_staking")), "invalid contact_staking") | |
125 | 127 | let votingAddressStr = valueOrErrorMessage(getString(configAddress, "contract_voting"), "voting address not found") | |
126 | 128 | let votingContract = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(configAddress, "contract_voting"), "no contract_voting")), "invalid contract_voting") | |
127 | 129 | let quorumRatio = valueOrErrorMessage(getInteger(configAddress, (votingAddressStr + "_proposal_quorum_ratio")), "proposalQuorumRatio is not defined") |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | func writeString (key,value) = StringEntry(key, value) | |
5 | 5 | ||
6 | 6 | ||
7 | 7 | func writeConstString (key,value) = if (!(isDefined(getString(this, key)))) | |
8 | 8 | then StringEntry(key, value) | |
9 | 9 | else throw(("already initialized: " + key)) | |
10 | 10 | ||
11 | 11 | ||
12 | 12 | let configAddress = this | |
13 | 13 | ||
14 | 14 | let HEIGHT = height | |
15 | 15 | ||
16 | 16 | func advise () = ("height: " + toString(HEIGHT)) | |
17 | 17 | ||
18 | 18 | ||
19 | - | let opsStr = "stake_power|mint" | |
20 | - | ||
21 | - | let ops = split_4C(opsStr, "|") | |
19 | + | let ops = ["governance_apply_tx", "mint", "stake", "stakeFor", "unstake", "unstakeFrom", "governance_propose_tx", "governance_vote_tx", "governance_retract_vote", "governance_retract_proposal_tx", "project", "commit", "undoCommit", "projectFor", "increraseSkin", "commitFor", "invest", "start", "slash", "complete", "cancel", "undoInvest", "undoCommitFor", "start", "slash"] | |
22 | 20 | ||
23 | 21 | let systemControlStore = "system_allowed" | |
24 | 22 | ||
25 | - | func opAllowedStore (op) = (("op_" + op) + "_allowed") | |
23 | + | func opAllowedStore (op) = if (isDefined(indexOf(ops, op))) | |
24 | + | then (("op_" + op) + "_allowed") | |
25 | + | else throw((("Invalid argument: op. " + op) + " operation is not registered")) | |
26 | 26 | ||
27 | 27 | ||
28 | 28 | func opUnpauseStore (op) = (("op_" + op) + "_unpaused_by") | |
29 | 29 | ||
30 | 30 | ||
31 | 31 | let sentinelsStore = "sentinels" | |
32 | 32 | ||
33 | 33 | let sentinelsStr = getString(this, sentinelsStore) | |
34 | 34 | ||
35 | 35 | let sentinels = if (isDefined(sentinelsStr)) | |
36 | 36 | then split(value(sentinelsStr), ",") | |
37 | 37 | else nil | |
38 | 38 | ||
39 | 39 | func assertSentinel (s) = valueOrErrorMessage(indexOf(sentinels, toString(s)), "pauseOp denied") | |
40 | 40 | ||
41 | 41 | ||
42 | 42 | func quorum () = { | |
43 | 43 | let all = size(sentinels) | |
44 | 44 | if ((3 > all)) | |
45 | 45 | then 1 | |
46 | 46 | else ((all / 2) + 1) | |
47 | 47 | } | |
48 | 48 | ||
49 | 49 | ||
50 | 50 | @Callable(i) | |
51 | - | func opAllowed (op) = if (contains(valueOrElse(getString(this, "stoplist"), ""), toString(i.originCaller))) | |
52 | - | then throw("operation paused by stoplist") | |
53 | - | else if (contains(valueOrElse(getString(this, "exc"), ""), toString(i.originCaller))) | |
54 | - | then $Tuple2(nil, true) | |
55 | - | else { | |
56 | - | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
57 | - | let enabled1 = valueOrElse(getBoolean(this, opAllowedStore(op)), true) | |
58 | - | if (!(enabled0)) | |
59 | - | then throw((("operation paused by " + systemControlStore) + "=false")) | |
60 | - | else if (!(enabled1)) | |
61 | - | then throw((("operation paused by " + opAllowedStore(op)) + "=false")) | |
62 | - | else $Tuple2(nil, true) | |
63 | - | } | |
51 | + | func opAllowed (op) = { | |
52 | + | let store = opAllowedStore(op) | |
53 | + | if (contains(valueOrElse(getString(this, "stoplist"), ""), toString(i.originCaller))) | |
54 | + | then throw("operation paused by stoplist") | |
55 | + | else if (contains(valueOrElse(getString(this, "exc"), ""), toString(i.originCaller))) | |
56 | + | then $Tuple2(nil, true) | |
57 | + | else { | |
58 | + | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
59 | + | let enabled1 = valueOrElse(getBoolean(this, store), true) | |
60 | + | if (!(enabled0)) | |
61 | + | then throw((("operation paused by " + systemControlStore) + "=false")) | |
62 | + | else if (!(enabled1)) | |
63 | + | then throw((("operation paused by " + store) + "=false")) | |
64 | + | else $Tuple2(nil, true) | |
65 | + | } | |
66 | + | } | |
64 | 67 | ||
65 | 68 | ||
66 | 69 | ||
67 | 70 | @Callable(i) | |
68 | 71 | func pauseOp (op) = { | |
69 | 72 | let s = assertSentinel(i.caller) | |
70 | 73 | if ((s == s)) | |
71 | 74 | then [BooleanEntry(opAllowedStore(op), false)] | |
72 | 75 | else throw("Strict value is not equal to itself.") | |
73 | 76 | } | |
74 | 77 | ||
75 | 78 | ||
76 | 79 | ||
77 | 80 | @Callable(i) | |
78 | 81 | func unpauseOp (op) = { | |
79 | 82 | let s = assertSentinel(i.caller) | |
80 | 83 | if ((s == s)) | |
81 | 84 | then { | |
82 | 85 | let store = opAllowedStore(op) | |
83 | 86 | if (valueOrElse(getBoolean(this, store), true)) | |
84 | 87 | then throw("unpauseOp: invalid operation") | |
85 | 88 | else { | |
86 | 89 | let sentinel = toString(i.caller) | |
87 | 90 | let unpausedByStore = opUnpauseStore(op) | |
88 | 91 | let unpausedByStr = getString(this, unpausedByStore) | |
89 | 92 | let q = quorum() | |
90 | 93 | if (isDefined(unpausedByStr)) | |
91 | 94 | then { | |
92 | 95 | let unpausedByValue = value(unpausedByStr) | |
93 | 96 | let unpausedBy = split(unpausedByValue, ",") | |
94 | 97 | if (isDefined(indexOf(unpausedBy, sentinel))) | |
95 | 98 | then throw("duplicate unpause") | |
96 | 99 | else if (((size(unpausedBy) + 1) == q)) | |
97 | 100 | then [DeleteEntry(store), DeleteEntry(unpausedByStore)] | |
98 | 101 | else [StringEntry(unpausedByStore, ((unpausedByValue + ",") + sentinel))] | |
99 | 102 | } | |
100 | 103 | else if ((q == 1)) | |
101 | 104 | then [DeleteEntry(store)] | |
102 | 105 | else [StringEntry(unpausedByStore, sentinel)] | |
103 | 106 | } | |
104 | 107 | } | |
105 | 108 | else throw("Strict value is not equal to itself.") | |
106 | 109 | } | |
107 | 110 | ||
108 | 111 | ||
109 | 112 | ||
110 | 113 | @Callable(i) | |
111 | 114 | func addSentinel (sentinel) = if ((i.caller != this)) | |
112 | 115 | then throw("only self can add sentinel") | |
113 | 116 | else if (!(isDefined(sentinelsStr))) | |
114 | 117 | then [StringEntry(sentinelsStore, sentinel)] | |
115 | 118 | else if (isDefined(indexOf(sentinels, sentinel))) | |
116 | 119 | then nil | |
117 | 120 | else [StringEntry(sentinelsStore, ((value(sentinelsStr) + ",") + sentinel))] | |
118 | 121 | ||
119 | 122 | ||
120 | 123 | @Verifier(tx) | |
121 | 124 | func verify () = { | |
122 | 125 | let thisAddress = toString(this) | |
123 | 126 | let BASE = 1000 | |
124 | - | let stakingContract = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(configAddress, "contact_staking"), "no contact_staking")), "invalid contact_staking") | |
125 | 127 | let votingAddressStr = valueOrErrorMessage(getString(configAddress, "contract_voting"), "voting address not found") | |
126 | 128 | let votingContract = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(configAddress, "contract_voting"), "no contract_voting")), "invalid contract_voting") | |
127 | 129 | let quorumRatio = valueOrErrorMessage(getInteger(configAddress, (votingAddressStr + "_proposal_quorum_ratio")), "proposalQuorumRatio is not defined") | |
128 | 130 | let passedRatio = valueOrErrorMessage(getInteger(configAddress, (votingAddressStr + "_proposal_passed_ratio")), "proposalThresholdRatio is not defined") | |
129 | 131 | let id = toBase58String(tx.id) | |
130 | 132 | let votesYes = valueOrElse(getInteger(votingContract, ("proposal_yes_" + id)), 0) | |
131 | 133 | let votesNo = valueOrElse(getInteger(votingContract, ("proposal_no_" + id)), 0) | |
132 | 134 | let proposalHeight = valueOrErrorMessage(getInteger(votingContract, ("proposal_height_" + id)), "proposal not registered") | |
133 | 135 | let applyStart = valueOrElse(getInteger(votingContract, ("proposal_applystart_" + id)), 0) | |
134 | 136 | let applyEnd = valueOrElse(getInteger(votingContract, ("proposal_applyend_" + id)), 0) | |
135 | 137 | let totalPower = valueOrElse(getInteger(votingContract, ("proposal_power_" + id)), 0) | |
136 | 138 | let enabled = valueOrElse(getBoolean(configAddress, opAllowedStore("governance_apply_tx")), true) | |
137 | 139 | let voteYes = votesYes | |
138 | 140 | let voteNo = votesNo | |
139 | 141 | let totalVotes = (voteYes + voteNo) | |
140 | 142 | let hasQuorum = (((totalVotes * BASE) / totalPower) >= quorumRatio) | |
141 | 143 | let hasPassed = (((voteYes * BASE) / totalVotes) >= passedRatio) | |
142 | 144 | let tooEarly = (applyStart >= HEIGHT) | |
143 | 145 | let tooLate = (HEIGHT >= applyEnd) | |
144 | 146 | let timeDebug = ((((((("proposalHeight = " + toString(proposalHeight)) + ", applyStart = ") + toString(applyStart)) + ", applyEnd = ") + toString(applyEnd)) + ", HEIGHT = ") + toString(HEIGHT)) | |
145 | 147 | let byVoting = if (!(enabled)) | |
146 | 148 | then throw("tx application throw governance not enabled") | |
147 | 149 | else if (tooEarly) | |
148 | 150 | then throw(("proposal can't be executed as it's too early: " + timeDebug)) | |
149 | 151 | else if (tooLate) | |
150 | 152 | then throw(("proposal can't be executed as it's too late:" + timeDebug)) | |
151 | 153 | else if (!(hasQuorum)) | |
152 | 154 | then throw((((((("no quorum: " + "totalVotes: ") + toString(totalVotes)) + ", totalPower: ") + toString(totalPower)) + ", quorumRatio: ") + toString(quorumRatio))) | |
153 | 155 | else if (!(hasPassed)) | |
154 | 156 | then throw((((((("no threshold achieved: " + "voteYes: ") + toString(voteYes)) + ", voteNo: ") + toString(voteNo)) + ", passedRatio: ") + toString(passedRatio))) | |
155 | 157 | else true | |
156 | 158 | if (sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)) | |
157 | 159 | then true | |
158 | 160 | else byVoting | |
159 | 161 | } | |
160 | 162 |
github/deemru/w8io/3ef1775 39.77 ms ◑