1

我的代码同时使用烧瓶(http,作为主线程)、ngrok 和 tcp 服务器(ngrok 和 tcp 服务器都在两个线程上运行)有问题(请参阅下面的错误消息)

Traceback (most recent call last):
  File "/home/monki/anaconda3/lib/python3.8/threading.py", line 932, in _bootstrap_inner
    self.run()
  File "/home/monki/anaconda3/lib/python3.8/threading.py", line 870, in run
    self._target(*self._args, **self._kwargs)
  File "/home/monki/VSCode/Python/Webhook/app.py", line 49, in launchServer
    s.bind((hostname, TCP_PORT))
OSError: [Errno 98] Address already in use

我已经使用检查了端口

ps -fA | grep python

发现没有端口被使用

root         910       1  0 15:02 ?        00:00:00 /usr/bin/python3 /usr/bin/networkd-dispatcher --run-startup-triggers
root        1036       1  0 15:02 ?        00:00:00 /usr/bin/python3 /usr/share/unattended-upgrades/unattended-upgrade-shutdown --wait-for-signal
x       6316    5499  0 15:55 pts/0    00:00:00 grep --color=auto python

这是我的代码

from flask import Flask, request, Response
import json
import socket
import threading  
from pyngrok import ngrok

from queue import Queue

#tcp server
TCP_IP = ''
TCP_PORT = 13370
BUFFER_SIZE  = 20

# http server
HOST_URL = 'http://xxx.ngrok.io'


def start_ngrok():
    ### sorry i have to blank out the account
    ngrok.set_auth_token("xxxxxx")
    url = ngrok.connect(5000, subdomain='xxx').public_url
    print(' * Tunnel URL:', url)


def launchServer(out_q):
    print("TCP thread started")
    hostname = socket.gethostname()
    local_ip = socket.gethostbyname(hostname)

    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)

    s.bind((hostname, TCP_PORT))
    s.listen(1)

    while True:
        print('waiting for a connection')
        conn, addr = s.accept()
        try:
            print(sys.stderr + 'client connected:' + addr)
            while True:
                out_q.put(data)
                data = conn.recv(1024)

                # print >>sys.stderr, 'received "%s"' % data
                if data:
                    conn.sendall(data)
                else:
                    break
        finally:
            conn.close()

app = Flask(__name__)

# route definition
@app.route('/<webhookAddr>', methods=['POST'])
def response(in_q, webhookAddr):

    if webhookAddr == 'getGithub':
        # data = json.loads(request.data)
        print(request.json)

        in_q.put(request.json)
        return Response(status=200)
    
    elif webhookAddr == 'getGitHub2':
        print(request.json)
        return Response(status=200)
    else:
        return Response(status=200)


if __name__ == '__main__':
   q = Queue()

   t1 = threading.Thread(target=launchServer, args =(q, ))
   t1.daemon = True
   t1.start()

   t2 = threading.Thread(target=start_ngrok)
   t2.daemon = False
   t2.start()

   app.run(debug=True)

但是,通过单独运行 launchServer 的代码部分,我不会收到此类错误。所以我错误地使用了ngrok。需要你的帮助来指出我的错误

问候

4

2 回答 2

3

也许您的端口 5000 已被其他进程占用。试试 netstat

netstat -anp  | grep 5000
于 2021-02-06T09:07:56.740 回答
1

这个问题也在我这边重现。从调试 Flask 代码来看,在调试模式下似乎没有使用 SO_REUSEADDR,因此下一次尝试绑定相同地址(您的 tcp 服务器)将失败。一种选择是关闭调试模式:

app.run(debug=False)

另一种选择是关闭重新加载器(如果您在服务器运行时修改应用程序,重新加载器很好):

app.run(debug=True, use_reloader=False)
于 2021-02-06T09:26:12.560 回答