2018-02-13 00:24:19 +00:00
|
|
|
pragma solidity ^0.4.18;
|
2018-02-10 14:14:52 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016, Jordi Baylina
|
|
|
|
Contributor: Adrià Massanet <adria@codecontext.io>
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// import "./Owned.sol";
|
|
|
|
import "giveth-common-contracts/contracts/ERC20.sol";
|
|
|
|
import "@aragon/os/contracts/apps/AragonApp.sol";
|
|
|
|
|
|
|
|
|
2018-02-13 00:24:19 +00:00
|
|
|
/// @dev `EscapableApp` is a base level contract; it creates an escape hatch
|
|
|
|
/// function that can be called in an
|
2018-02-10 14:14:52 +00:00
|
|
|
/// emergency that will allow designated addresses to send any ether or tokens
|
|
|
|
/// held in the contract to an `escapeHatchDestination` as long as they were
|
|
|
|
/// not blacklisted
|
|
|
|
contract EscapableApp is AragonApp {
|
|
|
|
// warning whoever has this role can move all funds to the `escapeHatchDestination`
|
2018-02-12 22:55:11 +00:00
|
|
|
bytes32 constant public ESCAPE_HATCH_CALLER_ROLE = keccak256("ESCAPE_HATCH_CALLER_ROLE");
|
2018-02-10 14:14:52 +00:00
|
|
|
|
2018-02-13 00:24:19 +00:00
|
|
|
event EscapeHatchBlackistedToken(address token);
|
|
|
|
event EscapeHatchCalled(address token, uint amount);
|
|
|
|
|
2018-02-10 14:14:52 +00:00
|
|
|
address public escapeHatchDestination;
|
|
|
|
mapping (address=>bool) private escapeBlacklist; // Token contract addresses
|
2018-02-15 23:42:19 +00:00
|
|
|
uint[20] private storageOffset; // reserve 20 slots for future upgrades
|
2018-02-10 14:14:52 +00:00
|
|
|
|
2018-03-27 17:55:37 +00:00
|
|
|
function EscapableApp(address _escapeHatchDestination) public {
|
|
|
|
_init(_escapeHatchDestination);
|
|
|
|
}
|
|
|
|
|
2018-02-10 14:14:52 +00:00
|
|
|
/// @param _escapeHatchDestination The address of a safe location (usu a
|
|
|
|
/// Multisig) to send the ether held in this contract; if a neutral address
|
|
|
|
/// is required, the WHG Multisig is an option:
|
|
|
|
/// 0x8Ff920020c8AD673661c8117f2855C384758C572
|
2018-02-12 22:55:11 +00:00
|
|
|
function initialize(address _escapeHatchDestination) onlyInit public {
|
2018-03-27 17:55:37 +00:00
|
|
|
_init(_escapeHatchDestination);
|
2018-02-10 14:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// @notice The `escapeHatch()` should only be called as a last resort if a
|
|
|
|
/// security issue is uncovered or something unexpected happened
|
|
|
|
/// @param _token to transfer, use 0x0 for ether
|
2018-03-27 17:55:37 +00:00
|
|
|
function escapeHatch(address _token) external authP(ESCAPE_HATCH_CALLER_ROLE, arr(_token)) {
|
2018-02-10 14:14:52 +00:00
|
|
|
require(escapeBlacklist[_token]==false);
|
|
|
|
|
|
|
|
uint256 balance;
|
|
|
|
|
|
|
|
/// @dev Logic for ether
|
|
|
|
if (_token == 0x0) {
|
|
|
|
balance = this.balance;
|
|
|
|
escapeHatchDestination.transfer(balance);
|
|
|
|
EscapeHatchCalled(_token, balance);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/// @dev Logic for tokens
|
|
|
|
ERC20 token = ERC20(_token);
|
|
|
|
balance = token.balanceOf(this);
|
|
|
|
require(token.transfer(escapeHatchDestination, balance));
|
|
|
|
EscapeHatchCalled(_token, balance);
|
|
|
|
}
|
|
|
|
|
2018-02-13 00:24:19 +00:00
|
|
|
/// @notice Checks to see if `_token` is in the blacklist of tokens
|
|
|
|
/// @param _token the token address being queried
|
|
|
|
/// @return False if `_token` is in the blacklist and can't be taken out of
|
|
|
|
/// the contract via the `escapeHatch()`
|
2018-03-27 17:55:37 +00:00
|
|
|
function isTokenEscapable(address _token) view external returns (bool) {
|
2018-02-13 00:24:19 +00:00
|
|
|
return !escapeBlacklist[_token];
|
|
|
|
}
|
|
|
|
|
2018-03-27 17:55:37 +00:00
|
|
|
function _init(address _escapeHatchDestination) internal {
|
|
|
|
initialized();
|
|
|
|
require(_escapeHatchDestination != 0x0);
|
|
|
|
|
|
|
|
escapeHatchDestination = _escapeHatchDestination;
|
|
|
|
}
|
|
|
|
|
2018-02-13 00:24:19 +00:00
|
|
|
/// @notice Creates the blacklist of tokens that are not able to be taken
|
|
|
|
/// out of the contract; can only be done at the deployment, and the logic
|
|
|
|
/// to add to the blacklist will be in the constructor of a child contract
|
|
|
|
/// @param _token the token contract address that is to be blacklisted
|
|
|
|
function _blacklistEscapeToken(address _token) internal {
|
|
|
|
escapeBlacklist[_token] = true;
|
|
|
|
EscapeHatchBlackistedToken(_token);
|
|
|
|
}
|
2018-02-10 14:14:52 +00:00
|
|
|
}
|