diff --git a/pkg/pool-weighted/contracts/managed/ControlledManagedPoolFactory.sol b/pkg/pool-weighted/contracts/managed/ControlledManagedPoolFactory.sol index 9aae2f7e47..6773cc9070 100644 --- a/pkg/pool-weighted/contracts/managed/ControlledManagedPoolFactory.sol +++ b/pkg/pool-weighted/contracts/managed/ControlledManagedPoolFactory.sol @@ -39,7 +39,8 @@ contract ControlledManagedPoolFactory { * @dev Deploys a new `ManagedPool`. */ function create( - ManagedPoolSettings.NewPoolParams memory poolParams, + ManagedPool.ManagedPoolParams memory params, + ManagedPoolSettings.ManagedPoolSettingsParams memory settingsParams, BasePoolController.BasePoolRights calldata basePoolRights, ManagedPoolController.ManagedPoolRights calldata managedPoolRights, uint256 minWeightChangeDuration, @@ -53,7 +54,7 @@ contract ControlledManagedPoolFactory { ); // Let the base factory deploy the pool (owner is the controller) - pool = ManagedPoolFactory(managedPoolFactory).create(poolParams, address(poolController)); + pool = ManagedPoolFactory(managedPoolFactory).create(params, settingsParams, address(poolController)); // Finally, initialize the controller poolController.initialize(pool); diff --git a/pkg/pool-weighted/contracts/managed/ManagedPool.sol b/pkg/pool-weighted/contracts/managed/ManagedPool.sol index cf2065d067..879515f4d9 100644 --- a/pkg/pool-weighted/contracts/managed/ManagedPool.sol +++ b/pkg/pool-weighted/contracts/managed/ManagedPool.sol @@ -58,32 +58,43 @@ contract ManagedPool is ManagedPoolSettings { uint256 private constant _PREMINTED_TOKEN_BALANCE = 2**(111); IExternalWeightedMath private immutable _weightedMath; + struct ManagedPoolParams { + string name; + string symbol; + address[] assetManagers; + } + + struct ManagedPoolConfigParams { + IVault vault; + IProtocolFeePercentagesProvider protocolFeeProvider; + IExternalWeightedMath weightedMath; + uint256 pauseWindowDuration; + uint256 bufferPeriodDuration; + } + constructor( - NewPoolParams memory params, - IVault vault, - IProtocolFeePercentagesProvider protocolFeeProvider, - IExternalWeightedMath weightedMath, - address owner, - uint256 pauseWindowDuration, - uint256 bufferPeriodDuration + ManagedPoolParams memory params, + ManagedPoolConfigParams memory configParams, + ManagedPoolSettingsParams memory settingsParams, + address owner ) NewBasePool( - vault, + configParams.vault, PoolRegistrationLib.registerComposablePool( - vault, + configParams.vault, IVault.PoolSpecialization.MINIMAL_SWAP_INFO, - params.tokens, + settingsParams.tokens, params.assetManagers ), params.name, params.symbol, - pauseWindowDuration, - bufferPeriodDuration, + configParams.pauseWindowDuration, + configParams.bufferPeriodDuration, owner ) - ManagedPoolSettings(params, protocolFeeProvider) + ManagedPoolSettings(settingsParams, configParams.protocolFeeProvider) { - _weightedMath = weightedMath; + _weightedMath = configParams.weightedMath; } function _getWeightedMath() internal view returns (IExternalWeightedMath) { diff --git a/pkg/pool-weighted/contracts/managed/ManagedPoolFactory.sol b/pkg/pool-weighted/contracts/managed/ManagedPoolFactory.sol index 7b0a3cdf6c..e8eb893249 100644 --- a/pkg/pool-weighted/contracts/managed/ManagedPoolFactory.sol +++ b/pkg/pool-weighted/contracts/managed/ManagedPoolFactory.sol @@ -50,23 +50,21 @@ contract ManagedPoolFactory is BasePoolFactory, FactoryWidePauseWindow { /** * @dev Deploys a new `ManagedPool`. The owner should be a contract, deployed by another factory. */ - function create(ManagedPoolSettings.NewPoolParams memory poolParams, address owner) - external - returns (address pool) - { + function create( + ManagedPool.ManagedPoolParams memory params, + ManagedPoolSettings.ManagedPoolSettingsParams memory settingsParams, + address owner + ) external returns (address pool) { (uint256 pauseWindowDuration, uint256 bufferPeriodDuration) = getPauseConfiguration(); - return - _create( - abi.encode( - poolParams, - getVault(), - getProtocolFeePercentagesProvider(), - _weightedMath, - owner, - pauseWindowDuration, - bufferPeriodDuration - ) - ); + ManagedPool.ManagedPoolConfigParams memory configParams = ManagedPool.ManagedPoolConfigParams({ + vault: getVault(), + protocolFeeProvider: getProtocolFeePercentagesProvider(), + weightedMath: _weightedMath, + pauseWindowDuration: pauseWindowDuration, + bufferPeriodDuration: bufferPeriodDuration + }); + + return _create(abi.encode(params, configParams, settingsParams, owner)); } } diff --git a/pkg/pool-weighted/contracts/managed/ManagedPoolSettings.sol b/pkg/pool-weighted/contracts/managed/ManagedPoolSettings.sol index 2dd6184b47..6dfdff31eb 100644 --- a/pkg/pool-weighted/contracts/managed/ManagedPoolSettings.sol +++ b/pkg/pool-weighted/contracts/managed/ManagedPoolSettings.sol @@ -91,12 +91,9 @@ abstract contract ManagedPoolSettings is NewBasePool, ProtocolFeeCache, IManaged // If mustAllowlistLPs is enabled, this is the list of addresses allowed to join the pool mapping(address => bool) private _allowedAddresses; - struct NewPoolParams { - string name; - string symbol; + struct ManagedPoolSettingsParams { IERC20[] tokens; uint256[] normalizedWeights; - address[] assetManagers; uint256 swapFeePercentage; bool swapEnabledOnStart; bool mustAllowlistLPs; @@ -104,7 +101,7 @@ abstract contract ManagedPoolSettings is NewBasePool, ProtocolFeeCache, IManaged uint256 aumFeeId; } - constructor(NewPoolParams memory params, IProtocolFeePercentagesProvider protocolFeeProvider) + constructor(ManagedPoolSettingsParams memory params, IProtocolFeePercentagesProvider protocolFeeProvider) ProtocolFeeCache( protocolFeeProvider, ProviderFeeIDs({ swap: ProtocolFeeType.SWAP, yield: ProtocolFeeType.YIELD, aum: params.aumFeeId }) @@ -114,7 +111,7 @@ abstract contract ManagedPoolSettings is NewBasePool, ProtocolFeeCache, IManaged _require(totalTokens >= _MIN_TOKENS, Errors.MIN_TOKENS); _require(totalTokens <= _MAX_TOKENS, Errors.MAX_TOKENS); - InputHelpers.ensureInputLengthMatch(totalTokens, params.normalizedWeights.length, params.assetManagers.length); + InputHelpers.ensureInputLengthMatch(totalTokens, params.normalizedWeights.length); // Validate and set initial fees _setManagementAumFeePercentage(params.managementAumFeePercentage); diff --git a/pkg/pool-weighted/contracts/test/MockManagedPool.sol b/pkg/pool-weighted/contracts/test/MockManagedPool.sol index b8ec7699c5..9aa45c4298 100644 --- a/pkg/pool-weighted/contracts/test/MockManagedPool.sol +++ b/pkg/pool-weighted/contracts/test/MockManagedPool.sol @@ -20,14 +20,11 @@ import "../ExternalWeightedMath.sol"; contract MockManagedPool is ManagedPool { constructor( - NewPoolParams memory params, - IVault vault, - IProtocolFeePercentagesProvider protocolFeeProvider, - ExternalWeightedMath weightedMath, - address owner, - uint256 pauseWindowDuration, - uint256 bufferPeriodDuration - ) ManagedPool(params, vault, protocolFeeProvider, weightedMath, owner, pauseWindowDuration, bufferPeriodDuration) { + ManagedPoolParams memory params, + ManagedPoolConfigParams memory configParams, + ManagedPoolSettingsParams memory settingsParams, + address owner + ) ManagedPool(params, configParams, settingsParams, owner) { // solhint-disable-previous-line no-empty-blocks } diff --git a/pkg/pool-weighted/contracts/test/MockManagedPoolSettings.sol b/pkg/pool-weighted/contracts/test/MockManagedPoolSettings.sol index d716316255..a9a9135ec5 100644 --- a/pkg/pool-weighted/contracts/test/MockManagedPoolSettings.sol +++ b/pkg/pool-weighted/contracts/test/MockManagedPoolSettings.sol @@ -24,29 +24,28 @@ contract MockManagedPoolSettings is ManagedPoolSettings { ExternalWeightedMath private immutable _weightedMath; constructor( - NewPoolParams memory params, + ManagedPoolSettingsParams memory settingsParams, IVault vault, IProtocolFeePercentagesProvider protocolFeeProvider, ExternalWeightedMath weightedMath, - address owner, - uint256 pauseWindowDuration, - uint256 bufferPeriodDuration + address[] memory asssetManagers, + address owner ) NewBasePool( vault, PoolRegistrationLib.registerPoolWithAssetManagers( vault, IVault.PoolSpecialization.MINIMAL_SWAP_INFO, - params.tokens, - params.assetManagers + settingsParams.tokens, + asssetManagers ), - params.name, - params.symbol, - pauseWindowDuration, - bufferPeriodDuration, + "MockManagedPoolName", + "MockManagedPoolSymbol", + 90 days, + 30 days, owner ) - ManagedPoolSettings(params, protocolFeeProvider) + ManagedPoolSettings(settingsParams, protocolFeeProvider) { _weightedMath = weightedMath; } @@ -55,7 +54,11 @@ contract MockManagedPoolSettings is ManagedPoolSettings { return _getVirtualSupply(); } - function _onInitializePool(address, address, bytes memory userData) internal override returns (uint256, uint256[] memory) { + function _onInitializePool( + address, + address, + bytes memory userData + ) internal override returns (uint256, uint256[] memory) { WeightedPoolUserData.JoinKind kind = userData.joinKind(); _require(kind == WeightedPoolUserData.JoinKind.INIT, Errors.UNINITIALIZED); diff --git a/pkg/pool-weighted/test/ControlledManagedPoolFactory.test.ts b/pkg/pool-weighted/test/ControlledManagedPoolFactory.test.ts index e74064b1bb..4670b97b17 100644 --- a/pkg/pool-weighted/test/ControlledManagedPoolFactory.test.ts +++ b/pkg/pool-weighted/test/ControlledManagedPoolFactory.test.ts @@ -73,12 +73,15 @@ describe('ControlledManagedPoolFactory', function () { const assetManagers: string[] = Array(tokens.length).fill(ZERO_ADDRESS); assetManagers[tokens.indexOf(tokens.DAI)] = assetManager.address; - const newPoolParams: ManagedPoolParams = { + const poolParams = { name: NAME, symbol: SYMBOL, + assetManagers, + }; + + const settingsParams: ManagedPoolParams = { tokens: tokens.addresses, normalizedWeights: WEIGHTS, - assetManagers: assetManagers, swapFeePercentage: POOL_SWAP_FEE_PERCENTAGE, swapEnabledOnStart: swapsEnabled, mustAllowlistLPs: mustAllowlistLPs, @@ -104,7 +107,14 @@ describe('ControlledManagedPoolFactory', function () { const receipt = await ( await controlledFactory .connect(manager) - .create(newPoolParams, basePoolRights, managedPoolRights, MIN_WEIGHT_CHANGE_DURATION, manager.address) + .create( + poolParams, + settingsParams, + basePoolRights, + managedPoolRights, + MIN_WEIGHT_CHANGE_DURATION, + manager.address + ) ).wait(); const event = expectEvent.inReceipt(receipt, 'ManagedPoolCreated'); diff --git a/pkg/pool-weighted/test/ManagedPool.test.ts b/pkg/pool-weighted/test/ManagedPool.test.ts index 1320534c24..f919fb9475 100644 --- a/pkg/pool-weighted/test/ManagedPool.test.ts +++ b/pkg/pool-weighted/test/ManagedPool.test.ts @@ -70,7 +70,6 @@ describe('ManagedPool', function () { owner: owner.address, aumFeeId: ProtocolFee.AUM, poolType: WeightedPoolType.MOCK_MANAGED_POOL, - mockContractName: 'MockManagedPool', ...overrides, }; return WeightedPool.create(params); diff --git a/pkg/pool-weighted/test/ManagedPoolFactory.test.ts b/pkg/pool-weighted/test/ManagedPoolFactory.test.ts index e67b65a033..3da9cb0fc3 100644 --- a/pkg/pool-weighted/test/ManagedPoolFactory.test.ts +++ b/pkg/pool-weighted/test/ManagedPoolFactory.test.ts @@ -59,12 +59,15 @@ describe('ManagedPoolFactory', function () { const assetManagers: string[] = Array(tokens.length).fill(ZERO_ADDRESS); assetManagers[tokens.indexOf(tokens.DAI)] = assetManager.address; - const newPoolParams: ManagedPoolParams = { + const poolParams = { name: NAME, symbol: SYMBOL, + assetManagers: assetManagers, + }; + + const settingsParams: ManagedPoolParams = { tokens: tokens.addresses, normalizedWeights: WEIGHTS, - assetManagers: assetManagers, swapFeePercentage: POOL_SWAP_FEE_PERCENTAGE, swapEnabledOnStart: swapsEnabled, mustAllowlistLPs: mustAllowlistLPs, @@ -72,7 +75,7 @@ describe('ManagedPoolFactory', function () { aumFeeId: ProtocolFee.AUM, }; - const receipt = await (await factory.connect(manager).create(newPoolParams, manager.address)).wait(); + const receipt = await (await factory.connect(manager).create(poolParams, settingsParams, manager.address)).wait(); const event = expectEvent.inReceipt(receipt, 'PoolCreated'); return deployedAt('ManagedPool', event.args.pool); diff --git a/pkg/pool-weighted/test/ManagedPoolSettings.test.ts b/pkg/pool-weighted/test/ManagedPoolSettings.test.ts index 3e6e408e8a..7e50f362b2 100644 --- a/pkg/pool-weighted/test/ManagedPoolSettings.test.ts +++ b/pkg/pool-weighted/test/ManagedPoolSettings.test.ts @@ -83,8 +83,7 @@ describe('ManagedPoolSettings', function () { const fullParams = { ...params, swapFeePercentage: INITIAL_SWAP_FEE, - poolType: WeightedPoolType.MOCK_MANAGED_POOL, - mockContractName: 'MockManagedPoolSettings', + poolType: WeightedPoolType.MOCK_MANAGED_POOL_SETTINGS, }; return WeightedPool.create(fullParams); } @@ -162,13 +161,6 @@ describe('ManagedPoolSettings', function () { expect(poolScalingFactors).to.deep.equal(tokenScalingFactors); }); - - it('sets asset managers', async () => { - await tokens.asyncEach(async (token, i) => { - const info = await pool.getTokenInfo(token); - expect(info.assetManager).to.eq(assetManagers[i]); - }); - }); }); }); } @@ -462,7 +454,6 @@ describe('ManagedPoolSettings', function () { context('when the sender is not the owner', () => { it('non-owners cannot update weights', async () => { const now = await currentTimestamp(); - await expect(pool.updateWeightsGradually(other, now, now, poolWeights)).to.be.revertedWith( 'SENDER_NOT_ALLOWED' ); diff --git a/pkg/pool-weighted/test/managed/ownerOnlyActions.test.ts b/pkg/pool-weighted/test/managed/ownerOnlyActions.test.ts index a7953734fb..4c45f94284 100644 --- a/pkg/pool-weighted/test/managed/ownerOnlyActions.test.ts +++ b/pkg/pool-weighted/test/managed/ownerOnlyActions.test.ts @@ -21,24 +21,24 @@ describe('ManagedPool owner only actions', () => { const circuitBreakerLib = await deploy('CircuitBreakerLib'); pool = await deploy('MockManagedPool', { args: [ + { name: '', symbol: '', assetManagers: new Array(2).fill(ZERO_ADDRESS) }, + { + vault: vault.address, + protocolFeeProvider: vault.getFeesProvider().address, + weightedMath: math.address, + pauseWindowDuration: 0, + bufferPeriodDuration: 0, + }, { - name: '', - symbol: '', tokens: tokens.addresses, normalizedWeights: [fp(0.5), fp(0.5)], - assetManagers: new Array(2).fill(ZERO_ADDRESS), swapFeePercentage: fp(0.05), swapEnabledOnStart: true, mustAllowlistLPs: false, managementAumFeePercentage: fp(0), aumFeeId: ProtocolFee.AUM, }, - vault.address, - vault.getFeesProvider().address, - math.address, ZERO_ADDRESS, - 0, - 0, ], libraries: { CircuitBreakerLib: circuitBreakerLib.address, diff --git a/pvt/benchmarks/misc.ts b/pvt/benchmarks/misc.ts index 5e1180d6b7..dc8a42b33c 100644 --- a/pvt/benchmarks/misc.ts +++ b/pvt/benchmarks/misc.ts @@ -81,12 +81,15 @@ export async function deployPool(vault: Vault, tokens: TokenList, poolName: Pool switch (poolName) { case 'ManagedPool': { - const newPoolParams: ManagedPoolParams = { + const userInputs = { name: name, symbol: symbol, + assetManagers: Array(tokens.length).fill(ZERO_ADDRESS), + }; + + const managedPoolSettings: ManagedPoolParams = { tokens: tokens.addresses, normalizedWeights: weights, - assetManagers: Array(tokens.length).fill(ZERO_ADDRESS), swapFeePercentage: swapFeePercentage, swapEnabledOnStart: true, mustAllowlistLPs: false, @@ -109,7 +112,7 @@ export async function deployPool(vault: Vault, tokens: TokenList, poolName: Pool canChangeMgmtFees: true, }; - params = [newPoolParams, basePoolRights, managedPoolRights, DAY, creator.address]; + params = [userInputs, managedPoolSettings, basePoolRights, managedPoolRights, DAY, creator.address]; break; } default: { diff --git a/pvt/helpers/src/models/pools/weighted/WeightedPool.ts b/pvt/helpers/src/models/pools/weighted/WeightedPool.ts index 410124d721..9c35692644 100644 --- a/pvt/helpers/src/models/pools/weighted/WeightedPool.ts +++ b/pvt/helpers/src/models/pools/weighted/WeightedPool.ts @@ -558,7 +558,11 @@ export default class WeightedPool extends BasePool { } private _isManagedPool() { - return this.poolType == WeightedPoolType.MANAGED_POOL || this.poolType == WeightedPoolType.MOCK_MANAGED_POOL; + return ( + this.poolType == WeightedPoolType.MANAGED_POOL || + this.poolType == WeightedPoolType.MOCK_MANAGED_POOL || + this.poolType == WeightedPoolType.MOCK_MANAGED_POOL_SETTINGS + ); } async setSwapEnabled(from: SignerWithAddress, swapEnabled: boolean): Promise { diff --git a/pvt/helpers/src/models/pools/weighted/WeightedPoolDeployer.ts b/pvt/helpers/src/models/pools/weighted/WeightedPoolDeployer.ts index f475e6481f..6416302a6c 100644 --- a/pvt/helpers/src/models/pools/weighted/WeightedPoolDeployer.ts +++ b/pvt/helpers/src/models/pools/weighted/WeightedPoolDeployer.ts @@ -76,7 +76,6 @@ export default { owner, from, aumFeeId, - mockContractName, } = params; let result: Promise; @@ -109,22 +108,26 @@ export default { { name: NAME, symbol: SYMBOL, + assetManagers: assetManagers, + }, + { + vault: vault.address, + protocolFeeProvider: vault.protocolFeesProvider.address, + weightedMath: math.address, + pauseWindowDuration, + bufferPeriodDuration, + }, + { tokens: tokens.addresses, normalizedWeights: weights, swapFeePercentage: swapFeePercentage, - assetManagers: assetManagers, swapEnabledOnStart: swapEnabledOnStart, mustAllowlistLPs: mustAllowlistLPs, managementAumFeePercentage: managementAumFeePercentage, aumProtocolFeesCollector: aumProtocolFeesCollector, aumFeeId: aumFeeId, }, - vault.address, - vault.protocolFeesProvider.address, - math.address, owner, - pauseWindowDuration, - bufferPeriodDuration, ], from, libraries: { @@ -135,22 +138,55 @@ export default { break; } case WeightedPoolType.MOCK_MANAGED_POOL: { - if (mockContractName == undefined) { - throw new Error('Mock contract name required to deploy mock base pool'); - } const addRemoveTokenLib = await deploy('v2-pool-weighted/ManagedPoolAddRemoveTokenLib'); const math = await deploy('v2-pool-weighted/ExternalWeightedMath'); const circuitBreakerLib = await deploy('v2-pool-weighted/CircuitBreakerLib'); - result = deploy(mockContractName, { + result = deploy('v2-pool-weighted/MockManagedPool', { args: [ { name: NAME, symbol: SYMBOL, + assetManagers: assetManagers, + }, + { + vault: vault.address, + protocolFeeProvider: vault.protocolFeesProvider.address, + weightedMath: math.address, + pauseWindowDuration, + bufferPeriodDuration, + }, + { + tokens: tokens.addresses, + normalizedWeights: weights, + swapFeePercentage: swapFeePercentage, + swapEnabledOnStart: swapEnabledOnStart, + mustAllowlistLPs: mustAllowlistLPs, + managementAumFeePercentage: managementAumFeePercentage, + aumProtocolFeesCollector: aumProtocolFeesCollector, + aumFeeId: aumFeeId, + }, + owner, + ], + from, + libraries: { + CircuitBreakerLib: circuitBreakerLib.address, + ManagedPoolAddRemoveTokenLib: addRemoveTokenLib.address, + }, + }); + break; + } + case WeightedPoolType.MOCK_MANAGED_POOL_SETTINGS: { + const addRemoveTokenLib = await deploy('v2-pool-weighted/ManagedPoolAddRemoveTokenLib'); + + const math = await deploy('v2-pool-weighted/ExternalWeightedMath'); + const circuitBreakerLib = await deploy('v2-pool-weighted/CircuitBreakerLib'); + result = deploy('v2-pool-weighted/MockManagedPoolSettings', { + args: [ + { tokens: tokens.addresses, normalizedWeights: weights, swapFeePercentage: swapFeePercentage, - assetManagers: assetManagers, swapEnabledOnStart: swapEnabledOnStart, mustAllowlistLPs: mustAllowlistLPs, managementAumFeePercentage: managementAumFeePercentage, @@ -160,9 +196,8 @@ export default { vault.address, vault.protocolFeesProvider.address, math.address, + assetManagers, owner, - pauseWindowDuration, - bufferPeriodDuration, ], from, libraries: { @@ -254,12 +289,15 @@ export default { from, }); - const newPoolParams: ManagedPoolParams = { + const poolParams = { name: NAME, symbol: SYMBOL, + assetManagers, + }; + + const settingsParams: ManagedPoolParams = { tokens: tokens.addresses, normalizedWeights: weights, - assetManagers, swapFeePercentage: swapFeePercentage, swapEnabledOnStart: swapEnabledOnStart, mustAllowlistLPs: mustAllowlistLPs, @@ -284,9 +322,10 @@ export default { const tx = await controlledFactory .connect(from || ZERO_ADDRESS) - .create(newPoolParams, basePoolRights, managedPoolRights, DAY, from?.address || ZERO_ADDRESS); + .create(poolParams, settingsParams, basePoolRights, managedPoolRights, DAY, from?.address || ZERO_ADDRESS); const receipt = await tx.wait(); const event = expectEvent.inReceipt(receipt, 'ManagedPoolCreated'); + result = deployedAt('v2-pool-weighted/ManagedPool', event.args.pool); break; } diff --git a/pvt/helpers/src/models/pools/weighted/types.ts b/pvt/helpers/src/models/pools/weighted/types.ts index dd7f173832..05848d5423 100644 --- a/pvt/helpers/src/models/pools/weighted/types.ts +++ b/pvt/helpers/src/models/pools/weighted/types.ts @@ -13,6 +13,7 @@ export enum WeightedPoolType { LIQUIDITY_BOOTSTRAPPING_POOL, MANAGED_POOL, MOCK_MANAGED_POOL, + MOCK_MANAGED_POOL_SETTINGS, } export type RawWeightedPoolDeployment = { @@ -54,7 +55,6 @@ export type WeightedPoolDeployment = { owner?: string; admin?: SignerWithAddress; from?: SignerWithAddress; - mockContractName?: string; }; export type SwapWeightedPool = { @@ -204,11 +204,8 @@ export type ManagedPoolRights = { }; export type ManagedPoolParams = { - name: string; - symbol: string; tokens: string[]; normalizedWeights: BigNumberish[]; - assetManagers: string[]; swapFeePercentage: BigNumberish; swapEnabledOnStart: boolean; mustAllowlistLPs: boolean;