-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathVault.sol
118 lines (81 loc) · 3.86 KB
/
Vault.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
// SPDX-License-Identifier: MIT
//EXPLICACIÓN: Para poder ejecutar el vault.sol se tiene que:
// 1. Deploy de Token.sol para obtener la direccion del contrato
// 2. Deploy de Vault.sol con la dirección del token establecida en el constructor
// 3. En el contrato token.sol, en la funcion approve, establecer la cantidad y la direccion del contrato vault.sol
// 4. La billetera que haya realizado el paso 3 y este inscrita en la whitelist (sino añadir) podra dipositar como maximo lo aprobado anteriormente.
// 5. Para retirar la cantidad tiene que ser igual a la dipositada previamente
// 6. Registro del balance de cada dirección y si pertenece en la whitelist, tenemos los dos mappings.
//IMPORTANTE !!!!! LA FUNCION DEPOSIT Y WITHDRAW ESTA PROGRAMADA COMO X/10**decimales es decir si queremos depositar 1000 tokens con 2 decimals debemos poner 100000
pragma solidity ^0.8.13;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
contract Vault {
using SafeERC20 for IERC20;
IERC20 public token;
uint public totalSupply;
// totalSupply se modifica segun la variable _shares ya sea en en funcion de burn o mint
address owner;
mapping(address => uint) public balanceOf;
constructor(address _token) {
token = IERC20(_token);
owner = msg.sender;
}
//WHITELIST
modifier onlyOwner(){
require(msg.sender == owner, "Not Owner!!");
_;
}
//A partir de la dirección retorna un T/F dependiendo de si esta a la WL
mapping(address =>bool) public whitelistedAddresses;
//Añadir usuario a la WL publica, uso exclusivo del owner
function addUser(address _addressToWL) public onlyOwner {
whitelistedAddresses[_addressToWL] = true;
}
function disableUser(address _addressOut) public onlyOwner {
whitelistedAddresses[_addressOut]= false;
}
// Verificar si el usuario pertenece en la WL, retorna T/F
function isInWhitelist(address _whitelistedAddress) public view returns(bool) {
bool userIsWhitelisted = whitelistedAddresses[_whitelistedAddress];
return userIsWhitelisted;
}
modifier isWhitelisted(address _address) {
require(whitelistedAddresses[_address], "You need to be whitelisted");
_;
}
/////////////////////////////////////////////////////////
//VAULT
function _mint(address _to, uint _shares) private {
totalSupply += _shares;
balanceOf[_to] += _shares;
}
function _burn(address _from, uint _shares) private {
totalSupply -= _shares;
balanceOf[_from] -= _shares;
}
//Añado funcion burn i mint para agregar o quitar tokens dels contrato
function deposit(uint _amount) external {
require(whitelistedAddresses[msg.sender] == true, "ERROR not in WL");
// shares puede ser sustituido por otros calculos distintos
uint shares;
if (totalSupply == 0) {
shares = _amount;
} else {
shares = (_amount * totalSupply) / token.balanceOf(address(this));
}
_mint(msg.sender, shares);
token.transferFrom(msg.sender, address(this), _amount);
}
//Falta añadir que solo pueden retirar lo que han depositado. --> require?
function withdraw(uint _shares) external {
//Quien ejecute tiene que estar dentro de la whitelist
require(whitelistedAddresses[msg.sender] == true, "ERROR not in WL");
// La cantidad a retirar tiene que ser igual a la dipositada anteriormente
// Si diposita multiples veces la cantidad a devolver es la total
require (balanceOf[msg.sender] == _shares, "not this amount" );
uint amount = (_shares * token.balanceOf(address(this))) / totalSupply;
_burn(msg.sender, _shares);
token.transfer(msg.sender, amount);
}
}