2022.07.28 14:45 [3225125] smart account 3P2rJGfb5MbeivZNSqdVmsD1Y2Mz8y6Jr3Z > SELF 0.00000000 Waves
{ "type": 13, "id": "G8QgYb4PJscacih1yaHQZ2Xhj5yo2bqi6ejnBJksHftB", "fee": 1000000, "feeAssetId": null, "timestamp": 1659008647549, "version": 1, "sender": "3P2rJGfb5MbeivZNSqdVmsD1Y2Mz8y6Jr3Z", "senderPublicKey": "3gQ8QUfoGQW6YVuhUv3zuqsbmxbV5F2FAuDXJqVKD6C9", "proofs": [ "64UyBKa4Jhr2Y1gpw4FpLuLi4jpRZpT7se3dqoQ5mnrbGiJWCGB1zZTkWFHDzy9LVPi4hByjkWSX5oC2YQqoJKYo" ], "script": "base64:BgIkCAISAwoBCBIDCgEIEgQKAggIEgMKAQgSABIDCgEIEgQKAggIFgELd3JpdGVTdHJpbmcCA2tleQV2YWx1ZQkBC1N0cmluZ0VudHJ5AgUDa2V5BQV2YWx1ZQEQd3JpdGVDb25zdFN0cmluZwIDa2V5BXZhbHVlAwkBASEBCQEJaXNEZWZpbmVkAQkAnQgCBQR0aGlzBQNrZXkJAQtTdHJpbmdFbnRyeQIFA2tleQUFdmFsdWUJAAIBCQCsAgICFWFscmVhZHkgaW5pdGlhbGl6ZWQ6IAUDa2V5AA1jb25maWdBZGRyZXNzBQR0aGlzAAphZG1pblN0b3JlAgVhZG1pbgAJbWFpblN0b3JlAgRtYWluAA9wYXVzYWJsZUJ5U3RvcmUCCnBhdXNhYmxlQnkBC0FQb2ludFN0b3JlAQdhc3NldElkCQCsAgIFB2Fzc2V0SWQCB19BUG9pbnQBC0JQb2ludFN0b3JlAQdhc3NldElkCQCsAgIFB2Fzc2V0SWQCB19CUG9pbnQBC0NQb2ludFN0b3JlAQdhc3NldElkCQCsAgIFB2Fzc2V0SWQCB19DUG9pbnQBC0RQb2ludFN0b3JlAQdhc3NldElkCQCsAgIFB2Fzc2V0SWQCB19EUG9pbnQBFWNvbGxhdGVyYWxGYWN0b3JTdG9yZQEHYXNzZXRJZAkArAICBQdhc3NldElkAhFfQ29sbGF0ZXJhbEZhY3RvcgEScmVzZXJ2ZUZhY3RvclN0b3JlAQdhc3NldElkCQCsAgIFB2Fzc2V0SWQCDl9SZXNlcnZlRmFjdG9yARlsaXF1aWRhdGlvblRocmVzaG9sZFN0b3JlAQdhc3NldElkCQCsAgIFB2Fzc2V0SWQCFV9MaXF1aWRhdGlvblRocmVzaG9sZAEXbGlxdWlkYXRpb25QZW5hbHR5U3RvcmUBB2Fzc2V0SWQJAKwCAgUHYXNzZXRJZAITX0xpcXVpZGF0aW9uUGVuYWx0eQAGb3BzU3RyCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgJxZGVwb3NpdHx3aXRoZHJhd3xib3Jyb3d8cmVwYXl8YXRva2Vuc3xjb2xsYXBzZXx0cmFuc2Zlcl9kZWJ0fHVzZV9hc19jb2x8Zm9yY2VfY29sbGFwc2V8c3Rha2VfdmlyZXN8dW5zdGFrZV92aXJlc3wCfGNsYWltX3ZpcmVzX2Rpc3RyaWJ1dGlvbnxjbGFpbV92aXJlc19lYnxyZWFsbG9jfGxvY2tfdmlyZXN8d2l0aGRyYXdfdW5sb2NrZWRfdmlyZXN8Y2xhaW1fZGl2aWRlbmRzfHNoYXJlX2RpdmlkZW5kc19tYW51YWxseXwCqgFnb3Zlcm5hbmNlX3Byb3Bvc2VfdHh8Z292ZXJuYW5jZV9yZXRyYWN0X3Byb3Bvc2FsX3R4fGdvdmVybmFuY2Vfdm90ZV90eHxnb3Zlcm5hbmNlX2FwcGx5X3R4fG1pbnRfYXRva2Vuc3xsb2NrX21pbnRfYXRva2Vuc3xyZWRlZW1fYXRva2Vuc3xyZXBsZW5pc2hfYXRva2Vuc3xyZXBheV9hdG9rZW5zfAJ8c3Rha2VyX21pZ3JhdGV8c3VwcGx5X3Byb3RlY3RlZF9jb2xsYXRlcmFsfHdpdGhkcmF3X3Byb3RlY3RlZF9jb2xsYXRlcmFsfGNvbGxhcHNlX3Byb3RlY3RlZF9jb2xsYXRlcmFsfHNldF9kZWxlZ2F0aW9uX2NsYWltfAJac3RvcF9kaXN0cmlidXRpb25fdW5sb2NrZWRfbHB8d2l0aGRyYXdfdW5sb2NrZWRfbHB8bHBfbG9ja19yZWFsbG9jfGxvY2tfbHB8Y2xhaW1fdmlyZXNfbHB8AjtsaXF1aWRhb19zdGFydFZlc3Rpbmd8bGlxdWlkYW9fd2l0aGRyYXdWZXN0ZWR8bGlxdWlkYW9fbW92ZQADb3BzCQC8CQIFBm9wc1N0cgIBfAEVcmVzZXJ2ZU9wQ29udHJvbFN0b3JlAgdhc3NldElkAm9wCQCsAgIJAKwCAgkArAICCQCsAgICA29wXwUCb3ACAV8FB2Fzc2V0SWQCB19wYXVzZWQBDm9wQ29udHJvbFN0b3JlAQJvcAkArAICCQCsAgICA29wXwUCb3ACB19wYXVzZWQAEnN5c3RlbUNvbnRyb2xTdG9yZQINc3lzdGVtX3BhdXNlZAEJYWRtaW5Pbmx5AQFpAwkBAiE9AgkApQgBCAUBaQZjYWxsZXIJAQt2YWx1ZU9yRWxzZQIJAJ0IAgUEdGhpcwUKYWRtaW5TdG9yZQIIbm8gYWRtaW4JAAIBAhFvbmx5IGFkbWluIGNhbiBkbwYBB2luUmFuZ2UDBXZhbHVlA21pbgNtYXgDAwkAZgIFA21pbgUFdmFsdWUGCQBmAgUFdmFsdWUFA21heAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAgd2YWx1ZTogCQCkAwEFBXZhbHVlAg8gbm90IGluIHJhbmdlOiAJAKQDAQUDbWluAgMgLSAJAKQDAQUDbWF4BgEIY2FuUGF1c2UBAWkJAQhjb250YWlucwIJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGlzBQ9wYXVzYWJsZUJ5U3RvcmUCHnBhdXNhYmxlQnlTdG9yZSBub3QgY29uZmlndXJlZAkApQgBCAUBaQZjYWxsZXIHAWkBCmluaXRpYWxpemUBBG1haW4JAMwIAgkBEHdyaXRlQ29uc3RTdHJpbmcCBQphZG1pblN0b3JlCQClCAEIBQFpBmNhbGxlcgkAzAgCCQEQd3JpdGVDb25zdFN0cmluZwIFCW1haW5TdG9yZQUEbWFpbgUDbmlsAWkBEWNoYW5nZVJlc3VtZUFkbWluAQluZXdSZXN1bWUEBmNoZWNrcwkBCWFkbWluT25seQEFAWkDCQAAAgUGY2hlY2tzBQZjaGVja3MJAMwIAgkBC3dyaXRlU3RyaW5nAgUKYWRtaW5TdG9yZQUJbmV3UmVzdW1lBQNuaWwJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4BaQEMcGF1c2VBc3NldE9wAgdhc3NldElkAm9wAwkBASEBCQEJaXNEZWZpbmVkAQkAzwgCBQNvcHMFAm9wCQACAQkArAICAgx1bmtub3duIG9wOiAFAm9wAwkBASEBCQEIY2FuUGF1c2UBBQFpCQACAQIWb25seSBwYXVzYWJsZUJ5IGNhbiBkbwkAzAgCCQEMQm9vbGVhbkVudHJ5AgkBFXJlc2VydmVPcENvbnRyb2xTdG9yZQIFB2Fzc2V0SWQFAm9wBwUDbmlsAWkBB3BhdXNlT3ABAm9wAwkBASEBCQEJaXNEZWZpbmVkAQkAzwgCBQNvcHMFAm9wCQACAQkArAICAgx1bmtub3duIG9wOiAFAm9wAwkBASEBCQEIY2FuUGF1c2UBBQFpCQACAQIWb25seSBwYXVzYWJsZUJ5IGNhbiBkbwkAzAgCCQEMQm9vbGVhbkVudHJ5AgkBDm9wQ29udHJvbFN0b3JlAQUCb3AHBQNuaWwBaQELcGF1c2VTeXN0ZW0AAwkBASEBCQEIY2FuUGF1c2UBBQFpCQACAQIWb25seSBwYXVzYWJsZUJ5IGNhbiBkbwkAzAgCCQEMQm9vbGVhbkVudHJ5AgUSc3lzdGVtQ29udHJvbFN0b3JlBwUDbmlsAWkBBnJlc3VtZQEDa2V5BAZjaGVja3MJAQlhZG1pbk9ubHkBBQFpAwkAAAIFBmNoZWNrcwUGY2hlY2tzAwMJAAACBQNrZXkFEnN5c3RlbUNvbnRyb2xTdG9yZQYDCQEIY29udGFpbnMCBQNrZXkCA29wXwkBCGNvbnRhaW5zAgUDa2V5AgdfcGF1c2VkBwkAzAgCCQEMQm9vbGVhbkVudHJ5AgUDa2V5BgUDbmlsCQACAQINaW5jb3JyZWN0IGtleQkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgFpAQlvcEFsbG93ZWQCB2Fzc2V0SWQCb3ADCQEIY29udGFpbnMCCQELdmFsdWVPckVsc2UCCQCdCAIFBHRoaXMCA2V4YwIACQClCAEIBQFpDG9yaWdpbkNhbGxlcgkAlAoCBQNuaWwGBAhlbmFibGVkMAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzBRJzeXN0ZW1Db250cm9sU3RvcmUGBAhlbmFibGVkMQkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzCQEOb3BDb250cm9sU3RvcmUBBQJvcAYECGVuYWJsZWQyCQELdmFsdWVPckVsc2UCCQCbCAIFBHRoaXMJARVyZXNlcnZlT3BDb250cm9sU3RvcmUCBQdhc3NldElkBQJvcAYDCQEBIQEFCGVuYWJsZWQwCQACAQkArAICAhRvcGVyYXRpb24gcGF1c2VkIGJ5IAUSc3lzdGVtQ29udHJvbFN0b3JlAwkBASEBBQhlbmFibGVkMQkAAgEJAKwCAgIUb3BlcmF0aW9uIHBhdXNlZCBieSAJAQ5vcENvbnRyb2xTdG9yZQEFAm9wAwkBASEBBQhlbmFibGVkMgkAAgEJAKwCAgIUb3BlcmF0aW9uIHBhdXNlZCBieSAJARVyZXNlcnZlT3BDb250cm9sU3RvcmUCBQdhc3NldElkBQJvcAkAlAoCBQNuaWwGAQJ0eAEGdmVyaWZ5AAQSbWF5YmVPcmFjbGVBZGRyZXNzBAckbWF0Y2gwCQCdCAIFDWNvbmZpZ0FkZHJlc3MCDm9yYWNsZV9hZGRyZXNzAwkAAQIFByRtYXRjaDACBlN0cmluZwQBcwUHJG1hdGNoMAkApggBBQFzBQR1bml0BAZIRUlHSFQFBmhlaWdodAQEQkFTRQDoBwQLcXVvcnVtUmF0aW8JARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQ1jb25maWdBZGRyZXNzAhVwcm9wb3NhbF9xdW9ydW1fcmF0aW8CInByb3Bvc2FsUXVvcnVtUmF0aW8gaXMgbm90IGRlZmluZWQEC3Bhc3NlZFJhdGlvCQETdmFsdWVPckVycm9yTWVzc2FnZQIJAJoIAgUNY29uZmlnQWRkcmVzcwIVcHJvcG9zYWxfcGFzc2VkX3JhdGlvAiVwcm9wb3NhbFRocmVzaG9sZFJhdGlvIGlzIG5vdCBkZWZpbmVkBA5nVmlyZXNDb250cmFjdAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQ1jb25maWdBZGRyZXNzAhJkaXZpZGVuZHNfY29udHJhY3QCFW5vIGRpdmlkZW5kc19jb250cmFjdAIaaW52YWxpZCBkaXZpZGVuZHNfY29udHJhY3QEDnZvdGluZ0NvbnRyYWN0CQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCdCAIFDWNvbmZpZ0FkZHJlc3MCD3ZvdGluZ19jb250cmFjdAISbm8gdm90aW5nX2NvbnRyYWN0AhdpbnZhbGlkIHZvdGluZ19jb250cmFjdAQCaWQJANgEAQgFAnR4AmlkBAh2b3Rlc1llcwkBC3ZhbHVlT3JFbHNlAgkAmggCBQ52b3RpbmdDb250cmFjdAkArAICAg1wcm9wb3NhbF95ZXNfBQJpZAAABAd2b3Rlc05vCQELdmFsdWVPckVsc2UCCQCaCAIFDnZvdGluZ0NvbnRyYWN0CQCsAgICDHByb3Bvc2FsX25vXwUCaWQAAAQOcHJvcG9zYWxIZWlnaHQJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAmggCBQ52b3RpbmdDb250cmFjdAkArAICAhBwcm9wb3NhbF9oZWlnaHRfBQJpZAIXcHJvcG9zYWwgbm90IHJlZ2lzdGVyZWQECmFwcGx5U3RhcnQJAQt2YWx1ZU9yRWxzZQIJAJoIAgUOdm90aW5nQ29udHJhY3QJAKwCAgIUcHJvcG9zYWxfYXBwbHlzdGFydF8FAmlkAAAECGFwcGx5RW5kCQELdmFsdWVPckVsc2UCCQCaCAIFDnZvdGluZ0NvbnRyYWN0CQCsAgICEnByb3Bvc2FsX2FwcGx5ZW5kXwUCaWQAAAQLdG90YWxHVmlyZXMJAQt2YWx1ZU9yRWxzZQIJAJoIAgUOdm90aW5nQ29udHJhY3QJAKwCAgIQcHJvcG9zYWxfZ3ZpcmVzXwUCaWQAAAQHZW5hYmxlZAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQ1jb25maWdBZGRyZXNzAh1vcF9nb3Zlcm5hbmNlX2FwcGx5X3R4X3BhdXNlZAcEB3ZvdGVZZXMFCHZvdGVzWWVzBAZ2b3RlTm8FB3ZvdGVzTm8ECnRvdGFsVm90ZXMJAGQCBQd2b3RlWWVzBQZ2b3RlTm8ECWhhc1F1b3J1bQkAZwIJAGkCCQBoAgUKdG90YWxWb3RlcwUEQkFTRQULdG90YWxHVmlyZXMFC3F1b3J1bVJhdGlvBAloYXNQYXNzZWQJAGcCCQBpAgkAaAIFB3ZvdGVZZXMFBEJBU0UFCnRvdGFsVm90ZXMFC3Bhc3NlZFJhdGlvBAh0b29FYXJseQkAZwIFCmFwcGx5U3RhcnQFBkhFSUdIVAQHdG9vTGF0ZQkAZwIFBkhFSUdIVAUIYXBwbHlFbmQECXRpbWVEZWJ1ZwkArAICCQCsAgIJAKwCAgkArAICCQCsAgIJAKwCAgkArAICAhFwcm9wb3NhbEhlaWdodCA9IAkApAMBBQ5wcm9wb3NhbEhlaWdodAIPLCBhcHBseVN0YXJ0ID0gCQCkAwEFCmFwcGx5U3RhcnQCDSwgYXBwbHlFbmQgPSAJAKQDAQUIYXBwbHlFbmQCCywgSEVJR0hUID0gCQCkAwEFBkhFSUdIVAQIYnlWb3RpbmcDCQEBIQEFB2VuYWJsZWQJAAIBAit0eCBhcHBsaWNhdGlvbiB0aHJvdyBnb3Zlcm5hbmNlIG5vdCBlbmFibGVkAwUIdG9vRWFybHkJAAIBCQCsAgICLnByb3Bvc2FsIGNhbid0IGJlIGV4ZWN1dGVkIGFzIGl0J3MgdG9vIGVhcmx5OiAFCXRpbWVEZWJ1ZwMFB3Rvb0xhdGUJAAIBCQCsAgICLHByb3Bvc2FsIGNhbid0IGJlIGV4ZWN1dGVkIGFzIGl0J3MgdG9vIGxhdGU6BQl0aW1lRGVidWcDCQEBIQEFCWhhc1F1b3J1bQkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICC25vIHF1b3J1bTogAgx0b3RhbFZvdGVzOiAJAKQDAQUKdG90YWxWb3RlcwIPLCB0b3RhbEdWaXJlczogCQCkAwEFC3RvdGFsR1ZpcmVzAg8sIHF1b3J1bVJhdGlvOiAJAKQDAQULcXVvcnVtUmF0aW8DCQEBIQEFCWhhc1Bhc3NlZAkAAgEJAKwCAgkArAICCQCsAgIJAKwCAgkArAICCQCsAgICF25vIHRocmVzaG9sZCBhY2hpZXZlZDogAgl2b3RlWWVzOiAJAKQDAQUHdm90ZVllcwIKLCB2b3RlTm86IAkApAMBBQZ2b3RlTm8CDywgcGFzc2VkUmF0aW86IAkApAMBBQtwYXNzZWRSYXRpbwYDCQD0AwMIBQJ0eAlib2R5Qnl0ZXMJAJEDAggFAnR4BnByb29mcwAACAUCdHgPc2VuZGVyUHVibGljS2V5BgUIYnlWb3Rpbmecr+BG", "chainId": 87, "height": 3225125, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 3ZfhvLtnXCM3w8ns85Y7Xv1a8BcyBpyER9nN5PnAWzuS Next: 5S3DmYDbRmVMGHE5rcNFm2RAomzSCT6ZTVzKLEz9nhBM Diff:
Old | New | Differences | |
---|---|---|---|
41 | 41 | func liquidationPenaltyStore (assetId) = (assetId + "_LiquidationPenalty") | |
42 | 42 | ||
43 | 43 | ||
44 | - | let opsStr = (((("deposit|withdraw|borrow|repay|atokens|collapse|transfer_debt|use_as_col|force_collapse|stake_vires|unstake_vires|" + "claim_vires_distribution|claim_vires_eb|realloc|lock_vires|withdraw_unlocked_vires|claim_dividends|share_dividends_manually|") + "governance_propose_tx|governance_retract_proposal_tx|governance_vote_tx|governance_apply_tx|mint_atokens|lock_mint_atokens|redeem_atokens|replenish_atokens|repay_atokens|") + "staker_migrate|supply_protected_collateral|withdraw_protected_collateral|collapse_protected_collateral|set_delegation_claim|") + "stop_distribution_unlocked_lp|withdraw_unlocked_lp|lp_lock_realloc|lock_lp|claim_vires_lp") | |
44 | + | let opsStr = ((((("deposit|withdraw|borrow|repay|atokens|collapse|transfer_debt|use_as_col|force_collapse|stake_vires|unstake_vires|" + "claim_vires_distribution|claim_vires_eb|realloc|lock_vires|withdraw_unlocked_vires|claim_dividends|share_dividends_manually|") + "governance_propose_tx|governance_retract_proposal_tx|governance_vote_tx|governance_apply_tx|mint_atokens|lock_mint_atokens|redeem_atokens|replenish_atokens|repay_atokens|") + "staker_migrate|supply_protected_collateral|withdraw_protected_collateral|collapse_protected_collateral|set_delegation_claim|") + "stop_distribution_unlocked_lp|withdraw_unlocked_lp|lp_lock_realloc|lock_lp|claim_vires_lp|") + "liquidao_startVesting|liquidao_withdrawVested|liquidao_move") | |
45 | 45 | ||
46 | 46 | let ops = split_4C(opsStr, "|") | |
47 | 47 | ||
125 | 125 | ||
126 | 126 | ||
127 | 127 | @Callable(i) | |
128 | - | func opAllowed (assetId,op) = { | |
129 | - | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
130 | - | let enabled1 = valueOrElse(getBoolean(this, opControlStore(op)), true) | |
131 | - | let enabled2 = valueOrElse(getBoolean(this, reserveOpControlStore(assetId, op)), true) | |
132 | - | if (!(enabled0)) | |
133 | - | then throw(("operation paused by " + systemControlStore)) | |
134 | - | else if (!(enabled1)) | |
135 | - | then throw(("operation paused by " + opControlStore(op))) | |
136 | - | else if (!(enabled2)) | |
137 | - | then throw(("operation paused by " + reserveOpControlStore(assetId, op))) | |
138 | - | else $Tuple2(nil, true) | |
139 | - | } | |
128 | + | func opAllowed (assetId,op) = if (contains(valueOrElse(getString(this, "exc"), ""), toString(i.originCaller))) | |
129 | + | then $Tuple2(nil, true) | |
130 | + | else { | |
131 | + | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
132 | + | let enabled1 = valueOrElse(getBoolean(this, opControlStore(op)), true) | |
133 | + | let enabled2 = valueOrElse(getBoolean(this, reserveOpControlStore(assetId, op)), true) | |
134 | + | if (!(enabled0)) | |
135 | + | then throw(("operation paused by " + systemControlStore)) | |
136 | + | else if (!(enabled1)) | |
137 | + | then throw(("operation paused by " + opControlStore(op))) | |
138 | + | else if (!(enabled2)) | |
139 | + | then throw(("operation paused by " + reserveOpControlStore(assetId, op))) | |
140 | + | else $Tuple2(nil, true) | |
141 | + | } | |
140 | 142 | ||
141 | 143 | ||
142 | 144 | @Verifier(tx) |
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 adminStore = "admin" | |
15 | 15 | ||
16 | 16 | let mainStore = "main" | |
17 | 17 | ||
18 | 18 | let pausableByStore = "pausableBy" | |
19 | 19 | ||
20 | 20 | func APointStore (assetId) = (assetId + "_APoint") | |
21 | 21 | ||
22 | 22 | ||
23 | 23 | func BPointStore (assetId) = (assetId + "_BPoint") | |
24 | 24 | ||
25 | 25 | ||
26 | 26 | func CPointStore (assetId) = (assetId + "_CPoint") | |
27 | 27 | ||
28 | 28 | ||
29 | 29 | func DPointStore (assetId) = (assetId + "_DPoint") | |
30 | 30 | ||
31 | 31 | ||
32 | 32 | func collateralFactorStore (assetId) = (assetId + "_CollateralFactor") | |
33 | 33 | ||
34 | 34 | ||
35 | 35 | func reserveFactorStore (assetId) = (assetId + "_ReserveFactor") | |
36 | 36 | ||
37 | 37 | ||
38 | 38 | func liquidationThresholdStore (assetId) = (assetId + "_LiquidationThreshold") | |
39 | 39 | ||
40 | 40 | ||
41 | 41 | func liquidationPenaltyStore (assetId) = (assetId + "_LiquidationPenalty") | |
42 | 42 | ||
43 | 43 | ||
44 | - | let opsStr = (((("deposit|withdraw|borrow|repay|atokens|collapse|transfer_debt|use_as_col|force_collapse|stake_vires|unstake_vires|" + "claim_vires_distribution|claim_vires_eb|realloc|lock_vires|withdraw_unlocked_vires|claim_dividends|share_dividends_manually|") + "governance_propose_tx|governance_retract_proposal_tx|governance_vote_tx|governance_apply_tx|mint_atokens|lock_mint_atokens|redeem_atokens|replenish_atokens|repay_atokens|") + "staker_migrate|supply_protected_collateral|withdraw_protected_collateral|collapse_protected_collateral|set_delegation_claim|") + "stop_distribution_unlocked_lp|withdraw_unlocked_lp|lp_lock_realloc|lock_lp|claim_vires_lp") | |
44 | + | let opsStr = ((((("deposit|withdraw|borrow|repay|atokens|collapse|transfer_debt|use_as_col|force_collapse|stake_vires|unstake_vires|" + "claim_vires_distribution|claim_vires_eb|realloc|lock_vires|withdraw_unlocked_vires|claim_dividends|share_dividends_manually|") + "governance_propose_tx|governance_retract_proposal_tx|governance_vote_tx|governance_apply_tx|mint_atokens|lock_mint_atokens|redeem_atokens|replenish_atokens|repay_atokens|") + "staker_migrate|supply_protected_collateral|withdraw_protected_collateral|collapse_protected_collateral|set_delegation_claim|") + "stop_distribution_unlocked_lp|withdraw_unlocked_lp|lp_lock_realloc|lock_lp|claim_vires_lp|") + "liquidao_startVesting|liquidao_withdrawVested|liquidao_move") | |
45 | 45 | ||
46 | 46 | let ops = split_4C(opsStr, "|") | |
47 | 47 | ||
48 | 48 | func reserveOpControlStore (assetId,op) = (((("op_" + op) + "_") + assetId) + "_paused") | |
49 | 49 | ||
50 | 50 | ||
51 | 51 | func opControlStore (op) = (("op_" + op) + "_paused") | |
52 | 52 | ||
53 | 53 | ||
54 | 54 | let systemControlStore = "system_paused" | |
55 | 55 | ||
56 | 56 | func adminOnly (i) = if ((toString(i.caller) != valueOrElse(getString(this, adminStore), "no admin"))) | |
57 | 57 | then throw("only admin can do") | |
58 | 58 | else true | |
59 | 59 | ||
60 | 60 | ||
61 | 61 | func inRange (value,min,max) = if (if ((min > value)) | |
62 | 62 | then true | |
63 | 63 | else (value > max)) | |
64 | 64 | then throw(((((("value: " + toString(value)) + " not in range: ") + toString(min)) + " - ") + toString(max))) | |
65 | 65 | else true | |
66 | 66 | ||
67 | 67 | ||
68 | 68 | func canPause (i) = contains(valueOrErrorMessage(getString(this, pausableByStore), "pausableByStore not configured"), toString(i.caller)) | |
69 | 69 | ||
70 | 70 | ||
71 | 71 | @Callable(i) | |
72 | 72 | func initialize (main) = [writeConstString(adminStore, toString(i.caller)), writeConstString(mainStore, main)] | |
73 | 73 | ||
74 | 74 | ||
75 | 75 | ||
76 | 76 | @Callable(i) | |
77 | 77 | func changeResumeAdmin (newResume) = { | |
78 | 78 | let checks = adminOnly(i) | |
79 | 79 | if ((checks == checks)) | |
80 | 80 | then [writeString(adminStore, newResume)] | |
81 | 81 | else throw("Strict value is not equal to itself.") | |
82 | 82 | } | |
83 | 83 | ||
84 | 84 | ||
85 | 85 | ||
86 | 86 | @Callable(i) | |
87 | 87 | func pauseAssetOp (assetId,op) = if (!(isDefined(indexOf(ops, op)))) | |
88 | 88 | then throw(("unknown op: " + op)) | |
89 | 89 | else if (!(canPause(i))) | |
90 | 90 | then throw("only pausableBy can do") | |
91 | 91 | else [BooleanEntry(reserveOpControlStore(assetId, op), false)] | |
92 | 92 | ||
93 | 93 | ||
94 | 94 | ||
95 | 95 | @Callable(i) | |
96 | 96 | func pauseOp (op) = if (!(isDefined(indexOf(ops, op)))) | |
97 | 97 | then throw(("unknown op: " + op)) | |
98 | 98 | else if (!(canPause(i))) | |
99 | 99 | then throw("only pausableBy can do") | |
100 | 100 | else [BooleanEntry(opControlStore(op), false)] | |
101 | 101 | ||
102 | 102 | ||
103 | 103 | ||
104 | 104 | @Callable(i) | |
105 | 105 | func pauseSystem () = if (!(canPause(i))) | |
106 | 106 | then throw("only pausableBy can do") | |
107 | 107 | else [BooleanEntry(systemControlStore, false)] | |
108 | 108 | ||
109 | 109 | ||
110 | 110 | ||
111 | 111 | @Callable(i) | |
112 | 112 | func resume (key) = { | |
113 | 113 | let checks = adminOnly(i) | |
114 | 114 | if ((checks == checks)) | |
115 | 115 | then if (if ((key == systemControlStore)) | |
116 | 116 | then true | |
117 | 117 | else if (contains(key, "op_")) | |
118 | 118 | then contains(key, "_paused") | |
119 | 119 | else false) | |
120 | 120 | then [BooleanEntry(key, true)] | |
121 | 121 | else throw("incorrect key") | |
122 | 122 | else throw("Strict value is not equal to itself.") | |
123 | 123 | } | |
124 | 124 | ||
125 | 125 | ||
126 | 126 | ||
127 | 127 | @Callable(i) | |
128 | - | func opAllowed (assetId,op) = { | |
129 | - | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
130 | - | let enabled1 = valueOrElse(getBoolean(this, opControlStore(op)), true) | |
131 | - | let enabled2 = valueOrElse(getBoolean(this, reserveOpControlStore(assetId, op)), true) | |
132 | - | if (!(enabled0)) | |
133 | - | then throw(("operation paused by " + systemControlStore)) | |
134 | - | else if (!(enabled1)) | |
135 | - | then throw(("operation paused by " + opControlStore(op))) | |
136 | - | else if (!(enabled2)) | |
137 | - | then throw(("operation paused by " + reserveOpControlStore(assetId, op))) | |
138 | - | else $Tuple2(nil, true) | |
139 | - | } | |
128 | + | func opAllowed (assetId,op) = if (contains(valueOrElse(getString(this, "exc"), ""), toString(i.originCaller))) | |
129 | + | then $Tuple2(nil, true) | |
130 | + | else { | |
131 | + | let enabled0 = valueOrElse(getBoolean(this, systemControlStore), true) | |
132 | + | let enabled1 = valueOrElse(getBoolean(this, opControlStore(op)), true) | |
133 | + | let enabled2 = valueOrElse(getBoolean(this, reserveOpControlStore(assetId, op)), true) | |
134 | + | if (!(enabled0)) | |
135 | + | then throw(("operation paused by " + systemControlStore)) | |
136 | + | else if (!(enabled1)) | |
137 | + | then throw(("operation paused by " + opControlStore(op))) | |
138 | + | else if (!(enabled2)) | |
139 | + | then throw(("operation paused by " + reserveOpControlStore(assetId, op))) | |
140 | + | else $Tuple2(nil, true) | |
141 | + | } | |
140 | 142 | ||
141 | 143 | ||
142 | 144 | @Verifier(tx) | |
143 | 145 | func verify () = { | |
144 | 146 | let maybeOracleAddress = match getString(configAddress, "oracle_address") { | |
145 | 147 | case s: String => | |
146 | 148 | addressFromString(s) | |
147 | 149 | case _ => | |
148 | 150 | unit | |
149 | 151 | } | |
150 | 152 | let HEIGHT = height | |
151 | 153 | let BASE = 1000 | |
152 | 154 | let quorumRatio = valueOrErrorMessage(getInteger(configAddress, "proposal_quorum_ratio"), "proposalQuorumRatio is not defined") | |
153 | 155 | let passedRatio = valueOrErrorMessage(getInteger(configAddress, "proposal_passed_ratio"), "proposalThresholdRatio is not defined") | |
154 | 156 | let gViresContract = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(configAddress, "dividends_contract"), "no dividends_contract")), "invalid dividends_contract") | |
155 | 157 | let votingContract = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(configAddress, "voting_contract"), "no voting_contract")), "invalid voting_contract") | |
156 | 158 | let id = toBase58String(tx.id) | |
157 | 159 | let votesYes = valueOrElse(getInteger(votingContract, ("proposal_yes_" + id)), 0) | |
158 | 160 | let votesNo = valueOrElse(getInteger(votingContract, ("proposal_no_" + id)), 0) | |
159 | 161 | let proposalHeight = valueOrErrorMessage(getInteger(votingContract, ("proposal_height_" + id)), "proposal not registered") | |
160 | 162 | let applyStart = valueOrElse(getInteger(votingContract, ("proposal_applystart_" + id)), 0) | |
161 | 163 | let applyEnd = valueOrElse(getInteger(votingContract, ("proposal_applyend_" + id)), 0) | |
162 | 164 | let totalGVires = valueOrElse(getInteger(votingContract, ("proposal_gvires_" + id)), 0) | |
163 | 165 | let enabled = valueOrElse(getBoolean(configAddress, "op_governance_apply_tx_paused"), false) | |
164 | 166 | let voteYes = votesYes | |
165 | 167 | let voteNo = votesNo | |
166 | 168 | let totalVotes = (voteYes + voteNo) | |
167 | 169 | let hasQuorum = (((totalVotes * BASE) / totalGVires) >= quorumRatio) | |
168 | 170 | let hasPassed = (((voteYes * BASE) / totalVotes) >= passedRatio) | |
169 | 171 | let tooEarly = (applyStart >= HEIGHT) | |
170 | 172 | let tooLate = (HEIGHT >= applyEnd) | |
171 | 173 | let timeDebug = ((((((("proposalHeight = " + toString(proposalHeight)) + ", applyStart = ") + toString(applyStart)) + ", applyEnd = ") + toString(applyEnd)) + ", HEIGHT = ") + toString(HEIGHT)) | |
172 | 174 | let byVoting = if (!(enabled)) | |
173 | 175 | then throw("tx application throw governance not enabled") | |
174 | 176 | else if (tooEarly) | |
175 | 177 | then throw(("proposal can't be executed as it's too early: " + timeDebug)) | |
176 | 178 | else if (tooLate) | |
177 | 179 | then throw(("proposal can't be executed as it's too late:" + timeDebug)) | |
178 | 180 | else if (!(hasQuorum)) | |
179 | 181 | then throw((((((("no quorum: " + "totalVotes: ") + toString(totalVotes)) + ", totalGVires: ") + toString(totalGVires)) + ", quorumRatio: ") + toString(quorumRatio))) | |
180 | 182 | else if (!(hasPassed)) | |
181 | 183 | then throw((((((("no threshold achieved: " + "voteYes: ") + toString(voteYes)) + ", voteNo: ") + toString(voteNo)) + ", passedRatio: ") + toString(passedRatio))) | |
182 | 184 | else true | |
183 | 185 | if (sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)) | |
184 | 186 | then true | |
185 | 187 | else byVoting | |
186 | 188 | } | |
187 | 189 |
github/deemru/w8io/786bc32 40.54 ms ◑