Skip to content

Commit 2a06101

Browse files
Merge pull request #19 from hokunet/sander/validator-gater
feat: validator gater contract
2 parents b299ff0 + a79a94b commit 2a06101

16 files changed

+692
-13
lines changed

.github/workflows/test.yml

+6-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ name: CI
33
on:
44
push:
55
pull_request:
6-
workflow_dispatch:
76

87
env:
98
FOUNDRY_PROFILE: ci
@@ -12,7 +11,6 @@ jobs:
1211
check:
1312
strategy:
1413
fail-fast: true
15-
1614
name: Foundry project
1715
runs-on: ubuntu-latest
1816
steps:
@@ -33,6 +31,12 @@ jobs:
3331
run: |
3432
forge fmt --check
3533
id: fmt
34+
35+
- name: Run Forge clean
36+
run: |
37+
rm -rf cache out
38+
forge clean
39+
id: clean
3640

3741
- name: Run Forge build
3842
run: |

hardhat.config.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ module.exports = {
2020
storageLayouts: ".storage-layouts",
2121
},
2222
storageLayoutConfig: {
23-
contracts: ['src/Hoku.sol:Hoku'],
23+
contracts: ['src/Hoku.sol:Hoku', 'src/ValidatorGater.sol:ValidatorGater'],
2424
fullPath: true
2525
},
2626
resolve: {

script/Faucet.s.sol

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
// SPDX-License-Identifier: UNLICENSED
2-
pragma solidity ^0.8.23;
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
33

44
import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
55
import {Script} from "forge-std/Script.sol";
@@ -10,7 +10,6 @@ import {Environment} from "../src/types/CommonTypes.sol";
1010

1111
contract DeployScript is Script {
1212
string constant PRIVATE_KEY = "PRIVATE_KEY";
13-
address public proxyAddress;
1413

1514
function setUp() public {}
1615

script/Hoku.s.sol

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
// SPDX-License-Identifier: UNLICENSED
2-
pragma solidity ^0.8.23;
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
33

44
import {Hoku} from "../src/Hoku.sol";
55
import {IInterchainTokenService} from

script/ValidatorGater.s.sol

+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
3+
4+
import {ValidatorGater} from "../src/ValidatorGater.sol";
5+
import {SubnetID} from "../src/structs/Subnet.sol";
6+
7+
import {Environment} from "../src/types/CommonTypes.sol";
8+
import {Options, Upgrades} from "@openzeppelin/foundry-upgrades/Upgrades.sol";
9+
import {Script, console2} from "forge-std/Script.sol";
10+
11+
contract DeployScript is Script {
12+
string constant PRIVATE_KEY = "PRIVATE_KEY";
13+
address public proxyAddress;
14+
15+
function setUp() public {}
16+
17+
function proxy() public view returns (address) {
18+
return proxyAddress;
19+
}
20+
21+
function run(Environment env) public returns (ValidatorGater) {
22+
if (vm.envExists(PRIVATE_KEY)) {
23+
uint256 privateKey = vm.envUint(PRIVATE_KEY);
24+
vm.startBroadcast(privateKey);
25+
} else if (env == Environment.Local) {
26+
vm.startBroadcast();
27+
} else {
28+
revert("PRIVATE_KEY not set in non-local environment");
29+
}
30+
Options memory options;
31+
options.unsafeAllow = "external-library-linking";
32+
33+
proxyAddress =
34+
Upgrades.deployUUPSProxy("ValidatorGater.sol", abi.encodeCall(ValidatorGater.initialize, ()), options);
35+
vm.stopBroadcast();
36+
37+
// Check implementation
38+
address implAddr = Upgrades.getImplementationAddress(proxyAddress);
39+
console2.log("Implementation address: ", implAddr);
40+
41+
ValidatorGater gater = ValidatorGater(proxyAddress);
42+
return gater;
43+
}
44+
}
45+
46+
contract UpgradeGaterProxyScript is Script {
47+
function setUp() public {}
48+
49+
function run() public {
50+
// Get proxy owner account
51+
uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY");
52+
53+
// Get proxy address
54+
address proxy = vm.envAddress("PROXY_ADDR");
55+
console2.log("proxy address: ", proxy);
56+
57+
// Check current implementation
58+
address implOld = Upgrades.getImplementationAddress(proxy);
59+
console2.log("Implementation address: ", implOld);
60+
61+
// Upgrade proxy to new implementation
62+
Options memory opts;
63+
opts.referenceContract = "ValidatorGater.sol";
64+
vm.startBroadcast(deployerPrivateKey);
65+
Upgrades.upgradeProxy(proxy, "ValidatorGater.sol", "", opts);
66+
vm.stopBroadcast();
67+
68+
// Check new implementation
69+
address implNew = Upgrades.getImplementationAddress(proxy);
70+
console2.log("Implementation address: ", implNew);
71+
72+
require(implOld != implNew, "Implementation address not changed");
73+
}
74+
}

src/Faucet.sol

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
// SPDX-License-Identifier: UNLICENSED
2-
pragma solidity ^0.8.23;
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
33

44
import {Ownable} from "@openzeppelin/contracts/access/Ownable.sol";
55
import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol";

src/Hoku.sol

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
// SPDX-License-Identifier: MIT
2-
pragma solidity ^0.8.23;
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
33

44
import {InterchainTokenStandard} from
55
"@axelar-network/interchain-token-service/contracts/interchain-token/InterchainTokenStandard.sol";

src/ValidatorGater.sol

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
3+
4+
import {InvalidSubnet, NotAuthorized, ValidatorPowerChangeDenied} from "./errors/IPCErrors.sol";
5+
import {IValidatorGater} from "./interfaces/IValidatorGater.sol";
6+
7+
import {SubnetIDHelper} from "./lib/SubnetIDHelper.sol";
8+
import {SubnetID} from "./structs/Subnet.sol";
9+
import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/contracts/access/OwnableUpgradeable.sol";
10+
import {UUPSUpgradeable} from "@openzeppelin/contracts-upgradeable/contracts/proxy/utils/UUPSUpgradeable.sol";
11+
12+
/// The power range that an approved validator can have.
13+
struct PowerRange {
14+
uint256 min;
15+
uint256 max;
16+
}
17+
18+
/// This is a simple implementation of `IValidatorGater`. It makes sure the exact power change
19+
/// request is approved. This is a very strict requirement.
20+
contract ValidatorGater is IValidatorGater, UUPSUpgradeable, OwnableUpgradeable {
21+
using SubnetIDHelper for SubnetID;
22+
23+
bool private _active;
24+
25+
SubnetID public subnet;
26+
mapping(address => PowerRange) public allowed;
27+
// New active status and who was the owner at change time
28+
29+
event ActiveStateChange(bool active, address account);
30+
31+
function initialize() public initializer {
32+
__Ownable_init(msg.sender);
33+
_active = true;
34+
}
35+
36+
/// @notice Indicates whether the gate is active or not
37+
function isActive() external view returns (bool) {
38+
return _active;
39+
}
40+
41+
/// @notice Sets the contract as active or inactive.
42+
/// @dev Only the owner can change the active state.
43+
function setActive(bool active) external onlyOwner {
44+
_active = active;
45+
emit ActiveStateChange(active, msg.sender);
46+
}
47+
48+
modifier whenActive() {
49+
if (!_active) {
50+
return; // Skip execution if not active
51+
}
52+
_; // Continue with function execution if active
53+
}
54+
55+
function setSubnet(SubnetID calldata id) external onlyOwner whenActive {
56+
subnet = id;
57+
}
58+
59+
function isAllow(address validator, uint256 power) public view whenActive returns (bool) {
60+
PowerRange memory range = allowed[validator];
61+
return range.min <= power && power <= range.max;
62+
}
63+
64+
/// Only owner can approve the validator join request
65+
function approve(address validator, uint256 minPower, uint256 maxPower) external onlyOwner whenActive {
66+
allowed[validator] = PowerRange({min: minPower, max: maxPower});
67+
}
68+
69+
/// Revoke approved power range
70+
function revoke(address validator) external onlyOwner whenActive {
71+
delete allowed[validator];
72+
}
73+
74+
function interceptPowerDelta(SubnetID memory id, address validator, uint256, /*prevPower*/ uint256 newPower)
75+
external
76+
view
77+
override
78+
whenActive
79+
{
80+
// unstake has checks to avoid newPower being zero, therefore zero means its leaving the network
81+
if (newPower == 0) return;
82+
83+
SubnetID memory targetSubnet = subnet;
84+
85+
if (!id.equals(targetSubnet)) {
86+
revert InvalidSubnet();
87+
}
88+
89+
if (msg.sender != targetSubnet.getAddress()) {
90+
revert NotAuthorized(msg.sender);
91+
}
92+
93+
if (!isAllow(validator, newPower)) {
94+
revert ValidatorPowerChangeDenied();
95+
}
96+
}
97+
98+
/// @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract
99+
/// @param newImplementation Address of the new implementation contract
100+
function _authorizeUpgrade(address newImplementation) internal view override onlyOwner {} // solhint-disable
101+
// no-empty-blocks
102+
}

src/errors/IPCErrors.sol

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
3+
4+
error InvalidSubnet();
5+
error NotAuthorized(address);
6+
error ValidatorPowerChangeDenied();

src/interfaces/IValidatorGater.sol

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
pragma solidity ^0.8.26;
3+
4+
import {SubnetID} from "../structs/Subnet.sol";
5+
6+
/// @title Validator Gater interface
7+
/// This interface introduces the ability to intercept validator power updates before it's executed. Power updates could
8+
/// come from staking, unstaking, and explicit validator membership adjustments (federated membership). With this
9+
/// interface,
10+
/// it introduces an extra layer of checking to directly allow or deny the action, according to a user-defined policy.
11+
interface IValidatorGater {
12+
/// This intercepts the power update call.
13+
/// @notice This method should revert if the power update is not allowed.
14+
function interceptPowerDelta(SubnetID memory id, address validator, uint256 prevPower, uint256 newPower) external;
15+
}

0 commit comments

Comments
 (0)