blob: b61ef7b15efe1ee2f570414f0cbeee44b75b68b6 [file] [log] [blame]
Scott Bakerb63ea792016-08-11 10:24:48 -07001import os
2import base64
3import struct
4import socket
5from collections import defaultdict
6from netaddr import IPAddress, IPNetwork
7from django.db.models import F, Q
8from xos.config import Config
9from synchronizers.base.openstacksyncstep import OpenStackSyncStep
10from synchronizers.base.syncstep import *
11from core.models.network import *
12from core.models.slice import *
13from core.models.instance import Instance
14from xos.logger import observer_logger as logger
15from synchronizers.base.ansible import *
16from openstack_xos.driver import OpenStackDriver
17from xos.config import Config
18import json
19
20import pdb
21
22class SyncControllerNetworks(OpenStackSyncStep):
23 requested_interval = 0
24 provides=[Network]
25 observes=ControllerNetwork
26 playbook='sync_controller_networks.yaml'
27
28 def alloc_subnet(self, uuid):
29 # 16 bits only
30 uuid_masked = uuid & 0xffff
31 a = 10
32 b = uuid_masked >> 8
33 c = uuid_masked & 0xff
34 d = 0
35
36 cidr = '%d.%d.%d.%d/24'%(a,b,c,d)
37 return cidr
38
39 def alloc_gateway(self, subnet):
40 # given a CIDR, allocate a default gateway using the .1 address within
41 # the subnet.
42 # 10.123.0.0/24 --> 10.123.0.1
43 # 207.141.192.128/28 --> 207.141.192.129
44 (network, bits) = subnet.split("/")
45 network=network.strip()
46 bits=int(bits.strip())
47 netmask = (~(pow(2,32-bits)-1) & 0xFFFFFFFF)
48 ip = struct.unpack("!L", socket.inet_aton(network))[0]
49 ip = ip & netmask | 1
50 return socket.inet_ntoa(struct.pack("!L", ip))
51
52 def save_controller_network(self, controller_network):
53 network_name = controller_network.network.name
54 subnet_name = '%s-%d'%(network_name,controller_network.pk)
55 if controller_network.subnet and controller_network.subnet.strip():
56 # If a subnet is already specified (pass in by the creator), then
57 # use that rather than auto-generating one.
58 cidr = controller_network.subnet.strip()
59 print "CIDR_MS", cidr
60 else:
61 cidr = self.alloc_subnet(controller_network.pk)
62 print "CIDR_AMS", cidr
63
64 if controller_network.network.start_ip and controller_network.network.start_ip.strip():
65 start_ip = controller_network.network.start_ip.strip()
66 else:
67 start_ip = None
68
69 if controller_network.network.end_ip and controller_network.network.end_ip.strip():
70 end_ip = controller_network.network.end_ip.strip()
71 else:
72 end_ip = None
73
74 self.cidr=cidr
75 slice = controller_network.network.owner
76
77 network_fields = {'endpoint':controller_network.controller.auth_url,
78 'endpoint_v3': controller_network.controller.auth_url_v3,
79 'admin_user':slice.creator.email,
80 'admin_password':slice.creator.remote_password,
81 'admin_project':slice.name,
82 'domain': controller_network.controller.domain,
83 'name':network_name,
84 'subnet_name':subnet_name,
85 'ansible_tag':'%s-%s@%s'%(network_name,slice.slicename,controller_network.controller.name),
86 'cidr':cidr,
87 'gateway':self.alloc_gateway(cidr),
88 'start_ip':start_ip,
89 'end_ip':end_ip,
90 'use_vtn':getattr(Config(), "networking_use_vtn", False),
91 'delete':False
92 }
93 return network_fields
94
95 def map_sync_outputs(self, controller_network,res):
96 network_id = res[0]['network']['id']
97 subnet_id = res[1]['subnet']['id']
98 controller_network.net_id = network_id
99 controller_network.subnet = self.cidr
100 controller_network.subnet_id = subnet_id
101 controller_network.backend_status = '1 - OK'
102 controller_network.save()
103
104
105 def map_sync_inputs(self, controller_network):
106 # XXX This check should really be made from booleans, rather than using hardcoded network names
107 #if (controller_network.network.template.name not in ['Private', 'Private-Indirect', 'Private-Direct', 'management_template'):
108 # logger.info("skipping network controller %s because it is not private" % controller_network)
109 # # We only sync private networks
110 # return SyncStep.SYNC_WITHOUT_RUNNING
111
112 # hopefully a better approach than above
113 if (controller_network.network.template.shared_network_name or controller_network.network.template.shared_network_id):
114 return SyncStep.SYNC_WITHOUT_RUNNING
115
116 if not controller_network.controller.admin_user:
117 logger.info("controller %r has no admin_user, skipping" % controller_network.controller)
118 return
119
120 if controller_network.network.owner and controller_network.network.owner.creator:
121 return self.save_controller_network(controller_network)
122 else:
123 raise Exception('Could not save network controller %s'%controller_network)
124
125 def map_delete_inputs(self, controller_network):
126 # XXX This check should really be made from booleans, rather than using hardcoded network names
127 if (controller_network.network.template.name not in ['Private', 'Private-Indirect', 'Private-Direct']):
128 # We only sync private networks
129 return
130 try:
131 slice = controller_network.network.owner # XXX: FIXME!!
132 except:
133 raise Exception('Could not get slice for Network %s'%controller_network.network.name)
134
135 network_name = controller_network.network.name
136 subnet_name = '%s-%d'%(network_name,controller_network.pk)
137 cidr = controller_network.subnet
138 network_fields = {'endpoint':controller_network.controller.auth_url,
139 'admin_user':slice.creator.email, # XXX: FIXME
140 'tenant_name':slice.name, # XXX: FIXME
141 'admin_password':slice.creator.remote_password,
142 'name':network_name,
143 'subnet_name':subnet_name,
144 'ansible_tag':'%s-%s@%s'%(network_name,slice.slicename,controller_network.controller.name),
145 'cidr':cidr,
146 'delete':True
147 }
148
149 return network_fields
150
151 """
152 driver = OpenStackDriver().client_driver(caller=controller_network.network.owner.creator,
153 tenant=controller_network.network.owner.name,
154 controller=controller_network.controller.name)
155 if (controller_network.router_id) and (controller_network.subnet_id):
156 driver.delete_router_interface(controller_network.router_id, controller_network.subnet_id)
157 if controller_network.subnet_id:
158 driver.delete_subnet(controller_network.subnet_id)
159 if controller_network.router_id:
160 driver.delete_router(controller_network.router_id)
161 if controller_network.net_id:
162 driver.delete_network(controller_network.net_id)
163 """