Skip to content
Permalink
f7b4cdb0cd
Switch branches/tags

Name already in use

A tag already exists with the provided branch name. Many Git commands accept both tag and branch names, so creating this branch may cause unexpected behavior. Are you sure you want to create this branch?
Go to file
 
 
Cannot retrieve contributors at this time
1299 lines (1189 sloc) 50.3 KB
/// GebProxyActions.sol
// Copyright (C) 2018-2020 Maker Ecosystem Growth Holdings, INC.
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
pragma solidity 0.6.7;
import "ds-auth/auth.sol";
abstract contract CollateralLike {
function approve(address, uint) virtual public;
function transfer(address, uint) virtual public;
function transferFrom(address, address, uint) virtual public;
function deposit() virtual public payable;
function withdraw(uint) virtual public;
}
abstract contract ManagerLike {
function safeCan(address, uint, address) virtual public view returns (uint);
function collateralTypes(uint) virtual public view returns (bytes32);
function ownsSAFE(uint) virtual public view returns (address);
function safes(uint) virtual public view returns (address);
function safeEngine() virtual public view returns (address);
function openSAFE(bytes32, address) virtual public returns (uint);
function transferSAFEOwnership(uint, address) virtual public;
function allowSAFE(uint, address, uint) virtual public;
function allowHandler(address, uint) virtual public;
function modifySAFECollateralization(uint, int, int) virtual public;
function transferCollateral(uint, address, uint) virtual public;
function transferInternalCoins(uint, address, uint) virtual public;
function quitSystem(uint, address) virtual public;
function enterSystem(address, uint) virtual public;
function moveSAFE(uint, uint) virtual public;
function protectSAFE(uint, address, address) virtual public;
}
abstract contract SAFEEngineLike {
function canModifySAFE(address, address) virtual public view returns (uint);
function collateralTypes(bytes32) virtual public view returns (uint, uint, uint, uint, uint);
function coinBalance(address) virtual public view returns (uint);
function safes(bytes32, address) virtual public view returns (uint, uint);
function modifySAFECollateralization(bytes32, address, address, address, int, int) virtual public;
function approveSAFEModification(address) virtual public;
function transferInternalCoins(address, address, uint) virtual public;
}
abstract contract CollateralJoinLike {
function decimals() virtual public returns (uint);
function collateral() virtual public returns (CollateralLike);
function join(address, uint) virtual public payable;
function exit(address, uint) virtual public;
}
abstract contract GNTJoinLike {
function bags(address) virtual public view returns (address);
function make(address) virtual public returns (address);
}
abstract contract DSTokenLike {
function balanceOf(address) virtual public view returns (uint);
function approve(address, uint) virtual public;
function transfer(address, uint) virtual public returns (bool);
function transferFrom(address, address, uint) virtual public returns (bool);
}
abstract contract WethLike {
function balanceOf(address) virtual public view returns (uint);
function approve(address, uint) virtual public;
function transfer(address, uint) virtual public;
function transferFrom(address, address, uint) virtual public;
function deposit() virtual public payable;
function withdraw(uint) virtual public;
}
abstract contract CoinJoinLike {
function safeEngine() virtual public returns (SAFEEngineLike);
function systemCoin() virtual public returns (DSTokenLike);
function join(address, uint) virtual public payable;
function exit(address, uint) virtual public;
}
abstract contract ApproveSAFEModificationLike {
function approveSAFEModification(address) virtual public;
function denySAFEModification(address) virtual public;
}
abstract contract GlobalSettlementLike {
function collateralCashPrice(bytes32) virtual public view returns (uint);
function redeemCollateral(bytes32, uint) virtual public;
function freeCollateral(bytes32) virtual public;
function prepareCoinsForRedeeming(uint) virtual public;
function processSAFE(bytes32, address) virtual public;
}
abstract contract TaxCollectorLike {
function taxSingle(bytes32) virtual public returns (uint);
}
abstract contract CoinSavingsAccountLike {
function savings(address) virtual public view returns (uint);
function updateAccumulatedRate() virtual public returns (uint);
function deposit(uint) virtual public;
function withdraw(uint) virtual public;
}
abstract contract ProxyRegistryLike {
function proxies(address) virtual public view returns (address);
function build(address) virtual public returns (address);
}
abstract contract ProxyLike {
function owner() virtual public view returns (address);
}
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
// WARNING: These functions meant to be used as a a library for a DSProxy. Some are unsafe if you call them directly.
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
contract Common {
uint256 constant RAY = 10 ** 27;
// Internal functions
function multiply(uint x, uint y) internal pure returns (uint z) {
require(y == 0 || (z = x * y) / y == x, "mul-overflow");
}
function _coinJoin_join(address apt, address safeHandler, uint wad) internal {
// Approves adapter to take the COIN amount
CoinJoinLike(apt).systemCoin().approve(apt, wad);
// Joins COIN into the safeEngine
CoinJoinLike(apt).join(safeHandler, wad);
}
// Public functions
function coinJoin_join(address apt, address safeHandler, uint wad) public {
// Gets COIN from the user's wallet
CoinJoinLike(apt).systemCoin().transferFrom(msg.sender, address(this), wad);
_coinJoin_join(apt, safeHandler, wad);
}
}
contract BasicActions is Common {
// Internal functions
/// @notice Safe subtraction
/// @dev Reverts on overflows
function subtract(uint x, uint y) internal pure returns (uint z) {
require((z = x - y) <= x, "sub-overflow");
}
/// @notice Safe conversion uint -> int
/// @dev Reverts on overflows
function toInt(uint x) internal pure returns (int y) {
y = int(x);
require(y >= 0, "int-overflow");
}
/// @notice Converts a wad (18 decimal places) to rad (45 decimal places)
function toRad(uint wad) internal pure returns (uint rad) {
rad = multiply(wad, 10 ** 27);
}
function convertTo18(address collateralJoin, uint256 amt) internal returns (uint256 wad) {
// For those collaterals that have other than 18 decimals precision we need to do the conversion before passing to modifySAFECollateralization function
// Adapters will automatically handle the difference of precision
uint decimals = CollateralJoinLike(collateralJoin).decimals();
wad = amt;
if (decimals < 18) {
wad = multiply(
amt,
10 ** (18 - decimals)
);
} else if (decimals > 18) {
wad = amt / 10 ** (decimals - 18);
}
}
/// @notice Gets delta debt generated (Total Safe debt minus available safeHandler COIN balance)
/// @param safeEngine address
/// @param taxCollector address
/// @param safeHandler address
/// @param collateralType bytes32
/// @return deltaDebt
function _getGeneratedDeltaDebt(
address safeEngine,
address taxCollector,
address safeHandler,
bytes32 collateralType,
uint wad
) internal returns (int deltaDebt) {
// Updates stability fee rate
uint rate = TaxCollectorLike(taxCollector).taxSingle(collateralType);
require(rate > 0, "invalid-collateral-type");
// Gets COIN balance of the handler in the safeEngine
uint coin = SAFEEngineLike(safeEngine).coinBalance(safeHandler);
// If there was already enough COIN in the safeEngine balance, just exits it without adding more debt
if (coin < multiply(wad, RAY)) {
// Calculates the needed deltaDebt so together with the existing coins in the safeEngine is enough to exit wad amount of COIN tokens
deltaDebt = toInt(subtract(multiply(wad, RAY), coin) / rate);
// This is neeeded due lack of precision. It might need to sum an extra deltaDebt wei (for the given COIN wad amount)
deltaDebt = multiply(uint(deltaDebt), rate) < multiply(wad, RAY) ? deltaDebt + 1 : deltaDebt;
}
}
/// @notice Gets repaid delta debt generated (rate adjusted debt)
/// @param safeEngine address
/// @param coin uint amount
/// @param safe uint - safeId
/// @param collateralType bytes32
/// @return deltaDebt
function _getRepaidDeltaDebt(
address safeEngine,
uint coin,
address safe,
bytes32 collateralType
) internal view returns (int deltaDebt) {
// Gets actual rate from the safeEngine
(, uint rate,,,) = SAFEEngineLike(safeEngine).collateralTypes(collateralType);
require(rate > 0, "invalid-collateral-type");
// Gets actual generatedDebt value of the safe
(, uint generatedDebt) = SAFEEngineLike(safeEngine).safes(collateralType, safe);
// Uses the whole coin balance in the safeEngine to reduce the debt
deltaDebt = toInt(coin / rate);
// Checks the calculated deltaDebt is not higher than safe.generatedDebt (total debt), otherwise uses its value
deltaDebt = uint(deltaDebt) <= generatedDebt ? - deltaDebt : - toInt(generatedDebt);
}
/// @notice Gets repaid debt (rate adjusted rate minus COIN balance available in usr's address)
/// @param safeEngine address
/// @param usr address
/// @param safe uint
/// @param collateralType address
/// @return wad
function _getRepaidAlDebt(
address safeEngine,
address usr,
address safe,
bytes32 collateralType
) internal view returns (uint wad) {
// Gets actual rate from the safeEngine
(, uint rate,,,) = SAFEEngineLike(safeEngine).collateralTypes(collateralType);
// Gets actual generatedDebt value of the safe
(, uint generatedDebt) = SAFEEngineLike(safeEngine).safes(collateralType, safe);
// Gets actual coin amount in the safe
uint coin = SAFEEngineLike(safeEngine).coinBalance(usr);
uint rad = subtract(multiply(generatedDebt, rate), coin);
wad = rad / RAY;
// If the rad precision has some dust, it will need to request for 1 extra wad wei
wad = multiply(wad, RAY) < rad ? wad + 1 : wad;
}
/// @notice Generates Debt (and sends coin balance to address to)
/// @param manager address
/// @param taxCollector address
/// @param coinJoin address
/// @param safe uint
/// @param wad uint - amount of debt to be generated
/// @param to address - receiver of the balance of generated COIN
function _generateDebt(address manager, address taxCollector, address coinJoin, uint safe, uint wad, address to) internal {
address safeHandler = ManagerLike(manager).safes(safe);
address safeEngine = ManagerLike(manager).safeEngine();
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
// Generates debt in the SAFE
modifySAFECollateralization(manager, safe, 0, _getGeneratedDeltaDebt(safeEngine, taxCollector, safeHandler, collateralType, wad));
// Moves the COIN amount (balance in the safeEngine in rad) to proxy's address
transferInternalCoins(manager, safe, address(this), toRad(wad));
// Allows adapter to access to proxy's COIN balance in the safeEngine
if (SAFEEngineLike(safeEngine).canModifySAFE(address(this), address(coinJoin)) == 0) {
SAFEEngineLike(safeEngine).approveSAFEModification(coinJoin);
}
// Exits COIN to this contract
CoinJoinLike(coinJoin).exit(to, wad);
}
/// @notice Generates Debt (and sends coin balance to address to)
/// @param manager address
/// @param ethJoin address
/// @param safe uint
/// @param value uint - amount of ETH to be locked in the Safe.
/// @dev Proxy needs to have enough balance (> value), public functions should handle this.
function _lockETH(
address manager,
address ethJoin,
uint safe,
uint value
) internal {
// Receives ETH amount, converts it to WETH and joins it into the safeEngine
ethJoin_join(ethJoin, address(this), value);
// Locks WETH amount into the SAFE
SAFEEngineLike(ManagerLike(manager).safeEngine()).modifySAFECollateralization(
ManagerLike(manager).collateralTypes(safe),
ManagerLike(manager).safes(safe),
address(this),
address(this),
toInt(value),
0
);
}
/// @notice Repays debt
/// @param manager address
/// @param coinJoin address
/// @param safe uint
/// @param wad uint - amount of debt to be repayed
function _repayDebt(
address manager,
address coinJoin,
uint safe,
uint wad,
bool transferFromCaller
) internal {
address safeEngine = ManagerLike(manager).safeEngine();
address safeHandler = ManagerLike(manager).safes(safe);
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
address own = ManagerLike(manager).ownsSAFE(safe);
if (own == address(this) || ManagerLike(manager).safeCan(own, safe, address(this)) == 1) {
// Joins COIN amount into the safeEngine
if (transferFromCaller) coinJoin_join(coinJoin, safeHandler, wad);
else _coinJoin_join(coinJoin, safeHandler, wad);
// // Paybacks debt to the SAFE
modifySAFECollateralization(manager, safe, 0, _getRepaidDeltaDebt(safeEngine, SAFEEngineLike(safeEngine).coinBalance(safeHandler), safeHandler, collateralType));
} else {
// Joins COIN amount into the safeEngine
if (transferFromCaller) coinJoin_join(coinJoin, address(this), wad);
else _coinJoin_join(coinJoin, address(this), wad);
// Paybacks debt to the SAFE
SAFEEngineLike(safeEngine).modifySAFECollateralization(
collateralType,
safeHandler,
address(this),
address(this),
0,
_getRepaidDeltaDebt(safeEngine, wad * RAY, safeHandler, collateralType)
);
}
}
/// @notice Repays debt and frees collateral ETH
/// @param manager address
/// @param ethJoin address
/// @param coinJoin address
/// @param safe uint
/// @param collateralWad uint - amount of ETH to free
/// @param deltaWad uint - amount of debt to be repayed
/// @param transferFromCaller True if transferring coin from caller, false if balance in the proxy
function _repayDebtAndFreeETH(
address manager,
address ethJoin,
address coinJoin,
uint safe,
uint collateralWad,
uint deltaWad,
bool transferFromCaller
) internal {
address safeHandler = ManagerLike(manager).safes(safe);
// Joins COIN amount into the safeEngine
if (transferFromCaller) coinJoin_join(coinJoin, safeHandler, deltaWad);
else _coinJoin_join(coinJoin, safeHandler, deltaWad);
// Paybacks debt to the SAFE and unlocks WETH amount from it
modifySAFECollateralization(
manager,
safe,
-toInt(collateralWad),
_getRepaidDeltaDebt(ManagerLike(manager).safeEngine(), SAFEEngineLike(ManagerLike(manager).safeEngine()).coinBalance(safeHandler), safeHandler, ManagerLike(manager).collateralTypes(safe))
);
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), collateralWad);
// Exits WETH amount to proxy address as a token
CollateralJoinLike(ethJoin).exit(address(this), collateralWad);
// Converts WETH to ETH
CollateralJoinLike(ethJoin).collateral().withdraw(collateralWad);
}
// Public functions
/// @notice ERC20 transfer
/// @param collateral address - address of ERC20 collateral
/// @param dst address - Transfer destination
/// @param amt address - Amount to transfer
function transfer(address collateral, address dst, uint amt) external {
CollateralLike(collateral).transfer(dst, amt);
}
/// @notice Joins the system with the full msg.value
/// @param apt address - Address of the adapter
/// @param safe uint - Safe Id
function ethJoin_join(address apt, address safe) external payable {
ethJoin_join(apt, safe, msg.value);
}
/// @notice Joins the system with the a specified value
/// @param apt address - Address of the adapter
/// @param safe uint - Safe Id
/// @param value uint - Value to join
function ethJoin_join(address apt, address safe, uint value) public payable {
// Wraps ETH in WETH
CollateralJoinLike(apt).collateral().deposit{value: value}();
// Approves adapter to take the WETH amount
CollateralJoinLike(apt).collateral().approve(address(apt), value);
// Joins WETH collateral into the safeEngine
CollateralJoinLike(apt).join(safe, value);
}
/// @notice Approves an address to modify the Safe
/// @param safeEngine address
/// @param usr address - Address allowed to modify Safe
function approveSAFEModification(
address safeEngine,
address usr
) external {
ApproveSAFEModificationLike(safeEngine).approveSAFEModification(usr);
}
/// @notice Denies an address to modify the Safe
/// @param safeEngine address
/// @param usr address - Address disallowed to modify Safe
function denySAFEModification(
address safeEngine,
address usr
) external {
ApproveSAFEModificationLike(safeEngine).denySAFEModification(usr);
}
/// @notice Opens a brand new Safe
/// @param manager address - Safe Manager
/// @param collateralType bytes32 - collateral type
/// @param usr address - Owner of the safe
function openSAFE(
address manager,
bytes32 collateralType,
address usr
) public returns (uint safe) {
safe = ManagerLike(manager).openSAFE(collateralType, usr);
}
/// @notice Transfer the ownership of a proxy owned Safe
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param usr address - Owner of the safe
function transferSAFEOwnership(
address manager,
uint safe,
address usr
) public {
ManagerLike(manager).transferSAFEOwnership(safe, usr);
}
/// @notice Transfer the ownership to a new proxy owned by a different address
/// @param proxyRegistry address - Safe Manager
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param dst address - Owner of the new proxy
function transferSAFEOwnershipToProxy(
address proxyRegistry,
address manager,
uint safe,
address dst
) external {
// Gets actual proxy address
address proxy = ProxyRegistryLike(proxyRegistry).proxies(dst);
// Checks if the proxy address already existed and dst address is still the owner
if (proxy == address(0) || ProxyLike(proxy).owner() != dst) {
uint csize;
assembly {
csize := extcodesize(dst)
}
// We want to avoid creating a proxy for a contract address that might not be able to handle proxies, then losing the SAFE
require(csize == 0, "dst-is-a-contract");
// Creates the proxy for the dst address
proxy = ProxyRegistryLike(proxyRegistry).build(dst);
}
// Transfers SAFE to the dst proxy
transferSAFEOwnership(manager, safe, proxy);
}
/// @notice Allow/disallow a usr address to manage the safe
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param usr address - usr address
/// uint ok - 1 for allowed
function allowSAFE(
address manager,
uint safe,
address usr,
uint ok
) external {
ManagerLike(manager).allowSAFE(safe, usr, ok);
}
/// @notice Allow/disallow a usr address to quit to the sender handler
/// @param manager address - Safe Manager
/// @param usr address - usr address
/// uint ok - 1 for allowed
function allowHandler(
address manager,
address usr,
uint ok
) external {
ManagerLike(manager).allowHandler(usr, ok);
}
/// @notice Transfer wad amount of safe collateral from the safe address to a dst address.
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param dst address - destination address
/// uint wad - amount
function transferCollateral(
address manager,
uint safe,
address dst,
uint wad
) public {
ManagerLike(manager).transferCollateral(safe, dst, wad);
}
/// @notice Transfer rad amount of COIN from the safe address to a dst address.
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param dst address - destination address
/// uint rad - amount
function transferInternalCoins(
address manager,
uint safe,
address dst,
uint rad
) public {
ManagerLike(manager).transferInternalCoins(safe, dst, rad);
}
/// @notice Modify a SAFE's collateralization ratio while keeping the generated COIN or collateral freed in the SAFE handler address.
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param deltaCollateral - int
/// @param deltaDebt - int
function modifySAFECollateralization(
address manager,
uint safe,
int deltaCollateral,
int deltaDebt
) public {
ManagerLike(manager).modifySAFECollateralization(safe, deltaCollateral, deltaDebt);
}
/// @notice Quit the system, migrating the safe (lockedCollateral, generatedDebt) to a different dst handler
/// @param manager address - Safe Manager
/// @param safe uint - Safe Id
/// @param dst - destination handler
function quitSystem(
address manager,
uint safe,
address dst
) external {
ManagerLike(manager).quitSystem(safe, dst);
}
/// @notice Import a position from src handler to the handler owned by safe
/// @param manager address - Safe Manager
/// @param src - source handler
/// @param safe uint - Safe Id
function enterSystem(
address manager,
address src,
uint safe
) external {
ManagerLike(manager).enterSystem(src, safe);
}
/// @notice Move a position from safeSrc handler to the safeDst handler
/// @param manager address - Safe Manager
/// @param safeSrc uint - Source Safe Id
/// @param safeDst uint - Destination Safe Id
function moveSAFE(
address manager,
uint safeSrc,
uint safeDst
) external {
ManagerLike(manager).moveSAFE(safeSrc, safeDst);
}
/// @notice Lock ETH (msg.value) as collateral in safe
/// @param manager address - Safe Manager
/// @param ethJoin address
/// @param safe uint - Safe Id
function lockETH(
address manager,
address ethJoin,
uint safe
) public payable {
_lockETH(manager, ethJoin, safe, msg.value);
}
/// @notice Free ETH (wad) from safe and sends it to msg.sender
/// @param manager address - Safe Manager
/// @param ethJoin address
/// @param safe uint - Safe Id
/// @param wad uint - Amount
function freeETH(
address manager,
address ethJoin,
uint safe,
uint wad
) public {
// Unlocks WETH amount from the SAFE
modifySAFECollateralization(manager, safe, -toInt(wad), 0);
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), wad);
// Exits WETH amount to proxy address as a token
CollateralJoinLike(ethJoin).exit(address(this), wad);
// Converts WETH to ETH
CollateralJoinLike(ethJoin).collateral().withdraw(wad);
// Sends ETH back to the user's wallet
msg.sender.transfer(wad);
}
/// @notice Exits ETH (wad) from balance available in the handler
/// @param manager address - Safe Manager
/// @param ethJoin address
/// @param safe uint - Safe Id
/// @param wad uint - Amount
function exitETH(
address manager,
address ethJoin,
uint safe,
uint wad
) external {
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), wad);
// Exits WETH amount to proxy address as a token
CollateralJoinLike(ethJoin).exit(address(this), wad);
// Converts WETH to ETH
CollateralJoinLike(ethJoin).collateral().withdraw(wad);
// Sends ETH back to the user's wallet
msg.sender.transfer(wad);
}
/// @notice Generates debt and sends COIN amount to msg.sender
/// @param manager address
/// @param taxCollector address
/// @param coinJoin address
/// @param safe uint - Safe Id
/// @param wad uint - Amount
function generateDebt(
address manager,
address taxCollector,
address coinJoin,
uint safe,
uint wad
) public {
_generateDebt(manager, taxCollector, coinJoin, safe, wad, msg.sender);
}
/// @notice Repays debt
/// @param manager address
/// @param coinJoin address
/// @param safe uint - Safe Id
/// @param wad uint - Amount
function repayDebt(
address manager,
address coinJoin,
uint safe,
uint wad
) public {
_repayDebt(manager, coinJoin, safe, wad, true);
}
/// @notice Locks Eth, generates debt and sends COIN amount (deltaWad) to msg.sender
/// @param manager address
/// @param taxCollector address
/// @param ethJoin address
/// @param coinJoin address
/// @param safe uint - Safe Id
/// @param deltaWad uint - Amount
function lockETHAndGenerateDebt(
address manager,
address taxCollector,
address ethJoin,
address coinJoin,
uint safe,
uint deltaWad
) public payable {
_lockETH(manager, ethJoin, safe, msg.value);
_generateDebt(manager, taxCollector, coinJoin, safe, deltaWad, msg.sender);
}
/// @notice Opens Safe, locks Eth, generates debt and sends COIN amount (deltaWad) to msg.sender
/// @param manager address
/// @param taxCollector address
/// @param ethJoin address
/// @param coinJoin address
/// @param deltaWad uint - Amount
function openLockETHAndGenerateDebt(
address manager,
address taxCollector,
address ethJoin,
address coinJoin,
bytes32 collateralType,
uint deltaWad
) external payable returns (uint safe) {
safe = openSAFE(manager, collateralType, address(this));
lockETHAndGenerateDebt(manager, taxCollector, ethJoin, coinJoin, safe, deltaWad);
}
/// @notice Repays debt and frees ETH (sends it to msg.sender)
/// @param manager address
/// @param ethJoin address
/// @param coinJoin address
/// @param safe uint - Safe Id
/// @param collateralWad uint - Amount of collateral to free
/// @param deltaWad uint - Amount of debt to repay
function repayDebtAndFreeETH(
address manager,
address ethJoin,
address coinJoin,
uint safe,
uint collateralWad,
uint deltaWad
) external {
_repayDebtAndFreeETH(manager, ethJoin, coinJoin, safe, collateralWad, deltaWad, true);
// Sends ETH back to the user's wallet
msg.sender.transfer(collateralWad);
}
}
contract GebProxyActions is BasicActions {
function tokenCollateralJoin_join(address apt, address safe, uint amt, bool transferFrom) public {
// Only executes for tokens that have approval/transferFrom implementation
if (transferFrom) {
// Gets token from the user's wallet
CollateralJoinLike(apt).collateral().transferFrom(msg.sender, address(this), amt);
// Approves adapter to take the token amount
CollateralJoinLike(apt).collateral().approve(apt, amt);
}
// Joins token collateral into the safeEngine
CollateralJoinLike(apt).join(safe, amt);
}
function protectSAFE(
address manager,
uint safe,
address liquidationEngine,
address saviour
) public {
ManagerLike(manager).protectSAFE(safe, liquidationEngine, saviour);
}
function makeCollateralBag(
address collateralJoin
) public returns (address bag) {
bag = GNTJoinLike(collateralJoin).make(address(this));
}
function safeLockETH(
address manager,
address ethJoin,
uint safe,
address owner
) public payable {
require(ManagerLike(manager).ownsSAFE(safe) == owner, "owner-missmatch");
lockETH(manager, ethJoin, safe);
}
function lockTokenCollateral(
address manager,
address collateralJoin,
uint safe,
uint amt,
bool transferFrom
) public {
// Takes token amount from user's wallet and joins into the safeEngine
tokenCollateralJoin_join(collateralJoin, address(this), amt, transferFrom);
// Locks token amount into the SAFE
SAFEEngineLike(ManagerLike(manager).safeEngine()).modifySAFECollateralization(
ManagerLike(manager).collateralTypes(safe),
ManagerLike(manager).safes(safe),
address(this),
address(this),
toInt(convertTo18(collateralJoin, amt)),
0
);
}
function safeLockTokenCollateral(
address manager,
address collateralJoin,
uint safe,
uint amt,
bool transferFrom,
address owner
) public {
require(ManagerLike(manager).ownsSAFE(safe) == owner, "owner-missmatch");
lockTokenCollateral(manager, collateralJoin, safe, amt, transferFrom);
}
function freeTokenCollateral(
address manager,
address collateralJoin,
uint safe,
uint amt
) public {
uint wad = convertTo18(collateralJoin, amt);
// Unlocks token amount from the SAFE
modifySAFECollateralization(manager, safe, -toInt(wad), 0);
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), wad);
// Exits token amount to the user's wallet as a token
CollateralJoinLike(collateralJoin).exit(msg.sender, amt);
}
function exitTokenCollateral(
address manager,
address collateralJoin,
uint safe,
uint amt
) public {
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), convertTo18(collateralJoin, amt));
// Exits token amount to the user's wallet as a token
CollateralJoinLike(collateralJoin).exit(msg.sender, amt);
}
function generateDebtAndProtectSAFE(
address manager,
address taxCollector,
address coinJoin,
uint safe,
uint wad,
address liquidationEngine,
address saviour
) external {
generateDebt(manager, taxCollector, coinJoin, safe, wad);
protectSAFE(manager, safe, liquidationEngine, saviour);
}
function safeRepayDebt(
address manager,
address coinJoin,
uint safe,
uint wad,
address owner
) public {
require(ManagerLike(manager).ownsSAFE(safe) == owner, "owner-missmatch");
repayDebt(manager, coinJoin, safe, wad);
}
function repayAllDebt(
address manager,
address coinJoin,
uint safe
) public {
address safeEngine = ManagerLike(manager).safeEngine();
address safeHandler = ManagerLike(manager).safes(safe);
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
(, uint generatedDebt) = SAFEEngineLike(safeEngine).safes(collateralType, safeHandler);
address own = ManagerLike(manager).ownsSAFE(safe);
if (own == address(this) || ManagerLike(manager).safeCan(own, safe, address(this)) == 1) {
// Joins COIN amount into the safeEngine
coinJoin_join(coinJoin, safeHandler, _getRepaidAlDebt(safeEngine, safeHandler, safeHandler, collateralType));
// Paybacks debt to the SAFE
modifySAFECollateralization(manager, safe, 0, -int(generatedDebt));
} else {
// Joins COIN amount into the safeEngine
coinJoin_join(coinJoin, address(this), _getRepaidAlDebt(safeEngine, address(this), safeHandler, collateralType));
// Paybacks debt to the SAFE
SAFEEngineLike(safeEngine).modifySAFECollateralization(
collateralType,
safeHandler,
address(this),
address(this),
0,
-int(generatedDebt)
);
}
}
function safeRepayAllDebt(
address manager,
address coinJoin,
uint safe,
address owner
) public {
require(ManagerLike(manager).ownsSAFE(safe) == owner, "owner-missmatch");
repayAllDebt(manager, coinJoin, safe);
}
function openLockETHGenerateDebtAndProtectSAFE(
address manager,
address taxCollector,
address ethJoin,
address coinJoin,
bytes32 collateralType,
uint deltaWad,
address liquidationEngine,
address saviour
) public payable returns (uint safe) {
safe = openSAFE(manager, collateralType, address(this));
lockETHAndGenerateDebt(manager, taxCollector, ethJoin, coinJoin, safe, deltaWad);
protectSAFE(manager, safe, liquidationEngine, saviour);
}
function lockTokenCollateralAndGenerateDebt(
address manager,
address taxCollector,
address collateralJoin,
address coinJoin,
uint safe,
uint collateralAmount,
uint deltaWad,
bool transferFrom
) public {
address safeHandler = ManagerLike(manager).safes(safe);
address safeEngine = ManagerLike(manager).safeEngine();
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
// Takes token amount from user's wallet and joins into the safeEngine
tokenCollateralJoin_join(collateralJoin, safeHandler, collateralAmount, transferFrom);
// Locks token amount into the SAFE and generates debt
modifySAFECollateralization(manager, safe, toInt(convertTo18(collateralJoin, collateralAmount)), _getGeneratedDeltaDebt(safeEngine, taxCollector, safeHandler, collateralType, deltaWad));
// Moves the COIN amount (balance in the safeEngine in rad) to proxy's address
transferInternalCoins(manager, safe, address(this), toRad(deltaWad));
// Allows adapter to access to proxy's COIN balance in the safeEngine
if (SAFEEngineLike(safeEngine).canModifySAFE(address(this), address(coinJoin)) == 0) {
SAFEEngineLike(safeEngine).approveSAFEModification(coinJoin);
}
// Exits COIN to the user's wallet as a token
CoinJoinLike(coinJoin).exit(msg.sender, deltaWad);
}
function lockTokenCollateralGenerateDebtAndProtectSAFE(
address manager,
address taxCollector,
address collateralJoin,
address coinJoin,
uint safe,
uint collateralAmount,
uint deltaWad,
bool transferFrom,
address liquidationEngine,
address saviour
) public {
lockTokenCollateralAndGenerateDebt(
manager,
taxCollector,
collateralJoin,
coinJoin,
safe,
collateralAmount,
deltaWad,
transferFrom
);
protectSAFE(manager, safe, liquidationEngine, saviour);
}
function openLockTokenCollateralAndGenerateDebt(
address manager,
address taxCollector,
address collateralJoin,
address coinJoin,
bytes32 collateralType,
uint collateralAmount,
uint deltaWad,
bool transferFrom
) public returns (uint safe) {
safe = openSAFE(manager, collateralType, address(this));
lockTokenCollateralAndGenerateDebt(manager, taxCollector, collateralJoin, coinJoin, safe, collateralAmount, deltaWad, transferFrom);
}
function openLockTokenCollateralGenerateDebtAndProtectSAFE(
address manager,
address taxCollector,
address collateralJoin,
address coinJoin,
bytes32 collateralType,
uint collateralAmount,
uint deltaWad,
bool transferFrom,
address liquidationEngine,
address saviour
) public returns (uint safe) {
safe = openSAFE(manager, collateralType, address(this));
lockTokenCollateralAndGenerateDebt(manager, taxCollector, collateralJoin, coinJoin, safe, collateralAmount, deltaWad, transferFrom);
protectSAFE(manager, safe, liquidationEngine, saviour);
}
function openLockGNTAndGenerateDebt(
address manager,
address taxCollector,
address gntJoin,
address coinJoin,
bytes32 collateralType,
uint collateralAmount,
uint deltaWad
) public returns (address bag, uint safe) {
// Creates bag (if doesn't exist) to hold GNT
bag = GNTJoinLike(gntJoin).bags(address(this));
if (bag == address(0)) {
bag = makeCollateralBag(gntJoin);
}
// Transfer funds to the funds which previously were sent to the proxy
CollateralLike(CollateralJoinLike(gntJoin).collateral()).transfer(bag, collateralAmount);
safe = openLockTokenCollateralAndGenerateDebt(manager, taxCollector, gntJoin, coinJoin, collateralType, collateralAmount, deltaWad, false);
}
function openLockGNTGenerateDebtAndProtectSAFE(
address manager,
address taxCollector,
address gntJoin,
address coinJoin,
bytes32 collateralType,
uint collateralAmount,
uint deltaWad,
address liquidationEngine,
address saviour
) public returns (address bag, uint safe) {
(bag, safe) = openLockGNTAndGenerateDebt(
manager,
taxCollector,
gntJoin,
coinJoin,
collateralType,
collateralAmount,
deltaWad
);
protectSAFE(manager, safe, liquidationEngine, saviour);
}
function repayAllDebtAndFreeETH(
address manager,
address ethJoin,
address coinJoin,
uint safe,
uint collateralWad
) public {
address safeEngine = ManagerLike(manager).safeEngine();
address safeHandler = ManagerLike(manager).safes(safe);
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
(, uint generatedDebt) = SAFEEngineLike(safeEngine).safes(collateralType, safeHandler);
// Joins COIN amount into the safeEngine
coinJoin_join(coinJoin, safeHandler, _getRepaidAlDebt(safeEngine, safeHandler, safeHandler, collateralType));
// Paybacks debt to the SAFE and unlocks WETH amount from it
modifySAFECollateralization(
manager,
safe,
-toInt(collateralWad),
-int(generatedDebt)
);
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), collateralWad);
// Exits WETH amount to proxy address as a token
CollateralJoinLike(ethJoin).exit(address(this), collateralWad);
// Converts WETH to ETH
CollateralJoinLike(ethJoin).collateral().withdraw(collateralWad);
// Sends ETH back to the user's wallet
msg.sender.transfer(collateralWad);
}
function repayDebtAndFreeTokenCollateral(
address manager,
address collateralJoin,
address coinJoin,
uint safe,
uint collateralAmount,
uint deltaWad
) external {
address safeHandler = ManagerLike(manager).safes(safe);
// Joins COIN amount into the safeEngine
coinJoin_join(coinJoin, safeHandler, deltaWad);
uint collateralWad = convertTo18(collateralJoin, collateralAmount);
// Paybacks debt to the SAFE and unlocks token amount from it
modifySAFECollateralization(
manager,
safe,
-toInt(collateralWad),
_getRepaidDeltaDebt(ManagerLike(manager).safeEngine(), SAFEEngineLike(ManagerLike(manager).safeEngine()).coinBalance(safeHandler), safeHandler, ManagerLike(manager).collateralTypes(safe))
);
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), collateralWad);
// Exits token amount to the user's wallet as a token
CollateralJoinLike(collateralJoin).exit(msg.sender, collateralAmount);
}
function repayAllDebtAndFreeTokenCollateral(
address manager,
address collateralJoin,
address coinJoin,
uint safe,
uint collateralAmount
) public {
address safeEngine = ManagerLike(manager).safeEngine();
address safeHandler = ManagerLike(manager).safes(safe);
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
(, uint generatedDebt) = SAFEEngineLike(safeEngine).safes(collateralType, safeHandler);
// Joins COIN amount into the safeEngine
coinJoin_join(coinJoin, safeHandler, _getRepaidAlDebt(safeEngine, safeHandler, safeHandler, collateralType));
uint collateralWad = convertTo18(collateralJoin, collateralAmount);
// Paybacks debt to the SAFE and unlocks token amount from it
modifySAFECollateralization(
manager,
safe,
-toInt(collateralWad),
-int(generatedDebt)
);
// Moves the amount from the SAFE handler to proxy's address
transferCollateral(manager, safe, address(this), collateralWad);
// Exits token amount to the user's wallet as a token
CollateralJoinLike(collateralJoin).exit(msg.sender, collateralAmount);
}
}
contract GebProxyActionsGlobalSettlement is Common {
// Internal functions
function _freeCollateral(
address manager,
address globalSettlement,
uint safe
) internal returns (uint lockedCollateral) {
bytes32 collateralType = ManagerLike(manager).collateralTypes(safe);
address safeHandler = ManagerLike(manager).safes(safe);
SAFEEngineLike safeEngine = SAFEEngineLike(ManagerLike(manager).safeEngine());
uint generatedDebt;
(lockedCollateral, generatedDebt) = safeEngine.safes(collateralType, safeHandler);
// If SAFE still has debt, it needs to be paid
if (generatedDebt > 0) {
GlobalSettlementLike(globalSettlement).processSAFE(collateralType, safeHandler);
(lockedCollateral,) = safeEngine.safes(collateralType, safeHandler);
}
// Approves the manager to transfer the position to proxy's address in the safeEngine
if (safeEngine.canModifySAFE(address(this), address(manager)) == 0) {
safeEngine.approveSAFEModification(manager);
}
// Transfers position from SAFE to the proxy address
ManagerLike(manager).quitSystem(safe, address(this));
// Frees the position and recovers the collateral in the safeEngine registry
GlobalSettlementLike(globalSettlement).freeCollateral(collateralType);
}
// Public functions
function freeETH(
address manager,
address ethJoin,
address globalSettlement,
uint safe
) external {
uint wad = _freeCollateral(manager, globalSettlement, safe);
// Exits WETH amount to proxy address as a token
CollateralJoinLike(ethJoin).exit(address(this), wad);
// Converts WETH to ETH
CollateralJoinLike(ethJoin).collateral().withdraw(wad);
// Sends ETH back to the user's wallet
msg.sender.transfer(wad);
}
function freeTokenCollateral(
address manager,
address collateralJoin,
address globalSettlement,
uint safe
) public {
uint amt = _freeCollateral(manager, globalSettlement, safe) / 10 ** (18 - CollateralJoinLike(collateralJoin).decimals());
// Exits token amount to the user's wallet as a token
CollateralJoinLike(collateralJoin).exit(msg.sender, amt);
}
function prepareCoinsForRedeeming(
address coinJoin,
address globalSettlement,
uint wad
) public {
coinJoin_join(coinJoin, address(this), wad);
SAFEEngineLike safeEngine = CoinJoinLike(coinJoin).safeEngine();
// Approves the globalSettlement to take out COIN from the proxy's balance in the safeEngine
if (safeEngine.canModifySAFE(address(this), address(globalSettlement)) == 0) {
safeEngine.approveSAFEModification(globalSettlement);
}
GlobalSettlementLike(globalSettlement).prepareCoinsForRedeeming(wad);
}
function redeemETH(
address ethJoin,
address globalSettlement,
bytes32 collateralType,
uint wad
) public {
GlobalSettlementLike(globalSettlement).redeemCollateral(collateralType, wad);
uint collateralWad = multiply(wad, GlobalSettlementLike(globalSettlement).collateralCashPrice(collateralType)) / RAY;
// Exits WETH amount to proxy address as a token
CollateralJoinLike(ethJoin).exit(address(this), collateralWad);
// Converts WETH to ETH
CollateralJoinLike(ethJoin).collateral().withdraw(collateralWad);
// Sends ETH back to the user's wallet
msg.sender.transfer(collateralWad);
}
function redeemTokenCollateral(
address collateralJoin,
address globalSettlement,
bytes32 collateralType,
uint wad
) public {
GlobalSettlementLike(globalSettlement).redeemCollateral(collateralType, wad);
// Exits token amount to the user's wallet as a token
uint amt = multiply(wad, GlobalSettlementLike(globalSettlement).collateralCashPrice(collateralType)) / RAY / 10 ** (18 - CollateralJoinLike(collateralJoin).decimals());
CollateralJoinLike(collateralJoin).exit(msg.sender, amt);
}
}
contract GebProxyActionsCoinSavingsAccount is Common {
function deposit(
address coinJoin,
address coinSavingsAccount,
uint wad
) public {
SAFEEngineLike safeEngine = CoinJoinLike(coinJoin).safeEngine();
// Executes updateAccumulatedRate to get the accumulatedRates updated to latestUpdateTime == now, otherwise join will fail
uint accumulatedRates = CoinSavingsAccountLike(coinSavingsAccount).updateAccumulatedRate();
// Joins wad amount to the safeEngine balance
coinJoin_join(coinJoin, address(this), wad);
// Approves the coinSavingsAccount to take out COIN from the proxy's balance in the safeEngine
if (safeEngine.canModifySAFE(address(this), address(coinSavingsAccount)) == 0) {
safeEngine.approveSAFEModification(coinSavingsAccount);
}
// Joins the savings value (equivalent to the COIN wad amount) in the coinSavingsAccount
CoinSavingsAccountLike(coinSavingsAccount).deposit(multiply(wad, RAY) / accumulatedRates);
}
function withdraw(
address coinJoin,
address coinSavingsAccount,
uint wad
) public {
SAFEEngineLike safeEngine = CoinJoinLike(coinJoin).safeEngine();
// Executes updateAccumulatedRate to count the savings accumulated until this moment
uint accumulatedRates = CoinSavingsAccountLike(coinSavingsAccount).updateAccumulatedRate();
// Calculates the savings value in the coinSavingsAccount equivalent to the COIN wad amount
uint savings = multiply(wad, RAY) / accumulatedRates;
// Exits COIN from the coinSavingsAccount
CoinSavingsAccountLike(coinSavingsAccount).withdraw(savings);
// Checks the actual balance of COIN in the safeEngine after the coinSavingsAccount exit
uint bal = CoinJoinLike(coinJoin).safeEngine().coinBalance(address(this));
// Allows adapter to access to proxy's COIN balance in the safeEngine
if (safeEngine.canModifySAFE(address(this), address(coinJoin)) == 0) {
safeEngine.approveSAFEModification(coinJoin);
}
// It is necessary to check if due rounding the exact wad amount can be exited by the adapter.
// Otherwise it will do the minimum COIN balance in the safeEngine
CoinJoinLike(coinJoin).exit(
msg.sender,
bal >= multiply(wad, RAY) ? wad : bal / RAY
);
}
function withdrawAll(
address coinJoin,
address coinSavingsAccount
) public {
SAFEEngineLike safeEngine = CoinJoinLike(coinJoin).safeEngine();
// Executes updateAccumulatedRate to count the savings accumulated until this moment
uint accumulatedRates = CoinSavingsAccountLike(coinSavingsAccount).updateAccumulatedRate();
// Gets the total savings belonging to the proxy address
uint savings = CoinSavingsAccountLike(coinSavingsAccount).savings(address(this));
// Exits COIN from the coinSavingsAccount
CoinSavingsAccountLike(coinSavingsAccount).withdraw(savings);
// Allows adapter to access to proxy's COIN balance in the safeEngine
if (safeEngine.canModifySAFE(address(this), address(coinJoin)) == 0) {
safeEngine.approveSAFEModification(coinJoin);
}
// Exits the COIN amount corresponding to the value of savings
CoinJoinLike(coinJoin).exit(msg.sender, multiply(accumulatedRates, savings) / RAY);
}
}