Skip to content

Commit

Permalink
Merge pull request #17 from bcnmy/feat/test-coverage
Browse files Browse the repository at this point in the history
feat: update test + dev notes + more tests WIP
  • Loading branch information
livingrockrises authored Oct 7, 2024
2 parents 5972f42 + 27a1527 commit 1d9364e
Show file tree
Hide file tree
Showing 10 changed files with 185 additions and 168 deletions.
9 changes: 7 additions & 2 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,13 @@ jobs:
- name: Checkout
uses: actions/checkout@v3

- name: Set up Node.js
uses: actions/setup-node@v4
with:
node-version: '22' # Specify the Node.js version you want to use

- name: Install dependencies
run: yarn install --frozen-lockfile
run: yarn cache clean && yarn install

- name: Install Foundry
uses: foundry-rs/foundry-toolchain@v1
Expand All @@ -30,7 +35,7 @@ jobs:
uses: actions/checkout@v3

- name: Install dependencies
run: yarn install --frozen-lockfile
run: yarn cache clean && yarn install

- name: Create a fake .secret file
run: echo "primary twist rack vendor diagram image used route theme frown either will" > .secret
Expand Down
6 changes: 3 additions & 3 deletions .github/workflows/coverage.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,15 @@ jobs:
uses: actions/checkout@v3

- name: Set up Node.js
uses: actions/setup-node@v3
uses: actions/setup-node@v4
with:
node-version: "20.x"
node-version: '22' # Specify the Node.js version you want to use

- name: Install lcov (for genhtml)
run: sudo apt-get update && sudo apt-get install -y lcov

- name: Install JavaScript Dependencies
run: yarn install --frozen-lockfile
run: yarn cache clean && yarn install

- name: Install Foundry
uses: foundry-rs/foundry-toolchain@v1
Expand Down
4 changes: 2 additions & 2 deletions .solhint.json
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@
"imports-on-top": "error",
"ordering": "error",
"visibility-modifier-order": "error",
"code-complexity": ["error", 7],
"function-max-lines": ["error", 80],
"code-complexity": ["error", 10],
"function-max-lines": ["error", 90],
"max-line-length": ["warn", 120],
"no-empty-blocks": "error",
"no-unused-vars": "error",
Expand Down
2 changes: 1 addition & 1 deletion contracts/interfaces/oracles/IOracle.sol
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
pragma solidity ^0.8.27;

interface IOracle {
function decimals() external view returns (uint8);
Expand Down
48 changes: 24 additions & 24 deletions contracts/references/SampleTokenPaymaster.sol
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity ^0.8.23;
pragma solidity ^0.8.27;

// Import the required libraries and contracts
import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
Expand Down Expand Up @@ -41,17 +41,17 @@ contract TokenPaymaster is BasePaymaster, UniswapHelper, OracleHelper {
uint48 priceMaxAge;
}

/// @notice All 'price' variables are multiplied by this value to avoid rounding up
uint256 private constant PRICE_DENOMINATOR = 1e26;

TokenPaymasterConfig public tokenPaymasterConfig;

event ConfigUpdated(TokenPaymasterConfig tokenPaymasterConfig);

event UserOperationSponsored(address indexed user, uint256 actualTokenCharge, uint256 actualGasCost, uint256 actualTokenPriceWithMarkup);

event Received(address indexed sender, uint256 value);

/// @notice All 'price' variables are multiplied by this value to avoid rounding up
uint256 private constant PRICE_DENOMINATOR = 1e26;

TokenPaymasterConfig public tokenPaymasterConfig;

/// @notice Initializes the TokenPaymaster contract with the given parameters.
/// @param _token The ERC20 token used for transaction fee payments.
/// @param _entryPoint The EntryPoint contract used in the Account Abstraction infrastructure.
Expand Down Expand Up @@ -88,15 +88,13 @@ contract TokenPaymaster is BasePaymaster, UniswapHelper, OracleHelper {
transferOwnership(_owner);
}

/// @notice Updates the configuration for the Token Paymaster.
/// @param _tokenPaymasterConfig The new configuration struct.
function setTokenPaymasterConfig(
TokenPaymasterConfig memory _tokenPaymasterConfig
) public onlyOwner {
require(_tokenPaymasterConfig.priceMarkup <= 2 * PRICE_DENOMINATOR, "TPM: price markup too high");
require(_tokenPaymasterConfig.priceMarkup >= PRICE_DENOMINATOR, "TPM: price markup too low");
tokenPaymasterConfig = _tokenPaymasterConfig;
emit ConfigUpdated(_tokenPaymasterConfig);
receive() external payable {
emit Received(msg.sender, msg.value);
}

function withdrawEth(address payable recipient, uint256 amount) external onlyOwner {
(bool success,) = recipient.call{value: amount}("");
require(success, "withdraw failed");
}

function setUniswapConfiguration(
Expand All @@ -112,6 +110,17 @@ contract TokenPaymaster is BasePaymaster, UniswapHelper, OracleHelper {
SafeERC20.safeTransfer(token, to, amount);
}

/// @notice Updates the configuration for the Token Paymaster.
/// @param _tokenPaymasterConfig The new configuration struct.
function setTokenPaymasterConfig(
TokenPaymasterConfig memory _tokenPaymasterConfig
) public onlyOwner {
require(_tokenPaymasterConfig.priceMarkup <= 2 * PRICE_DENOMINATOR, "TPM: price markup too high");
require(_tokenPaymasterConfig.priceMarkup >= PRICE_DENOMINATOR, "TPM: price markup too low");
tokenPaymasterConfig = _tokenPaymasterConfig;
emit ConfigUpdated(_tokenPaymasterConfig);
}

/// @notice Validates a paymaster user operation and calculates the required token amount for the transaction.
/// @param userOp The user operation data.
/// @param requiredPreFund The maximum cost (in native token) the paymaster has to prefund.
Expand Down Expand Up @@ -205,13 +214,4 @@ contract TokenPaymaster is BasePaymaster, UniswapHelper, OracleHelper {
entryPoint.depositTo{value: address(this).balance}(address(this));
}
}

receive() external payable {
emit Received(msg.sender, msg.value);
}

function withdrawEth(address payable recipient, uint256 amount) external onlyOwner {
(bool success,) = recipient.call{value: amount}("");
require(success, "withdraw failed");
}
}
16 changes: 8 additions & 8 deletions contracts/sponsorship/BiconomySponsorshipPaymaster.sol
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,7 @@ contract BiconomySponsorshipPaymaster is
* @notice If _newVerifyingSigner is set to zero address, it will revert with an error.
* After setting the new signer address, it will emit an event VerifyingSignerChanged.
*/
function setSigner(address _newVerifyingSigner) external payable override onlyOwner {
function setSigner(address _newVerifyingSigner) external payable onlyOwner {
if (_isContract(_newVerifyingSigner)) revert VerifyingSignerCanNotBeContract();
if (_newVerifyingSigner == address(0)) {
revert VerifyingSignerCanNotBeZero();
Expand Down Expand Up @@ -124,7 +124,7 @@ contract BiconomySponsorshipPaymaster is
* @param value The new value to be set as the unaccountedEPGasOverhead.
* @notice only to be called by the owner of the contract.
*/
function setUnaccountedGas(uint256 value) external payable override onlyOwner {
function setUnaccountedGas(uint256 value) external payable onlyOwner {
if (value > UNACCOUNTED_GAS_LIMIT) {
revert UnaccountedGasTooHigh();
}
Expand Down Expand Up @@ -338,12 +338,6 @@ contract BiconomySponsorshipPaymaster is
return (context, _packValidationData(false, validUntil, validAfter));
}

function _withdrawERC20(IERC20 token, address target, uint256 amount) private {
if (target == address(0)) revert CanNotWithdrawToZeroAddress();
SafeTransferLib.safeTransfer(address(token), target, amount);
emit TokensWithdrawn(address(token), target, amount, msg.sender);
}

function _checkConstructorArgs(
address _verifyingSigner,
address _feeCollector,
Expand All @@ -364,4 +358,10 @@ contract BiconomySponsorshipPaymaster is
revert UnaccountedGasTooHigh();
}
}

function _withdrawERC20(IERC20 token, address target, uint256 amount) private {
if (target == address(0)) revert CanNotWithdrawToZeroAddress();
SafeTransferLib.safeTransfer(address(token), target, amount);
emit TokensWithdrawn(address(token), target, amount, msg.sender);
}
}
74 changes: 39 additions & 35 deletions contracts/token/BiconomyTokenPaymaster.sol
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,6 @@ import "./swaps/Uniswapper.sol";
* applied, and how
* to manage the assets received by the paymaster.
*/

// TODO: Review unnecessary overrides

contract BiconomyTokenPaymaster is
IBiconomyTokenPaymaster,
BasePaymaster,
Expand All @@ -55,14 +52,20 @@ contract BiconomyTokenPaymaster is
uint256 public independentPriceMarkup; // price markup used for independent mode
uint256 public priceExpiryDuration; // oracle price expiry duration
IOracle public nativeAssetToUsdOracle; // ETH -> USD price oracle
mapping(address => TokenInfo) independentTokenDirectory; // mapping of token address => info for tokens supported in
mapping(address => TokenInfo) public independentTokenDirectory; // mapping of token address => info for tokens supported in
// independent mode

// PAYMASTER_ID_OFFSET
uint256 private constant UNACCOUNTED_GAS_LIMIT = 50_000; // Limit for unaccounted gas cost
uint256 private constant PRICE_DENOMINATOR = 1e6; // Denominator used when calculating cost with price markup
uint256 private constant MAX_PRICE_MARKUP = 2e6; // 100% premium on price (2e6/PRICE_DENOMINATOR)

// Note: _priceExpiryDuration is common for all the feeds.
// Note: _independentPriceMarkup is common for all the independent tokens.
// Todo: add cases when uniswap is not available
// Note: swapTokenAndDeposit: we may not need to keep this onlyOwner


constructor(
address _owner,
address _verifyingSigner,
Expand Down Expand Up @@ -122,25 +125,6 @@ contract BiconomyTokenPaymaster is
}
}

/**
* Add a deposit in native currency for this paymaster, used for paying for transaction fees.
* This is ideally done by the entity who is managing the received ERC20 gas tokens.
*/
function deposit() public payable virtual override nonReentrant {
entryPoint.depositTo{ value: msg.value }(address(this));
}

/**
* @dev Withdraws the specified amount of gas tokens from the paymaster's balance and transfers them to the
* specified address.
* @param withdrawAddress The address to which the gas tokens should be transferred.
* @param amount The amount of gas tokens to withdraw.
*/
function withdrawTo(address payable withdrawAddress, uint256 amount) public override onlyOwner nonReentrant {
if (withdrawAddress == address(0)) revert CanNotWithdrawToZeroAddress();
entryPoint.withdrawTo(withdrawAddress, amount);
}

/**
* @dev pull tokens out of paymaster in case they were sent to the paymaster at any point.
* @param token the token deposit to withdraw
Expand Down Expand Up @@ -218,7 +202,7 @@ contract BiconomyTokenPaymaster is
* @notice If _newVerifyingSigner is set to zero address, it will revert with an error.
* After setting the new signer address, it will emit an event VerifyingSignerChanged.
*/
function setSigner(address _newVerifyingSigner) external payable override onlyOwner {
function setSigner(address _newVerifyingSigner) external payable onlyOwner {
if (_isContract(_newVerifyingSigner)) revert VerifyingSignerCanNotBeContract();
if (_newVerifyingSigner == address(0)) {
revert VerifyingSignerCanNotBeZero();
Expand All @@ -235,7 +219,7 @@ contract BiconomyTokenPaymaster is
* @param _newUnaccountedGas The new value to be set as the unaccounted gas value
* @notice only to be called by the owner of the contract.
*/
function setUnaccountedGas(uint256 _newUnaccountedGas) external payable override onlyOwner {
function setUnaccountedGas(uint256 _newUnaccountedGas) external payable onlyOwner {
if (_newUnaccountedGas > UNACCOUNTED_GAS_LIMIT) {
revert UnaccountedGasTooHigh();
}
Expand All @@ -251,7 +235,7 @@ contract BiconomyTokenPaymaster is
* @param _newIndependentPriceMarkup The new value to be set as the price markup
* @notice only to be called by the owner of the contract.
*/
function setPriceMarkup(uint256 _newIndependentPriceMarkup) external payable override onlyOwner {
function setPriceMarkup(uint256 _newIndependentPriceMarkup) external payable onlyOwner {
if (_newIndependentPriceMarkup > MAX_PRICE_MARKUP || _newIndependentPriceMarkup < PRICE_DENOMINATOR) {
// Not between 0% and 100% markup
revert InvalidPriceMarkup();
Expand All @@ -268,7 +252,7 @@ contract BiconomyTokenPaymaster is
* @param _newPriceExpiryDuration The new value to be set as the unaccounted gas value
* @notice only to be called by the owner of the contract.
*/
function setPriceExpiryDuration(uint256 _newPriceExpiryDuration) external payable override onlyOwner {
function setPriceExpiryDuration(uint256 _newPriceExpiryDuration) external payable onlyOwner {
uint256 oldPriceExpiryDuration = priceExpiryDuration;
assembly ("memory-safe") {
sstore(priceExpiryDuration.slot, _newPriceExpiryDuration)
Expand Down Expand Up @@ -301,7 +285,7 @@ contract BiconomyTokenPaymaster is
* @param _oracle The oracle to use for the specified token
* @notice only to be called by the owner of the contract.
*/
function updateTokenDirectory(address _tokenAddress, IOracle _oracle) external payable override onlyOwner {
function updateTokenDirectory(address _tokenAddress, IOracle _oracle) external payable onlyOwner {
if (_oracle.decimals() != 8) {
// Token -> USD will always have 8 decimals
revert InvalidOracleDecimals();
Expand Down Expand Up @@ -360,6 +344,25 @@ contract BiconomyTokenPaymaster is
entryPoint.depositTo{ value: amountOut }(address(this));
}

/**
* Add a deposit in native currency for this paymaster, used for paying for transaction fees.
* This is ideally done by the entity who is managing the received ERC20 gas tokens.
*/
function deposit() public payable virtual override nonReentrant {
entryPoint.depositTo{ value: msg.value }(address(this));
}

/**
* @dev Withdraws the specified amount of gas tokens from the paymaster's balance and transfers them to the
* specified address.
* @param withdrawAddress The address to which the gas tokens should be transferred.
* @param amount The amount of gas tokens to withdraw.
*/
function withdrawTo(address payable withdrawAddress, uint256 amount) public override onlyOwner nonReentrant {
if (withdrawAddress == address(0)) revert CanNotWithdrawToZeroAddress();
entryPoint.withdrawTo(withdrawAddress, amount);
}

/**
* return the hash we're going to sign off-chain (and validate on-chain)
* this method is called by the off-chain service, to sign the request.
Expand Down Expand Up @@ -524,7 +527,7 @@ contract BiconomyTokenPaymaster is

// Calculate the actual cost in tokens based on the actual gas cost and the token price
uint256 actualTokenAmount = (
(actualGasCost + (unaccountedGas) * actualUserOpFeePerGas) * appliedPriceMarkup * tokenPrice
(actualGasCost + (unaccountedGas * actualUserOpFeePerGas)) * appliedPriceMarkup * tokenPrice
) / (1e18 * PRICE_DENOMINATOR);

if (prechargedAmount > actualTokenAmount) {
Expand All @@ -539,12 +542,6 @@ contract BiconomyTokenPaymaster is
);
}

function _withdrawERC20(IERC20 token, address target, uint256 amount) private {
if (target == address(0)) revert CanNotWithdrawToZeroAddress();
SafeTransferLib.safeTransfer(address(token), target, amount);
emit TokensWithdrawn(address(token), target, amount, msg.sender);
}

/// @notice Fetches the latest token price.
/// @return price The latest token price fetched from the oracles.
function getPrice(address tokenAddress) internal view returns (uint192 price) {
Expand Down Expand Up @@ -578,4 +575,11 @@ contract BiconomyTokenPaymaster is
}
price = uint192(int192(answer));
}

function _withdrawERC20(IERC20 token, address target, uint256 amount) private {
if (target == address(0)) revert CanNotWithdrawToZeroAddress();
SafeTransferLib.safeTransfer(address(token), target, amount);
emit TokensWithdrawn(address(token), target, amount, msg.sender);
}

}
Loading

0 comments on commit 1d9364e

Please sign in to comment.