zoukankan      html  css  js  c++  java
  • 【区块链】关于ETH/BTC区块的监控

    此次我写的是一个小型的shell, 链接钉钉的机器人, 使用过的应该会比较娴熟的了,下面就简述一下把

    主要的功能就是, 当发现本地数据库区块跟网络上的区块差距相差较大的时候就代表, 数据同步有问题, 这个时候, 发出一条告警出来,

    对于比特网络来说,我用的是https://blockchair.com/

    注意到的哥们已经发现了, 在首页的下方就是它的API, 我用的是比较愚蠢的方式去调用它, 直接用curl吧, 躁一把.

    也可以点进去查询它的相关调用, 这里面所调用的币种也是换汤不换药, 比如以下的, 就是取到指定币种的网络信息, 先拿下来分析一下把

    https://api.blockchair.com/bitcoin/stats
    https://api.blockchair.com/bitcoin-cash/stats
    https://api.blockchair.com/litecoin/stats
    https://api.blockchair.com/bitcoin-sv/stats
    https://api.blockchair.com/dogecoin/stats
    https://api.blockchair.com/dash/stats
    https://api.blockchair.com/groestlcoin/stats
    https://api.blockchair.com/bitcoin/testnet/stats我

    直接用curl了,取到信息了, 如下

    但是有时候使用curl会给我弹出total这些渣渣字段, 很不喜欢, 所以, 发招用s参数吧, 以防万一

    curl -s https://api.blockchair.com/ethereum/stats

    取到的数据很明显了, 就是一个json格式的, 难道我还要去弄一个工具去格式化它? 非也

    在linux中, 想必大家都知道,jq命令就能显示格式化json, 如果没有的话安装一下

    yum  install jq -y

     这样就好办很多了, 我们可以先把curl定义下来的东西重定向到一个文件中去啊, 然后用jq去分析那个文件不就行了吗, 接下来数据就变成这样了

    [root@tx ~]# curl -s https://api.blockchair.com/ethereum/stats > block
    [root@tx ~]# jq ".data.blocks" block 
    9551780
    

    你看, 直接取到网络上最新块的高度了, 但是, 为什么会这样取呢, 其实用了jq之后格式是这样的了, 用过的应该就很清楚

     看, 活生生就是一个json, 我用的是.data.blocks, 就是直接把高度过滤出来而已.

    那么接下来就是, 去取本地数据库中的高度了, 我这边用的是mysql, 直接就拿他开刀

    [root@tx ~]# /data/tools/mysql -uroot -prkm2020 -S /tmp/mysql.sock -P 3306 -e "select cfg_val from db_config.tb_sysconfig where cfg_name='cache.usdt_erc20_sync_bcnum';"
    mysql: [Warning] Using a password on the command line interface can be insecure.
    +---------+
    | cfg_val |
    +---------+
    | 9519439 |
    +---------+
    [root@tx ~]# 
    [root@tx ~]#  

    看, 这就是我本地的高度, 但是, 如果你想拿这个9519439跟上面网络上取到的高度去对比, 那么就不得不去掉mysql查询格式出来的框框, 其实这里可以使用-N -s参数, 到底有什么用呢, 自行百度

    [root@tx ~]# /data/tools/mysql -uroot -prkm2020 -S /tmp/mysql.sock -P 3306 -N -s -e "select cfg_val from db_config.tb_sysconfig where cfg_name='cache.usdt_erc20_sync_bcnum';"
    mysql: [Warning] Using a password on the command line interface can be insecure.
    9519439
    [root@tx ~]# 
    [root@tx ~]#
    

    看, 直接抠出来了, 上面还有一个提醒是吧, 其实不影响, 你直接把值取出来, 然后扔给一个变量去做对比, 是没问题的

    那么直接就看shell吧

    这是一个完整的shell !
    解析器在写的时候可以自行添加.

    currencyType="ethereum"
    MYSQL_CONN="/data/tools/mysql -uroot -prkm2020 -S /tmp/mysql.sock -P 3306 -N -s"
    N=0
    
    function dingding(){
        hostname=`hostname`
        webhook="https://oapi.dingtalk.com/robot/send?access_token=4db55f9cd1f96921acd6187d4431641e68bc39923d84d24fe0dbd"
    
        currTime1=`echo $(date +"%Y-%m-%d.%T")`
        curl ''$webhook'' 
       -H 'Content-Type: application/json' 
       -d '{"msgtype": "text", 
            "text": {
                 "content": "
    服务器:'$hostname' 发生: '$currencyType'区块数据同步异常!
    
    [
    提醒主机:'$hostname'
    提醒信息: 区块数据同步异常
    监控币种:'$currencyType'
    进程同步区块:'$local_height'
    网络最新区块:'$netwo_height'
    区块差距:'$value'
    提醒时间:'$currTime1'
    ]
    "
            }
          }'
    }
    
    function check(){
        local_height=`$MYSQL_CONN -e "select cfg_val from db_config.tb_sysconfig where cfg_name='cache.usdt_erc20_sync_bcnum';"`
        curl -s https://api.blockchair.com/${currencyType}/stats > /tmp/netwo_height
        netwo_height=`jq ".data.blocks" /tmp/netwo_height`
        value=$[netwo_height - local_height]
        echo $(date +"%Y-%m-%d.%T"),$value
        if [ $value -ge 6 ];then
            dingding
            N=$[N+1]
            sleep 600;
        else
            N=0
        fi
    }
    function main(){
        while :
        do
            sleep 180;
            if [ $N == 0 ];then
                check
            else
                if [ $N -ge 2 ];then
                    sleep 300;
                    check
                else
                    sleep 50;
                    check
                fi
            fi
        done
    }
    main
    

    首先, 我把币种声明变量扔在了文件的开头位置, 下面就直接去识别, 就是这个

    currencyType="ethereum"
    

    需要监控其他的可以换成其他的, 比如bitcoin等

    然后我为了偷懒, 把mysql的连接code, 塞进一个变量里面去, 为了下面方便调用它

    dingding模块应该没什么好说的了, 检查模块, 无非就是拿出数据库中的高度, 跟网络上重定向分析出来的高度做对比, 用最新的高度减去本地区块, 得出差距, 我这边设置是在6个节点, 如果大于等于6, 那发个信息出来吧, 

    肯定有人疑问, N是什么鬼, N就是一个统计告警数, 为了不让告警太频繁

    比如我下面的main的主体调用模块, 3分钟检测一下,如果N等于0, 那就是说, 一切正常,  那么如果说, 发现了一次告警, 那么N+1, 发了一次信息, 等待10分钟, 再检测, 如果10分钟后还是没有恢复, 还来告警, 那么N再+1, 那就是跑了下面的大于或等于2的时候那个条件了, 再等待5分钟然后再检查, 如果说, 在这次死循环中, 告警过后恢复正常了, 然后N会被重置为0, 一切还是原来的模样.

    那么钉钉出来是怎么样的呢, 这样的

    目的是什么, 主要是为了让服务器本地环境与网络上的区块环境实现一致同步, 如果服务器的程序出现问题, 那么就能第一时间知道, 然后着手处理,
    对于为什么不用golang去写, 因为golang不会写, bash更简单, 但最不完美的就是, 可能死循环造成的资源消耗远比golang小工具的多?
    有待研究。
    就到这里。

  • 相关阅读:
    C++ XML解析之TinyXML篇[转]
    TinyXML:一个优秀的C++ XML解析器[转]
    nginx 出现413 Request Entity Too Large问题的解决方法
    redis配置认证密码
    《Discuz安装时候出现乱码 -- 问题解决方法》
    MySQL创建用户与授权
    CentOS 7 安装mysql
    setfacl命令 来自: http://man.linuxde.net/setfacl
    install pip3 for python 3.x
    自己制作ssl证书:自己签发免费ssl证书,为nginx生成自签名ssl证书
  • 原文地址:https://www.cnblogs.com/chenglee/p/12362906.html
Copyright © 2011-2022 走看看