- Update dbis_core, cross-chain-pmm-lps, explorer-monorepo, metamask-integration, pr-workspace/chains - Omit embedded publish git dirs and empty placeholders from index Made-with: Cursor
633 lines
29 KiB
Solidity
633 lines
29 KiB
Solidity
// SPDX-License-Identifier: MIT OR Apache-2.0
|
|
pragma solidity ^0.8.13;
|
|
|
|
import {VmSafe} from "./Vm.sol";
|
|
import {Variable, Type, TypeKind, LibVariable} from "./LibVariable.sol";
|
|
|
|
/// @notice A contract that parses a toml configuration file and loads its
|
|
/// variables into storage, automatically casting them, on deployment.
|
|
///
|
|
/// @dev This contract assumes a toml structure where top-level keys
|
|
/// represent chain ids or aliases. Under each chain key, variables are
|
|
/// organized by type in separate sub-tables like `[<chain>.<type>]`, where
|
|
/// type must be: `bool`, `address`, `bytes32`, `uint`, `int`, `string`, or `bytes`.
|
|
///
|
|
/// Supported format:
|
|
/// ```
|
|
/// [mainnet]
|
|
/// endpoint_url = "${MAINNET_RPC}"
|
|
///
|
|
/// [mainnet.bool]
|
|
/// is_live = true
|
|
///
|
|
/// [mainnet.address]
|
|
/// weth = "0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2"
|
|
/// whitelisted_admins = [
|
|
/// "${MAINNET_ADMIN}",
|
|
/// "0x00000000000000000000000000000000deadbeef",
|
|
/// "0x000000000000000000000000000000c0ffeebabe"
|
|
/// ]
|
|
///
|
|
/// [mainnet.uint]
|
|
/// important_number = 123
|
|
/// ```
|
|
contract StdConfig {
|
|
using LibVariable for Type;
|
|
using LibVariable for TypeKind;
|
|
|
|
VmSafe private constant vm = VmSafe(address(uint160(uint256(keccak256("hevm cheat code")))));
|
|
|
|
/// @dev Types: `bool`, `address`, `bytes32`, `uint`, `int`, `string`, `bytes`.
|
|
uint8 private constant NUM_TYPES = 7;
|
|
|
|
// -- ERRORS ---------------------------------------------------------------
|
|
|
|
error AlreadyInitialized(string key);
|
|
error InvalidChainKey(string aliasOrId);
|
|
error ChainNotInitialized(uint256 chainId);
|
|
error UnableToParseVariable(string key);
|
|
error WriteToFileInForbiddenCtxt();
|
|
|
|
// -- STORAGE (CACHE FROM CONFIG FILE) ------------------------------------
|
|
|
|
/// @dev Path to the loaded TOML configuration file.
|
|
string private _filePath;
|
|
|
|
/// @dev List of top-level keys found in the TOML file, assumed to be chain names/aliases.
|
|
string[] private _chainKeys;
|
|
|
|
/// @dev Storage for the configured RPC URL for each chain.
|
|
mapping(uint256 => string) private _rpcOf;
|
|
|
|
/// @dev Storage for values, organized by chain ID and variable key.
|
|
mapping(uint256 => mapping(string => bytes)) private _dataOf;
|
|
|
|
/// @dev Type cache for runtime checking when casting.
|
|
mapping(uint256 => mapping(string => Type)) private _typeOf;
|
|
|
|
/// @dev When enabled, `set` will always write updates back to the configuration file.
|
|
/// Can only be enabled in a scripting context to prevent file corruption from
|
|
/// concurrent I/O access, as tests run in parallel.
|
|
bool private _writeToFile;
|
|
|
|
// -- CONSTRUCTOR ----------------------------------------------------------
|
|
|
|
/// @notice Reads the TOML file and iterates through each top-level key, which is
|
|
/// assumed to be a chain name or ID. For each chain, it caches its RPC
|
|
/// endpoint and all variables defined in typed sub-tables like `[<chain>.<type>]`,
|
|
/// where type must be: `bool`, `address`, `bytes32`, `uint`, `int`, `string`, or `bytes`.
|
|
///
|
|
/// The constructor attempts to parse each variable first as a single value,
|
|
/// and if that fails, as an array of that type. If a variable cannot be
|
|
/// parsed as either, the constructor will revert with an error.
|
|
///
|
|
/// @param configFilePath: The local path to the TOML configuration file.
|
|
/// @param writeToFile: Whether to write updates back to the TOML file. Only for scripts.
|
|
constructor(string memory configFilePath, bool writeToFile) {
|
|
if (writeToFile && !vm.isContext(VmSafe.ForgeContext.ScriptGroup)) {
|
|
revert WriteToFileInForbiddenCtxt();
|
|
}
|
|
|
|
_filePath = configFilePath;
|
|
_writeToFile = writeToFile;
|
|
string memory content = vm.resolveEnv(vm.readFile(configFilePath));
|
|
string[] memory chain_keys = vm.parseTomlKeys(content, "$");
|
|
|
|
// Cache the entire configuration to storage
|
|
for (uint256 i = 0; i < chain_keys.length; i++) {
|
|
string memory chain_key = chain_keys[i];
|
|
// Ignore top-level keys that are not tables
|
|
if (vm.parseTomlKeys(content, string.concat("$.", chain_key)).length == 0) {
|
|
continue;
|
|
}
|
|
uint256 chainId = resolveChainId(chain_key);
|
|
_chainKeys.push(chain_key);
|
|
|
|
// Cache the configured RPC endpoint for that chain.
|
|
// Falls back to `[rpc_endpoints]`. Panics if no rpc endpoint is configured.
|
|
try vm.parseTomlString(content, string.concat("$.", chain_key, ".endpoint_url")) returns (
|
|
string memory url
|
|
) {
|
|
_rpcOf[chainId] = vm.resolveEnv(url);
|
|
} catch {
|
|
_rpcOf[chainId] = vm.resolveEnv(vm.rpcUrl(chain_key));
|
|
}
|
|
|
|
// Iterate through all the available `TypeKind`s (except `None`) to create the sub-section paths
|
|
for (uint8 t = 1; t <= NUM_TYPES; t++) {
|
|
TypeKind ty = TypeKind(t);
|
|
string memory typePath = string.concat("$.", chain_key, ".", ty.toTomlKey());
|
|
|
|
try vm.parseTomlKeys(content, typePath) returns (string[] memory keys) {
|
|
for (uint256 j = 0; j < keys.length; j++) {
|
|
string memory key = keys[j];
|
|
if (_typeOf[chainId][key].kind == TypeKind.None) {
|
|
_loadAndCacheValue(content, string.concat(typePath, ".", key), chainId, key, ty);
|
|
} else {
|
|
revert AlreadyInitialized(key);
|
|
}
|
|
}
|
|
} catch {}
|
|
}
|
|
}
|
|
}
|
|
|
|
function _loadAndCacheValue(
|
|
string memory content,
|
|
string memory path,
|
|
uint256 chainId,
|
|
string memory key,
|
|
TypeKind ty
|
|
) private {
|
|
bool success = false;
|
|
if (ty == TypeKind.Bool) {
|
|
try vm.parseTomlBool(content, path) returns (bool val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Bool, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlBoolArray(content, path) returns (bool[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Bool, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
} else if (ty == TypeKind.Address) {
|
|
try vm.parseTomlAddress(content, path) returns (address val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Address, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlAddressArray(content, path) returns (address[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Address, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
} else if (ty == TypeKind.Bytes32) {
|
|
try vm.parseTomlBytes32(content, path) returns (bytes32 val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Bytes32, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlBytes32Array(content, path) returns (bytes32[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Bytes32, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
} else if (ty == TypeKind.Uint256) {
|
|
try vm.parseTomlUint(content, path) returns (uint256 val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Uint256, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlUintArray(content, path) returns (uint256[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Uint256, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
} else if (ty == TypeKind.Int256) {
|
|
try vm.parseTomlInt(content, path) returns (int256 val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Int256, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlIntArray(content, path) returns (int256[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Int256, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
} else if (ty == TypeKind.Bytes) {
|
|
try vm.parseTomlBytes(content, path) returns (bytes memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Bytes, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlBytesArray(content, path) returns (bytes[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.Bytes, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
} else if (ty == TypeKind.String) {
|
|
try vm.parseTomlString(content, path) returns (string memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.String, false);
|
|
success = true;
|
|
} catch {
|
|
try vm.parseTomlStringArray(content, path) returns (string[] memory val) {
|
|
_dataOf[chainId][key] = abi.encode(val);
|
|
_typeOf[chainId][key] = Type(TypeKind.String, true);
|
|
success = true;
|
|
} catch {}
|
|
}
|
|
}
|
|
|
|
if (!success) {
|
|
revert UnableToParseVariable(key);
|
|
}
|
|
}
|
|
|
|
// -- HELPER FUNCTIONS -----------------------------------------------------
|
|
|
|
/// @notice Enable or disable automatic writing to the TOML file on `set`.
|
|
/// Can only be enabled when scripting.
|
|
function writeUpdatesBackToFile(bool enabled) public {
|
|
if (enabled && !vm.isContext(VmSafe.ForgeContext.ScriptGroup)) {
|
|
revert WriteToFileInForbiddenCtxt();
|
|
}
|
|
|
|
_writeToFile = enabled;
|
|
}
|
|
|
|
/// @notice Resolves a chain alias or a chain id string to its numerical chain id.
|
|
/// @param aliasOrId The string representing the chain alias (i.e. "mainnet") or a numerical ID (i.e. "1").
|
|
/// @return The numerical chain ID.
|
|
/// @dev It first attempts to parse the input as a number. If that fails, it uses `vm.getChain` to resolve a named alias.
|
|
/// Reverts if the alias is not valid or not a number.
|
|
function resolveChainId(string memory aliasOrId) public view returns (uint256) {
|
|
try vm.parseUint(aliasOrId) returns (uint256 chainId) {
|
|
return chainId;
|
|
} catch {
|
|
try vm.getChain(aliasOrId) returns (VmSafe.Chain memory chainInfo) {
|
|
return chainInfo.chainId;
|
|
} catch {
|
|
revert InvalidChainKey(aliasOrId);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// @dev Retrieves the chain key/alias from the configuration based on the chain ID.
|
|
function _getChainKeyFromId(uint256 chainId) private view returns (string memory) {
|
|
for (uint256 i = 0; i < _chainKeys.length; i++) {
|
|
if (resolveChainId(_chainKeys[i]) == chainId) {
|
|
return _chainKeys[i];
|
|
}
|
|
}
|
|
revert ChainNotInitialized(chainId);
|
|
}
|
|
|
|
/// @dev Ensures type consistency when setting a value - prevents changing types unless uninitialized.
|
|
/// Updates type only when the previous type was `None`.
|
|
function _ensureTypeConsistency(uint256 chainId, string memory key, Type memory ty) private {
|
|
Type memory current = _typeOf[chainId][key];
|
|
|
|
if (current.kind == TypeKind.None) {
|
|
_typeOf[chainId][key] = ty;
|
|
} else {
|
|
current.assertEq(ty);
|
|
}
|
|
}
|
|
|
|
/// @dev Wraps a string in double quotes for JSON compatibility.
|
|
function _quote(string memory s) private pure returns (string memory) {
|
|
return string.concat('"', s, '"');
|
|
}
|
|
|
|
/// @dev Writes a JSON-formatted value to a specific key in the TOML file.
|
|
/// @param chainId The chain id to write under.
|
|
/// @param ty The type category ('bool', 'address', 'uint', 'bytes32', 'string', or 'bytes').
|
|
/// @param key The variable key name.
|
|
/// @param jsonValue The JSON-formatted value to write.
|
|
function _writeToToml(uint256 chainId, string memory ty, string memory key, string memory jsonValue) private {
|
|
string memory chainKey = _getChainKeyFromId(chainId);
|
|
string memory valueKey = string.concat("$.", chainKey, ".", ty, ".", key);
|
|
vm.writeToml(jsonValue, _filePath, valueKey);
|
|
}
|
|
|
|
// -- GETTER FUNCTIONS -----------------------------------------------------
|
|
|
|
/// @dev Reads a variable for a given chain id and key, and returns it in a generic container.
|
|
/// The caller should use `LibVariable` to safely coerce the type.
|
|
/// Example: `uint256 myVar = config.get("my_key").toUint256();`
|
|
///
|
|
/// @param chain_id The chain ID to read from.
|
|
/// @param key The key of the variable to retrieve.
|
|
/// @return `Variable` struct containing the type and the ABI-encoded value.
|
|
function get(uint256 chain_id, string memory key) public view returns (Variable memory) {
|
|
return Variable(_typeOf[chain_id][key], _dataOf[chain_id][key]);
|
|
}
|
|
|
|
/// @dev Reads a variable for the current chain and a given key, and returns it in a generic container.
|
|
/// The caller should use `LibVariable` to safely coerce the type.
|
|
/// Example: `uint256 myVar = config.get("my_key").toUint256();`
|
|
///
|
|
/// @param key The key of the variable to retrieve.
|
|
/// @return `Variable` struct containing the type and the ABI-encoded value.
|
|
function get(string memory key) public view returns (Variable memory) {
|
|
return get(vm.getChainId(), key);
|
|
}
|
|
|
|
/// @dev Checks the existence of a variable for a given chain ID and key, and returns a boolean.
|
|
/// Example: `bool hasKey = config.exists(1, "my_key");`
|
|
///
|
|
/// @param chain_id The chain ID to check.
|
|
/// @param key The variable key name.
|
|
/// @return `bool` indicating whether a variable with the given key exists.
|
|
function exists(uint256 chain_id, string memory key) public view returns (bool) {
|
|
return _dataOf[chain_id][key].length > 0;
|
|
}
|
|
|
|
/// @dev Checks the existence of a variable for the current chain id and a given key, and returns a boolean.
|
|
/// Example: `bool hasKey = config.exists("my_key");`
|
|
///
|
|
/// @param key The variable key name.
|
|
/// @return `bool` indicating whether a variable with the given key exists.
|
|
function exists(string memory key) public view returns (bool) {
|
|
return exists(vm.getChainId(), key);
|
|
}
|
|
|
|
/// @notice Returns the numerical chain ids for all configured chains.
|
|
function getChainIds() public view returns (uint256[] memory) {
|
|
string[] memory keys = _chainKeys;
|
|
|
|
uint256[] memory ids = new uint256[](keys.length);
|
|
for (uint256 i = 0; i < keys.length; i++) {
|
|
ids[i] = resolveChainId(keys[i]);
|
|
}
|
|
|
|
return ids;
|
|
}
|
|
|
|
/// @notice Reads the RPC URL for a specific chain id.
|
|
function getRpcUrl(uint256 chainId) public view returns (string memory) {
|
|
return _rpcOf[chainId];
|
|
}
|
|
|
|
/// @notice Reads the RPC URL for the current chain.
|
|
function getRpcUrl() public view returns (string memory) {
|
|
return _rpcOf[vm.getChainId()];
|
|
}
|
|
|
|
// -- SETTER FUNCTIONS (SINGLE VALUES) -------------------------------------
|
|
|
|
/// @notice Sets a boolean value for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, bool value) public {
|
|
Type memory ty = Type(TypeKind.Bool, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, vm.toString(value));
|
|
}
|
|
|
|
/// @notice Sets a boolean value for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, bool value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets an address value for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, address value) public {
|
|
Type memory ty = Type(TypeKind.Address, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, _quote(vm.toString(value)));
|
|
}
|
|
|
|
/// @notice Sets an address value for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, address value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a bytes32 value for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, bytes32 value) public {
|
|
Type memory ty = Type(TypeKind.Bytes32, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, _quote(vm.toString(value)));
|
|
}
|
|
|
|
/// @notice Sets a bytes32 value for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, bytes32 value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a uint256 value for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, uint256 value) public {
|
|
Type memory ty = Type(TypeKind.Uint256, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, vm.toString(value));
|
|
}
|
|
|
|
/// @notice Sets a uint256 value for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, uint256 value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets an int256 value for a given key and chain ID.
|
|
function set(uint256 chainId, string memory key, int256 value) public {
|
|
Type memory ty = Type(TypeKind.Int256, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, vm.toString(value));
|
|
}
|
|
|
|
/// @notice Sets an int256 value for a given key on the current chain.
|
|
function set(string memory key, int256 value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a string value for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, string memory value) public {
|
|
Type memory ty = Type(TypeKind.String, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, _quote(value));
|
|
}
|
|
|
|
/// @notice Sets a string value for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, string memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a bytes value for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, bytes memory value) public {
|
|
Type memory ty = Type(TypeKind.Bytes, false);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) _writeToToml(chainId, ty.kind.toTomlKey(), key, _quote(vm.toString(value)));
|
|
}
|
|
|
|
/// @notice Sets a bytes value for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, bytes memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
// -- SETTER FUNCTIONS (ARRAYS) --------------------------------------------
|
|
|
|
/// @notice Sets a boolean array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, bool[] memory value) public {
|
|
Type memory ty = Type(TypeKind.Bool, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, vm.toString(value[i]));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets a boolean array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, bool[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets an address array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, address[] memory value) public {
|
|
Type memory ty = Type(TypeKind.Address, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, _quote(vm.toString(value[i])));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets an address array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, address[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a bytes32 array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, bytes32[] memory value) public {
|
|
Type memory ty = Type(TypeKind.Bytes32, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, _quote(vm.toString(value[i])));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets a bytes32 array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, bytes32[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a uint256 array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, uint256[] memory value) public {
|
|
Type memory ty = Type(TypeKind.Uint256, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, vm.toString(value[i]));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets a uint256 array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, uint256[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a int256 array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, int256[] memory value) public {
|
|
Type memory ty = Type(TypeKind.Int256, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, vm.toString(value[i]));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets a int256 array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, int256[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a string array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, string[] memory value) public {
|
|
Type memory ty = Type(TypeKind.String, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, _quote(value[i]));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets a string array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, string[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
|
|
/// @notice Sets a bytes array for a given key and chain ID.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(uint256 chainId, string memory key, bytes[] memory value) public {
|
|
Type memory ty = Type(TypeKind.Bytes, true);
|
|
_ensureTypeConsistency(chainId, key, ty);
|
|
_dataOf[chainId][key] = abi.encode(value);
|
|
if (_writeToFile) {
|
|
string memory json = "[";
|
|
for (uint256 i = 0; i < value.length; i++) {
|
|
json = string.concat(json, _quote(vm.toString(value[i])));
|
|
if (i < value.length - 1) json = string.concat(json, ",");
|
|
}
|
|
json = string.concat(json, "]");
|
|
_writeToToml(chainId, ty.kind.toTomlKey(), key, json);
|
|
}
|
|
}
|
|
|
|
/// @notice Sets a bytes array for a given key on the current chain.
|
|
/// @dev Sets the cached value in storage and writes the change back to the TOML file if `autoWrite` is enabled.
|
|
function set(string memory key, bytes[] memory value) public {
|
|
set(vm.getChainId(), key, value);
|
|
}
|
|
}
|