6
 
from optparse import OptionParser
 
10
 
import gnutls.connection
 
17
 
    def __init__(self, name=None, options=None, dn=None,
 
18
 
                 password=None, passfile=None, fqdn=None,
 
19
 
                 timeout=None, interval=-1):
 
23
 
            self.password = password
 
25
 
            self.password = open(passfile).readall()
 
27
 
            print "No Password or Passfile in client config file"
 
28
 
            # raise RuntimeError XXX
 
29
 
            self.password = "gazonk"
 
31
 
        self.created = datetime.datetime.now()
 
34
 
            timeout = options.timeout
 
35
 
        self.timeout = timeout
 
37
 
            interval = options.interval
 
38
 
        self.interval = interval
 
39
 
        self.next_check = datetime.datetime.now()
 
42
 
class server_metaclass(type):
 
43
 
    "Common behavior for the UDP and TCP server classes"
 
44
 
    def __new__(cls, name, bases, attrs):
 
45
 
        attrs["address_family"] = socket.AF_INET6
 
46
 
        attrs["allow_reuse_address"] = True
 
47
 
        def server_bind(self):
 
48
 
            if self.options.interface:
 
49
 
                if not hasattr(socket, "SO_BINDTODEVICE"):
 
50
 
                    # From /usr/include/asm-i486/socket.h
 
51
 
                    socket.SO_BINDTODEVICE = 25
 
53
 
                    self.socket.setsockopt(socket.SOL_SOCKET,
 
54
 
                                           socket.SO_BINDTODEVICE,
 
55
 
                                           self.options.interface)
 
56
 
                except socket.error, error:
 
57
 
                    if error[0] == errno.EPERM:
 
58
 
                        print "Warning: No permission to bind to interface", \
 
59
 
                              self.options.interface
 
62
 
            return super(type(self), self).server_bind()
 
63
 
        attrs["server_bind"] = server_bind
 
64
 
        def init(self, *args, **kwargs):
 
65
 
            if "options" in kwargs:
 
66
 
                self.options = kwargs["options"]
 
68
 
            if "clients" in kwargs:
 
69
 
                self.clients = kwargs["clients"]
 
71
 
            if "credentials" in kwargs:
 
72
 
                self.credentials = kwargs["credentials"]
 
73
 
                del kwargs["credentials"]
 
74
 
            return super(type(self), self).__init__(*args, **kwargs)
 
75
 
        attrs["__init__"] = init
 
76
 
        return type.__new__(cls, name, bases, attrs)
 
79
 
class udp_handler(SocketServer.DatagramRequestHandler, object):
 
81
 
        self.wfile.write("Polo")
 
82
 
        print "UDP request answered"
 
85
 
class IPv6_UDPServer(SocketServer.UDPServer, object):
 
86
 
    __metaclass__ = server_metaclass
 
87
 
    def verify_request(self, request, client_address):
 
88
 
        print "UDP request came"
 
89
 
        return request[0] == "Marco"
 
92
 
class tcp_handler(SocketServer.BaseRequestHandler, object):
 
94
 
        print "TCP request came"
 
95
 
        print "Request:", self.request
 
96
 
        print "Client Address:", self.client_address
 
97
 
        print "Server:", self.server
 
98
 
        session = gnutls.connection.ServerSession(self.request,
 
99
 
                                                  self.server.credentials)
 
101
 
        if session.peer_certificate:
 
102
 
            print "DN:", session.peer_certificate.subject
 
104
 
            session.verify_peer()
 
105
 
        except gnutls.errors.CertificateError, error:
 
106
 
            print "Verify failed", error
 
110
 
            session.send(dict((client.dn, client.password)
 
111
 
                              for client in self.server.clients)
 
112
 
                         [session.peer_certificate.subject])
 
114
 
            session.send("gazonk")
 
119
 
class IPv6_TCPServer(SocketServer.ForkingTCPServer, object):
 
120
 
    __metaclass__ = server_metaclass
 
121
 
    request_queue_size = 1024
 
128
 
def string_to_delta(interval):
 
129
 
    """Parse a string and return a datetime.timedelta
 
131
 
    >>> string_to_delta('7d')
 
132
 
    datetime.timedelta(7)
 
133
 
    >>> string_to_delta('60s')
 
134
 
    datetime.timedelta(0, 60)
 
135
 
    >>> string_to_delta('60m')
 
136
 
    datetime.timedelta(0, 3600)
 
137
 
    >>> string_to_delta('24h')
 
138
 
    datetime.timedelta(1)
 
139
 
    >>> string_to_delta(u'1w')
 
140
 
    datetime.timedelta(7)
 
143
 
        suffix=unicode(interval[-1])
 
144
 
        value=int(interval[:-1])
 
146
 
            delta = datetime.timedelta(value)
 
148
 
            delta = datetime.timedelta(0, value)
 
150
 
            delta = datetime.timedelta(0, 0, 0, 0, value)
 
152
 
            delta = datetime.timedelta(0, 0, 0, 0, 0, value)
 
154
 
            delta = datetime.timedelta(0, 0, 0, 0, 0, 0, value)
 
157
 
    except (ValueError, IndexError):
 
162
 
    parser = OptionParser()
 
163
 
    parser.add_option("-i", "--interface", type="string",
 
164
 
                      default="eth0", metavar="IF",
 
165
 
                      help="Interface to bind to")
 
166
 
    parser.add_option("--cert", type="string", default="cert.pem",
 
168
 
                      help="Public key certificate to use")
 
169
 
    parser.add_option("--key", type="string", default="key.pem",
 
171
 
                      help="Private key to use")
 
172
 
    parser.add_option("--ca", type="string", default="ca.pem",
 
174
 
                      help="Certificate Authority certificate to use")
 
175
 
    parser.add_option("--crl", type="string", default="crl.pem",
 
177
 
                      help="Certificate Revokation List to use")
 
178
 
    parser.add_option("-p", "--port", type="int", default=49001,
 
179
 
                      help="Port number to receive requests on")
 
180
 
    parser.add_option("--dh", type="int", metavar="BITS",
 
181
 
                      help="DH group to use")
 
182
 
    parser.add_option("-t", "--timeout", type="string", # Parsed later
 
184
 
                      help="Amount of downtime allowed for clients")
 
185
 
    parser.add_option("--interval", type="string", # Parsed later
 
187
 
                      help="How often to check that a client is up")
 
188
 
    parser.add_option("--check", action="store_true", default=False,
 
189
 
                      help="Run self-test")
 
190
 
    (options, args) = parser.parse_args()
 
197
 
    # Parse the time arguments
 
199
 
        options.timeout = string_to_delta(options.timeout)
 
201
 
        parser.error("option --timeout: Unparseable time")
 
204
 
        options.interval = string_to_delta(options.interval)
 
206
 
        parser.error("option --interval: Unparseable time")
 
208
 
    cert = gnutls.crypto.X509Certificate(open(options.cert).read())
 
209
 
    key = gnutls.crypto.X509PrivateKey(open(options.key).read())
 
210
 
    ca = gnutls.crypto.X509Certificate(open(options.ca).read())
 
211
 
    crl = gnutls.crypto.X509CRL(open(options.crl).read())
 
212
 
    cred = gnutls.connection.X509Credentials(cert, key, [ca], [crl])
 
216
 
    client_config_object = ConfigParser.SafeConfigParser(defaults)
 
217
 
    client_config_object.read("mandos-clients.conf")
 
218
 
    clients = [Client(name=section, options=options,
 
219
 
                      **(dict(client_config_object.items(section))))
 
220
 
               for section in client_config_object.sections()]
 
222
 
    udp_server = IPv6_UDPServer((in6addr_any, options.port),
 
226
 
    tcp_server = IPv6_TCPServer((in6addr_any, options.port),
 
233
 
        in_, out, err = select.select((udp_server,
 
236
 
            server.handle_request()
 
239
 
if __name__ == "__main__":