Python:BaseHTTPRequestHandler 实例在 GET 请求后重置
Python: Instance of BaseHTTPRequestHandler resets after GET Request
我有一个 python BaseHTTPRequestHandler
class 被 HTTPServer
class 调用。基本上 BaseHTTPRequestHandler
只是运行一个基本算法,然后响应 Get
请求。问题是每次我发出 Get
请求时,我都会得到正确的响应,但是 BaseHTTPRequestHandler
中收集的所有数据都会重置,就像每次向 HTTPServer
发送请求一样创建 BaseHTTPRequestHandler
的新实例。我在网上找不到任何能真正解释幕后发生的事情。我附上了我的代码的简化版本。任何帮助或解释将不胜感激。
在有人建议创建 class 或全局变量之前,我正在使用线程一次创建此 class 的多个实例,这样做会使每个线程上的所有实例共享并替换彼此的数据。
CODE(复制和粘贴时关闭缩进)
BaseHTTPRequestHandler
此简化版本仅跟踪已发生的警报数量。问题是当我调用 Get
请求时,计数总是重置为 0,就好像 class 的实例被重置一样。
class SimulationServer(BaseHTTPRequestHandler):
def __init__(self, address, port, randomNumberMax, *args):
self.IP_ADDRESS = address
self.PORT = port
self.RANDOM_NUMBER_MAX = randomNumberMax
self.COUNT = 0
BaseHTTPRequestHandler.__init__(self, *args)
def do_GET(self):
if self.headers['Authorization'] == 'Basic ' + str(key):
print("send response")
self.do_HEAD()
randomNumberMax = self.RANDOM_NUMBER_MAX
response = ""
if randint(0, randomNumberMax) == 0:
self.generateAlert()
base_path = urlparse(self.path).path
print('base_path: ' + base_path)
if base_path == '/count':
response = self.getCount()
self.wfile.write(bytes(response, 'utf-8'))
def getCount(self):
count = self.COUNT
jsonString = '{"_sig": "","count": ' + str(count) + '}'
return jsonString
def generateAlert(self):
newAlert = {}
newAlert['siteId'] = "siteId"+ str(self.COUNT)
newAlert['mesg'] = "Simulated Alert"
newAlert['when'] = int(time.time())
self.COUNT += 1
HTTPServer
class CustomHTTPServer(HTTPServer):
key = ''
def __init__(self, address, handlerClass=SimulationServer):
super().__init__(address, handlerClass)
def set_auth(self, username, password):
self.key = base64.b64encode(
bytes('%s:%s' % (username, password), 'utf-8')).decode('ascii')
def get_auth_key(self):
return self.key
主要
此 class 创建 HTTPServer
并附加 handler
class RunSimulator(object):
def run(self, alertFrequency=50, port=9000):
ipAddress="127.0.0.1"
def handler(*args):
SimulationServer(ipAddress, port, alertFrequency, *args)
simulationServer = CustomHTTPServer((ipAddress, port), handler)
simulationServer.set_auth('username', 'password')
try:
simulationServer.serve_forever()
except KeyboardInterrupt:
pass
simulationServer.server_close()
print(time.asctime(), "Server Stops - %s:%s" % (ipAddress, port))
if __name__ == "__main__":
from sys import argv
simu = RunSimulator()
simu.run()
所以简短的回答是每次发送请求时它都会创建一个新实例。
BaseHTTPServer.HTTPServer
是 SocketServer.TCPServer
的子类,它本身是 SocketServer.BaseServer
的子类。每次有请求进来它都会处理请求process_request
然后调用finish_request
。 finish_request
将实例化您的请求处理程序的新实例。
我有一个 python BaseHTTPRequestHandler
class 被 HTTPServer
class 调用。基本上 BaseHTTPRequestHandler
只是运行一个基本算法,然后响应 Get
请求。问题是每次我发出 Get
请求时,我都会得到正确的响应,但是 BaseHTTPRequestHandler
中收集的所有数据都会重置,就像每次向 HTTPServer
发送请求一样创建 BaseHTTPRequestHandler
的新实例。我在网上找不到任何能真正解释幕后发生的事情。我附上了我的代码的简化版本。任何帮助或解释将不胜感激。
在有人建议创建 class 或全局变量之前,我正在使用线程一次创建此 class 的多个实例,这样做会使每个线程上的所有实例共享并替换彼此的数据。
CODE(复制和粘贴时关闭缩进)
BaseHTTPRequestHandler
此简化版本仅跟踪已发生的警报数量。问题是当我调用 Get
请求时,计数总是重置为 0,就好像 class 的实例被重置一样。
class SimulationServer(BaseHTTPRequestHandler):
def __init__(self, address, port, randomNumberMax, *args):
self.IP_ADDRESS = address
self.PORT = port
self.RANDOM_NUMBER_MAX = randomNumberMax
self.COUNT = 0
BaseHTTPRequestHandler.__init__(self, *args)
def do_GET(self):
if self.headers['Authorization'] == 'Basic ' + str(key):
print("send response")
self.do_HEAD()
randomNumberMax = self.RANDOM_NUMBER_MAX
response = ""
if randint(0, randomNumberMax) == 0:
self.generateAlert()
base_path = urlparse(self.path).path
print('base_path: ' + base_path)
if base_path == '/count':
response = self.getCount()
self.wfile.write(bytes(response, 'utf-8'))
def getCount(self):
count = self.COUNT
jsonString = '{"_sig": "","count": ' + str(count) + '}'
return jsonString
def generateAlert(self):
newAlert = {}
newAlert['siteId'] = "siteId"+ str(self.COUNT)
newAlert['mesg'] = "Simulated Alert"
newAlert['when'] = int(time.time())
self.COUNT += 1
HTTPServer
class CustomHTTPServer(HTTPServer):
key = ''
def __init__(self, address, handlerClass=SimulationServer):
super().__init__(address, handlerClass)
def set_auth(self, username, password):
self.key = base64.b64encode(
bytes('%s:%s' % (username, password), 'utf-8')).decode('ascii')
def get_auth_key(self):
return self.key
主要
此 class 创建 HTTPServer
并附加 handler
class RunSimulator(object):
def run(self, alertFrequency=50, port=9000):
ipAddress="127.0.0.1"
def handler(*args):
SimulationServer(ipAddress, port, alertFrequency, *args)
simulationServer = CustomHTTPServer((ipAddress, port), handler)
simulationServer.set_auth('username', 'password')
try:
simulationServer.serve_forever()
except KeyboardInterrupt:
pass
simulationServer.server_close()
print(time.asctime(), "Server Stops - %s:%s" % (ipAddress, port))
if __name__ == "__main__":
from sys import argv
simu = RunSimulator()
simu.run()
所以简短的回答是每次发送请求时它都会创建一个新实例。
BaseHTTPServer.HTTPServer
是 SocketServer.TCPServer
的子类,它本身是 SocketServer.BaseServer
的子类。每次有请求进来它都会处理请求process_request
然后调用finish_request
。 finish_request
将实例化您的请求处理程序的新实例。