3

所以我一直在尝试创建一个在后台运行服务器的进程,并使用守护进程启动它。所以我的代码是:

class App():
  def __init__(self):
    self.stdin_path = '/dev/null'
    self.stdout_path = '/dev/tty'
    self.stderr_path = '/dev/tty'
    self.pidfile_path = '/tmp/foo.pid'
    self.pidfile_timeout = 5
  def run(self):
    server = WSGIServer(('localhost',28080),handle_request)
    server.serve_forever()

if __name__ == '__main__':
  app = App()
  daemon_runner = runner.DaemonRunner(app)
  daemon_runner.do_action()

但是,这给了我错误:

[warn] Epoll ADD(1) on fd 5 failed.  Old events were 0; read change was 1 (add); write change was 0 (none): Invalid argument
Traceback (most recent call last):
  File "enrollmentrunner2.py", line 110, in <module>
    daemon_runner.do_action()
  File "/usr/lib/python2.7/dist-packages/daemon/runner.py", line 186, in do_action
    func(self)
  File "/usr/lib/python2.7/dist-packages/daemon/runner.py", line 131, in _start
    self.app.run()
  File "enrollmentrunner2.py", line 105, in run
    server.serve_forever()
  File "/usr/lib/python2.7/dist-packages/gevent/baseserver.py", line 188, in serve_forever
    self.start()
  File "/usr/lib/python2.7/dist-packages/gevent/baseserver.py", line 149, in start
    self.start_accepting()
  File "/usr/lib/python2.7/dist-packages/gevent/server.py", line 99, in start_accepting
    self._accept_event = core.read_event(self.socket.fileno(), self._do_accept, persist=True)
  File "core.pyx", line 308, in gevent.core.read_event.__init__ (gevent/core.c:3960)
  File "core.pyx", line 252, in gevent.core.event.add (gevent/core.c:2952)
IOError: [Errno 22] Invalid argument

我在网上查找了警告,但在任何地方都找不到,而且该错误并没有真正给我太多有用的信息。我已经运行了这里描述的程序https://stackoverflow.com/a/9047339,我已经单独运行了程序,只需将其放入 main 等。但是,当我将它们组合起来时,它似乎把事情搞砸了向上。有人知道为什么会这样吗?

4

1 回答 1

7

您遇到了 fork 和 epoll(或 kqueue)之间交互不良的问题。一般情况下,基于 epoll 的事件循环在 fork 之后很难可靠地工作,最好重新创建一个新的事件循环。

有几种方法可以解决您的问题:

  • 升级到 gevent 1.0。与 gevent 0.x 不同,它在第一次使用时创建事件循环,而不是在第一次导入时创建,从而避免了这个问题。它还使用 libev,它对 fork 的处理比 libevent 稍好一些,尽管不是 100% 可靠。升级 gevent 后,您的问题很可能会消失。

  • 将 gevent 包的导入延迟到 fork 之后。

于 2012-07-27T11:59:47.053 回答