[Solidity] 이더리움 체인 ERC-20 토큰(코인) 락 걸기 & 락 해제 하기
OwnerHelper / tokenLock / tokenPersonalLock / onlyOwner / removePersonalTokenLock / removeTokenLock
OwnerHelper
-
특정 함수를 관리자만 사용할 수 있도록 설정하는
OwnerHelper
함수 -
OwnerHelper
를 사용하여public
으로 공개되어 있는 함수 중에, 관리자만 접근 가능한 함수를 만들수 있다 -
OwnerHelper
컨트랙트는 abstract contract 라고 하는 추상 컨트랙트이다
tokenLock
-
tokenLock
은 토큰의 전체 락에 대한 처리이고 -
tokenPersonalLock
은 토큰의 개인 락에 대한 처리이다 -
함수
isTokenLock
은 전체 락과 / 보내는 사람의 락 / 받는 사람의 락을 검사하여 ‘락이 걸려 있는지 확인’한다 -
이 락들을 제거 할 수 있는
removeTokenLock
과removePersonalTokenLock
도 존재한다 -
이 함수들은
onlyOwner
를 적용하여 관리자만 락을 해제할 수 있도록 해야 한다
👉 락을 적용하게 되면 모든 락을 해제할 때만 토큰의 이동이 가능하게된다
👉 예를 들어 거래소 입출금 금지(?)
Ropsten 테스트넷에 ERC-20 토큰 락 걸기 & 락 해제 해보기
-1. 메타마스크에 접속하고, Ropsten 테스트 네트워크를 선택한다
-2. Remix IDE 에서 SimpleToken.sol
파일을 생성한다
-3. Remix IDE에 OwnerHelper
코드를 추가한다
-4. SimpleToken.sol
파일을 컴파일한다
-5. DEPLOY & RUN TRANSACTIONS 항목으로 이동해서
Ropsten 네트워크에 OwnerHelper
가 적용된 SimpleToken
을 배포한다
5-1. ENVIRONMENT 부분을 Injected Web3로 변경하여 ‘메타마스크’와 연동한다
5-2. DEPLOY에 들어갈 인자값을 입력한다
- GETNAME 👉 코인이름
- GETSYMBOL 👉 코인이름약자
5-3. 설정이 완료되면 transact 버튼을 누른다
-6. 메타마스크 하단의 토큰 가져오기 (Import tokens)를 클릭하고
토큰 계약 주소에는 Remix IDE에서 복사한 컨트랙트 주소를 입력한 후,
‘맞춤형 토큰 추가(Add Custom Token)’ 버튼을 누른다
-7. 다른 테스트 계정 주소로 토큰을 전송해(보내)본다
하지만 락이 걸려 있어서 토큰이 전송이 안된다
-8. 이더스캔에서 트랜잭션이 실패한 원인을 확인해본다
-9. 락을 해제하기 위해
removePersonalTokenLock
함수에 전송할 계정(내 주소)을 입력하고
removeTokenLock
함수를 실행시킨다
-10. removeTokenLock
락을 해제한 내역이 보이고
토큰도 전송이 되는걸 확인 할 수 있다
정상적인 트랜잭션 👆
ERC-20 토큰(코인) 락 걸고 & 락 해제 코드
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.5.0 <0.9.0;
interface ERC20Interface {
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address recipient, uint256 amount) external returns (bool);
function approve(address spender, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function transferFrom(address spender, address recipient, uint256 amount) external returns (bool);
event Transfer(address indexed from, address indexed to, uint256 amount);
event Transfer(address indexed spender, address indexed from, address indexed to, uint256 amount);
event Approval(address indexed owner, address indexed spender, uint256 oldAmount, uint256 amount);
}
//----------------------------------------------------------------
library SafeMath { // 👉 기존 ERC-20 코드에서 SafeMath 라이브러리를 추가
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a * b;
assert(a == 0 || c / a == b);
return c;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a / b;
return c;
}
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
assert(b <= a);
return a - b;
}
function add(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a + b;
assert(c >= a);
return c;
}
}
// 'internal'은 함수가 컨트랙트 내부에서만 사용될 수 있도록 제한할 때 사용한다
// 'pure' 는 함수가 상태 변수를 읽거나 쓰지 않을 때 사용한다
// 'SafeMath' 라이브러리의 함수에서는 단순 연산의 결과값을 반환하기 때문에 상태 변수를 읽거나 쓰지 않는다
// ⭐ 'assert'는 false를 리턴하지만 계약을 실행 시키기 전에 확인을 할 수 없고,
// ⭐ 'require'는 계약을 실행 시키기 전에 확인을 할 수 있는것으로 알 수 있다
//----------------------------------------------------------------
abstract contract OwnerHelper {
// 👉 'OwnerHelper' 컨트랙트는 'abstract contract'라고 하는 추상 컨트랙트
// abstract contract는 contract의 구현된 기능과 interface의 추상화 기능 모두를 포함한다
// abstract contract는 만약 실제 contract에서 사용하지 않는다면 추상으로 표시되어 사용되지 않는다
address private _owner;
// 👉 _owner는 관리자를 나타냄
event OwnershipTransferred(address indexed preOwner, address indexed nextOwner);
// 👉 'OwnershipTransferred' 이벤트는 관리자가 변경되었을때 이전 관리자의 주소와 새로운 관리자의 주소 로그를 남긴다
modifier onlyOwner {
require(msg.sender == _owner, "OwnerHelper: caller is not owner");
_;
}
// 👉 'onlyOwner' 함수 변경자는 함수 실행 이전에 함수를 실행시키는 사람이 관리자인지 확인한다
constructor() {
_owner = msg.sender;
}
function owner() public view virtual returns (address) {
return _owner;
}
function transferOwnership(address newOwner) onlyOwner public {
require(newOwner != _owner);
require(newOwner != address(0x0));
address preOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(preOwner, newOwner);
}
}
//----------------------------------------------------------------
contract SimpleToken is ERC20Interface, OwnerHelper {
using SafeMath for uint256;
// 👉 메인 컨트랙트인 SimpleToken에서는 자료형 'uint256'에 대해서 'SafeMath' 라이브러리를 사용하도록 선언해준다
// 'uint256'으로 선언된 함수에 대해서 'SafeMath Library'를 이용해서 'transferFrom'과 '_transfer' 함수를 사용을 할 수 있다
// 'transferFrom'과 '_transfer'에서 사용되는 연산자(+, -)를 'SafeMath' 라이브러리 함수를 사용해 안전한 연산자로 변경할 수 있음
mapping (address => uint256) private _balances;
mapping (address => mapping (address => uint256)) public _allowances;
uint256 public _totalSupply;
string public _name;
string public _symbol;
uint8 public _decimals;
bool public _tokenLock;
mapping (address => bool) public _personalTokenLock;
constructor(string memory getName, string memory getSymbol) {
_name = getName;
_symbol = getSymbol;
_decimals = 18;
_totalSupply = 100000000e18;
_balances[msg.sender] = _totalSupply;
_tokenLock = true;
}
function name() public view returns (string memory) {
return _name;
}
function symbol() public view returns (string memory) {
return _symbol;
}
function decimals() public view returns (uint8) {
return _decimals;
}
function totalSupply() external view virtual override returns (uint256) {
return _totalSupply;
}
function balanceOf(address account) external view virtual override returns (uint256) {
return _balances[account];
}
function transfer(address recipient, uint amount) public virtual override returns (bool) {
_transfer(msg.sender, recipient, amount);
emit Transfer(msg.sender, recipient, amount);
return true;
}
function allowance(address owner, address spender) external view override returns (uint256) {
return _allowances[owner][spender];
}
function approve(address spender, uint amount) external virtual override returns (bool) {
// uint256 currentAllownace = _allowances[spender][msg.sender]; // 삭제
uint256 currentAllowance = _allowances[msg.sender][spender]; // 추가
require(_balances[msg.sender] >= amount,"ERC20: The amount to be transferred exceeds the amount of tokens held by the owner.");
_approve(msg.sender, spender, currentAllowance, amount);
return true;
}
function transferFrom(address sender, address recipient, uint256 amount) external virtual override returns (bool) {
_transfer(sender, recipient, amount);
emit Transfer(msg.sender, sender, recipient, amount);
uint256 currentAllowance = _allowances[sender][msg.sender];
require(currentAllowance >= amount, "ERC20: transfer amount exceeds allowance");
// 다음의 코드에서 currentAllowance.sub(amount)이 SafeMath 라이브러리 함수를 사용한 예시
_approve(sender, msg.sender, currentAllowance, currentAllowance - amount);
return true;
}
function _transfer(address sender, address recipient, uint256 amount) internal virtual {
require(sender != address(0), "ERC20: transfer from the zero address");
require(recipient != address(0), "ERC20: transfer to the zero address");
require(isTokenLock(sender, recipient) == false, "TokenLock: invalid token transfer");
uint256 senderBalance = _balances[sender];
require(senderBalance >= amount, "ERC20: transfer amount exceeds balance");
_balances[sender] = senderBalance.sub(amount);
_balances[recipient] = _balances[recipient].add(amount);
}
// uint256 으로 선언했던 'currentAllowance'와 'senderBalance'에 'sub' 함수를 사용 가능한 모습을 볼 수 있다
// 'Mapping' 으로 선언했던 '_balances' 도 uint256 으로 받아오는 값에서 add 함수를 사용 가능하다
// 👉 '_transfer'에 검사를 추가해, 보내는 사람과 받는 사람 중 락이 걸려있다면 토큰은 이동이 불가능하다
function isTokenLock(address from, address to) public view returns (bool lock) {
// 👉 함수 'isTokenLock' 은 전체 락과, 보내는 사람의 락, 받는 사람의 락을 검사하여 락이 걸려 있는지 확인한다
lock = false;
if(_tokenLock == true)
{
lock = true;
}
if(_personalTokenLock[from] == true || _personalTokenLock[to] == true) {
lock = true;
}
}
// 'tokenLock' 은 토큰의 전체 락에 대한 처리, 'tokenPersonalLock' 은 토큰의 개인 락에 대한 처리이다
// 다음의 코드에서 함수로 전달되는 파라미터 브라켓 뒤에 오는 onlyOwner가 예시이다
function removeTokenLock() onlyOwner public {
require(_tokenLock == true);
_tokenLock = false;
}
// 락들을 제거 할 수 있는 removeTokenLock
// 다음의 코드에서 함수로 전달되는 파라미터 브라켓 뒤에 오는 onlyOwner가 예시이다
function removePersonalTokenLock(address _who) onlyOwner public {
require(_personalTokenLock[_who] == true);
_personalTokenLock[_who] = false;
}
// 락들을 제거 할 수 있는 removePersonalTokenLock
// 이 함수들은 'onlyOwner'를 적용하여 관리자만 락을 해제할 수 있도록 해야 한다
// 이렇게 락을 적용하게 되면 모든 락을 해제할 때만 토큰의 이동이 가능하다
function _approve(address owner, address spender, uint256 currentAmount, uint256 amount) internal virtual {
require(owner != address(0), "ERC20: approve from the zero address");
require(spender != address(0), "ERC20: approve to the zero address");
require(currentAmount == _allowances[owner][spender], "ERC20: invalid currentAmount");
_allowances[owner][spender] = amount; // 삭제
emit Approval(owner, spender, currentAmount, amount);
}
}
더 공부할 내용 📃
-
관리자가 한명인 컨트랙트를 세명인 컨트랙트로 변경하여
투표를 통해 관리자 변경으로 만들어 보기 -
락을 일회성이 아닌 재사용 가능한 형태의 함수로 변경해 보기