缩短xfrp客户端断线重连时间

试用一段时间xfrp以后发现这货不是很稳定,容易任性断开,重连的时间不定,可能是秒级,也可能是分钟级甚至小时级,实在不能忍,于是迫切希望能够想个法子缩短这个断线重连的问题。

于是简单想了一个思路:

因为云主机不太可能任性开放端口,所以常规端口扫描的方法并不可行

  1. 客户端先检测自己能否正常上网
  2. 确定上网OK后,客户端向xfrps的api发起请求查看返回的端口号
  3. 根据返回端口号来判断是否重启xfrpc强制重连xfrps服务

但实际发现xfrps自带的api有个bug,即使是客户端断开连接,依然返回原来的端口号,这就比较扯,于是给原项目提了个Issues:设备断开后调用api依然可以获取remote port,在得到原项目解决方案之前,只能自己简单撸个api来完成检测端口的任务曲线救国。

Python Flask甩起来:

 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
#!/usr/bin/env python3
# coding=utf-8
import json
import socket

import requests
from flask import Flask, jsonify

app = Flask(__name__)


def get_port(sn):
    # 利用原api获取设备占用端口
    url = "http://127.0.0.1:7500/api/port/tcp/getport/" + str(sn)
    try:
        r = requests.request("GET", url)
        result = r.text
        port = json.loads(result).get("port")
        return port
    except:
        pass


def is_used(ip, port):
    is_used = False
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    try:
        s.bind((ip, int(port)))
    except socket.error as e:
        #print(e)
        is_used = True
    finally:
        s.close()
    return is_used


@app.route('/api/port/tcp/getport/<sn>')
def is_online(sn):
    port = get_port(sn)
    # 当xfrps端口在线且该端口确实有占用才能确定该客户端在线,其他均为设备断开
    if port:
        if is_used("127.0.0.1", port):
            return jsonify({"code": 0, "msg": "", "port": port}), 200
    return jsonify({"code": 1, "msg": "can not get port by its runid", "port": 0}), 200


if __name__ == '__main__':
    # app.debug = True
    app.run(host='0.0.0.0')

测试

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
# 客户端在线,返回端口号
curl http://host:5000/api/port/tcp/getport/2076932757AC
{
  "code": 0, 
  "msg": "", 
  "port": 62058
}
# 不在线返回0
curl http://host:5000/api/port/tcp/getport/2076932757A
{
  "code": 1, 
  "msg": "can not get port by its runid", 
  "port": 0
}

客户端搞个shell脚本检测

 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
PROG_NAME="xfrpc"
PROG_UCI_CONF="$PROG_NAME"
PROG_COMMAND=$(which "$PROG_NAME")
JQ_NAME="jq"
JQ_COMMAND=$(which "$JQ_NAME")
PROG_INITD="/etc/init.d/$PROG_NAME"
# 拿到remote server地址
SERVER_ADDR=$($UCI get $PROG_UCI_CONF.common.server_addr)
REMOTE_PORT_API="${SERVER_ADDR}:5000/api/port/tcp/getport"
GENERATE_204="http://g.cn/generate_204"
RUN_ID=$($PROG_COMMAND -r | cut -d ":" -f2)

if [ -z "$JQ_COMMAND" ]; then
    echo "error: $JQ_NAME not found!"
    exit 1
fi

get_remote_port() {
    PORT=$(curl --connect-timeout 3 -m 3 "$REMOTE_PORT_API/$RUN_ID" 2>/dev/null | $JQ_COMMAND '.port') || exit 1
    echo $PORT
}

CODE=$(curl --connect-timeout 3 -m 3 -sk $GENERATE_204 -w '%{http_code}') || exit 1
if [ "$CODE"x = "204"x ]; then
    echo "Network ok, check xfrps remote port..."
    PORT=$(get_remote_port)
    if [ "$PORT"x == "0"x ] || [ $(echo "${PORT}" | grep -E -q '^[0-9]{4,}$') ]; then
        echo "xfrps remote port not available, reconnect..."
        $PROG_INITD restart
    else
        echo "Got remote port:$PORT, do nothing."
    fi
else
    echo "Network down, no need to run xfrpc, stop it."
    $PROG_INITD stop
fi

扔crontab任务计划,每五分钟检测一次

1
2
# crontab -e
*/5 * * * * /root/check_xfrpc.sh

重启crontab生效

1
/etc/init.d/cron restart

查看log有无是否调用成功

1
2
# logread | grep check_xfrpc
Fri Jan 19 16:10:01 2018 cron.info crond[2056]: crond: USER root pid 15405 cmd /root/check_xfrpc.sh
加载评论