Contract Name:
TokenController
Contract Source Code:
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.18;
import "../interfaces/ISAFURAMaster.sol";
import "../interfaces/IMasterAwareV2.sol";
import "../interfaces/IMemberRoles.sol";
abstract contract MasterAwareV2 is IMasterAwareV2 {
ISAFURAMaster public master;
mapping(uint => address payable) public internalContracts;
modifier onlyMember {
require(
IMemberRoles(internalContracts[uint(ID.MR)]).checkRole(
msg.sender,
uint(IMemberRoles.Role.Member)
),
"Caller is not a member"
);
_;
}
modifier onlyAdvisoryBoard {
require(
IMemberRoles(internalContracts[uint(ID.MR)]).checkRole(
msg.sender,
uint(IMemberRoles.Role.AdvisoryBoard)
),
"Caller is not an advisory board member"
);
_;
}
modifier onlyInternal {
require(master.isInternal(msg.sender), "Caller is not an internal contract");
_;
}
modifier onlyMaster {
if (address(master) != address(0)) {
require(address(master) == msg.sender, "Not master");
}
_;
}
modifier onlyGovernance {
require(
master.checkIsAuthToGoverned(msg.sender),
"Caller is not authorized to govern"
);
_;
}
modifier onlyEmergencyAdmin {
require(
msg.sender == master.emergencyAdmin(),
"Caller is not emergency admin"
);
_;
}
modifier whenPaused {
require(master.isPause(), "System is not paused");
_;
}
modifier whenNotPaused {
require(!master.isPause(), "System is paused");
_;
}
function getInternalContractAddress(ID id) internal view returns (address payable) {
return internalContracts[uint(id)];
}
function changeMasterAddress(address masterAddress) public onlyMaster {
master = ISAFURAMaster(masterAddress);
}
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface IAssessment {
/* ========== DATA STRUCTURES ========== */
enum UintParams {
minVotingPeriodInDays,
stakeLockupPeriodInDays,
payoutCooldownInDays,
silentEndingPeriodInDays
}
struct Configuration {
// The minimum number of days the users can vote on polls
uint8 minVotingPeriodInDays;
// Number of days the users must wait from their last vote to withdraw their stake.
uint8 stakeLockupPeriodInDays;
// Number of days the users must wait after a poll closes to redeem payouts.
uint8 payoutCooldownInDays;
// Number of days representing the silence period. It is used to extend a poll's end date when
// a vote is cast during the silence period before the end date.
uint8 silentEndingPeriodInDays;
}
struct Stake {
uint96 amount;
uint104 rewardsWithdrawableFromIndex;
uint16 fraudCount;
/*uint32 unused,*/
}
// Holds data for a vote belonging to an assessor.
//
// The structure is used to keep track of user's votes. Each vote is used to determine
// a user's share of rewards or to create a fraud resolution which excludes fraudulent votes
// from the initial poll.
struct Vote {
// Identifier of the claim or incident
uint80 assessmentId;
// If the assessor votes to accept the event it's true otherwise it's false
bool accepted;
// Date and time when the vote was cast
uint32 timestamp;
// How many tokens were staked when the vote was cast
uint96 stakedAmount;
}
// Holds poll results for an assessment.
//
// The structure is used to keep track of all votes on a given assessment such as how many NXM were
// used to cast accept and deny votes as well as when the poll started and when it ends.
struct Poll {
// The amount of NXM from accept votes
uint96 accepted;
// The amount of NXM from deny votes
uint96 denied;
// Timestamp of when the poll started.
uint32 start;
// Timestamp of when the poll ends.
uint32 end;
}
// Holds data for an assessment belonging to an assessable event (individual claims, yield token
// incidents etc.).
//
// The structure is used to keep track of the total reward that should be distributed to
// assessors, the assessment deposit the claimants made to start the assessment, and the poll
// coresponding to this assessment.
struct Assessment {
// See Poll struct
Poll poll;
// The amount of NXM representing the assessment reward which is split among those who voted.
uint128 totalRewardInNXM;
// An amount of ETH which is sent back to the claimant when the poll result is positive,
// otherwise it is kep it the pool to back the assessment rewards. This allows claimants to
// open an unlimited amount of claims and prevents unbacked NXM to be minted through the
// assessment process.
uint128 assessmentDepositInETH;
}
/* ========== VIEWS ========== */
function getAssessmentsCount() external view returns (uint);
function assessments(uint id) external view
returns (Poll memory poll, uint128 totalReward, uint128 assessmentDeposit);
function getPoll(uint assessmentId) external view returns (Poll memory);
function getRewards(address user) external view returns (
uint totalPendingAmount,
uint withdrawableAmount,
uint withdrawableUntilIndex
);
function getVoteCountOfAssessor(address assessor) external view returns (uint);
function votesOf(address user, uint id) external view
returns (uint80 assessmentId, bool accepted, uint32 timestamp, uint96 stakedAmount);
function stakeOf(address user) external view
returns (uint96 amount, uint104 rewardsWithdrawableFromIndex, uint16 fraudCount);
function config() external view returns (
uint8 minVotingPeriodInDays,
uint8 stakeLockupPeriodInDays,
uint8 payoutCooldownInDays,
uint8 silentEndingPeriodInDays
);
function hasAlreadyVotedOn(address voter, uint pollId) external view returns (bool);
/* === MUTATIVE FUNCTIONS ==== */
function stake(uint96 amount) external;
function unstake(uint96 amount, address to) external;
function withdrawRewards(
address user,
uint104 batchSize
) external returns (uint withdrawn, uint withdrawnUntilIndex);
function withdrawRewardsTo(
address destination,
uint104 batchSize
) external returns (uint withdrawn, uint withdrawnUntilIndex);
function startAssessment(uint totalReward, uint assessmentDeposit) external
returns (uint);
function castVotes(
uint[] calldata assessmentIds,
bool[] calldata votes,
string[] calldata ipfsAssessmentDataHashes,
uint96 stakeIncrease
) external;
function submitFraud(bytes32 root) external;
function processFraud(
uint256 rootIndex,
bytes32[] calldata proof,
address assessor,
uint256 lastFraudulentVoteIndex,
uint96 burnAmount,
uint16 fraudCount,
uint256 voteBatchSize
) external;
function updateUintParameters(UintParams[] calldata paramNames, uint[] calldata values) external;
/* ========== EVENTS ========== */
event StakeDeposited(address user, uint104 amount);
event StakeWithdrawn(address indexed user, address to, uint96 amount);
event VoteCast(address indexed user, uint256 assessmentId, uint96 stakedAmount, bool accepted, string ipfsAssessmentDataHash);
event RewardWithdrawn(address user, address to, uint256 amount);
event FraudProcessed(uint assessmentId, address assessor, Poll poll);
event FraudSubmitted(bytes32 root);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface IGovernance {
event Proposal(
address indexed proposalOwner,
uint256 indexed proposalId,
uint256 dateAdd,
string proposalTitle,
string proposalSD,
string proposalDescHash
);
event Solution(
uint256 indexed proposalId,
address indexed solutionOwner,
uint256 indexed solutionId,
string solutionDescHash,
uint256 dateAdd
);
event Vote(
address indexed from,
uint256 indexed proposalId,
uint256 indexed voteId,
uint256 dateAdd,
uint256 solutionChosen
);
event RewardClaimed(
address indexed member,
uint gbtReward
);
/// @dev VoteCast event is called whenever a vote is cast that can potentially close the proposal.
event VoteCast (uint256 proposalId);
/// @dev ProposalAccepted event is called when a proposal is accepted so that a server can listen that can
/// call any offchain actions
event ProposalAccepted (uint256 proposalId);
/// @dev CloseProposalOnTime event is called whenever a proposal is created or updated to close it on time.
event CloseProposalOnTime (
uint256 indexed proposalId,
uint256 time
);
/// @dev ActionSuccess event is called whenever an onchain action is executed.
event ActionSuccess (
uint256 proposalId
);
struct DelegateVote {
address follower;
address leader;
uint lastUpd;
}
/// @dev Creates a new proposal
/// @param _proposalDescHash Proposal description hash through IPFS having Short and long description of proposal
/// @param _categoryId This id tells under which the proposal is categorized i.e. Proposal's Objective
function createProposal(
string calldata _proposalTitle,
string calldata _proposalSD,
string calldata _proposalDescHash,
uint _categoryId
)
external;
/// @dev Edits the details of an existing proposal and creates new version
/// @param _proposalId Proposal id that details needs to be updated
/// @param _proposalDescHash Proposal description hash having long and short description of proposal.
function updateProposal(
uint _proposalId,
string calldata _proposalTitle,
string calldata _proposalSD,
string calldata _proposalDescHash
)
external;
/// @dev Categorizes proposal to proceed further. Categories shows the proposal objective.
function categorizeProposal(
uint _proposalId,
uint _categoryId,
uint _incentives
)
external;
/// @dev Submit proposal with solution
/// @param _proposalId Proposal id
/// @param _solutionHash Solution hash contains parameters, values and description needed according to proposal
function submitProposalWithSolution(
uint _proposalId,
string calldata _solutionHash,
bytes calldata _action
) external;
/// @dev Creates a new proposal with solution and votes for the solution
/// @param _proposalDescHash Proposal description hash through IPFS having Short and long description of proposal
/// @param _categoryId This id tells under which the proposal is categorized i.e. Proposal's Objective
/// @param _solutionHash Solution hash contains parameters, values and description needed according to proposal
function createProposalwithSolution(
string calldata _proposalTitle,
string calldata _proposalSD,
string calldata _proposalDescHash,
uint _categoryId,
string calldata _solutionHash,
bytes calldata _action
) external;
/// @dev Casts vote using total balance including delegated tokens via managed staking pools
/// @param _proposalId Proposal id
/// @param _solutionChosen solution chosen while voting
function submitVote(uint _proposalId, uint _solutionChosen) external;
/// @dev Casts vote using total balance without delegated tokens via managed staking pools
/// @param _proposalId Proposal id
/// @param _solutionChosen solution chosen while voting
function submitVoteWithoutDelegations(uint _proposalId, uint _solutionChosen) external;
function closeProposal(uint _proposalId) external;
function tokenHoldingTime() external returns (uint);
function claimReward(address _memberAddress, uint _maxRecords) external returns (uint pendingDAppReward);
function proposal(uint _proposalId) external view returns (
uint proposalId,
uint category,
uint status,
uint finalVerdict,
uint totalReward
);
function canCloseProposal(uint _proposalId) external view returns (uint closeValue);
function allowedToCatgorize() external view returns (uint roleId);
function getPendingReward(address _memberAddress) external view returns (uint pendingDAppReward);
function getFollowers(address _add) external view returns (uint[] memory);
function followerDelegation(address _add) external view returns (uint delegationId);
function allDelegation(uint _delegationId) external view returns (address follower, address leader, uint lastUpd);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface IMasterAwareV2 {
// TODO: if you update this enum, update lib/constants.js as well
enum ID {
TC, // TokenController.sol
P1, // Pool.sol
MR, // MemberRoles.sol
MC, // MCR.sol
CO, // Cover.sol
SP, // StakingProducts.sol
PS, // LegacyPooledStaking.sol
GV, // Governance.sol
GW, // LegacyGateway.sol - removed
CL, // CoverMigrator.sol - removed
AS, // Assessment.sol
CI, // IndividualClaims.sol - Claims for Individuals
CG, // YieldTokenIncidents.sol - Claims for Groups
RA, // Ramm.sol
ST, // SafeTracker.sol
CP // CoverProducts.sol
}
function changeMasterAddress(address masterAddress) external;
function changeDependentContractAddress() external;
function internalContracts(uint) external view returns (address payable);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface IMemberRoles {
enum Role {Unassigned, AdvisoryBoard, Member, Owner, Auditor}
function join(address _userAddress, uint nonce, bytes calldata signature) external payable;
function switchMembership(address _newAddress) external;
function switchMembershipAndAssets(
address newAddress,
uint[] calldata coverIds,
uint[] calldata stakingTokenIds
) external;
function switchMembershipOf(address member, address _newAddress) external;
function totalRoles() external view returns (uint256);
function changeAuthorized(uint _roleId, address _newAuthorized) external;
function setKycAuthAddress(address _add) external;
function members(uint _memberRoleId) external view returns (uint, address[] memory memberArray);
function numberOfMembers(uint _memberRoleId) external view returns (uint);
function authorized(uint _memberRoleId) external view returns (address);
function roles(address _memberAddress) external view returns (uint[] memory);
function checkRole(address _memberAddress, uint _roleId) external view returns (bool);
function getMemberLengthForAllRoles() external view returns (uint[] memory totalMembers);
function memberAtIndex(uint _memberRoleId, uint index) external view returns (address, bool);
function membersLength(uint _memberRoleId) external view returns (uint);
event MemberRole(uint256 indexed roleId, bytes32 roleName, string roleDescription);
event MemberJoined(address indexed newMember, uint indexed nonce);
event switchedMembership(address indexed previousMember, address indexed newMember, uint timeStamp);
event MembershipWithdrawn(address indexed member, uint timestamp);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
import "./IPriceFeedOracle.sol";
struct SwapDetails {
uint104 minAmount;
uint104 maxAmount;
uint32 lastSwapTime;
// 2 decimals of precision. 0.01% -> 0.0001 -> 1e14
uint16 maxSlippageRatio;
}
struct Asset {
address assetAddress;
bool isCoverAsset;
bool isAbandoned;
}
interface IPool {
function swapOperator() external view returns (address);
function getAsset(uint assetId) external view returns (Asset memory);
function getAssets() external view returns (Asset[] memory);
function transferAssetToSwapOperator(address asset, uint amount) external;
function setSwapDetailsLastSwapTime(address asset, uint32 lastSwapTime) external;
function getAssetSwapDetails(address assetAddress) external view returns (SwapDetails memory);
function sendPayout(uint assetIndex, address payable payoutAddress, uint amount, uint ethDepositAmount) external;
function sendEth(address payoutAddress, uint amount) external;
function upgradeCapitalPool(address payable newPoolAddress) external;
function priceFeedOracle() external view returns (IPriceFeedOracle);
function getPoolValueInEth() external view returns (uint);
function calculateMCRRatio(uint totalAssetValue, uint mcrEth) external pure returns (uint);
function getInternalTokenPriceInAsset(uint assetId) external view returns (uint tokenPrice);
function getInternalTokenPriceInAssetAndUpdateTwap(uint assetId) external returns (uint tokenPrice);
function getTokenPrice() external view returns (uint tokenPrice);
function getMCRRatio() external view returns (uint);
function setSwapValue(uint value) external;
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface IPooledStaking {
struct Staker {
uint deposit; // total amount of deposit nxm
uint reward; // total amount that is ready to be claimed
address[] contracts; // list of contracts the staker has staked on
// staked amounts for each contract
mapping(address => uint) stakes;
// amount pending to be subtracted after all unstake requests will be processed
mapping(address => uint) pendingUnstakeRequestsTotal;
// flag to indicate the presence of this staker in the array of stakers of each contract
mapping(address => bool) isInContractStakers;
}
struct Burn {
uint amount;
uint burnedAt;
address contractAddress;
}
struct Reward {
uint amount;
uint rewardedAt;
address contractAddress;
}
struct UnstakeRequest {
uint amount;
uint unstakeAt;
address contractAddress;
address stakerAddress;
uint next; // id of the next unstake request in the linked list
}
struct ContractReward {
uint amount;
uint lastDistributionRound;
}
function accumulateReward(address contractAddress, uint amount) external;
function pushBurn(address contractAddress, uint amount) external;
function hasPendingActions() external view returns (bool);
function processPendingActions(uint maxIterations) external returns (bool finished);
function contractStake(address contractAddress) external view returns (uint);
function stakerReward(address staker) external view returns (uint);
function stakerDeposit(address staker) external view returns (uint);
function stakerContractStake(address staker, address contractAddress) external view returns (uint);
function withdraw(uint amount) external;
function withdrawForUser(address user) external;
function stakerMaxWithdrawable(address stakerAddress) external view returns (uint);
function withdrawReward(address stakerAddress) external;
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface Aggregator {
function decimals() external view returns (uint8);
function latestAnswer() external view returns (int);
}
interface IPriceFeedOracle {
struct OracleAsset {
Aggregator aggregator;
uint8 decimals;
}
function ETH() external view returns (address);
function assets(address) external view returns (Aggregator, uint8);
function getAssetToEthRate(address asset) external view returns (uint);
function getAssetForEth(address asset, uint ethIn) external view returns (uint);
function getEthForAsset(address asset, uint amount) external view returns (uint);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface IQuotationData {
function authQuoteEngine() external view returns (address);
function stlp() external view returns (uint);
function stl() external view returns (uint);
function pm() external view returns (uint);
function minDays() external view returns (uint);
function tokensRetained() external view returns (uint);
function kycAuthAddress() external view returns (address);
function refundEligible(address) external view returns (bool);
function holdedCoverIDStatus(uint) external view returns (uint);
function timestampRepeated(uint) external view returns (bool);
enum HCIDStatus {NA, kycPending, kycPass, kycFailedOrRefunded, kycPassNoCover}
enum CoverStatus {Active, ClaimAccepted, ClaimDenied, CoverExpired, ClaimSubmitted, Requested}
function addInTotalSumAssuredSC(address _add, bytes4 _curr, uint _amount) external;
function subFromTotalSumAssuredSC(address _add, bytes4 _curr, uint _amount) external;
function subFromTotalSumAssured(bytes4 _curr, uint _amount) external;
function addInTotalSumAssured(bytes4 _curr, uint _amount) external;
function setTimestampRepeated(uint _timestamp) external;
/// @dev Creates a blank new cover.
function addCover(
uint16 _coverPeriod,
uint _sumAssured,
address payable _userAddress,
bytes4 _currencyCode,
address _scAddress,
uint premium,
uint premiumNXM
) external;
function addHoldCover(
address payable from,
address scAddress,
bytes4 coverCurr,
uint[] calldata coverDetails,
uint16 coverPeriod
) external;
function setRefundEligible(address _add, bool status) external;
function setHoldedCoverIDStatus(uint holdedCoverID, uint status) external;
function setKycAuthAddress(address _add) external;
function changeAuthQuoteEngine(address _add) external;
function getUintParameters(bytes8 code) external view returns (bytes8 codeVal, uint val);
function getProductDetails()
external
view
returns (
uint _minDays,
uint _pm,
uint _stl,
uint _stlp
);
function getCoverLength() external view returns (uint len);
function getAuthQuoteEngine() external view returns (address _add);
function getTotalSumAssured(bytes4 _curr) external view returns (uint amount);
function getAllCoversOfUser(address _add) external view returns (uint[] memory allCover);
function getUserCoverLength(address _add) external view returns (uint len);
function getCoverStatusNo(uint _cid) external view returns (uint8);
function getCoverPeriod(uint _cid) external view returns (uint32 cp);
function getCoverSumAssured(uint _cid) external view returns (uint sa);
function getCurrencyOfCover(uint _cid) external view returns (bytes4 curr);
function getValidityOfCover(uint _cid) external view returns (uint date);
function getscAddressOfCover(uint _cid) external view returns (uint, address);
function getCoverMemberAddress(uint _cid) external view returns (address payable _add);
function getCoverPremiumNXM(uint _cid) external view returns (uint _premiumNXM);
function getCoverDetailsByCoverID1(
uint _cid
)
external
view
returns (
uint cid,
address _memberAddress,
address _scAddress,
bytes4 _currencyCode,
uint _sumAssured,
uint premiumNXM
);
function getCoverDetailsByCoverID2(
uint _cid
)
external
view
returns (
uint cid,
uint8 status,
uint sumAssured,
uint16 coverPeriod,
uint validUntil
);
function getHoldedCoverDetailsByID1(
uint _hcid
)
external
view
returns (
uint hcid,
address scAddress,
bytes4 coverCurr,
uint16 coverPeriod
);
function getUserHoldedCoverLength(address _add) external view returns (uint);
function getUserHoldedCoverByIndex(address _add, uint index) external view returns (uint);
function getHoldedCoverDetailsByID2(
uint _hcid
)
external
view
returns (
uint hcid,
address payable memberAddress,
uint[] memory coverDetails
);
function getTotalSumAssuredSC(address _add, bytes4 _curr) external view returns (uint amount);
function changeCoverStatusNo(uint _cid, uint8 _stat) external;
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface ISAFURAMaster {
function tokenAddress() external view returns (address);
function owner() external view returns (address);
function emergencyAdmin() external view returns (address);
function masterInitialized() external view returns (bool);
function isInternal(address _add) external view returns (bool);
function isPause() external view returns (bool check);
function isMember(address _add) external view returns (bool);
function checkIsAuthToGoverned(address _add) external view returns (bool);
function getLatestAddress(bytes2 _contractName) external view returns (address payable contractAddress);
function contractAddresses(bytes2 code) external view returns (address payable);
function upgradeMultipleContracts(
bytes2[] calldata _contractCodes,
address payable[] calldata newAddresses
) external;
function removeContracts(bytes2[] calldata contractCodesToRemove) external;
function addNewInternalContracts(
bytes2[] calldata _contractCodes,
address payable[] calldata newAddresses,
uint[] calldata _types
) external;
function updateOwnerParameters(bytes8 code, address payable val) external;
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
interface ISAFURAToken {
function burn(uint256 amount) external returns (bool);
function burnFrom(address from, uint256 value) external returns (bool);
function operatorTransfer(address from, uint256 value) external returns (bool);
function mint(address account, uint256 amount) external;
function isLockedForMV(address member) external view returns (uint);
function whiteListed(address member) external view returns (bool);
function addToWhiteList(address _member) external returns (bool);
function removeFromWhiteList(address _member) external returns (bool);
function changeOperator(address _newOperator) external returns (bool);
function lockForMemberVote(address _of, uint _days) external;
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `recipient`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address recipient, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `sender` to `recipient` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 value);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
/* structs for io */
struct AllocationRequest {
uint productId;
uint coverId;
uint allocationId;
uint period;
uint gracePeriod;
bool useFixedPrice;
uint previousStart;
uint previousExpiration;
uint previousRewardsRatio;
uint globalCapacityRatio;
uint capacityReductionRatio;
uint rewardRatio;
uint globalMinPrice;
}
struct BurnStakeParams {
uint allocationId;
uint productId;
uint start;
uint period;
uint deallocationAmount;
}
interface IStakingPool {
/* structs for storage */
// stakers are grouped in tranches based on the timelock expiration
// tranche index is calculated based on the expiration date
// the initial proposal is to have 4 tranches per year (1 tranche per quarter)
struct Tranche {
uint128 stakeShares;
uint128 rewardsShares;
}
struct ExpiredTranche {
uint96 accNxmPerRewardShareAtExpiry;
uint96 stakeAmountAtExpiry; // nxm total supply is 6.7e24 and uint96.max is 7.9e28
uint128 stakeSharesSupplyAtExpiry;
}
struct Deposit {
uint96 lastAccNxmPerRewardShare;
uint96 pendingRewards;
uint128 stakeShares;
uint128 rewardsShares;
}
function initialize(
bool isPrivatePool,
uint initialPoolFee,
uint maxPoolFee,
uint _poolId,
string memory ipfsDescriptionHash
) external;
function processExpirations(bool updateUntilCurrentTimestamp) external;
function requestAllocation(
uint amount,
uint previousPremium,
AllocationRequest calldata request
) external returns (uint premium, uint allocationId);
function burnStake(uint amount, BurnStakeParams calldata params) external;
function depositTo(
uint amount,
uint trancheId,
uint requestTokenId,
address destination
) external returns (uint tokenId);
function withdraw(
uint tokenId,
bool withdrawStake,
bool withdrawRewards,
uint[] memory trancheIds
) external returns (uint withdrawnStake, uint withdrawnRewards);
function isPrivatePool() external view returns (bool);
function isHalted() external view returns (bool);
function manager() external view returns (address);
function getPoolId() external view returns (uint);
function getPoolFee() external view returns (uint);
function getMaxPoolFee() external view returns (uint);
function getActiveStake() external view returns (uint);
function getStakeSharesSupply() external view returns (uint);
function getRewardsSharesSupply() external view returns (uint);
function getRewardPerSecond() external view returns (uint);
function getAccNxmPerRewardsShare() external view returns (uint);
function getLastAccNxmUpdate() external view returns (uint);
function getFirstActiveTrancheId() external view returns (uint);
function getFirstActiveBucketId() external view returns (uint);
function getNextAllocationId() external view returns (uint);
function getDeposit(uint tokenId, uint trancheId) external view returns (
uint lastAccNxmPerRewardShare,
uint pendingRewards,
uint stakeShares,
uint rewardsShares
);
function getTranche(uint trancheId) external view returns (
uint stakeShares,
uint rewardsShares
);
function getExpiredTranche(uint trancheId) external view returns (
uint accNxmPerRewardShareAtExpiry,
uint stakeAmountAtExpiry,
uint stakeShareSupplyAtExpiry
);
function setPoolFee(uint newFee) external;
function setPoolPrivacy(bool isPrivatePool) external;
function getActiveAllocations(
uint productId
) external view returns (uint[] memory trancheAllocations);
function getTrancheCapacities(
uint productId,
uint firstTrancheId,
uint trancheCount,
uint capacityRatio,
uint reductionRatio
) external view returns (uint[] memory trancheCapacities);
/* ========== EVENTS ========== */
event StakeDeposited(address indexed user, uint256 amount, uint256 trancheId, uint256 tokenId);
event DepositExtended(address indexed user, uint256 tokenId, uint256 initialTrancheId, uint256 newTrancheId, uint256 topUpAmount);
event PoolPrivacyChanged(address indexed manager, bool isPrivate);
event PoolFeeChanged(address indexed manager, uint newFee);
event PoolDescriptionSet(string ipfsDescriptionHash);
event Withdraw(address indexed user, uint indexed tokenId, uint tranche, uint amountStakeWithdrawn, uint amountRewardsWithdrawn);
event StakeBurned(uint amount);
event Deallocated(uint productId);
event BucketExpired(uint bucketId);
event TrancheExpired(uint trancheId);
// Auth
error OnlyCoverContract();
error OnlyStakingProductsContract();
error OnlyManager();
error PrivatePool();
error SystemPaused();
error PoolHalted();
// Fees
error PoolFeeExceedsMax();
error MaxPoolFeeAbove100();
// Voting
error NxmIsLockedForGovernanceVote();
error ManagerNxmIsLockedForGovernanceVote();
// Deposit
error InsufficientDepositAmount();
error RewardRatioTooHigh();
// Staking NFTs
error InvalidTokenId();
error NotTokenOwnerOrApproved();
error InvalidStakingPoolForToken();
// Tranche & capacity
error NewTrancheEndsBeforeInitialTranche();
error RequestedTrancheIsNotYetActive();
error RequestedTrancheIsExpired();
error InsufficientCapacity();
// Allocation
error AlreadyDeallocated(uint allocationId);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity >=0.5.0;
import "./ISAFURAToken.sol";
interface ITokenController {
struct StakingPoolNXMBalances {
uint128 rewards;
uint128 deposits;
}
struct CoverInfo {
uint16 claimCount;
bool hasOpenClaim;
bool hasAcceptedClaim;
uint96 requestedPayoutAmount;
// note: still 128 bits available here, can be used later
}
struct StakingPoolOwnershipOffer {
address proposedManager;
uint96 deadline;
}
function coverInfo(uint id) external view returns (
uint16 claimCount,
bool hasOpenClaim,
bool hasAcceptedClaim,
uint96 requestedPayoutAmount
);
function withdrawCoverNote(
address _of,
uint[] calldata _coverIds,
uint[] calldata _indexes
) external;
function changeOperator(address _newOperator) external;
function operatorTransfer(address _from, address _to, uint _value) external returns (bool);
function burnFrom(address _of, uint amount) external returns (bool);
function addToWhitelist(address _member) external;
function removeFromWhitelist(address _member) external;
function mint(address _member, uint _amount) external;
function lockForMemberVote(address _of, uint _days) external;
function withdrawClaimAssessmentTokens(address[] calldata users) external;
function getLockReasons(address _of) external view returns (bytes32[] memory reasons);
function totalSupply() external view returns (uint);
function totalBalanceOf(address _of) external view returns (uint amount);
function totalBalanceOfWithoutDelegations(address _of) external view returns (uint amount);
function getTokenPrice() external view returns (uint tokenPrice);
function token() external view returns (ISAFURAToken);
function getStakingPoolManager(uint poolId) external view returns (address manager);
function getManagerStakingPools(address manager) external view returns (uint[] memory poolIds);
function isStakingPoolManager(address member) external view returns (bool);
function getStakingPoolOwnershipOffer(uint poolId) external view returns (address proposedManager, uint deadline);
function transferStakingPoolsOwnership(address from, address to) external;
function assignStakingPoolManager(uint poolId, address manager) external;
function createStakingPoolOwnershipOffer(uint poolId, address proposedManager, uint deadline) external;
function acceptStakingPoolOwnershipOffer(uint poolId) external;
function cancelStakingPoolOwnershipOffer(uint poolId) external;
function mintStakingPoolNXMRewards(uint amount, uint poolId) external;
function burnStakingPoolNXMRewards(uint amount, uint poolId) external;
function depositStakedNXM(address from, uint amount, uint poolId) external;
function withdrawNXMStakeAndRewards(address to, uint stakeToWithdraw, uint rewardsToWithdraw, uint poolId) external;
function burnStakedNXM(uint amount, uint poolId) external;
function stakingPoolNXMBalances(uint poolId) external view returns(uint128 rewards, uint128 deposits);
function tokensLocked(address _of, bytes32 _reason) external view returns (uint256 amount);
function getWithdrawableCoverNotes(
address coverOwner
) external view returns (
uint[] memory coverIds,
bytes32[] memory lockReasons,
uint withdrawableAmount
);
function getPendingRewards(address member) external view returns (uint);
}
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;
/**
* @dev Wrappers over Solidity's uintXX casting operators with added overflow
* checks.
*
* Downcasting from uint256 in Solidity does not revert on overflow. This can
* easily result in undesired exploitation or bugs, since developers usually
* assume that overflows raise errors. `SafeCast` restores this intuition by
* reverting the transaction when such an operation overflows.
*
* Using this library instead of the unchecked operations eliminates an entire
* class of bugs, so it's recommended to use it always.
*/
library SafeUintCast {
/**
* @dev Returns the downcasted uint248 from uint256, reverting on
* overflow (when the input is greater than largest uint248).
*
* Counterpart to Solidity's `uint248` operator.
*
* Requirements:
*
* - input must fit into 248 bits
*/
function toUint248(uint256 value) internal pure returns (uint248) {
require(value < 2**248, "SafeCast: value doesn\'t fit in 248 bits");
return uint248(value);
}
/**
* @dev Returns the downcasted uint240 from uint256, reverting on
* overflow (when the input is greater than largest uint240).
*
* Counterpart to Solidity's `uint240` operator.
*
* Requirements:
*
* - input must fit into 240 bits
*/
function toUint240(uint256 value) internal pure returns (uint240) {
require(value < 2**240, "SafeCast: value doesn\'t fit in 240 bits");
return uint240(value);
}
/**
* @dev Returns the downcasted uint232 from uint256, reverting on
* overflow (when the input is greater than largest uint232).
*
* Counterpart to Solidity's `uint232` operator.
*
* Requirements:
*
* - input must fit into 232 bits
*/
function toUint232(uint256 value) internal pure returns (uint232) {
require(value < 2**232, "SafeCast: value doesn\'t fit in 232 bits");
return uint232(value);
}
/**
* @dev Returns the downcasted uint224 from uint256, reverting on
* overflow (when the input is greater than largest uint224).
*
* Counterpart to Solidity's `uint224` operator.
*
* Requirements:
*
* - input must fit into 224 bits
*/
function toUint224(uint256 value) internal pure returns (uint224) {
require(value < 2**224, "SafeCast: value doesn\'t fit in 224 bits");
return uint224(value);
}
/**
* @dev Returns the downcasted uint216 from uint256, reverting on
* overflow (when the input is greater than largest uint216).
*
* Counterpart to Solidity's `uint216` operator.
*
* Requirements:
*
* - input must fit into 216 bits
*/
function toUint216(uint256 value) internal pure returns (uint216) {
require(value < 2**216, "SafeCast: value doesn\'t fit in 216 bits");
return uint216(value);
}
/**
* @dev Returns the downcasted uint208 from uint256, reverting on
* overflow (when the input is greater than largest uint208).
*
* Counterpart to Solidity's `uint208` operator.
*
* Requirements:
*
* - input must fit into 208 bits
*/
function toUint208(uint256 value) internal pure returns (uint208) {
require(value < 2**208, "SafeCast: value doesn\'t fit in 208 bits");
return uint208(value);
}
/**
* @dev Returns the downcasted uint200 from uint256, reverting on
* overflow (when the input is greater than largest uint200).
*
* Counterpart to Solidity's `uint200` operator.
*
* Requirements:
*
* - input must fit into 200 bits
*/
function toUint200(uint256 value) internal pure returns (uint200) {
require(value < 2**200, "SafeCast: value doesn\'t fit in 200 bits");
return uint200(value);
}
/**
* @dev Returns the downcasted uint192 from uint256, reverting on
* overflow (when the input is greater than largest uint192).
*
* Counterpart to Solidity's `uint192` operator.
*
* Requirements:
*
* - input must fit into 192 bits
*/
function toUint192(uint256 value) internal pure returns (uint192) {
require(value < 2**192, "SafeCast: value doesn\'t fit in 192 bits");
return uint192(value);
}
/**
* @dev Returns the downcasted uint184 from uint256, reverting on
* overflow (when the input is greater than largest uint184).
*
* Counterpart to Solidity's `uint184` operator.
*
* Requirements:
*
* - input must fit into 184 bits
*/
function toUint184(uint256 value) internal pure returns (uint184) {
require(value < 2**184, "SafeCast: value doesn\'t fit in 184 bits");
return uint184(value);
}
/**
* @dev Returns the downcasted uint176 from uint256, reverting on
* overflow (when the input is greater than largest uint176).
*
* Counterpart to Solidity's `uint176` operator.
*
* Requirements:
*
* - input must fit into 176 bits
*/
function toUint176(uint256 value) internal pure returns (uint176) {
require(value < 2**176, "SafeCast: value doesn\'t fit in 176 bits");
return uint176(value);
}
/**
* @dev Returns the downcasted uint168 from uint256, reverting on
* overflow (when the input is greater than largest uint168).
*
* Counterpart to Solidity's `uint168` operator.
*
* Requirements:
*
* - input must fit into 168 bits
*/
function toUint168(uint256 value) internal pure returns (uint168) {
require(value < 2**168, "SafeCast: value doesn\'t fit in 168 bits");
return uint168(value);
}
/**
* @dev Returns the downcasted uint160 from uint256, reverting on
* overflow (when the input is greater than largest uint160).
*
* Counterpart to Solidity's `uint160` operator.
*
* Requirements:
*
* - input must fit into 160 bits
*/
function toUint160(uint256 value) internal pure returns (uint160) {
require(value < 2**160, "SafeCast: value doesn\'t fit in 160 bits");
return uint160(value);
}
/**
* @dev Returns the downcasted uint152 from uint256, reverting on
* overflow (when the input is greater than largest uint152).
*
* Counterpart to Solidity's `uint152` operator.
*
* Requirements:
*
* - input must fit into 152 bits
*/
function toUint152(uint256 value) internal pure returns (uint152) {
require(value < 2**152, "SafeCast: value doesn\'t fit in 152 bits");
return uint152(value);
}
/**
* @dev Returns the downcasted uint144 from uint256, reverting on
* overflow (when the input is greater than largest uint144).
*
* Counterpart to Solidity's `uint144` operator.
*
* Requirements:
*
* - input must fit into 144 bits
*/
function toUint144(uint256 value) internal pure returns (uint144) {
require(value < 2**144, "SafeCast: value doesn\'t fit in 144 bits");
return uint144(value);
}
/**
* @dev Returns the downcasted uint136 from uint256, reverting on
* overflow (when the input is greater than largest uint136).
*
* Counterpart to Solidity's `uint136` operator.
*
* Requirements:
*
* - input must fit into 136 bits
*/
function toUint136(uint256 value) internal pure returns (uint136) {
require(value < 2**136, "SafeCast: value doesn\'t fit in 136 bits");
return uint136(value);
}
/**
* @dev Returns the downcasted uint128 from uint256, reverting on
* overflow (when the input is greater than largest uint128).
*
* Counterpart to Solidity's `uint128` operator.
*
* Requirements:
*
* - input must fit into 128 bits
*/
function toUint128(uint256 value) internal pure returns (uint128) {
require(value < 2**128, "SafeCast: value doesn\'t fit in 128 bits");
return uint128(value);
}
/**
* @dev Returns the downcasted uint120 from uint256, reverting on
* overflow (when the input is greater than largest uint120).
*
* Counterpart to Solidity's `uint120` operator.
*
* Requirements:
*
* - input must fit into 120 bits
*/
function toUint120(uint256 value) internal pure returns (uint120) {
require(value < 2**120, "SafeCast: value doesn\'t fit in 120 bits");
return uint120(value);
}
/**
* @dev Returns the downcasted uint112 from uint256, reverting on
* overflow (when the input is greater than largest uint112).
*
* Counterpart to Solidity's `uint112` operator.
*
* Requirements:
*
* - input must fit into 112 bits
*/
function toUint112(uint256 value) internal pure returns (uint112) {
require(value < 2**112, "SafeCast: value doesn\'t fit in 112 bits");
return uint112(value);
}
/**
* @dev Returns the downcasted uint104 from uint256, reverting on
* overflow (when the input is greater than largest uint104).
*
* Counterpart to Solidity's `uint104` operator.
*
* Requirements:
*
* - input must fit into 104 bits
*/
function toUint104(uint256 value) internal pure returns (uint104) {
require(value < 2**104, "SafeCast: value doesn\'t fit in 104 bits");
return uint104(value);
}
/**
* @dev Returns the downcasted uint96 from uint256, reverting on
* overflow (when the input is greater than largest uint96).
*
* Counterpart to Solidity's `uint104` operator.
*
* Requirements:
*
* - input must fit into 96 bits
*/
function toUint96(uint256 value) internal pure returns (uint96) {
require(value < 2**96, "SafeCast: value doesn\'t fit in 96 bits");
return uint96(value);
}
/**
* @dev Returns the downcasted uint88 from uint256, reverting on
* overflow (when the input is greater than largest uint88).
*
* Counterpart to Solidity's `uint104` operator.
*
* Requirements:
*
* - input must fit into 88 bits
*/
function toUint88(uint256 value) internal pure returns (uint88) {
require(value < 2**88, "SafeCast: value doesn\'t fit in 88 bits");
return uint88(value);
}
/**
* @dev Returns the downcasted uint80 from uint256, reverting on
* overflow (when the input is greater than largest uint80).
*
* Counterpart to Solidity's `uint104` operator.
*
* Requirements:
*
* - input must fit into 80 bits
*/
function toUint80(uint256 value) internal pure returns (uint80) {
require(value < 2**80, "SafeCast: value doesn\'t fit in 80 bits");
return uint80(value);
}
/**
* @dev Returns the downcasted uint64 from uint256, reverting on
* overflow (when the input is greater than largest uint64).
*
* Counterpart to Solidity's `uint64` operator.
*
* Requirements:
*
* - input must fit into 64 bits
*/
function toUint64(uint256 value) internal pure returns (uint64) {
require(value < 2**64, "SafeCast: value doesn\'t fit in 64 bits");
return uint64(value);
}
/**
* @dev Returns the downcasted uint56 from uint256, reverting on
* overflow (when the input is greater than largest uint56).
*
* Counterpart to Solidity's `uint56` operator.
*
* Requirements:
*
* - input must fit into 56 bits
*/
function toUint56(uint256 value) internal pure returns (uint56) {
require(value < 2**56, "SafeCast: value doesn\'t fit in 56 bits");
return uint56(value);
}
/**
* @dev Returns the downcasted uint48 from uint256, reverting on
* overflow (when the input is greater than largest uint48).
*
* Counterpart to Solidity's `uint48` operator.
*
* Requirements:
*
* - input must fit into 48 bits
*/
function toUint48(uint256 value) internal pure returns (uint48) {
require(value < 2**48, "SafeCast: value doesn\'t fit in 48 bits");
return uint48(value);
}
/**
* @dev Returns the downcasted uint40 from uint256, reverting on
* overflow (when the input is greater than largest uint40).
*
* Counterpart to Solidity's `uint40` operator.
*
* Requirements:
*
* - input must fit into 40 bits
*/
function toUint40(uint256 value) internal pure returns (uint40) {
require(value < 2**40, "SafeCast: value doesn\'t fit in 40 bits");
return uint40(value);
}
/**
* @dev Returns the downcasted uint32 from uint256, reverting on
* overflow (when the input is greater than largest uint32).
*
* Counterpart to Solidity's `uint32` operator.
*
* Requirements:
*
* - input must fit into 32 bits
*/
function toUint32(uint256 value) internal pure returns (uint32) {
require(value < 2**32, "SafeCast: value doesn\'t fit in 32 bits");
return uint32(value);
}
/**
* @dev Returns the downcasted uint24 from uint256, reverting on
* overflow (when the input is greater than largest uint24).
*
* Counterpart to Solidity's `uint24` operator.
*
* Requirements:
*
* - input must fit into 24 bits
*/
function toUint24(uint256 value) internal pure returns (uint24) {
require(value < 2**24, "SafeCast: value doesn\'t fit in 24 bits");
return uint24(value);
}
/**
* @dev Returns the downcasted uint16 from uint256, reverting on
* overflow (when the input is greater than largest uint16).
*
* Counterpart to Solidity's `uint16` operator.
*
* Requirements:
*
* - input must fit into 16 bits
*/
function toUint16(uint256 value) internal pure returns (uint16) {
require(value < 2**16, "SafeCast: value doesn\'t fit in 16 bits");
return uint16(value);
}
/**
* @dev Returns the downcasted uint8 from uint256, reverting on
* overflow (when the input is greater than largest uint8).
*
* Counterpart to Solidity's `uint8` operator.
*
* Requirements:
*
* - input must fit into 8 bits.
*/
function toUint8(uint256 value) internal pure returns (uint8) {
require(value < 2**8, "SafeCast: value doesn\'t fit in 8 bits");
return uint8(value);
}
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.18;
/**
* @dev Simple library to derive the staking pool address from the pool id without external calls
*/
library StakingPoolLibrary {
function getAddress(address factory, uint poolId) internal pure returns (address) {
bytes32 hash = keccak256(
abi.encodePacked(
hex'ff',
factory,
poolId, // salt
// init code hash of the MinimalBeaconProxy
// updated using patch-staking-pool-library.js script
hex'1eb804b66941a2e8465fa0951be9c8b855b7794ee05b0789ab22a02ee1298ebe' // init code hash
)
);
// cast last 20 bytes of hash to address
return address(uint160(uint(hash)));
}
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.18;
/**
* @title ERC1132 interface
* @dev see https://github.com/ethereum/EIPs/issues/1132
*/
contract LockHandler {
/**
* @dev Reasons why a user's tokens have been locked
*/
mapping(address => bytes32[]) public lockReason;
/**
* @dev locked token structure
*/
struct LockToken {
uint256 amount;
uint256 validity;
bool claimed;
}
/**
* @dev Holds number & validity of tokens locked for a given reason for
* a specified address
*/
mapping(address => mapping(bytes32 => LockToken)) public locked;
event Locked(address indexed _of, bytes32 indexed _reason, uint256 _amount, uint256 _validity);
event Unlocked(address indexed _of, bytes32 indexed _reason, uint256 _amount);
}
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.18;
import "../../interfaces/IAssessment.sol";
import "../../interfaces/IGovernance.sol";
import "../../interfaces/ISAFURAToken.sol";
import "../../interfaces/IPool.sol";
import "../../interfaces/IPooledStaking.sol";
import "../../interfaces/IQuotationData.sol";
import "../../interfaces/IStakingPool.sol";
import "../../interfaces/ITokenController.sol";
import "../../libraries/SafeUintCast.sol";
import "../../libraries/StakingPoolLibrary.sol";
import "../../abstract/MasterAwareV2.sol";
import "./external/LockHandler.sol";
contract TokenController is ITokenController, LockHandler, MasterAwareV2 {
using SafeUintCast for uint;
address public _unused_token;
address public _unused_pooledStaking;
uint public _unused_minCALockTime;
uint public _unused_claimSubmissionGracePeriod;
// coverId => CoverInfo
mapping(uint => CoverInfo) public override coverInfo;
// pool id => { rewards, deposits }
mapping(uint => StakingPoolNXMBalances) public override stakingPoolNXMBalances;
// pool id => manager
mapping(uint => address) internal stakingPoolManagers;
// pool id => offer
mapping(uint => StakingPoolOwnershipOffer) internal stakingPoolOwnershipOffers;
// manager => pool ids
mapping(address => uint[]) internal managerStakingPools;
ISAFURAToken public immutable token;
IQuotationData public immutable quotationData;
address public immutable claimsReward;
address public immutable stakingPoolFactory;
constructor(
address quotationDataAddress,
address claimsRewardAddress,
address stakingPoolFactoryAddress,
address tokenAddress
) {
quotationData = IQuotationData(quotationDataAddress);
claimsReward = claimsRewardAddress;
stakingPoolFactory = stakingPoolFactoryAddress;
token = ISAFURAToken(tokenAddress);
}
/* ========== DEPENDENCIES ========== */
function pooledStaking() internal view returns (IPooledStaking) {
return IPooledStaking(internalContracts[uint(ID.PS)]);
}
function assessment() internal view returns (IAssessment) {
return IAssessment(internalContracts[uint(ID.AS)]);
}
function governance() internal view returns (IGovernance) {
return IGovernance(internalContracts[uint(ID.GV)]);
}
function pool() internal view returns (IPool) {
return IPool(internalContracts[uint(ID.P1)]);
}
function changeDependentContractAddress() public override {
internalContracts[uint(ID.PS)] = master.getLatestAddress("PS");
internalContracts[uint(ID.AS)] = master.getLatestAddress("AS");
internalContracts[uint(ID.GV)] = master.getLatestAddress("GV");
internalContracts[uint(ID.P1)] = master.getLatestAddress("P1");
}
/**
* @dev to change the operator address
* @param _newOperator is the new address of operator
*/
function changeOperator(address _newOperator) public override onlyGovernance {
token.changeOperator(_newOperator);
}
/**
* @dev Proxies token transfer through this contract to allow staking when members are locked for voting
* @param _from Source address
* @param _to Destination address
* @param _value Amount to transfer
*/
function operatorTransfer(
address _from,
address _to,
uint _value
) external override onlyInternal returns (bool) {
token.operatorTransfer(_from, _value);
if (_to != address(this)) {
token.transfer(_to, _value);
}
return true;
}
/**
* @dev burns tokens of an address
* @param _of is the address to burn tokens of
* @param amount is the amount to burn
* @return the boolean status of the burning process
*/
function burnFrom(address _of, uint amount) public override onlyInternal returns (bool) {
return token.burnFrom(_of, amount);
}
/**
* @dev Adds an address to whitelist maintained in the contract
* @param _member address to add to whitelist
*/
function addToWhitelist(address _member) public virtual override onlyInternal {
token.addToWhiteList(_member);
}
/**
* @dev Removes an address from the whitelist in the token
* @param _member address to remove
*/
function removeFromWhitelist(address _member) public override onlyInternal {
token.removeFromWhiteList(_member);
}
/**
* @dev Mints new tokens for an address and checks if the address is a member
* @param _member address to send the minted tokens to
* @param _amount number of tokens to mint
*/
function mint(address _member, uint _amount) public override onlyInternal {
_mint(_member, _amount);
}
/**
* @dev Internal function to mint new tokens for an address and checks if the address is a member
* @dev Other internal functions in this contract should use _mint and never token.mint directly
* @param _member address to send the minted tokens to
* @param _amount number of tokens to mint
*/
function _mint(address _member, uint _amount) internal {
require(
_member == address(this) || token.whiteListed(_member),
"TokenController: Address is not a member"
);
token.mint(_member, _amount);
}
/**
* @dev Lock the user's tokens
* @param _of user's address.
*/
function lockForMemberVote(address _of, uint _days) public override onlyInternal {
token.lockForMemberVote(_of, _days);
}
/**
* @dev Unlocks the withdrawable tokens against CLA of a specified addresses
* @param users Addresses of users for whom the tokens are unlocked
*/
function withdrawClaimAssessmentTokens(address[] calldata users) external whenNotPaused {
for (uint256 i = 0; i < users.length; i++) {
if (locked[users[i]]["CLA"].claimed) {
continue;
}
uint256 amount = locked[users[i]]["CLA"].amount;
if (amount > 0) {
locked[users[i]]["CLA"].claimed = true;
emit Unlocked(users[i], "CLA", amount);
token.transfer(users[i], amount);
}
}
}
/**
* @dev Updates Uint Parameters of a code
* @param code whose details we want to update
* @param value value to set
*/
function updateUintParameters(bytes8 code, uint value) external view onlyGovernance {
// silence compiler warnings
code;
value;
revert("TokenController: invalid param code");
}
function getLockReasons(address _of) external override view returns (bytes32[] memory reasons) {
return lockReason[_of];
}
function totalSupply() public override view returns (uint256) {
return token.totalSupply();
}
/// Returns the base voting power. It is used in governance and snapshot voting.
/// Includes the delegated tokens via staking pools.
///
/// @param _of The member address for which the base voting power is calculated.
function totalBalanceOf(address _of) public override view returns (uint) {
return _totalBalanceOf(_of, true);
}
/// Returns the base voting power. It is used in governance and snapshot voting.
/// Does not include the delegated tokens via staking pools in order to act as a fallback if
/// voting including delegations fails for whatever reason.
///
/// @param _of The member address for which the base voting power is calculated.
function totalBalanceOfWithoutDelegations(address _of) public override view returns (uint) {
return _totalBalanceOf(_of, false);
}
function _totalBalanceOf(address _of, bool includeManagedStakingPools) internal view returns (uint) {
uint amount = token.balanceOf(_of);
// This loop can be removed once all cover notes are withdrawn
for (uint256 i = 0; i < lockReason[_of].length; i++) {
amount = amount + tokensLocked(_of, lockReason[_of][i]);
}
// TODO: can be removed after PooledStaking is decommissioned
amount += pooledStaking().stakerReward(_of);
amount += pooledStaking().stakerDeposit(_of);
(uint assessmentStake,,) = assessment().stakeOf(_of);
amount += assessmentStake;
if (includeManagedStakingPools) {
uint managedStakingPoolCount = managerStakingPools[_of].length;
for (uint i = 0; i < managedStakingPoolCount; i++) {
uint poolId = managerStakingPools[_of][i];
amount += stakingPoolNXMBalances[poolId].deposits;
}
}
return amount;
}
/// Returns the NXM price in ETH. To be use by external protocols.
///
/// @dev Intended for external protocols - this is a proxy and the contract address won't change
function getTokenPrice() public override view returns (uint tokenPrice) {
// get spot price from ramm
return pool().getTokenPrice();
}
/// Withdraws governance rewards for the given member address
/// @dev This function requires a batchSize that fits in one block. It cannot be 0.
function withdrawGovernanceRewards(
address memberAddress,
uint batchSize
) public whenNotPaused {
uint governanceRewards = governance().claimReward(memberAddress, batchSize);
require(governanceRewards > 0, "TokenController: No withdrawable governance rewards");
token.transfer(memberAddress, governanceRewards);
}
/// Withdraws governance rewards to the destination address. It can only be called by the owner
/// of the rewards.
/// @dev This function requires a batchSize that fits in one block. It cannot be 0.
function withdrawGovernanceRewardsTo(
address destination,
uint batchSize
) public whenNotPaused {
uint governanceRewards = governance().claimReward(msg.sender, batchSize);
require(governanceRewards > 0, "TokenController: No withdrawable governance rewards");
token.transfer(destination, governanceRewards);
}
function getPendingRewards(address member) public view returns (uint) {
(uint totalPendingAmountInNXM,,) = assessment().getRewards(member);
uint governanceRewards = governance().getPendingReward(member);
return totalPendingAmountInNXM + governanceRewards;
}
/// Function used to claim all pending rewards in one tx. It can be used to selectively withdraw
/// rewards.
///
/// @param forUser The address for whom the governance and/or assessment rewards are
/// withdrawn.
/// @param fromGovernance When true, governance rewards are withdrawn.
/// @param fromAssessment When true, assessment rewards are withdrawn.
/// @param batchSize The maximum number of iterations to avoid unbounded loops when
/// withdrawing governance and/or assessment rewards.
function withdrawPendingRewards(
address forUser,
bool fromGovernance,
bool fromAssessment,
uint batchSize
) external whenNotPaused {
if (fromAssessment) {
assessment().withdrawRewards(forUser, batchSize.toUint104());
}
if (fromGovernance) {
uint governanceRewards = governance().claimReward(forUser, batchSize);
require(governanceRewards > 0, "TokenController: No withdrawable governance rewards");
token.transfer(forUser, governanceRewards);
}
}
/**
* @dev Returns tokens locked for a specified address for a
* specified reason
*
* @param _of The address whose tokens are locked
* @param _reason The reason to query the lock tokens for
*/
function tokensLocked(
address _of,
bytes32 _reason
) public view returns (uint256 amount) {
if (!locked[_of][_reason].claimed) {
amount = locked[_of][_reason].amount;
}
}
// Can be removed once all cover notes are withdrawn
function getWithdrawableCoverNotes(
address coverOwner
) public view returns (
uint[] memory coverIds,
bytes32[] memory lockReasons,
uint withdrawableAmount
) {
uint[] memory allCoverIds = quotationData.getAllCoversOfUser(coverOwner);
uint[] memory idsQueue = new uint[](allCoverIds.length);
bytes32[] memory lockReasonsQueue = new bytes32[](allCoverIds.length);
uint idsQueueLength = 0;
for (uint i = 0; i < allCoverIds.length; i++) {
uint coverId = allCoverIds[i];
bytes32 lockReason = keccak256(abi.encodePacked("CN", coverOwner, coverId));
uint coverNoteAmount = tokensLocked(coverOwner, lockReason);
if (coverNoteAmount > 0) {
idsQueue[idsQueueLength] = coverId;
lockReasonsQueue[idsQueueLength] = lockReason;
withdrawableAmount += coverNoteAmount;
idsQueueLength++;
}
}
coverIds = new uint[](idsQueueLength);
lockReasons = new bytes32[](idsQueueLength);
for (uint i = 0; i < idsQueueLength; i++) {
coverIds[i] = idsQueue[i];
lockReasons[i] = lockReasonsQueue[i];
}
}
// Can be removed once all cover notes are withdrawn
function withdrawCoverNote(
address user,
uint[] calldata coverIds,
uint[] calldata indexes
) external whenNotPaused override {
uint reasonCount = lockReason[user].length;
require(reasonCount > 0, "TokenController: No locked cover notes found");
uint lastReasonIndex = reasonCount - 1;
uint totalAmount = 0;
// The iteration is done from the last to first to prevent reason indexes from
// changing due to the way we delete the items (copy last to current and pop last).
// The provided indexes array must be ordered, otherwise reason index checks will fail.
for (uint i = coverIds.length; i > 0; i--) {
// note: cover owner is implicitly checked using the reason hash
bytes32 _reason = keccak256(abi.encodePacked("CN", user, coverIds[i - 1]));
uint _reasonIndex = indexes[i - 1];
require(lockReason[user][_reasonIndex] == _reason, "TokenController: Bad reason index");
uint amount = locked[user][_reason].amount;
totalAmount = totalAmount + amount;
delete locked[user][_reason];
if (lastReasonIndex != _reasonIndex) {
lockReason[user][_reasonIndex] = lockReason[user][lastReasonIndex];
}
lockReason[user].pop();
emit Unlocked(user, _reason, amount);
if (lastReasonIndex > 0) {
lastReasonIndex = lastReasonIndex - 1;
}
}
token.transfer(user, totalAmount);
}
function getStakingPoolManager(uint poolId) external override view returns (address) {
return stakingPoolManagers[poolId];
}
function getManagerStakingPools(address manager) external override view returns (uint[] memory) {
return managerStakingPools[manager];
}
function isStakingPoolManager(address member) external override view returns (bool) {
return managerStakingPools[member].length > 0;
}
function getStakingPoolOwnershipOffer(
uint poolId
) external override view returns (address proposedManager, uint deadline) {
return (
stakingPoolOwnershipOffers[poolId].proposedManager,
stakingPoolOwnershipOffers[poolId].deadline
);
}
/// Transfer ownership of all staking pools managed by a member to a new address. Used when switching membership.
///
/// @param from address of the member whose pools are being transferred
/// @param to the new address of the member
function transferStakingPoolsOwnership(address from, address to) external override onlyInternal {
uint stakingPoolCount = managerStakingPools[from].length;
if (stakingPoolCount == 0) {
return;
}
while (stakingPoolCount > 0) {
// remove from old
uint poolId = managerStakingPools[from][stakingPoolCount - 1];
managerStakingPools[from].pop();
// add to new and update manager
managerStakingPools[to].push(poolId);
stakingPoolManagers[poolId] = to;
stakingPoolCount--;
}
}
function _assignStakingPoolManager(uint poolId, address manager) internal {
address previousManager = stakingPoolManagers[poolId];
// remove previous manager
if (previousManager != address(0)) {
uint managedPoolCount = managerStakingPools[previousManager].length;
// find staking pool id index and remove from previous manager's list
// on-chain iteration is expensive, but we don't expect to have many pools per manager
for (uint i = 0; i < managedPoolCount; i++) {
if (managerStakingPools[previousManager][i] == poolId) {
uint lastIndex = managedPoolCount - 1;
managerStakingPools[previousManager][i] = managerStakingPools[previousManager][lastIndex];
managerStakingPools[previousManager].pop();
break;
}
}
}
// add staking pool id to new manager's list
managerStakingPools[manager].push(poolId);
stakingPoolManagers[poolId] = manager;
}
/// Transfers the ownership of a staking pool to a new address
/// Used by PooledStaking during the migration
///
/// @param poolId id of the staking pool
/// @param manager address of the new manager of the staking pool
function assignStakingPoolManager(uint poolId, address manager) external override onlyInternal {
_assignStakingPoolManager(poolId, manager);
}
/// Creates a ownership transfer offer for a staking pool
/// The offer can be accepted by the proposed manager before the deadline expires
///
/// @param poolId id of the staking pool
/// @param proposedManager address of the proposed manager
/// @param deadline timestamp after which the offer expires
function createStakingPoolOwnershipOffer(
uint poolId,
address proposedManager,
uint deadline
) external override {
require(msg.sender == stakingPoolManagers[poolId], "TokenController: Caller is not staking pool manager");
require(block.timestamp < deadline, "TokenController: Deadline cannot be in the past");
stakingPoolOwnershipOffers[poolId] = StakingPoolOwnershipOffer(proposedManager, deadline.toUint96());
}
/// Accepts a staking pool ownership offer
///
/// @param poolId id of the staking pool
function acceptStakingPoolOwnershipOffer(uint poolId) external override {
address oldManager = stakingPoolManagers[poolId];
require(
block.timestamp > token.isLockedForMV(oldManager),
"TokenController: Current manager is locked for voting in governance"
);
require(
msg.sender == stakingPoolOwnershipOffers[poolId].proposedManager,
"TokenController: Caller is not the proposed manager"
);
require(
stakingPoolOwnershipOffers[poolId].deadline > block.timestamp,
"TokenController: Ownership offer has expired"
);
_assignStakingPoolManager(poolId, msg.sender);
delete stakingPoolOwnershipOffers[poolId];
}
/// Cancels a staking pool ownership offer
///
/// @param poolId id of the staking pool
function cancelStakingPoolOwnershipOffer(uint poolId) external override {
require(msg.sender == stakingPoolManagers[poolId], "TokenController: Caller is not staking pool manager");
delete stakingPoolOwnershipOffers[poolId];
}
function _stakingPool(uint poolId) internal view returns (address) {
return StakingPoolLibrary.getAddress(stakingPoolFactory, poolId);
}
function mintStakingPoolNXMRewards(uint amount, uint poolId) external {
require(msg.sender == _stakingPool(poolId), "TokenController: Caller not a staking pool");
_mint(address(this), amount);
stakingPoolNXMBalances[poolId].rewards += amount.toUint128();
}
function burnStakingPoolNXMRewards(uint amount, uint poolId) external {
require(msg.sender == _stakingPool(poolId), "TokenController: Caller not a staking pool");
stakingPoolNXMBalances[poolId].rewards -= amount.toUint128();
token.burn(amount);
}
function depositStakedNXM(address from, uint amount, uint poolId) external {
require(msg.sender == _stakingPool(poolId), "TokenController: Caller not a staking pool");
stakingPoolNXMBalances[poolId].deposits += amount.toUint128();
token.operatorTransfer(from, amount);
}
function withdrawNXMStakeAndRewards(
address to,
uint stakeToWithdraw,
uint rewardsToWithdraw,
uint poolId
) external {
require(msg.sender == _stakingPool(poolId), "TokenController: Caller not a staking pool");
StakingPoolNXMBalances memory poolBalances = stakingPoolNXMBalances[poolId];
poolBalances.deposits -= stakeToWithdraw.toUint128();
poolBalances.rewards -= rewardsToWithdraw.toUint128();
stakingPoolNXMBalances[poolId] = poolBalances;
token.transfer(to, stakeToWithdraw + rewardsToWithdraw);
}
function burnStakedNXM(uint amount, uint poolId) external {
require(msg.sender == _stakingPool(poolId), "TokenController: Caller not a staking pool");
stakingPoolNXMBalances[poolId].deposits -= amount.toUint128();
token.burn(amount);
}
}