Pool Factory: Overview
The TwoCrypto-NG Factory allows the permissionless deployment of two-coin volatile asset pools, as well as gauges. The liquidity pool and LP token share the same contract.
Additionally, the Factory contract is the direct admin and fee receiver of all pools. In turn, the Factory is controlled by the CurveDAO.
Contract Source & Deployment
Source code for the Factory is available on Github. 
 A full list of all deployments can be found here.
Implementations¶
The TwoCrypto-NG Factory makes use of blueprint contracts (EIP-5202) to deploy liquidity pools and gauges.
Warning
Implementation contracts are upgradable. They can either be replaced, or additional implementation contracts can be added. Therefore, always make sure to check the most recent ones.
It utilizes four different implementations:
- pool_implementations, containing multiple blueprint contracts that are used to deploy the pools.
- gauge_implementation, containing a blueprint contract that is used when deploying gauges for pools. This is only available on Ethereum Mainnet.
- views_implementation, containing a view methods contract relevant for integrators and users looking to interact with the AMMs.
- math_implementation, containing math functions used in the AMM.
More on the Math Implementation and Views Implementation.
Query Implementations¶
pool_implementations¶
 Factory.pool_implementations(arg0: uint256) -> address: view
Getter for the pool implementation at index arg0.
Returns: Pool implementation (address).
| Input | Type | Description | 
|---|---|---|
| arg0 | uint256 | Index of pool implementation | 
gauge_implementation¶
 Factory.gauge_implementation() -> address: view
Getter for the current gauge implementation. Only Ethereum mainnet has a valid gauge implementation; on other chains, the implementation is set to ZERO_ADDRESS, as sidechain gauges need to be deployed via the RootChainGaugeFactory.
Returns: Gauge implementation (address).
views_implementation¶
 Factory.views_implementation() -> address: view
Getter for the current views contract implementation.
Returns: Views contract implementation (address).
math_implementation¶
 Factory.math_implementation() -> address: view
Getter for the current math contract implementation.
Returns: Math contract implementation (address).
Set New Implementations¶
New implementations can be set via the following admin-only functions:
set_pool_implementation¶
 Factory.set_pool_implementation(_pool_implementation: address, _implementation_index: uint256):
Guarded Method
This function is only callable by the admin of the contract.
Function to set a new pool implementation at a certain index. The Factory allows multiple pool implementations as some pools might require a different one.
Emits: UpdatePoolImplementation
| Input | Type | Description | 
|---|---|---|
| _pool_implementation | address | New pool implementation | 
| _implementation_index | uint256 | Index for the implementation | 
Source code
event UpdatePoolImplementation:
    _implemention_id: uint256
    _old_pool_implementation: address
    _new_pool_implementation: address
pool_implementations: public(HashMap[uint256, address])
@external
def set_pool_implementation(
    _pool_implementation: address, _implementation_index: uint256
):
    """
    @notice Set pool implementation
    @dev Set to empty(address) to prevent deployment of new pools
    @param _pool_implementation Address of the new pool implementation
    @param _implementation_index Index of the pool implementation
    """
    assert msg.sender == self.admin, "dev: admin only"
    log UpdatePoolImplementation(
        _implementation_index,
        self.pool_implementations[_implementation_index],
        _pool_implementation
    )
    self.pool_implementations[_implementation_index] = _pool_implementation
set_gauge_implementation¶
 Factory.set_gauge_implementation(_gauge_implementation: address):
Guarded Method
This function is only callable by the admin of the contract.
Function to set a new gauge implementation (blueprint contract). This implementation is only available on Ethereum mainnet. To deploy a gauge on a sidechain, this needs to be done through the RootChainGaugeFactory.
Emits: UpdateGaugeImplementation
| Input | Type | Description | 
|---|---|---|
| _gauge_implementation | address | New gauge implementation | 
Source code
event UpdateGaugeImplementation:
    _old_gauge_implementation: address
    _new_gauge_implementation: address
gauge_implementation: public(address)
@external
def set_gauge_implementation(_gauge_implementation: address):
    """
    @notice Set gauge implementation
    @dev Set to empty(address) to prevent deployment of new gauges
    @param _gauge_implementation Address of the new token implementation
    """
    assert msg.sender == self.admin, "dev: admin only"
    log UpdateGaugeImplementation(self.gauge_implementation, _gauge_implementation)
    self.gauge_implementation = _gauge_implementation
set_views_implementation¶
 Factory.set_views_implementation(_views_implementation: address):
Guarded Method
This function is only callable by the admin of the contract.
Function to set a new views contract.
Emits: UpdateViewsImplementation
| Input | Type | Description | 
|---|---|---|
| _views_implementation | address | New views contract implementation | 
Source code
event UpdateViewsImplementation:
    _old_views_implementation: address
    _new_views_implementation: address
views_implementation: public(address)
@external
def set_views_implementation(_views_implementation: address):
    """
    @notice Set views contract implementation
    @param _views_implementation Address of the new views contract
    """
    assert msg.sender == self.admin,  "dev: admin only"
    log UpdateViewsImplementation(self.views_implementation, _views_implementation)
    self.views_implementation = _views_implementation
set_math_implementation¶
 Factory.set_math_implementation(_math_implementation: address):
Guarded Method
This function is only callable by the admin of the contract.
Function to set a new math contract.
Emits: UpdateMathImplementation
| Input | Type | Description | 
|---|---|---|
| _math_implementation | address | New math contract implementation | 
Source code
event UpdateMathImplementation:
    _old_math_implementation: address
    _new_math_implementation: address
math_implementation: public(address)
@external
def set_math_implementation(_math_implementation: address):
    """
    @notice Set math implementation
    @param _math_implementation Address of the new math contract
    """
    assert msg.sender == self.admin, "dev: admin only"
    log UpdateMathImplementation(self.math_implementation, _math_implementation)
    self.math_implementation = _math_implementation
Fee Receiver¶
fee_receiver¶
 Factory.fee_receiver() -> address: view
Getter for the fee receiver address of the admin fee. The fee receiver is initially set by calling the initialize_ownership function. It can later be changed via the set_fee_receiver method.
Returns: fee receiver (address).
Source code
fee_receiver: public(address)
@external
def initialise_ownership(_fee_receiver: address, _admin: address):
    assert msg.sender == self.deployer
    assert self.admin == empty(address)
    self.fee_receiver = _fee_receiver
    self.admin = _admin
    log UpdateFeeReceiver(empty(address), _fee_receiver)
    log TransferOwnership(empty(address), _admin)
set_fee_receiver¶
 Factory.set_fee_receiver(_fee_receiver: address):
Guarded Method
This function is only callable by the admin of the contract.
Function to set a new fee receiver address.
Emits: UpdateFeeReceiver
| Input | Type | Description | 
|---|---|---|
| _fee_receiver | uint256 | New fee receiver address | 
Source code
event UpdateFeeReceiver:
    _old_fee_receiver: address
    _new_fee_receiver: address
fee_receiver: public(address)
@external
def set_fee_receiver(_fee_receiver: address):
    """
    @notice Set fee receiver
    @param _fee_receiver Address that fees are sent to
    """
    assert msg.sender == self.admin, "dev: admin only"
    log UpdateFeeReceiver(self.fee_receiver, _fee_receiver)
    self.fee_receiver = _fee_receiver