blob: 22aa45caaaef00eb142c1e32f0f28ec6efd76cd0 [file] [log] [blame]
Sapan Bhatia24836f12013-08-27 10:16:05 -04001import os
2import base64
Sapan Bhatia3346abe2015-01-23 16:26:46 +00003import socket
Tony Mackae7f30c2013-09-25 12:46:50 -04004from django.db.models import F, Q
Scott Baker86e132c2015-02-11 21:38:09 -08005from xos.config import Config
Scott Baker107b3e22015-03-09 16:24:45 -07006from xos.settings import RESTAPI_HOSTNAME, RESTAPI_PORT
Sapan Bhatia16be1432016-01-14 11:41:38 -05007from synchronizers.base.openstacksyncstep import OpenStackSyncStep
Tony Mackd8515472015-08-19 11:58:18 -04008from core.models.instance import Instance
Tony Mack3066a952015-01-05 22:48:11 -05009from core.models.slice import Slice, SlicePrivilege, ControllerSlice
10from core.models.network import Network, NetworkSlice, ControllerNetwork
Sapan Bhatiaf0538b82016-01-15 11:05:52 -050011from synchronizers.base.ansible import *
12from synchronizers.base.syncstep import *
Scott Baker3586c552016-01-14 15:30:20 -080013from xos.logger import observer_logger as logger
Sapan Bhatia24836f12013-08-27 10:16:05 -040014
Sapan Bhatia14356b72014-11-05 10:32:41 -050015def escape(s):
16 s = s.replace('\n',r'\n').replace('"',r'\"')
17 return s
Sapan6a975622014-12-03 00:01:23 -050018
Tony Mackd8515472015-08-19 11:58:18 -040019class SyncInstances(OpenStackSyncStep):
20 provides=[Instance]
Tony Mackdacfb982013-09-24 21:57:16 -040021 requested_interval=0
Tony Mackd8515472015-08-19 11:58:18 -040022 observes=Instance
Sapan Bhatiac239f672015-08-19 12:20:47 -040023 playbook='sync_instances.yaml'
Tony Mack66646d52013-09-24 21:47:12 -040024
Scott Bakerdcf9e0d2015-11-09 16:17:11 -080025 def fetch_pending(self, deletion=False):
26 objs = super(SyncInstances, self).fetch_pending(deletion)
27 objs = [x for x in objs if x.isolation=="vm"]
28 return objs
29
Tony Mackd8515472015-08-19 11:58:18 -040030 def get_userdata(self, instance, pubkeys):
31 userdata = '#cloud-config\n\nopencloud:\n slicename: "%s"\n hostname: "%s"\n restapi_hostname: "%s"\n restapi_port: "%s"\n' % (instance.slice.name, instance.node.name, RESTAPI_HOSTNAME, str(RESTAPI_PORT))
Andy Bavier9473f5e2015-04-16 13:41:50 -040032 userdata += 'ssh_authorized_keys:\n'
Andy Bavier65b22de2015-03-18 15:59:19 -040033 for key in pubkeys:
34 userdata += ' - %s\n' % key
Scott Bakerdb0a1102014-08-08 09:57:28 -070035 return userdata
36
Scott Bakerc9b0ec72016-02-03 18:31:37 -080037 def sort_controller_networks(self, nets):
38 nets = list(nets)
39 result = []
40
41 # Enforce VTN's network order requirement. The access network must be
42 # inserted into the first slot. The management network must be inserted
43 # into the second slot.
44
45 # move the private and/or access network to the first spot
46 for net in nets[:]:
47 tem = net.network.template
48 if (tem.visibility == "private") and (tem.translation=="none") and ("management" not in tem.name):
49 result.append(net)
50 nets.remove(net)
51
52 # move the management network to the second spot
53 for net in nets[:]:
54 tem = net.network.template
55 if (tem.visibility == "private") and (tem.translation=="none") and ("management" in tem.name):
56 if len(result)!=1:
57 raise Exception("Management network needs to be inserted in slot 1, but there are %d private nets" % len(result))
58 result.append(net)
59 nets.remove(net)
60
61 # add everything else. For VTN there probably shouldn't be any more.
62 result.extend(nets)
63
64 return result
65
Sapan Bhatiac239f672015-08-19 12:20:47 -040066 def map_sync_inputs(self, instance):
67 inputs = {}
68 metadata_update = {}
Tony Mackd8515472015-08-19 11:58:18 -040069 if (instance.numberCores):
70 metadata_update["cpu_cores"] = str(instance.numberCores)
Tony Mack6fd3ef82013-10-08 22:17:54 -040071
Tony Mackd8515472015-08-19 11:58:18 -040072 for tag in instance.slice.tags.all():
Tony Mack6fd3ef82013-10-08 22:17:54 -040073 if tag.name.startswith("sysctl-"):
74 metadata_update[tag.name] = tag.value
75
Sapan Bhatiac239f672015-08-19 12:20:47 -040076 slice_memberships = SlicePrivilege.objects.filter(slice=instance.slice)
Sapan Bhatia14356b72014-11-05 10:32:41 -050077 pubkeys = set([sm.user.public_key for sm in slice_memberships if sm.user.public_key])
Tony Mackd8515472015-08-19 11:58:18 -040078 if instance.creator.public_key:
79 pubkeys.add(instance.creator.public_key)
Tony Macked9925a2014-04-14 22:15:02 -040080
Tony Mackd8515472015-08-19 11:58:18 -040081 if instance.slice.creator.public_key:
82 pubkeys.add(instance.slice.creator.public_key)
Scott Baker9c7a1e12014-05-30 14:42:42 -070083
Tony Mackd8515472015-08-19 11:58:18 -040084 if instance.slice.service and instance.slice.service.public_key:
85 pubkeys.add(instance.slice.service.public_key)
Scott Baker5b044612015-04-30 14:30:56 -070086
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050087 nics = []
Tony Mackd8515472015-08-19 11:58:18 -040088 networks = [ns.network for ns in NetworkSlice.objects.filter(slice=instance.slice)]
Tony Mack3066a952015-01-05 22:48:11 -050089 controller_networks = ControllerNetwork.objects.filter(network__in=networks,
Tony Mackd8515472015-08-19 11:58:18 -040090 controller=instance.node.site_deployment.controller)
Scott Bakerdb0a1102014-08-08 09:57:28 -070091
Scott Bakerc9b0ec72016-02-03 18:31:37 -080092 controller_networks = self.sort_controller_networks(controller_networks)
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050093 for controller_network in controller_networks:
Sapan Bhatia7a4ebb32015-10-14 19:40:10 +020094 # Lenient exception - causes slow backoff
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050095 if controller_network.network.template.visibility == 'private' and \
Scott Baker5fb68322015-04-30 14:56:46 -070096 controller_network.network.template.translation == 'none':
97 if not controller_network.net_id:
Scott Bakerad3f78d2015-11-24 22:47:54 -080098 raise DeferredException("Instance %s Private Network %s has no id; Try again later" % (instance, controller_network.network.name))
Scott Baker5fb68322015-04-30 14:56:46 -070099 nics.append(controller_network.net_id)
Sapan Bhatia14356b72014-11-05 10:32:41 -0500100
Sapan Bhatiac239f672015-08-19 12:20:47 -0400101 # now include network template
Scott Baker369f9b92015-01-03 12:03:38 -0800102 network_templates = [network.template.shared_network_name for network in networks \
103 if network.template.shared_network_name]
Sapan Bhatia14356b72014-11-05 10:32:41 -0500104
Tony Mackd8515472015-08-19 11:58:18 -0400105 #driver = self.driver.client_driver(caller=instance.creator, tenant=instance.slice.name, controller=instance.controllerNetwork)
106 driver = self.driver.admin_driver(tenant='admin', controller=instance.node.site_deployment.controller)
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500107 nets = driver.shell.quantum.list_networks()['networks']
108 for net in nets:
109 if net['name'] in network_templates:
110 nics.append(net['id'])
Sapan Bhatia14356b72014-11-05 10:32:41 -0500111
Sapan Bhatiac239f672015-08-19 12:20:47 -0400112 if (not nics):
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500113 for net in nets:
114 if net['name']=='public':
115 nics.append(net['id'])
Sapan Bhatia14356b72014-11-05 10:32:41 -0500116
Sapan Bhatia7dac9662015-05-13 18:49:55 +0200117 image_name = None
Tony Mackd8515472015-08-19 11:58:18 -0400118 controller_images = instance.image.controllerimages.filter(controller=instance.node.site_deployment.controller)
Scott Baker5b044612015-04-30 14:30:56 -0700119 if controller_images:
Sapan Bhatia7dac9662015-05-13 18:49:55 +0200120 image_name = controller_images[0].image.name
121 logger.info("using image from ControllerImage object: " + str(image_name))
Scott Baker5b044612015-04-30 14:30:56 -0700122
Sapan Bhatia7dac9662015-05-13 18:49:55 +0200123 if image_name is None:
Tony Mackd8515472015-08-19 11:58:18 -0400124 controller_driver = self.driver.admin_driver(controller=instance.node.site_deployment.controller)
Sapan Bhatia655b93e2015-03-16 12:55:23 -0400125 images = controller_driver.shell.glanceclient.images.list()
126 for image in images:
Tony Mackd8515472015-08-19 11:58:18 -0400127 if image.name == instance.image.name or not image_name:
Sapan Bhatia7dac9662015-05-13 18:49:55 +0200128 image_name = image.name
129 logger.info("using image from glance: " + str(image_name))
Sapan Bhatia14356b72014-11-05 10:32:41 -0500130
Sapan Bhatiac239f672015-08-19 12:20:47 -0400131 try:
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500132 legacy = Config().observer_legacy
133 except:
134 legacy = False
135
136 if (legacy):
Tony Mackd8515472015-08-19 11:58:18 -0400137 host_filter = instance.node.name.split('.',1)[0]
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500138 else:
Tony Mackd8515472015-08-19 11:58:18 -0400139 host_filter = instance.node.name.strip()
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500140
141 availability_zone_filter = 'nova:%s'%host_filter
Tony Mackd8515472015-08-19 11:58:18 -0400142 instance_name = '%s-%d'%(instance.slice.name,instance.id)
Sapan Bhatia2aa1c032015-09-16 19:52:40 +0200143 self.instance_name = instance_name
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500144
Tony Mackd8515472015-08-19 11:58:18 -0400145 userData = self.get_userdata(instance, pubkeys)
146 if instance.userData:
Sapan Bhatia9ee3b152015-10-11 21:11:47 +0200147 userData += instance.userData
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500148
Tony Mackd8515472015-08-19 11:58:18 -0400149 controller = instance.node.site_deployment.controller
Sapan Bhatiac239f672015-08-19 12:20:47 -0400150 fields = {'endpoint':controller.auth_url,
Tony Mack3ceb16f2015-09-14 00:53:39 +0000151 'endpoint_v3': controller.auth_url_v3,
152 'domain': controller.domain,
Tony Mackd8515472015-08-19 11:58:18 -0400153 'admin_user': instance.creator.email,
154 'admin_password': instance.creator.remote_password,
155 'admin_tenant': instance.slice.name,
156 'tenant': instance.slice.name,
157 'tenant_description': instance.slice.description,
158 'name':instance_name,
159 'ansible_tag':instance_name,
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500160 'availability_zone': availability_zone_filter,
Sapan Bhatia7dac9662015-05-13 18:49:55 +0200161 'image_name':image_name,
Tony Mackd8515472015-08-19 11:58:18 -0400162 'flavor_name':instance.flavor.name,
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500163 'nics':nics,
164 'meta':metadata_update,
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500165 'user_data':r'%s'%escape(userData)}
Sapan Bhatiac239f672015-08-19 12:20:47 -0400166 return fields
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500167
Sapan Bhatiac239f672015-08-19 12:20:47 -0400168
169 def map_sync_outputs(self, instance, res):
170 instance_id = res[0]['info']['OS-EXT-SRV-ATTR:instance_name']
Tony Mackd8515472015-08-19 11:58:18 -0400171 instance_uuid = res[0]['id']
Sapan Bhatia68b9c1c2015-01-20 20:59:24 +0000172
Sapan Bhatiac239f672015-08-19 12:20:47 -0400173 try:
Andy Baviera4025e92015-04-17 14:40:57 -0400174 hostname = res[0]['info']['OS-EXT-SRV-ATTR:hypervisor_hostname']
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000175 ip = socket.gethostbyname(hostname)
Tony Mackd8515472015-08-19 11:58:18 -0400176 instance.ip = ip
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000177 except:
178 pass
Sapan Bhatia14356b72014-11-05 10:32:41 -0500179
Tony Mackd8515472015-08-19 11:58:18 -0400180 instance.instance_id = instance_id
181 instance.instance_uuid = instance_uuid
Sapan Bhatia2aa1c032015-09-16 19:52:40 +0200182 instance.instance_name = self.instance_name
Tony Mackd8515472015-08-19 11:58:18 -0400183 instance.save()
Sapan Bhatiac239f672015-08-19 12:20:47 -0400184
185
186 def map_delete_inputs(self, instance):
Tony Mackd8515472015-08-19 11:58:18 -0400187 controller_register = json.loads(instance.node.site_deployment.controller.backend_register)
Sapan Bhatia06b1a882015-05-09 18:14:40 +0200188
189 if (controller_register.get('disabled',False)):
Tony Mackd8515472015-08-19 11:58:18 -0400190 raise InnocuousException('Controller %s is disabled'%instance.node.site_deployment.controller.name)
Sapan Bhatia06b1a882015-05-09 18:14:40 +0200191
Tony Mackd8515472015-08-19 11:58:18 -0400192 instance_name = '%s-%d'%(instance.slice.name,instance.id)
193 controller = instance.node.site_deployment.controller
Sapan Bhatiac239f672015-08-19 12:20:47 -0400194 input = {'endpoint':controller.auth_url,
Tony Mackd8515472015-08-19 11:58:18 -0400195 'admin_user': instance.creator.email,
196 'admin_password': instance.creator.remote_password,
197 'admin_tenant': instance.slice.name,
198 'tenant': instance.slice.name,
199 'tenant_description': instance.slice.description,
200 'name':instance_name,
201 'ansible_tag':instance_name,
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000202 'delete': True}
Sapan Bhatiac239f672015-08-19 12:20:47 -0400203 return input