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