179 lines
6.3 KiB
Solidity
179 lines
6.3 KiB
Solidity
pragma solidity ^0.8.0;
|
|
|
|
import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol";
|
|
|
|
interface IToken {
|
|
function mintFor(address contributorAccount, uint256 amount) external;
|
|
function balanceOf(address contributorAccount) external view returns (uint256);
|
|
}
|
|
interface IContributionBalance {
|
|
function totalKreditsEarnedByContributor(uint32 contributorId, bool confirmedOnly) external view returns (uint32 amount);
|
|
function balanceOf(address owner) external view returns (uint256);
|
|
}
|
|
|
|
contract Contributor is Initializable {
|
|
address public deployer;
|
|
IContributionBalance public contributionContract;
|
|
IToken public tokenContract;
|
|
|
|
struct Contributor {
|
|
address account;
|
|
bytes32 hashDigest;
|
|
uint8 hashFunction;
|
|
uint8 hashSize;
|
|
bool exists;
|
|
uint32 kreditsWithdrawn;
|
|
}
|
|
|
|
mapping (address => uint32) public contributorIds;
|
|
mapping (uint32 => Contributor) public contributors;
|
|
uint32 public contributorsCount;
|
|
|
|
address public profileManager;
|
|
|
|
event ContributorProfileUpdated(uint32 id, bytes32 oldHashDigest, bytes32 newHashDigest); // what should be logged
|
|
event ContributorAccountUpdated(uint32 id, address oldAccount, address newAccount);
|
|
event ContributorAdded(uint32 id, address account);
|
|
|
|
modifier onlyCore {
|
|
require(addressIsCore(tx.origin), "Core only");
|
|
_;
|
|
}
|
|
|
|
modifier onlyContributors {
|
|
require(addressExists(msg.sender) && contributionContract.balanceOf(msg.sender) > 0, "Contributors only");
|
|
_;
|
|
}
|
|
|
|
function initialize(address profileManagerAddress) public initializer {
|
|
deployer = msg.sender;
|
|
profileManager = profileManagerAddress;
|
|
}
|
|
|
|
function reinitialize(address profileManagerAddress) public reinitializer(2) {
|
|
profileManager = profileManagerAddress;
|
|
}
|
|
|
|
function setContributionContract(address contribution) public onlyCore {
|
|
require(address(contributionContract) == address(0) || addressIsCore(msg.sender), "Core only");
|
|
contributionContract = IContributionBalance(contribution);
|
|
}
|
|
|
|
function setTokenContract(address token) public onlyCore {
|
|
require(address(tokenContract) == address(0) || addressIsCore(msg.sender), "Core only");
|
|
tokenContract = IToken(token);
|
|
}
|
|
|
|
function coreContributorsCount() public view returns (uint32) {
|
|
uint32 count = 0;
|
|
for (uint32 i = 1; i <= contributorsCount; i++) {
|
|
if (isCoreTeam(i)) {
|
|
count += 1;
|
|
}
|
|
}
|
|
return count;
|
|
}
|
|
|
|
function updateContributorAccount(uint32 id, address oldAccount, address newAccount) public onlyCore {
|
|
require(newAccount != address(0), "invalid new account address");
|
|
require(getContributorAddressById(id) == oldAccount, "contributor does not exist");
|
|
|
|
contributorIds[oldAccount] = 0;
|
|
contributorIds[newAccount] = id;
|
|
contributors[id].account = newAccount;
|
|
emit ContributorAccountUpdated(id, oldAccount, newAccount);
|
|
}
|
|
|
|
function updateContributorProfileHash(uint32 id, bytes32 hashDigest, uint8 hashFunction, uint8 hashSize) public onlyCore {
|
|
Contributor storage c = contributors[id];
|
|
bytes32 oldHashDigest = c.hashDigest;
|
|
c.hashDigest = hashDigest;
|
|
c.hashFunction = hashFunction;
|
|
c.hashSize = hashSize;
|
|
|
|
emit ContributorProfileUpdated(id, oldHashDigest, c.hashDigest);
|
|
}
|
|
|
|
function addContributor(address account, bytes32 hashDigest, uint8 hashFunction, uint8 hashSize) public {
|
|
require(!addressExists(account), "Address already in use");
|
|
require((msg.sender == profileManager) || addressIsCore(msg.sender), "Only core and profile manager");
|
|
uint32 _id = contributorsCount + 1;
|
|
assert(!contributors[_id].exists); // this can not be acually
|
|
Contributor storage c = contributors[_id];
|
|
c.exists = true;
|
|
c.hashDigest = hashDigest;
|
|
c.hashFunction = hashFunction;
|
|
c.hashSize = hashSize;
|
|
c.account = account;
|
|
c.kreditsWithdrawn = 0;
|
|
contributorIds[account] = _id;
|
|
|
|
contributorsCount += 1;
|
|
emit ContributorAdded(_id, account);
|
|
}
|
|
|
|
function isCoreTeam(uint32 id) public view returns (bool) {
|
|
// TODO: for simplicity we simply define the first contributors as core
|
|
// later this needs to be changed to something more dynamic
|
|
return id > 0 && id < 7;
|
|
}
|
|
|
|
function exists(uint32 id) public view returns (bool) {
|
|
return contributors[id].exists;
|
|
}
|
|
|
|
function addressIsCore(address account) public view returns (bool) {
|
|
// the deployer is always core
|
|
if(account == deployer) {
|
|
return true;
|
|
}
|
|
uint32 id = getContributorIdByAddress(account);
|
|
return isCoreTeam(id);
|
|
}
|
|
|
|
function addressExists(address account) public view returns (bool) {
|
|
return getContributorByAddress(account).exists;
|
|
}
|
|
|
|
function getContributorIdByAddress(address account) public view returns (uint32) {
|
|
return contributorIds[account];
|
|
}
|
|
|
|
function getContributorAddressById(uint32 id) public view returns (address) {
|
|
return contributors[id].account;
|
|
}
|
|
|
|
function getContributorByAddress(address account) internal view returns (Contributor memory) {
|
|
uint32 id = contributorIds[account];
|
|
return contributors[id];
|
|
}
|
|
|
|
function getContributorById(uint32 _id) public view returns (uint32 id, address account, bytes32 hashDigest, uint8 hashFunction, uint8 hashSize, bool isCore, uint256 balance, uint32 totalKreditsEarned, uint256 contributionsCount, bool exists, uint256 kreditsWithdrawn) {
|
|
id = _id;
|
|
Contributor storage c = contributors[_id];
|
|
account = c.account;
|
|
hashDigest = c.hashDigest;
|
|
hashFunction = c.hashFunction;
|
|
hashSize = c.hashSize;
|
|
isCore = isCoreTeam(id);
|
|
balance = tokenContract.balanceOf(c.account);
|
|
totalKreditsEarned = contributionContract.totalKreditsEarnedByContributor(_id, true);
|
|
contributionsCount = contributionContract.balanceOf(c.account);
|
|
exists = c.exists;
|
|
kreditsWithdrawn = c.kreditsWithdrawn;
|
|
}
|
|
|
|
function withdraw() public onlyContributors {
|
|
uint32 id = getContributorIdByAddress(msg.sender);
|
|
Contributor storage c = contributors[id];
|
|
|
|
// TODO check if we need a failsafe for unconfirmed or malicious txs
|
|
uint32 confirmedKredits = contributionContract.totalKreditsEarnedByContributor(id, true);
|
|
uint32 amountWithdrawable = confirmedKredits - c.kreditsWithdrawn;
|
|
require (amountWithdrawable > 0, "No kredits available");
|
|
|
|
c.kreditsWithdrawn += amountWithdrawable;
|
|
tokenContract.mintFor(msg.sender, amountWithdrawable);
|
|
}
|
|
}
|