bancor去中心化交易所解析(3)Bancor交易所上币及合约
介绍
接下来,我会分析在Bancor交易所上币的过程和带来的影响,以及相关的合约解析
上币流程
在Bancor交易所上币需要手动提交申请,提交申请后,需要质押一定数量的BNT和自己的代币,来锚定价格(同时通过广大人民群众的上币热情,来增加BNT的需求,来提高BNT的价格)。然后交易所就会为你创建连接BNT和你自己代币的中继代币,同时会在合约上注册你的代币。完成这一切后就可以在交易所交易你的币了
合约解析
SmartToken
SmartToken就是智能代币合约
contract SmartToken is ISmartToken, Owned, ERC20Token, TokenHolder {
}
从声明可以看出,SmartToken本质上也是ERC20Token,只不过多实现了几个方法
function issue(address _to, uint256 _amount)
public
ownerOnly
validAddress(_to)
notThis(_to)
{
totalSupply = safeAdd(totalSupply, _amount);
balanceOf[_to] = safeAdd(balanceOf[_to], _amount);
emit Issuance(_amount);
emit Transfer(this, _to, _amount);
}
function destroy(address _from, uint256 _amount) public {
require(msg.sender == _from || msg.sender == owner); // validate input
balanceOf[_from] = safeSub(balanceOf[_from], _amount);
totalSupply = safeSub(totalSupply, _amount);
emit Transfer(_from, this, _amount);
emit Destruction(_amount);
}
从代码可以看出,issue方法可以增加总供应量,相当于创建智能代币, destroy可以减少总供应量,相当于可以销毁智能代币
BancorConverter
BancorConverter
是Bancor合约中非常重要的一个合约,这个合约的作用是用来兑换(基于某一种智能代币),可以用某一种连接器代币购买某一种中继代币,也可以用中继代币来买某一种代币,还有就是实现基于这个智能代币的两种代币的转换。
根据之前说的流程,把智能代币创建好后,就需要在BancorConverter这个合约中创建两个连接器,一个连接至新代币,另一个连接到BNT
首先来看Connector
在合约中的定义
struct Connector {
uint256 virtualBalance; // connector virtual balance
uint32 weight; // connector weight, represented in ppm, 1-1000000 ppm = 百万分之一
bool isVirtualBalanceEnabled; // true if virtual balance is enabled, false if not
bool isPurchaseEnabled; // is purchase of the smart token enabled with the connector, can be set by the owner
bool isSet; // used to tell if the mapping element is defined
}
mapping (address => Connector) public connectors;
首先Connetor是一个结构体,里面存储这对某个代币的必要信息,还有一个map,用来根据地址找到Connetor
这里需要注意的重点是这个virtualBalance
,根据之前我们队协议的了解,我们可以知道,要计算价格,是必须要知道连接器代币储备量的,而且,这个储备量还要能动态调整。这个virtualBalance就是用来记外面的代币的储备量,交易的时候,储备量的变化,就会提现在这个值里。如果不设置virtualBalance,那么储备量会去直接读当前合约地址的balance
那么现在需要添加Connetor
function addConnector(IERC20Token _token, uint32 _weight, bool _enableVirtualBalance)
public
ownerOnly
inactive
validAddress(_token)
notThis(_token)
validConnectorWeight(_weight)
{
// 连接器token不等于smartToken, 连接器token没有被初始化过, 当前cw和加上传进来的cw不超过最大max_cw
require(_token != token && !connectors[_token].isSet && totalConnectorWeight + _weight <= MAX_WEIGHT); // validate input
connectors[_token].virtualBalance = 0;
connectors[_token].weight = _weight;
connectors[_token].isVirtualBalanceEnabled = _enableVirtualBalance;
connectors[_token].isPurchaseEnabled = true;
connectors[_token].isSet = true;
connectorTokens.push(_token);
totalConnectorWeight += _weight;
}
可以看到,默认virtualBalance是0,如果是外部的代币,那么需要更改
function updateConnector(IERC20Token _connectorToken, uint32 _weight, bool _enableVirtualBalance, uint256 _virtualBalance)
public
ownerOnly
validConnector(_connectorToken)
validConnectorWeight(_weight)
{
Connector storage connector = connectors[_connectorToken];
require(totalConnectorWeight - connector.weight + _weight <= MAX_WEIGHT); // validate input
totalConnectorWeight = totalConnectorWeight - connector.weight + _weight;
connector.weight = _weight;
connector.isVirtualBalanceEnabled = _enableVirtualBalance;
connector.virtualBalance = _virtualBalance;
}
获取连接器余额
function getConnectorBalance(IERC20Token _connectorToken)
public
view
validConnector(_connectorToken)
returns (uint256)
{
Connector storage connector = connectors[_connectorToken];
// 如果设置了虚拟余额,则取虚拟余额,如果没有,就直接从合约地址取余额
return connector.isVirtualBalanceEnabled ? connector.virtualBalance : _connectorToken.balanceOf(this);
}
现在看看转化过程virtualBalance的变化
function convertInternal(IERC20Token _fromToken, IERC20Token _toToken, uint256 _amount, uint256 _minReturn)
public
bancorNetworkOnly
conversionsAllowed
greaterThanZero(_minReturn)
returns (uint256)
{
...
// update the source token virtual balance if relevant
Connector storage fromConnector = connectors[_fromToken];
if (fromConnector.isVirtualBalanceEnabled)
fromConnector.virtualBalance = safeAdd(fromConnector.virtualBalance, _amount);
// update the target token virtual balance if relevant
Connector storage toConnector = connectors[_toToken];
if (toConnector.isVirtualBalanceEnabled)
toConnector.virtualBalance = safeSub(toConnector.virtualBalance, amount);
...
}
增加代币会对原来的代币产生影响吗
笔者在看合约之前一直有一个疑惑,就是Bancor网络里所有的代币都连接到了BNT,那么我加入一个新币的时候,会不会吧BNT打到Bancor网络里,导致BNT的储备量一下子飙升,附带所有代币大幅贬值。
实际上,这个情况是不存在的回到,之前贴的代码片段,我们知道货币兑换其实是发生在一个连接器里的,一个连接器的决定因素是由智能代币和连接器里记录的代币余额,所以实际的图是这样的
image.png看到了吗,每个连接器的BNT的储备量是独立的,也就是我新加了一个币,创建的是一个新的连接器,并不会对原来的连接器的BNT余额产生任何影响。
那当然BNT的价格是会受外部影响的(包括别的交易所,或者本身合约兑ETH发生了变化),就有可能产生套利,这样就促进了所有玩家,自己通过套利来调整价格
在Bancor网络里,对于大部分代币来说,BNT只是普通的ERC20代币,而对于合约本身来说BNT就是锚定在ETH上的一种智能代币。
这也是为什么在BNT即是智能代币,也是普通的流动代币的原因