我有以下conf文件供主管使用:

[program:backend]
command=../../../venv/bin/python backend.py

[program:celeryd]
command=../../../venv/bin/celery worker --app=tasks -l debug

[program:memcached]
command=memcached


我的backend.py是一个非常标准的烧瓶应用程序,带有debug = True。它不是守护进程(在控制台中运行python backend.py不会返回提示)

执行supervisord时,出现以下提示:

[venv] supervisor$ supervisord
2013-05-08 18:42:22,148 INFO RPC interface 'supervisor' initialized
2013-05-08 18:42:22,148 CRIT Server 'unix_http_server' running without any HTTP authentication checking
2013-05-08 18:42:22,149 INFO supervisord started with pid 6778
2013-05-08 18:42:23,152 INFO spawned: 'celeryd' with pid 6781
2013-05-08 18:42:23,156 INFO spawned: 'memcached' with pid 6782
2013-05-08 18:42:23,159 INFO spawned: 'backend' with pid 6783
2013-05-08 18:42:24,314 INFO success: celeryd entered RUNNING state, process has stayed up for > than 1 seconds (startsecs)
2013-05-08 18:42:24,314 INFO success: memcached entered RUNNING state, process has stayed up for > than 1 seconds (startsecs)
2013-05-08 18:42:24,314 INFO success: backend entered RUNNING state, process has stayed up for > than 1 seconds (startsecs)


这是pstree的样子:

 | | \-+= 91117 vng -/bin/bash
 | |   \-+= 06778 vng /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/python /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/supervisord
 | |     |-+= 06781 vng /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/python ../../../venv/bin/celery worker --app=tasks -l debug
 | |     | |--- 06795 vng /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/python ../../../venv/bin/celery worker --app=tasks -l debug
 | |     | |--- 06796 vng /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/python ../../../venv/bin/celery worker --app=tasks -l debug
 | |     | |--- 06797 vng /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/python ../../../venv/bin/celery worker --app=tasks -l debug
 | |     | \--- 06798 vng /Volumes/Data2/Dropbox/projects/Giordano/venv/bin/python ../../../venv/bin/celery worker --app=tasks -l debug
 | |     |--= 06782 vng memcached
 | |     \-+= 06783 vng ../../../venv/bin/python backend.py
 | |       \--- 06790 vng /Volumes/Data2/Dropbox/projects/Giordano/src/giordano/web/../../../venv/bin/python backend.py


杀死主管会导致:

^C2013-05-08 18:43:18,305 WARN received SIGINT indicating exit request
2013-05-08 18:43:18,305 INFO waiting for celeryd, memcached, backend to die
2013-05-08 18:43:18,352 INFO stopped: backend (exit status 0)
2013-05-08 18:43:18,353 INFO stopped: memcached (terminated by SIGTERM)
2013-05-08 18:43:18,414 INFO stopped: celeryd (exit status 0)


毫无疑问,backend.py在pstree中被替换了:

 |--= 41659 root /usr/libexec/taskgated -s
 |--- 42779 vng /usr/local/Cellar/erlang/R15B03-1/lib/erlang/erts-5.9.3.1/bin/epmd -daemon
 |--= 88518 root /usr/sbin/ocspd
 |--- 97815 vng /Applications/Dropbox.app/Contents/MacOS/Dropbox /firstrunupdate 403
 |--= 97905 _usbmuxd /System/Library/PrivateFrameworks/MobileDevice.framework/Versions/A/Resources/usbmuxd -launchd
 \--- 06790 vng /Volumes/Data2/Dropbox/projects/Giordano/src/giordano/web/../../../venv/bin/python backend.py


我对此感到恼火,因为每次我重新启动supervisor时,都会收到一个错误消息,指出该端口已被使用。

这里有人知道导致此问题的原因吗?

编辑:

我的backend.py

app = Flask(__name__)
app.config.from_object(__name__)
app.secret_key = 'asgasdasdgasd'
app.debug = bool(int(os.environ.get("DEBUG", 1)))   # DEBUG=0
if app.debug:
  from giordano.web import colorer # Needed for coloring logging

app.config.update(
  MAIL_SERVER='smtp.gmail.com',
  MAIL_DEBUG=False,
  MAIL_PORT=465,
  MAIL_USE_SSL=True,
  MAIL_USERNAME = '[email protected]',
  MAIL_PASSWORD = 'asdfasdfasdfa'
)
mail = Mail(app)    # Very important to do it after!!

log_level = logging.DEBUG if app.debug else logging.INFO
logging.basicConfig(level=log_level, format='%(asctime)s %(levelname)s %(message)s')
logging.info("Launching app with debug=%s" % app.debug)

....

class TimedRequestHandler(BaseRequestHandler):
  """http://blog.sendhub.com/post/48832423565/hacking-flask-a-success-story
  """
  def handle(self):
    self.fancyStarted = time.time()
    rv = super(TimedRequestHandler, self).handle()
    return rv

  def send_response(self, code, message=None):
    self.fancyProcessed = time.time()
    super(TimedRequestHandler, self).send_response(code, message=None)

  def log_request(self, code='-', size='-'):
    duration = int((self.fancyProcessed - self.fancyStarted) * 1000)
    line = '"{0}" {1} {2} [{3}ms]'.format(self.requestline, code, size, duration)
    self.log('info', line)


if __name__ == "__main__":
   init_db() # run if you need to create tables
   app.run(host='0.0.0.0', port=8888, request_handler=TimedRequestHandler)

最佳答案

在您的配置中添加stopasgroup=true,它应该可以工作。当您执行stoprestart时,此配置将杀死所有子进程。

If true, the flag causes supervisor to send the stop signal to the whole process group and implies killasgroup is true. This is useful for programs, such as Flask in debug mode, that do not propagate stop signals to their children, leaving them orphaned.


http://supervisord.org/configuration.html

关于python - 主管不会终止 flask 的应用,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/16453052/

10-10 19:02