Benjamin Peterson | e711caf | 2008-06-11 16:44:04 +0000 | [diff] [blame] | 1 | # |
| 2 | # Example where a pool of http servers share a single listening socket |
| 3 | # |
| 4 | # On Windows this module depends on the ability to pickle a socket |
| 5 | # object so that the worker processes can inherit a copy of the server |
| 6 | # object. (We import `multiprocessing.reduction` to enable this pickling.) |
| 7 | # |
| 8 | # Not sure if we should synchronize access to `socket.accept()` method by |
| 9 | # using a process-shared lock -- does not seem to be necessary. |
| 10 | # |
Benjamin Peterson | 4469d0c | 2008-11-30 22:46:23 +0000 | [diff] [blame] | 11 | # Copyright (c) 2006-2008, R Oudkerk |
| 12 | # All rights reserved. |
| 13 | # |
Benjamin Peterson | e711caf | 2008-06-11 16:44:04 +0000 | [diff] [blame] | 14 | |
| 15 | import os |
| 16 | import sys |
| 17 | |
| 18 | from multiprocessing import Process, current_process, freeze_support |
Christian Heimes | aae1b70 | 2008-11-28 11:23:26 +0000 | [diff] [blame] | 19 | from http.server import HTTPServer |
| 20 | from http.server import SimpleHTTPRequestHandler |
Benjamin Peterson | e711caf | 2008-06-11 16:44:04 +0000 | [diff] [blame] | 21 | |
| 22 | if sys.platform == 'win32': |
| 23 | import multiprocessing.reduction # make sockets pickable/inheritable |
| 24 | |
| 25 | |
| 26 | def note(format, *args): |
Benjamin Peterson | 58ea9fe | 2008-08-19 19:17:39 +0000 | [diff] [blame] | 27 | sys.stderr.write('[%s]\t%s\n' % (current_process().name, format%args)) |
Benjamin Peterson | e711caf | 2008-06-11 16:44:04 +0000 | [diff] [blame] | 28 | |
| 29 | |
| 30 | class RequestHandler(SimpleHTTPRequestHandler): |
| 31 | # we override log_message() to show which process is handling the request |
| 32 | def log_message(self, format, *args): |
| 33 | note(format, *args) |
| 34 | |
| 35 | def serve_forever(server): |
| 36 | note('starting server') |
| 37 | try: |
| 38 | server.serve_forever() |
| 39 | except KeyboardInterrupt: |
| 40 | pass |
| 41 | |
| 42 | |
| 43 | def runpool(address, number_of_processes): |
| 44 | # create a single server object -- children will each inherit a copy |
| 45 | server = HTTPServer(address, RequestHandler) |
| 46 | |
| 47 | # create child processes to act as workers |
| 48 | for i in range(number_of_processes-1): |
| 49 | Process(target=serve_forever, args=(server,)).start() |
| 50 | |
| 51 | # main process also acts as a worker |
| 52 | serve_forever(server) |
| 53 | |
| 54 | |
| 55 | def test(): |
| 56 | DIR = os.path.join(os.path.dirname(__file__), '..') |
| 57 | ADDRESS = ('localhost', 8000) |
| 58 | NUMBER_OF_PROCESSES = 4 |
| 59 | |
Christian Heimes | aae1b70 | 2008-11-28 11:23:26 +0000 | [diff] [blame] | 60 | print('Serving at http://%s:%d using %d worker processes' % \ |
| 61 | (ADDRESS[0], ADDRESS[1], NUMBER_OF_PROCESSES)) |
| 62 | print('To exit press Ctrl-' + ['C', 'Break'][sys.platform=='win32']) |
Benjamin Peterson | e711caf | 2008-06-11 16:44:04 +0000 | [diff] [blame] | 63 | |
| 64 | os.chdir(DIR) |
| 65 | runpool(ADDRESS, NUMBER_OF_PROCESSES) |
| 66 | |
| 67 | |
| 68 | if __name__ == '__main__': |
| 69 | freeze_support() |
| 70 | test() |