| Jeremy Mowery | 0427ff8 | 2016-04-02 10:32:11 -0700 | [diff] [blame] | 1 | from subprocess import PIPE, Popen |
| 2 | |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 3 | from core.models import Service, TenantWithContainer |
| 4 | from django.db import transaction |
| Jeremy Mowery | 9d718b5 | 2016-04-02 10:16:21 -0700 | [diff] [blame] | 5 | from xos.exceptions import XOSConfigurationError, XOSValidationError |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 6 | |
| 7 | VPN_KIND = "vpn" |
| 8 | |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 9 | |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 10 | class VPNService(Service): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 11 | """Defines the Service for creating VPN servers.""" |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 12 | KIND = VPN_KIND |
| Jeremy Mowery | 9d718b5 | 2016-04-02 10:16:21 -0700 | [diff] [blame] | 13 | OPENVPN_PREFIX = "/opt/openvpn/" |
| 14 | VARS = OPENVPN_PREFIX + "vars" |
| 15 | EASYRSA_LOC = OPENVPN_PREFIX + "easyrsa3/easyrsa" |
| 16 | EASYRSA_COMMAND = EASYRSA_LOC + " --vars=" + VARS |
| 17 | |
| 18 | @classmethod |
| Jeremy Mowery | b46eada | 2016-04-02 11:00:10 -0700 | [diff] [blame] | 19 | def execute_easyrsa_command(cls, pki_dir, command): |
| Jeremy Mowery | 9d718b5 | 2016-04-02 10:16:21 -0700 | [diff] [blame] | 20 | full_command = ( |
| 21 | VPNService.EASYRSA_COMMAND + " --pki-dir=" + |
| 22 | pki_dir + " " + command) |
| Jeremy Mowery | 19d1ad6 | 2016-04-02 11:17:55 -0700 | [diff] [blame] | 23 | proc = Popen( |
| 24 | full_command, shell=True, stdout=PIPE, stderr=PIPE |
| Jeremy Mowery | 9d718b5 | 2016-04-02 10:16:21 -0700 | [diff] [blame] | 25 | ) |
| Jeremy Mowery | 19d1ad6 | 2016-04-02 11:17:55 -0700 | [diff] [blame] | 26 | (stdout, stderr) = proc.communicate() |
| 27 | if (proc.returncode != 0): |
| Jeremy Mowery | 9d718b5 | 2016-04-02 10:16:21 -0700 | [diff] [blame] | 28 | raise XOSConfigurationError( |
| 29 | full_command + " failed with standard out:" + str(stdout) + |
| 30 | " and stderr: " + str(stderr)) |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 31 | |
| 32 | class Meta: |
| 33 | proxy = True |
| 34 | # The name used to find this service, all directories are named this |
| 35 | app_label = "vpn" |
| 36 | verbose_name = "VPN Service" |
| 37 | |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 38 | default_attributes = {'exposed_ports': None, |
| 39 | 'exposed_ports_str': None} |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 40 | |
| 41 | @property |
| 42 | def exposed_ports(self): |
| 43 | return self.get_attribute("exposed_ports", |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 44 | self.default_attributes["exposed_ports"]) |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 45 | |
| 46 | @exposed_ports.setter |
| 47 | def exposed_ports(self, value): |
| 48 | self.set_attribute("exposed_ports", value) |
| 49 | |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 50 | @property |
| 51 | def exposed_ports_str(self): |
| 52 | return self.get_attribute("exposed_ports_str", |
| 53 | self.default_attributes["exposed_ports_str"]) |
| 54 | |
| 55 | @exposed_ports_str.setter |
| 56 | def exposed_ports_str(self, value): |
| 57 | self.set_attribute("exposed_ports_str", value) |
| 58 | |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 59 | def get_next_available_port(self, protocol): |
| 60 | if protocol != "udp" and protocol != "tcp": |
| Jeremy Mowery | 30bdab9 | 2016-03-29 23:52:46 -0700 | [diff] [blame] | 61 | raise XOSValidationError("Port protocol must be udp or tcp") |
| Jeremy Mowery | abccb8a | 2016-03-27 22:20:57 -0700 | [diff] [blame] | 62 | if not self.exposed_ports[protocol]: |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 63 | raise XOSValidationError( |
| 64 | "No availble ports for protocol: " + protocol) |
| 65 | tenants = [ |
| 66 | tenant for tenant in VPNTenant.get_tenant_objects().all() |
| 67 | if tenant.protocol == protocol] |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 68 | port_numbers = self.exposed_ports[protocol] |
| 69 | for port_number in port_numbers: |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 70 | if ( |
| 71 | len([ |
| 72 | tenant for tenant in tenants |
| 73 | if tenant.port_number == port_number]) == 0): |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 74 | return port_number |
| 75 | |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 76 | |
| Jeremy Mowery | 7ced738 | 2015-12-04 23:58:38 -0700 | [diff] [blame] | 77 | class VPNTenant(TenantWithContainer): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 78 | """Defines the Tenant for creating VPN servers.""" |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 79 | |
| 80 | class Meta: |
| 81 | proxy = True |
| 82 | verbose_name = "VPN Tenant" |
| 83 | |
| 84 | KIND = VPN_KIND |
| 85 | |
| 86 | sync_attributes = ("nat_ip", "nat_mac",) |
| 87 | |
| Jeremy Mowery | 3f6adcd | 2016-02-21 15:36:32 -0700 | [diff] [blame] | 88 | default_attributes = {'vpn_subnet': None, |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 89 | 'server_network': None, |
| 90 | 'clients_can_see_each_other': True, |
| Jeremy Mowery | e02d4b6 | 2016-01-10 15:21:52 -0700 | [diff] [blame] | 91 | 'is_persistent': True, |
| Jeremy Mowery | 290714e | 2016-02-25 14:36:29 -0700 | [diff] [blame] | 92 | 'ca_crt': None, |
| Jeremy Mowery | 47c9374 | 2016-03-02 00:01:25 -0700 | [diff] [blame] | 93 | 'port': None, |
| Jeremy Mowery | 65c6070 | 2016-03-16 11:22:02 -0700 | [diff] [blame] | 94 | 'script_text': None, |
| Jeremy Mowery | b52c49a | 2016-04-05 23:32:10 -0700 | [diff] [blame] | 95 | 'pki_dir': None, |
| 96 | 'use_ca_from': list(), |
| Jeremy Mowery | 7b54e14 | 2016-04-06 11:33:43 -0700 | [diff] [blame] | 97 | 'failover_servers': list(), |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 98 | 'protocol': None} |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 99 | |
| 100 | def __init__(self, *args, **kwargs): |
| 101 | vpn_services = VPNService.get_service_objects().all() |
| 102 | if vpn_services: |
| 103 | self._meta.get_field( |
| 104 | "provider_service").default = vpn_services[0].id |
| 105 | super(VPNTenant, self).__init__(*args, **kwargs) |
| 106 | |
| 107 | def save(self, *args, **kwargs): |
| 108 | super(VPNTenant, self).save(*args, **kwargs) |
| 109 | model_policy_vpn_tenant(self.pk) |
| 110 | |
| 111 | def delete(self, *args, **kwargs): |
| 112 | self.cleanup_container() |
| 113 | super(VPNTenant, self).delete(*args, **kwargs) |
| 114 | |
| 115 | @property |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 116 | def protocol(self): |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 117 | return self.get_attribute( |
| 118 | "protocol", self.default_attributes["protocol"]) |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 119 | |
| 120 | @protocol.setter |
| 121 | def protocol(self, value): |
| 122 | self.set_attribute("protocol", value) |
| 123 | |
| 124 | @property |
| Jeremy Mowery | b52c49a | 2016-04-05 23:32:10 -0700 | [diff] [blame] | 125 | def use_ca_from(self): |
| 126 | return self.get_attribute( |
| 127 | "use_ca_from", self.default_attributes["use_ca_from"]) |
| 128 | |
| 129 | @use_ca_from.setter |
| 130 | def use_ca_from(self, value): |
| 131 | self.set_attribute("use_ca_from", value) |
| 132 | |
| 133 | @property |
| 134 | def pki_dir(self): |
| 135 | return self.get_attribute( |
| 136 | "pki_dir", self.default_attributes["pki_dir"]) |
| 137 | |
| 138 | @pki_dir.setter |
| 139 | def pki_dir(self, value): |
| 140 | self.set_attribute("pki_dir", value) |
| 141 | |
| 142 | @property |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 143 | def addresses(self): |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 144 | """Mapping[str, str]: The ip, mac address, and subnet of the NAT |
| 145 | network of this Tenant.""" |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 146 | if (not self.id) or (not self.instance): |
| 147 | return {} |
| 148 | |
| 149 | addresses = {} |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 150 | for ns in self.instance.ports.all(): |
| 151 | if "nat" in ns.network.name.lower(): |
| Jeremy Mowery | 5055c7b | 2016-01-08 17:25:33 -0700 | [diff] [blame] | 152 | addresses["ip"] = ns.ip |
| 153 | addresses["mac"] = ns.mac |
| Jeremy Mowery | 5055c7b | 2016-01-08 17:25:33 -0700 | [diff] [blame] | 154 | break |
| 155 | |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 156 | return addresses |
| 157 | |
| 158 | # This getter is necessary because nat_ip is a sync_attribute |
| 159 | @property |
| 160 | def nat_ip(self): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 161 | """str: The IP of this Tenant on the NAT network.""" |
| Jeremy Mowery | 5055c7b | 2016-01-08 17:25:33 -0700 | [diff] [blame] | 162 | return self.addresses.get("ip", None) |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 163 | |
| 164 | # This getter is necessary because nat_mac is a sync_attribute |
| 165 | @property |
| 166 | def nat_mac(self): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 167 | """str: The MAC address of this Tenant on the NAT network.""" |
| Jeremy Mowery | 5055c7b | 2016-01-08 17:25:33 -0700 | [diff] [blame] | 168 | return self.addresses.get("mac", None) |
| Jeremy Mowery | a9b673b | 2016-01-07 21:25:50 -0700 | [diff] [blame] | 169 | |
| 170 | @property |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 171 | def server_network(self): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 172 | """str: The IP address of the server on the VPN.""" |
| Jeremy Mowery | bd2ed3a | 2016-01-05 16:52:43 -0700 | [diff] [blame] | 173 | return self.get_attribute( |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 174 | 'server_network', |
| 175 | self.default_attributes['server_network']) |
| Jeremy Mowery | bd2ed3a | 2016-01-05 16:52:43 -0700 | [diff] [blame] | 176 | |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 177 | @server_network.setter |
| 178 | def server_network(self, value): |
| 179 | self.set_attribute("server_network", value) |
| Jeremy Mowery | bd2ed3a | 2016-01-05 16:52:43 -0700 | [diff] [blame] | 180 | |
| 181 | @property |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 182 | def vpn_subnet(self): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 183 | """str: The IP address of the client on the VPN.""" |
| Jeremy Mowery | bd2ed3a | 2016-01-05 16:52:43 -0700 | [diff] [blame] | 184 | return self.get_attribute( |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 185 | 'vpn_subnet', |
| 186 | self.default_attributes['vpn_subnet']) |
| Jeremy Mowery | bd2ed3a | 2016-01-05 16:52:43 -0700 | [diff] [blame] | 187 | |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 188 | @vpn_subnet.setter |
| 189 | def vpn_subnet(self, value): |
| 190 | self.set_attribute("vpn_subnet", value) |
| Jeremy Mowery | bd2ed3a | 2016-01-05 16:52:43 -0700 | [diff] [blame] | 191 | |
| 192 | @property |
| Jeremy Mowery | 4a23e7d | 2016-01-06 15:16:33 -0700 | [diff] [blame] | 193 | def is_persistent(self): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 194 | """bool: True if the VPN connection is persistence, false otherwise.""" |
| Jeremy Mowery | 4a23e7d | 2016-01-06 15:16:33 -0700 | [diff] [blame] | 195 | return self.get_attribute( |
| 196 | "is_persistent", |
| 197 | self.default_attributes['is_persistent']) |
| 198 | |
| 199 | @is_persistent.setter |
| 200 | def is_persistent(self, value): |
| 201 | self.set_attribute("is_persistent", value) |
| 202 | |
| 203 | @property |
| Jeremy Mowery | 65c6070 | 2016-03-16 11:22:02 -0700 | [diff] [blame] | 204 | def failover_servers(self): |
| Jeremy Mowery | a74c31d | 2016-04-04 22:30:44 -0700 | [diff] [blame] | 205 | return self.get_attribute( |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 206 | "failover_servers", self.default_attributes["failover_servers"]) |
| Jeremy Mowery | 65c6070 | 2016-03-16 11:22:02 -0700 | [diff] [blame] | 207 | |
| 208 | @failover_servers.setter |
| 209 | def failover_servers(self, value): |
| 210 | self.set_attribute("failover_servers", value) |
| 211 | |
| 212 | @property |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 213 | def clients_can_see_each_other(self): |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 214 | """bool: True if the client can see the subnet of the server, false |
| 215 | otherwise.""" |
| Jeremy Mowery | 4a23e7d | 2016-01-06 15:16:33 -0700 | [diff] [blame] | 216 | return self.get_attribute( |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 217 | "clients_can_see_each_other", |
| 218 | self.default_attributes['clients_can_see_each_other']) |
| Jeremy Mowery | 4a23e7d | 2016-01-06 15:16:33 -0700 | [diff] [blame] | 219 | |
| Jeremy Mowery | fd08129 | 2016-02-07 17:07:55 -0700 | [diff] [blame] | 220 | @clients_can_see_each_other.setter |
| 221 | def clients_can_see_each_other(self, value): |
| 222 | self.set_attribute("clients_can_see_each_other", value) |
| Jeremy Mowery | 4a23e7d | 2016-01-06 15:16:33 -0700 | [diff] [blame] | 223 | |
| Jeremy Mowery | e02d4b6 | 2016-01-10 15:21:52 -0700 | [diff] [blame] | 224 | @property |
| Jeremy Mowery | 3b1caba | 2016-02-02 23:53:50 -0700 | [diff] [blame] | 225 | def ca_crt(self): |
| Jeremy Mowery | 1eab4fa | 2016-02-02 17:17:20 -0700 | [diff] [blame] | 226 | """str: the string for the ca certificate""" |
| Jeremy Mowery | 3b1caba | 2016-02-02 23:53:50 -0700 | [diff] [blame] | 227 | return self.get_attribute("ca_crt", self.default_attributes['ca_crt']) |
| Jeremy Mowery | 1eab4fa | 2016-02-02 17:17:20 -0700 | [diff] [blame] | 228 | |
| Jeremy Mowery | 3b1caba | 2016-02-02 23:53:50 -0700 | [diff] [blame] | 229 | @ca_crt.setter |
| 230 | def ca_crt(self, value): |
| 231 | self.set_attribute("ca_crt", value) |
| 232 | |
| Jeremy Mowery | 290714e | 2016-02-25 14:36:29 -0700 | [diff] [blame] | 233 | @property |
| 234 | def port_number(self): |
| 235 | """int: the integer representing the port number for this server""" |
| 236 | return self.get_attribute("port", self.default_attributes['port']) |
| 237 | |
| 238 | @port_number.setter |
| 239 | def port_number(self, value): |
| 240 | self.set_attribute("port", value) |
| 241 | |
| Jeremy Mowery | 47c9374 | 2016-03-02 00:01:25 -0700 | [diff] [blame] | 242 | @property |
| 243 | def script_text(self): |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 244 | return self.get_attribute( |
| 245 | "script_text", self.default_attributes['script_text']) |
| Jeremy Mowery | 47c9374 | 2016-03-02 00:01:25 -0700 | [diff] [blame] | 246 | |
| 247 | @script_text.setter |
| 248 | def script_text(self, value): |
| 249 | self.set_attribute("script_text", value) |
| 250 | |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 251 | def create_client_script(self, client_name): |
| Jeremy Mowery | bd3dfae | 2016-03-01 19:15:40 -0700 | [diff] [blame] | 252 | script = "" |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 253 | # write the configuration portion |
| Jeremy Mowery | bd3dfae | 2016-03-01 19:15:40 -0700 | [diff] [blame] | 254 | script += ("printf \"%b\" \"") |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 255 | script += self.generate_client_conf(client_name) |
| Jeremy Mowery | bd3dfae | 2016-03-01 19:15:40 -0700 | [diff] [blame] | 256 | script += ("\" > client.conf\n") |
| 257 | script += ("printf \"%b\" \"") |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 258 | for line in self.ca_crt: |
| Jeremy Mowery | bd3dfae | 2016-03-01 19:15:40 -0700 | [diff] [blame] | 259 | script += (line.rstrip() + r"\n") |
| 260 | script += ("\" > ca.crt\n") |
| Jeremy Mowery | cf1c519 | 2016-03-15 20:28:04 -0700 | [diff] [blame] | 261 | script += ("printf \"%b\" \"") |
| Jeremy Mowery | 30bdab9 | 2016-03-29 23:52:46 -0700 | [diff] [blame] | 262 | for line in self.get_client_cert(client_name): |
| Jeremy Mowery | cf1c519 | 2016-03-15 20:28:04 -0700 | [diff] [blame] | 263 | script += (line.rstrip() + r"\n") |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 264 | script += ("\" > " + client_name + ".crt\n") |
| Jeremy Mowery | 30bdab9 | 2016-03-29 23:52:46 -0700 | [diff] [blame] | 265 | for line in self.get_client_key(client_name): |
| Jeremy Mowery | cf1c519 | 2016-03-15 20:28:04 -0700 | [diff] [blame] | 266 | script += (line.rstrip() + r"\n") |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 267 | script += ("\" > " + client_name + ".key\n") |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 268 | # make sure openvpn is installed |
| Jeremy Mowery | bd3dfae | 2016-03-01 19:15:40 -0700 | [diff] [blame] | 269 | script += ("apt-get update\n") |
| 270 | script += ("apt-get install openvpn\n") |
| 271 | script += ("openvpn client.conf &\n") |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 272 | # close the script |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 273 | return script |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 274 | |
| Jeremy Mowery | 30bdab9 | 2016-03-29 23:52:46 -0700 | [diff] [blame] | 275 | def get_client_cert(self, client_name): |
| Jeremy Mowery | b52c49a | 2016-04-05 23:32:10 -0700 | [diff] [blame] | 276 | with open(self.pki_dir + "/issued/" + client_name + ".crt", 'r') as f: |
| 277 | return f.readlines() |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 278 | |
| Jeremy Mowery | 30bdab9 | 2016-03-29 23:52:46 -0700 | [diff] [blame] | 279 | def get_client_key(self, client_name): |
| Jeremy Mowery | b52c49a | 2016-04-05 23:32:10 -0700 | [diff] [blame] | 280 | with open(self.pki_dir + "/private/" + client_name + ".key", 'r') as f: |
| 281 | return f.readlines() |
| Jeremy Mowery | cf1c519 | 2016-03-15 20:28:04 -0700 | [diff] [blame] | 282 | |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 283 | def generate_client_conf(self, client_name): |
| Jeremy Mowery | 3bcea4b | 2016-04-01 21:54:29 -0700 | [diff] [blame] | 284 | """str: Generates the client configuration to use to connect to this |
| 285 | VPN server. |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 286 | """ |
| 287 | conf = ("client\n" + |
| Jeremy Mowery | 6d49f1a | 2016-03-21 17:56:25 -0700 | [diff] [blame] | 288 | "dev tun\n" + |
| Jeremy Mowery | a74c31d | 2016-04-04 22:30:44 -0700 | [diff] [blame] | 289 | self.get_remote_line( |
| 290 | self.nat_ip, self.port_number, self.protocol)) |
| 291 | for remote in self.failover_servers: |
| 292 | conf += self.get_remote_line( |
| 293 | remote.nat_ip, remote.port_number, remote.protocol) |
| 294 | |
| 295 | conf += ("resolv-retry 60\n" + |
| 296 | "nobind\n" + |
| 297 | "ca ca.crt\n" + |
| 298 | "cert " + client_name + ".crt\n" + |
| 299 | "key " + client_name + ".key\n" + |
| 300 | "comp-lzo\n" + |
| 301 | "verb 3\n") |
| Jeremy Mowery | 01e3a5b | 2016-02-25 14:50:58 -0700 | [diff] [blame] | 302 | |
| 303 | if self.is_persistent: |
| 304 | conf += "persist-tun\n" |
| 305 | conf += "persist-key\n" |
| 306 | |
| 307 | return conf |
| 308 | |
| Jeremy Mowery | a74c31d | 2016-04-04 22:30:44 -0700 | [diff] [blame] | 309 | def get_remote_line(self, host, port_number, protocol): |
| 310 | return ("remote " + str(host) + " " + str(port_number) + " " + |
| 311 | str(protocol) + "\n") |
| 312 | |
| Jeremy Mowery | 290714e | 2016-02-25 14:36:29 -0700 | [diff] [blame] | 313 | |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 314 | def model_policy_vpn_tenant(pk): |
| Jeremy Mowery | 8276082 | 2016-01-08 16:36:22 -0700 | [diff] [blame] | 315 | """Manages the contain for the VPN Tenant.""" |
| Jeremy Mowery | 8b664f7 | 2015-12-04 11:52:16 -0700 | [diff] [blame] | 316 | # This section of code is atomic to prevent race conditions |
| 317 | with transaction.atomic(): |
| 318 | # We find all of the tenants that are waiting to update |
| 319 | tenant = VPNTenant.objects.select_for_update().filter(pk=pk) |
| 320 | if not tenant: |
| 321 | return |
| 322 | # Since this code is atomic it is safe to always use the first tenant |
| 323 | tenant = tenant[0] |
| 324 | tenant.manage_container() |