Jean-Paul Calderone | 3de9f62 | 2008-03-12 14:12:19 -0400 | [diff] [blame] | 1 | # -*- coding: latin-1 -*- |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 2 | # |
Jean-Paul Calderone | de0a71e | 2011-03-02 19:55:11 -0500 | [diff] [blame] | 3 | # Copyright (C) AB Strakt |
| 4 | # Copyright (C) Jean-Paul Calderone |
| 5 | # See LICENSE for details. |
Jean-Paul Calderone | 8b63d45 | 2008-03-21 18:31:12 -0400 | [diff] [blame] | 6 | |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 7 | """ |
| 8 | Simple echo server, using nonblocking I/O |
| 9 | """ |
| 10 | |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 11 | from __future__ import print_function |
| 12 | |
| 13 | import os |
| 14 | import select |
| 15 | import socket |
| 16 | import sys |
| 17 | |
Jim Shaver | b2ff5be | 2015-04-30 08:26:29 -0400 | [diff] [blame] | 18 | from OpenSSL import SSL, crypto |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 19 | |
| 20 | |
| 21 | def verify_cb(conn, cert, errnum, depth, ok): |
Jim Shaver | b2ff5be | 2015-04-30 08:26:29 -0400 | [diff] [blame] | 22 | certsubject = crypto.X509Name(cert.get_subject()) |
| 23 | commonname = certsubject.commonName |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 24 | print('Got certificate: ' + commonname) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 25 | return ok |
| 26 | |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 27 | |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 28 | if len(sys.argv) < 2: |
Jim Shaver | 8a4a7ae | 2015-04-29 01:17:33 -0400 | [diff] [blame] | 29 | print('Usage: python server.py PORT') |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 30 | sys.exit(1) |
| 31 | |
| 32 | dir = os.path.dirname(sys.argv[0]) |
| 33 | if dir == '': |
| 34 | dir = os.curdir |
| 35 | |
| 36 | # Initialize context |
| 37 | ctx = SSL.Context(SSL.SSLv23_METHOD) |
| 38 | ctx.set_options(SSL.OP_NO_SSLv2) |
Daniƫl van Eeden | ae8243d | 2016-01-16 18:00:52 +0100 | [diff] [blame] | 39 | ctx.set_options(SSL.OP_NO_SSLv3) |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 40 | ctx.set_verify( |
| 41 | SSL.VERIFY_PEER | SSL.VERIFY_FAIL_IF_NO_PEER_CERT, verify_cb |
| 42 | ) # Demand a certificate |
| 43 | ctx.use_privatekey_file(os.path.join(dir, 'server.pkey')) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 44 | ctx.use_certificate_file(os.path.join(dir, 'server.cert')) |
| 45 | ctx.load_verify_locations(os.path.join(dir, 'CA.cert')) |
| 46 | |
| 47 | # Set up server |
| 48 | server = SSL.Connection(ctx, socket.socket(socket.AF_INET, socket.SOCK_STREAM)) |
| 49 | server.bind(('', int(sys.argv[1]))) |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 50 | server.listen(3) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 51 | server.setblocking(0) |
| 52 | |
| 53 | clients = {} |
| 54 | writers = {} |
| 55 | |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 56 | |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 57 | def dropClient(cli, errors=None): |
| 58 | if errors: |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 59 | print('Client %s left unexpectedly:' % (clients[cli],)) |
| 60 | print(' ', errors) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 61 | else: |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 62 | print('Client %s left politely' % (clients[cli],)) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 63 | del clients[cli] |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 64 | if cli in writers: |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 65 | del writers[cli] |
| 66 | if not errors: |
| 67 | cli.shutdown() |
| 68 | cli.close() |
| 69 | |
| 70 | while 1: |
| 71 | try: |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 72 | r, w, _ = select.select( |
| 73 | [server] + list(clients.keys()), list(writers.keys()), [] |
| 74 | ) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 75 | except: |
| 76 | break |
| 77 | |
| 78 | for cli in r: |
| 79 | if cli == server: |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 80 | cli, addr = server.accept() |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 81 | print('Connection from %s' % (addr,)) |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 82 | clients[cli] = addr |
| 83 | |
| 84 | else: |
| 85 | try: |
Jim Shaver | b2ff5be | 2015-04-30 08:26:29 -0400 | [diff] [blame] | 86 | ret = cli.recv(1024).decode('utf-8') |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 87 | except (SSL.WantReadError, |
| 88 | SSL.WantWriteError, |
| 89 | SSL.WantX509LookupError): |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 90 | pass |
| 91 | except SSL.ZeroReturnError: |
| 92 | dropClient(cli) |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 93 | except SSL.Error as errors: |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 94 | dropClient(cli, errors) |
| 95 | else: |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 96 | if cli not in writers: |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 97 | writers[cli] = '' |
| 98 | writers[cli] = writers[cli] + ret |
| 99 | |
| 100 | for cli in w: |
| 101 | try: |
| 102 | ret = cli.send(writers[cli]) |
Hynek Schlawack | 8b7e455 | 2016-03-13 07:51:09 +0100 | [diff] [blame^] | 103 | except (SSL.WantReadError, |
| 104 | SSL.WantWriteError, |
| 105 | SSL.WantX509LookupError): |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 106 | pass |
| 107 | except SSL.ZeroReturnError: |
| 108 | dropClient(cli) |
Jim Shaver | 71ad368 | 2015-04-29 00:09:14 -0400 | [diff] [blame] | 109 | except SSL.Error as errors: |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 110 | dropClient(cli, errors) |
| 111 | else: |
| 112 | writers[cli] = writers[cli][ret:] |
| 113 | if writers[cli] == '': |
| 114 | del writers[cli] |
| 115 | |
Jim Shaver | 473fe6a | 2015-04-29 09:42:39 -0400 | [diff] [blame] | 116 | for cli in clients.keys(): |
Jean-Paul Calderone | 897bc25 | 2008-02-18 20:50:23 -0500 | [diff] [blame] | 117 | cli.close() |
| 118 | server.close() |