codex-contracts-eth/contracts/Marketplace.sol

135 lines
3.7 KiB
Solidity
Raw Normal View History

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "./Collateral.sol";
contract Marketplace is Collateral {
uint256 public immutable collateral;
MarketplaceFunds private funds;
mapping(bytes32 => Request) private requests;
2022-02-21 11:31:37 +01:00
mapping(bytes32 => RequestState) private requestState;
mapping(bytes32 => Offer) private offers;
constructor(IERC20 _token, uint256 _collateral)
Collateral(_token)
marketplaceInvariant
{
collateral = _collateral;
}
function requestStorage(Request calldata request)
public
marketplaceInvariant
{
2022-02-17 11:00:18 +01:00
require(request.client == msg.sender, "Invalid client address");
2022-02-21 12:55:00 +01:00
bytes32 id = keccak256(abi.encode(request));
require(requests[id].client == address(0), "Request already exists");
2022-02-21 12:55:00 +01:00
requests[id] = request;
2022-02-21 12:55:00 +01:00
_createLock(id, request.expiry);
2022-02-21 12:55:00 +01:00
funds.received += request.maxPrice;
funds.balance += request.maxPrice;
2022-02-21 12:55:00 +01:00
transferFrom(msg.sender, request.maxPrice);
emit StorageRequested(id, request);
}
function offerStorage(Offer calldata offer) public marketplaceInvariant {
2022-02-21 12:55:00 +01:00
require(offer.host == msg.sender, "Invalid host address");
require(balanceOf(msg.sender) >= collateral, "Insufficient collateral");
2022-02-21 12:55:00 +01:00
Request storage request = requests[offer.requestId];
require(request.client != address(0), "Unknown request");
2022-02-21 12:16:27 +01:00
// solhint-disable-next-line not-rely-on-time
require(request.expiry > block.timestamp, "Request expired");
2022-02-21 12:55:00 +01:00
require(offer.price <= request.maxPrice, "Price too high");
2022-02-21 12:55:00 +01:00
bytes32 id = keccak256(abi.encode(offer));
require(offers[id].host == address(0), "Offer already exists");
offers[id] = offer;
2022-02-21 12:55:00 +01:00
_lock(msg.sender, offer.requestId);
2022-02-21 12:55:00 +01:00
emit StorageOffered(id, offer, offer.requestId);
}
2022-02-21 11:31:37 +01:00
function selectOffer(bytes32 id) public marketplaceInvariant {
Offer storage offer = offers[id];
require(offer.host != address(0), "Unknown offer");
// solhint-disable-next-line not-rely-on-time
require(offer.expiry > block.timestamp, "Offer expired");
2022-02-21 12:55:00 +01:00
2022-02-21 11:31:37 +01:00
Request storage request = requests[offer.requestId];
require(request.client == msg.sender, "Only client can select offer");
2022-02-21 12:55:00 +01:00
2022-02-21 11:31:37 +01:00
RequestState storage state = requestState[offer.requestId];
require(!state.offerSelected, "Offer already selected");
2022-02-21 12:55:00 +01:00
2022-02-21 11:31:37 +01:00
state.offerSelected = true;
2022-02-21 12:55:00 +01:00
2022-02-21 11:31:37 +01:00
_createLock(id, offer.expiry);
_lock(offer.host, id);
_unlock(offer.requestId);
2022-02-21 12:55:00 +01:00
2022-02-21 11:31:37 +01:00
uint256 difference = request.maxPrice - offer.price;
funds.sent += difference;
funds.balance -= difference;
token.transfer(request.client, difference);
2022-02-21 14:00:59 +01:00
emit OfferSelected(id, offer.requestId);
2022-02-21 11:31:37 +01:00
}
function _request(bytes32 id) internal view returns (Request storage) {
return requests[id];
}
function _offer(bytes32 id) internal view returns (Offer storage) {
return offers[id];
}
struct Request {
2022-02-17 11:00:18 +01:00
address client;
uint256 duration;
uint256 size;
bytes32 contentHash;
uint256 maxPrice;
2022-02-17 12:24:27 +01:00
uint256 expiry;
bytes32 nonce;
}
2022-02-21 11:31:37 +01:00
struct RequestState {
bool offerSelected;
}
struct Offer {
2022-02-17 11:06:14 +01:00
address host;
bytes32 requestId;
uint256 price;
uint256 expiry;
}
event StorageRequested(bytes32 requestId, Request request);
event StorageOffered(bytes32 offerId, Offer offer, bytes32 indexed requestId);
event OfferSelected(bytes32 offerId, bytes32 indexed requestId);
modifier marketplaceInvariant() {
MarketplaceFunds memory oldFunds = funds;
_;
assert(funds.received >= oldFunds.received);
assert(funds.sent >= oldFunds.sent);
assert(funds.received == funds.balance + funds.sent);
}
struct MarketplaceFunds {
uint256 balance;
uint256 received;
uint256 sent;
}
}