Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor: avoid micropayments #58

Merged
merged 11 commits into from
Dec 20, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
332 changes: 332 additions & 0 deletions contracts/BalanceTrackerFixedPrice.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,332 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.28;

import {IMech} from "./interfaces/IMech.sol";

interface IMechMarketplace {
function fee() external returns(uint256);
}

interface IToken {
/// @dev Transfers the token amount.
/// @param to Address to transfer to.
/// @param amount The amount to transfer.
/// @return True if the function execution is successful.
function transfer(address to, uint256 amount) external returns (bool);

/// @dev Transfers the token amount that was previously approved up until the maximum allowance.
/// @param from Account address to transfer from.
/// @param to Account address to transfer to.
/// @param amount Amount to transfer to.
/// @return True if the function execution is successful.
function transferFrom(address from, address to, uint256 amount) external returns (bool);
}

interface IWrappedToken {
function deposit() external payable;
}

/// @dev Only `manager` has a privilege, but the `sender` was provided.
/// @param sender Sender address.
/// @param manager Required sender address as a manager.
error ManagerOnly(address sender, address manager);

/// @dev Provided zero address.
error ZeroAddress();

/// @dev Provided zero value.
error ZeroValue();

/// @dev Not enough balance to cover costs.
/// @param current Current balance.
/// @param required Required balance.
error InsufficientBalance(uint256 current, uint256 required);

/// @dev Caught reentrancy violation.
error ReentrancyGuard();

/// @dev Account is unauthorized.
/// @param account Account address.
error UnauthorizedAccount(address account);

/// @dev No incoming msg.value is allowed.
/// @param amount Value amount.
error NoDepositAllowed(uint256 amount);

/// @dev Payload length is incorrect.
/// @param provided Provided payload length.
/// @param expected Expected payload length.
error InvalidPayloadLength(uint256 provided, uint256 expected);

/// @dev Failure of a transfer.
/// @param token Address of a token.
/// @param from Address `from`.
/// @param to Address `to`.
/// @param amount Amount value.
error TransferFailed(address token, address from, address to, uint256 amount);

contract BalanceTrackerFixedPrice {
event MechPaymentCalculated(address indexed mech, uint256 indexed requestId, uint256 deliveryRate, uint256 rateDiff);
event Deposit(address indexed account, address indexed token, uint256 amount);
event Withdraw(address indexed account, address indexed token, uint256 amount);
event Drained(address indexed token, uint256 collectedFees);

// Mech marketplace address
address public immutable mechMarketplace;

Check warning on line 75 in contracts/BalanceTrackerFixedPrice.sol

View workflow job for this annotation

GitHub Actions / build

Immutable variables name are set to be in capitalized SNAKE_CASE
// Wrapped native token address
address public immutable wrappedNativeToken;

Check warning on line 77 in contracts/BalanceTrackerFixedPrice.sol

View workflow job for this annotation

GitHub Actions / build

Immutable variables name are set to be in capitalized SNAKE_CASE
// Buy back burner address
address public immutable buyBackBurner;

Check warning on line 79 in contracts/BalanceTrackerFixedPrice.sol

View workflow job for this annotation

GitHub Actions / build

Immutable variables name are set to be in capitalized SNAKE_CASE
// Reentrancy lock
uint256 internal _locked = 1;

// Map of requester => map of (token => current debit balance)
mapping(address => mapping(address => uint256)) public mapRequesterBalances;
// Map of mech => map of (token => current debit balance)
mapping(address => mapping(address => uint256)) public mapMechBalances;
// Map of token => collected fees
mapping(address => uint256) public mapCollectedFees;
// Map of requestId => token
mapping(uint256 => address) public mapRequestIdTokens;

/// @dev BalanceTrackerFixedPrice constructor.
/// @param _mechMarketplace Mech marketplace address.
/// @param _wrappedNativeToken Wrapped native token address.
/// @param _buyBackBurner Buy back burner address.
constructor(address _mechMarketplace, address _wrappedNativeToken, address _buyBackBurner) {
// Check for zero address
if (_mechMarketplace == address(0) || _wrappedNativeToken == address(0) || _buyBackBurner == address(0)) {
revert ZeroAddress();
}

mechMarketplace = _mechMarketplace;
wrappedNativeToken = _wrappedNativeToken;
buyBackBurner = _buyBackBurner;
}

function _wrap(uint256 amount) internal virtual {
IWrappedToken(wrappedNativeToken).deposit{value: amount}();
}

// Check and record delivery rate
function checkAndRecordDeliveryRate(
address mech,
address requester,
uint256 requestId,
bytes memory paymentData
) external payable {
// Check for marketplace access
if (msg.sender != mechMarketplace) {
revert ManagerOnly(msg.sender, mechMarketplace);
}

// Get mech max delivery rate
uint256 maxDeliveryRate = IMech(mech).maxDeliveryRate();

// Get payment token
address token;
if (paymentData.length == 32) {
// Extract token address
token = abi.decode(paymentData, (address));
if (token != address(0) && msg.value > 0) {
revert NoDepositAllowed(msg.value);
}
} else if (paymentData.length > 0) {
revert InvalidPayloadLength(paymentData.length, 32);
}

// Get account balance
uint256 balance = mapRequesterBalances[requester][token];
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Get requester balance - it must be sufficient to cover request.


// Check the request delivery rate for a fixed price
if (balance < maxDeliveryRate) {
revert InsufficientBalance(balance, maxDeliveryRate);
}

// Record request token
mapRequestIdTokens[requestId] = token;

// Adjust account balance
balance -= maxDeliveryRate;
mapRequesterBalances[requester][token] = balance;
Comment on lines +150 to +151
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Requester balance is adjusted with the mech max delivery rate

}

/// @dev Finalizes mech delivery rate based on requested and actual ones.
/// @param mech Delivery mech address.
/// @param requester Requester address.
/// @param requestId Request Id.
/// @param maxDeliveryRate Requested max delivery rate.
function finalizeDeliveryRate(address mech, address requester, uint256 requestId, uint256 maxDeliveryRate) external {
// Check for marketplace access
if (msg.sender != mechMarketplace) {
revert ManagerOnly(msg.sender, mechMarketplace);
}

// Get actual delivery rate
uint256 actualDeliveryRate = IMech(mech).getFinalizedDeliveryRate(requestId);
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Gets the actual rate from the mech for the request delivery


// Check for zero value
if (actualDeliveryRate == 0) {
revert ZeroValue();
}

// Get token associated with request Id
address token = mapRequestIdTokens[requestId];

uint256 rateDiff;
if (maxDeliveryRate > actualDeliveryRate) {
// Return back requester overpayment debit
rateDiff = maxDeliveryRate - actualDeliveryRate;
mapRequesterBalances[requester][token] += rateDiff;
} else {
actualDeliveryRate = maxDeliveryRate;
}
Comment on lines +176 to +183
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now the delivery rate is exactly adjusted with wither the leftovers, or max being a cap.


// Record payment into mech balance
mapMechBalances[mech][token] += actualDeliveryRate;

emit MechPaymentCalculated(mech, requestId, actualDeliveryRate, rateDiff);
}

// TODO buyBackBurner does not account for other tokens but WETH, OLAS
/// @dev Drains collected fees by sending them to a Buy back burner contract.
function drain(address token) external {
// Reentrancy guard
if (_locked > 1) {
revert ReentrancyGuard();
}
_locked = 2;

uint256 localCollectedFees = mapCollectedFees[token];

// TODO Limits
// Check for zero value
if (localCollectedFees == 0) {
revert ZeroValue();
}

mapCollectedFees[token] = 0;

// Check token address
if (token == address (0)) {
// Wrap native tokens
_wrap(localCollectedFees);
// Transfer to Buy back burner
IToken(wrappedNativeToken).transfer(buyBackBurner, localCollectedFees);
} else {
IToken(token).transfer(buyBackBurner, localCollectedFees);
}

emit Drained(token, localCollectedFees);

_locked = 1;

Check warning on line 222 in contracts/BalanceTrackerFixedPrice.sol

View workflow job for this annotation

GitHub Actions / build

Possible reentrancy vulnerabilities. Avoid state changes after transfer
}

function _withdraw(address token, uint256 balance) internal {
bool success;
// Transfer mech balance
if (token == address(0)) {
// solhint-disable-next-line avoid-low-level-calls
(success, ) = msg.sender.call{value: balance}("");
} else {
IToken(token).transfer(msg.sender, balance);
}

// Check transfer
if (!success) {
revert TransferFailed(token, address(this), msg.sender, balance);
}
}

/// @dev Processes mech payment by withdrawing funds.
function processPayment(address token) external returns (uint256 mechPayment, uint256 marketplaceFee) {
// Reentrancy guard
if (_locked > 1) {
revert ReentrancyGuard();
}
_locked = 2;

// Get mech balance
uint256 balance = mapMechBalances[msg.sender][token];
// TODO limits?
if (balance == 0) {
revert ZeroValue();
}

// Calculate mech payment and marketplace fee
uint256 fee = IMechMarketplace(mechMarketplace).fee();
marketplaceFee = (balance * fee) / 10_000;
mechPayment = balance - marketplaceFee;
Comment on lines +258 to +259
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Calculate fee on withdraw only


// Check for zero value, although this must never happen
if (mechPayment == 0) {
revert ZeroValue();
}

// Adjust marketplace fee
mapCollectedFees[token] += marketplaceFee;

// Clear balances
mapMechBalances[msg.sender][token] = 0;

// Process withdraw
_withdraw(token, balance);

emit Withdraw(msg.sender, token, balance);

_locked = 1;
}

/// @dev Withdraws funds for a specific requester account.
function withdraw(address token) external {
// Reentrancy guard
if (_locked > 1) {
revert ReentrancyGuard();
}
_locked = 2;

// Get account balance
uint256 balance = mapRequesterBalances[msg.sender][token];
// TODO limits?
if (balance == 0) {
revert ZeroValue();
}

// Clear balances
mapRequesterBalances[msg.sender][token] = 0;

// Process withdraw
_withdraw(token, balance);

emit Withdraw(msg.sender, token, balance);

_locked = 1;
}

// Deposits token funds for requester.
function deposit(address token, uint256 amount) external {
// TODO Accept deposits from mechs as well?

if (token == address(0)) {
revert ZeroAddress();
}

// TODO: safe transfer?
IToken(token).transferFrom(msg.sender, address(0), amount);

// Update account balances
mapRequesterBalances[msg.sender][address(0)] += amount;

emit Deposit(msg.sender, token, amount);
}

// Deposits native funds for requester.
receive() external payable {

Check warning on line 324 in contracts/BalanceTrackerFixedPrice.sol

View workflow job for this annotation

GitHub Actions / build

Fallback function must be simple
// TODO Accept deposits from mechs as well?

// Update account balances
mapRequesterBalances[msg.sender][address(0)] += msg.value;

emit Deposit(msg.sender, address(0), msg.value);
}
}
56 changes: 0 additions & 56 deletions contracts/EscrowBase.sol

This file was deleted.

Loading
Loading