您正在查看: surou 发布的文章

EOS 整个链停止,重新强制引导启动

BP节点全部停止,各节点重新启动后,无法正常出块

最近因为非技术原因,导致测试链所有BP节点都停止了。
由于producer_plugin中的安全检查,导致BP节点启动后,一直处于start_block_result::waiting
导致整个链网络瘫痪。

强制引导启动

先准备一台非目前BP节点所在的机器做强制引导节点。然后把目前出块账户的 2/3 +1 个BP账号都配置到此节点。
并把目前其他的BP节点都关闭pkill nodeos

修改强制引导节点的链程序,绕过安全检查

https://github.com/EOSIO/eos/blob/3c553db73864ea19458512b4669cf9942dc59f57/plugins/producer_plugin/producer_plugin.cpp#L1350
将下面代码注释掉,并重新编译链程序

if (_pending_block_mode == pending_block_mode::speculating) {
      auto head_block_age = now - chain.head_block_time();
      if (head_block_age > fc::seconds(5))
         return start_block_result::waiting;
   }

修改BP节点的配置为同步节点的配置

先将其余BP节点的config,改为同步节点的配置,也就是把强制引导节点的p2p-peer-address加到各BP节点的config,并把各BP节点的producer-namesignature-provider都注释掉。

强制启动引导节点

添加-e,强制启动出块

nodeos -e

此时强制出块节点应该已出块。

启动各BP节点

将其余BP节点分别启动,此时开始同步区块,等到追上强制出块节点高度。

迁移BP账号

关闭强制出块节点,把BP账号从config里面每注释掉一部分,就把对应的BP节点停止,并修改config,把对应的BP账号放开。此时重新启动强制出块节点和新迁移BP的节点,现在强制出块节点除去注释掉的部分出块账号。其余BP账号出块正常,且能从新迁移BP节点同步对应的出块BP账号的出块。新迁移的BP节点,出块也正常。
依次类推,迁移其余BP节点账号,等其余BP节点都出块正常后,及此次强制引导启动链网络结束。

备注

能出现全部节点停止的情况,基本都是特殊公链-“私链”,所以具有大部分BP的掌控权,我们先只针对此情景提出修复方案。

EOS 同步Mongo,各表数据解释

使用场景

由于需要对链上数据进行大量自定义得查询,一般来说还是通过部署同步节点,然后使用mongo插件将链上数据同步到Mongo进行使用更直接和方便。(虽然有其他得替代方案比如(Hyperion-History-API)(history-tools),测试下来,mongo更直接些,如果以后有其他方案更新,我们在做文章跟进,此处我们只讨论同步mongo得方案)

mongo 插件配置

必须开启以下配置,才能获取以下讨论中谈及得数据

  • mongodb-store-block-states = true
  • mongodb-store-blocks = true
  • mongodb-store-transactions = true
  • mongodb-store-transaction-traces = true
  • mongodb-store-action-traces = true

Mongo 数据表介绍

使用官方得mongo插件同步后,库数据表如下

  • account_controls
  • accounts
  • action_traces
  • block_states
  • blocks
  • pub_keys
  • transaction_traces
  • transactions

// 稍后补充

附加

action_traces表默认没有"trx_id"得索引,如果想根据"trx_id"查询得话,可以自行添加索引,不然会很慢。

db.action_traces.getIndexes()
db.action_traces.createIndex({"trx_id":1, "background":true})

curl is not working on Ubuntu 18.04 LTS

解决办法

sudo apt remove -y libcurl4
sudo apt install -y libcurl4 curl

参考

https://askubuntu.com/questions/1029273/curl-is-not-working-on-ubuntu-18-04-lts

安装mongodb,并切换dbpath

卸载旧版本(按需要)

sudo apt-get --purge remove mongodb mongodb-clients mongodb-server

安装新版本

sudo apt-get install mongodb

查看安装版本

mongo -version

测试启动

sudo service mongodb start

查看进程

ps -ef | grep mongo
mongodb  25630     1  0 13:13 ?        00:00:00 /usr/bin/mongod --unixSocketPrefix=/run/mongodb --config /etc/mongodb.conf

确认服务开机启动正常

pgrep mongo -l

停止服务

sudo service mongodb stop

修改自定义数据目录

假设我们想从默认位置,换到 /home/surou/mongodb

先新建此目录

mkdir /home/surou/mongodb

设置mongo组访问权限

cd /home/surou/mongodb
chown -R mongodb:mongodb /home/surou/mongodb
sudo chmod 777 mongodb

修改配置

sudo vi /etc/mongodb.conf
# Where to store the data.
dbpath=/home/surou/mongodb

#where to log
logpath=/home/surou/mongodb/mongodb.log

logappend=true

bind_ip = 0.0.0.0

重新启动

sudo service mongodb start

查看状态

sudo service mongodb status

创建自己的数字货币(ERC20 代币)

本文从技术角度详细介绍如何基于以太坊 ERC20 创建代币的流程.

写在前面

本文所讲的代币是使用以太坊智能合约创建,阅读本文前,你应该对以太坊、智能合约有所了解,如果你还不了解,建议你先看以太坊是什么

代币 Token

如果不那么追求精确的定义,代币就是数字货币,比特币、以太币就是一个代币。
利用以太坊的智能合约可以轻松编写出属于自己的代币,代币可以代表任何可以交易的东西,如:积分、财产、证书等等。
因此不管是出于商业,还是学习很多人想创建一个自己的代币,先贴一个图看看创建的代币是什么样子。

今天我们就来详细讲一讲怎样创建一个这样的代币。

ERC20 Token

也许你经常看到 ERC20 和代币一同出现, ERC20 是以太坊定义的一个代币标准。
要求我们在实现代币的时候必须要遵守的协议,如指定代币名称、总量、实现代币交易函数等,只有支持了协议才能被以太坊钱包支持。
其接口如下:

contract ERC20Interface {

    string public constant name = "Token Name";
    string public constant symbol = "SYM";
    uint8 public constant decimals = 18;  // 18 is the most common number of decimal places

    function totalSupply() public constant returns (uint);
    function balanceOf(address tokenOwner) public constant returns (uint balance);
    function allowance(address tokenOwner, address spender) public constant returns (uint remaining);
    function transfer(address to, uint tokens) public returns (bool success);
    function approve(address spender, uint tokens) public returns (bool success);
    function transferFrom(address from, address to, uint tokens) public returns (bool success);

    event Transfer(address indexed from, address indexed to, uint tokens);
    event Approval(address indexed tokenOwner, address indexed spender, uint tokens);
}

简单说明一下:

  • name : 代币名称
  • symbol: 代币符号
  • decimals: 代币小数点位数,代币的最小单位, 18 表示我们可以拥有 .0000000000000000001 单位个代币。
  • totalSupply () : 发行代币总量。
  • balanceOf (): 查看对应账号的代币余额。
  • transfer (): 实现代币转账交易,用于给用户发送代币(从我们的账户里)。
  • transferFrom (): 给被授权的用户使用,他可以从我们(参数 from)的账户里发送代币给其他用户(参数 to)。
  • allowance (): 返回授权花费的代币数。
  • approve (): 授权用户可代表我们花费的代币数。

编写代币合约代码

代币合约代码:

pragma solidity ^0.4.16;

interface tokenRecipient { function receiveApproval(address _from, uint256 _value, address _token, bytes _extraData) public; }

contract TokenERC20 {
    string public name;
    string public symbol;
    uint8 public decimals = 18;  // 18 是建议的默认值
    uint256 public totalSupply;

    mapping (address => uint256) public balanceOf;  //
    mapping (address => mapping (address => uint256)) public allowance;

    event Transfer(address indexed from, address indexed to, uint256 value);

    event Burn(address indexed from, uint256 value);


    function TokenERC20(uint256 initialSupply, string tokenName, string tokenSymbol) public {
        totalSupply = initialSupply * 10 ** uint256(decimals);
        balanceOf[msg.sender] = totalSupply;
        name = tokenName;
        symbol = tokenSymbol;
    }


    function _transfer(address _from, address _to, uint _value) internal {
        require(_to != 0x0);
        require(balanceOf[_from] >= _value);
        require(balanceOf[_to] + _value > balanceOf[_to]);
        uint previousBalances = balanceOf[_from] + balanceOf[_to];
        balanceOf[_from] -= _value;
        balanceOf[_to] += _value;
        Transfer(_from, _to, _value);
        assert(balanceOf[_from] + balanceOf[_to] == previousBalances);
    }

    function transfer(address _to, uint256 _value) public returns (bool) {
        _transfer(msg.sender, _to, _value);
        return true;
    }

    function transferFrom(address _from, address _to, uint256 _value) public returns (bool success) {
        require(_value <= allowance[_from][msg.sender]);     // Check allowance
        allowance[_from][msg.sender] -= _value;
        _transfer(_from, _to, _value);
        return true;
    }

    function approve(address _spender, uint256 _value) public
        returns (bool success) {
        allowance[msg.sender][_spender] = _value;
        return true;
    }

    function approveAndCall(address _spender, uint256 _value, bytes _extraData) public returns (bool success) {
        tokenRecipient spender = tokenRecipient(_spender);
        if (approve(_spender, _value)) {
            spender.receiveApproval(msg.sender, _value, this, _extraData);
            return true;
        }
    }

    function burn(uint256 _value) public returns (bool success) {
        require(balanceOf[msg.sender] >= _value);
        balanceOf[msg.sender] -= _value;
        totalSupply -= _value;
        Burn(msg.sender, _value);
        return true;
    }

    function burnFrom(address _from, uint256 _value) public returns (bool success) {
        require(balanceOf[_from] >= _value);
        require(_value <= allowance[_from][msg.sender]);
        balanceOf[_from] -= _value;
        allowance[_from][msg.sender] -= _value;
        totalSupply -= _value;
        Burn(_from, _value);
        return true;
    }
}

部署

在开发测试智能合约时,MetaMaskRemix Solidity IDE 是两个非常好用的工具,今天就用他们来完成部署。

  1. 安装和配置 MetaMask 请参考开发、部署第一个去中心化应用,不同的上本文选择了以太坊的测试网络 Ropsten,如果你没有余额请点击购买 buy,进入的网站可以送一些测试以太币给你,配置好之后,界面应该如下:

  2. 浏览器打开 Remix Solidity IDE,复制以上源码粘贴上,在右侧选项参考如图的设置:

    注意 Environment 和 Account 和 MetaMask 保持一致,然后选择合约 TokenERC20,填入你想要的发行量,名称及代号,就可以创建合约了。
    这时 MetaMask 会弹出一个交易确认框,点 SUBMIT。待合约部署交易确认之后,复制合约地址。

  3. 打开 Metamask 界面,切换到 TOKENS,点添加合约,出现如下对话框:

    填入刚刚复制的地址,点 ADD,这时你就可以看到你创建的代币了,如图:

    哈哈,你已经完成了代币的创建和部署 (正式网络和测试网络部署方法一样),可以在 Etherscan 查询到我们刚刚部署的代币。可以用它进行 ICO 了,从此走上人生巅峰(玩笑话,不鼓励大家发行无意义的代币)。

代币交易

由于 MetaMask 插件没有提供代币交易功能,同时考虑到很多人并没有以太坊钱包或是被以太坊钱包网络同步问题折磨,今天我用网页钱包来讲解代币交易。

  1. 进入网页钱包地址 , 第一次进入有一些安全提示需要用户确认。
  2. 进入之后,按照下图进行设置:
  3. 连接上之后,如图

    需要添加代币,填入代币合约地址。
  4. 进行代币转账交易

    在接下来的交易确认也,点击确认即可。
  5. 交易完成后,可以看到 MetaMask 中代币余额减少了,如图:

    代币交易是不是很简单,只要明白了交易流程,使用其他的钱包也是一样的道理。

转载自:https://learnblockchain.cn/2018/01/12/create_token/