blob: b42b092cbf9d1d84d2baee5838df131af9fc97ee [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 Bhatia04c94ad2013-09-02 18:00:28 -04007from observer.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 Bhatia14356b72014-11-05 10:32:41 -050011from observer.ansible import *
Sapan Bhatia01298ed2015-05-13 15:51:03 +020012from observer.syncstep import *
Tony Mack4d0d75c2015-03-29 08:32:21 -040013from util.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
Tony Mack66646d52013-09-24 21:47:12 -040023
Tony Mackd8515472015-08-19 11:58:18 -040024 def get_userdata(self, instance, pubkeys):
25 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 -040026 userdata += 'ssh_authorized_keys:\n'
Andy Bavier65b22de2015-03-18 15:59:19 -040027 for key in pubkeys:
28 userdata += ' - %s\n' % key
Scott Bakerdb0a1102014-08-08 09:57:28 -070029 return userdata
30
Tony Mackd8515472015-08-19 11:58:18 -040031 def sync_record(self, instance):
32 logger.info("sync'ing instance:%s slice:%s controller:%s " % (instance, instance.slice.name, instance.node.site_deployment.controller))
33 controller_register = json.loads(instance.node.site_deployment.controller.backend_register)
Sapan Bhatia06b1a882015-05-09 18:14:40 +020034
35 if (controller_register.get('disabled',False)):
Tony Mackd8515472015-08-19 11:58:18 -040036 raise InnocuousException('Controller %s is disabled'%instance.node.site_deployment.controller.name)
Sapan Bhatia14356b72014-11-05 10:32:41 -050037
Tony Mack6fd3ef82013-10-08 22:17:54 -040038 metadata_update = {}
Tony Mackd8515472015-08-19 11:58:18 -040039 if (instance.numberCores):
40 metadata_update["cpu_cores"] = str(instance.numberCores)
Tony Mack6fd3ef82013-10-08 22:17:54 -040041
Tony Mackd8515472015-08-19 11:58:18 -040042 for tag in instance.slice.tags.all():
Tony Mack6fd3ef82013-10-08 22:17:54 -040043 if tag.name.startswith("sysctl-"):
44 metadata_update[tag.name] = tag.value
45
Sapan Bhatia14356b72014-11-05 10:32:41 -050046 # public keys
Tony Mackd8515472015-08-19 11:58:18 -040047 slice_memberships = SlicePrivilege.objects.filter(slice=instance.slice)
Sapan Bhatia14356b72014-11-05 10:32:41 -050048 pubkeys = set([sm.user.public_key for sm in slice_memberships if sm.user.public_key])
Tony Mackd8515472015-08-19 11:58:18 -040049 if instance.creator.public_key:
50 pubkeys.add(instance.creator.public_key)
Tony Macked9925a2014-04-14 22:15:02 -040051
Tony Mackd8515472015-08-19 11:58:18 -040052 if instance.slice.creator.public_key:
53 pubkeys.add(instance.slice.creator.public_key)
Scott Baker9c7a1e12014-05-30 14:42:42 -070054
Tony Mackd8515472015-08-19 11:58:18 -040055 if instance.slice.service and instance.slice.service.public_key:
56 pubkeys.add(instance.slice.service.public_key)
Scott Baker5b044612015-04-30 14:30:56 -070057
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050058 nics = []
Tony Mackd8515472015-08-19 11:58:18 -040059 networks = [ns.network for ns in NetworkSlice.objects.filter(slice=instance.slice)]
Tony Mack3066a952015-01-05 22:48:11 -050060 controller_networks = ControllerNetwork.objects.filter(network__in=networks,
Tony Mackd8515472015-08-19 11:58:18 -040061 controller=instance.node.site_deployment.controller)
Scott Bakerdb0a1102014-08-08 09:57:28 -070062
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050063 for controller_network in controller_networks:
64 if controller_network.network.template.visibility == 'private' and \
Scott Baker5fb68322015-04-30 14:56:46 -070065 controller_network.network.template.translation == 'none':
66 if not controller_network.net_id:
67 raise Exception("Private Network %s has no id; Try again later" % controller_network.network.name)
68 nics.append(controller_network.net_id)
Sapan Bhatia14356b72014-11-05 10:32:41 -050069
Sapan6a975622014-12-03 00:01:23 -050070 # now include network template
Scott Baker369f9b92015-01-03 12:03:38 -080071 network_templates = [network.template.shared_network_name for network in networks \
72 if network.template.shared_network_name]
Sapan Bhatia14356b72014-11-05 10:32:41 -050073
Tony Mackd8515472015-08-19 11:58:18 -040074 #driver = self.driver.client_driver(caller=instance.creator, tenant=instance.slice.name, controller=instance.controllerNetwork)
75 driver = self.driver.admin_driver(tenant='admin', controller=instance.node.site_deployment.controller)
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050076 nets = driver.shell.quantum.list_networks()['networks']
77 for net in nets:
78 if net['name'] in network_templates:
79 nics.append(net['id'])
Sapan Bhatia14356b72014-11-05 10:32:41 -050080
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -050081 if (not nics):
82 for net in nets:
83 if net['name']=='public':
84 nics.append(net['id'])
Sapan Bhatia14356b72014-11-05 10:32:41 -050085
Sapan Bhatia7dac9662015-05-13 18:49:55 +020086 image_name = None
Tony Mackd8515472015-08-19 11:58:18 -040087 controller_images = instance.image.controllerimages.filter(controller=instance.node.site_deployment.controller)
Scott Baker5b044612015-04-30 14:30:56 -070088 if controller_images:
Sapan Bhatia7dac9662015-05-13 18:49:55 +020089 image_name = controller_images[0].image.name
90 logger.info("using image from ControllerImage object: " + str(image_name))
Scott Baker5b044612015-04-30 14:30:56 -070091
Sapan Bhatia7dac9662015-05-13 18:49:55 +020092 if image_name is None:
Tony Mackd8515472015-08-19 11:58:18 -040093 controller_driver = self.driver.admin_driver(controller=instance.node.site_deployment.controller)
Sapan Bhatia655b93e2015-03-16 12:55:23 -040094 images = controller_driver.shell.glanceclient.images.list()
95 for image in images:
Tony Mackd8515472015-08-19 11:58:18 -040096 if image.name == instance.image.name or not image_name:
Sapan Bhatia7dac9662015-05-13 18:49:55 +020097 image_name = image.name
98 logger.info("using image from glance: " + str(image_name))
Sapan Bhatia14356b72014-11-05 10:32:41 -050099
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500100 try:
101 legacy = Config().observer_legacy
102 except:
103 legacy = False
104
105 if (legacy):
Tony Mackd8515472015-08-19 11:58:18 -0400106 host_filter = instance.node.name.split('.',1)[0]
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500107 else:
Tony Mackd8515472015-08-19 11:58:18 -0400108 host_filter = instance.node.name.strip()
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500109
110 availability_zone_filter = 'nova:%s'%host_filter
Tony Mackd8515472015-08-19 11:58:18 -0400111 instance_name = '%s-%d'%(instance.slice.name,instance.id)
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500112
Tony Mackd8515472015-08-19 11:58:18 -0400113 userData = self.get_userdata(instance, pubkeys)
114 if instance.userData:
115 userData = instance.userData
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500116
Tony Mackd8515472015-08-19 11:58:18 -0400117 controller = instance.node.site_deployment.controller
Sapan Bhatia6bc43282014-12-22 13:01:18 -0500118 tenant_fields = {'endpoint':controller.auth_url,
Tony Mackd8515472015-08-19 11:58:18 -0400119 'admin_user': instance.creator.email,
120 'admin_password': instance.creator.remote_password,
121 'admin_tenant': instance.slice.name,
122 'tenant': instance.slice.name,
123 'tenant_description': instance.slice.description,
124 'name':instance_name,
125 'ansible_tag':instance_name,
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500126 'availability_zone': availability_zone_filter,
Sapan Bhatia7dac9662015-05-13 18:49:55 +0200127 'image_name':image_name,
Tony Mackd8515472015-08-19 11:58:18 -0400128 'flavor_name':instance.flavor.name,
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500129 'nics':nics,
130 'meta':metadata_update,
Sapan Bhatiaeeaf5fd2014-12-22 11:50:47 -0500131 'user_data':r'%s'%escape(userData)}
132
Tony Mackd8515472015-08-19 11:58:18 -0400133 res = run_template('sync_instances.yaml', tenant_fields,path='instances', expected_num=1)
134 instance_id = res[0]['info']['OS-EXT-SRV-ATTR:instance_name']
135 instance_uuid = res[0]['id']
Sapan Bhatia68b9c1c2015-01-20 20:59:24 +0000136
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000137 try:
Andy Baviera4025e92015-04-17 14:40:57 -0400138 hostname = res[0]['info']['OS-EXT-SRV-ATTR:hypervisor_hostname']
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000139 ip = socket.gethostbyname(hostname)
Tony Mackd8515472015-08-19 11:58:18 -0400140 instance.ip = ip
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000141 except:
142 pass
Sapan Bhatia14356b72014-11-05 10:32:41 -0500143
Tony Mackd8515472015-08-19 11:58:18 -0400144 instance.instance_id = instance_id
145 instance.instance_uuid = instance_uuid
146 instance.instance_name = instance_name
147 instance.save()
Sapan Bhatia24836f12013-08-27 10:16:05 -0400148
Tony Mackd8515472015-08-19 11:58:18 -0400149 def delete_record(self, instance):
150 controller_register = json.loads(instance.node.site_deployment.controller.backend_register)
Sapan Bhatia06b1a882015-05-09 18:14:40 +0200151
152 if (controller_register.get('disabled',False)):
Tony Mackd8515472015-08-19 11:58:18 -0400153 raise InnocuousException('Controller %s is disabled'%instance.node.site_deployment.controller.name)
Sapan Bhatia06b1a882015-05-09 18:14:40 +0200154
Tony Mackd8515472015-08-19 11:58:18 -0400155 instance_name = '%s-%d'%(instance.slice.name,instance.id)
156 controller = instance.node.site_deployment.controller
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000157 tenant_fields = {'endpoint':controller.auth_url,
Tony Mackd8515472015-08-19 11:58:18 -0400158 'admin_user': instance.creator.email,
159 'admin_password': instance.creator.remote_password,
160 'admin_tenant': instance.slice.name,
161 'tenant': instance.slice.name,
162 'tenant_description': instance.slice.description,
163 'name':instance_name,
164 'ansible_tag':instance_name,
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000165 'delete': True}
166
Scott Baker107b3e22015-03-09 16:24:45 -0700167 try:
Tony Mackd8515472015-08-19 11:58:18 -0400168 res = run_template('sync_instances.yaml', tenant_fields,path='instances', expected_num=1)
Scott Baker107b3e22015-03-09 16:24:45 -0700169 except Exception,e:
Tony Mackd8515472015-08-19 11:58:18 -0400170 print "Could not sync %s"%instance_name
Sapan Bhatia5d518072015-05-09 18:18:56 +0200171 #import traceback
172 #traceback.print_exc()
173 raise e
Scott Baker107b3e22015-03-09 16:24:45 -0700174
Sapan Bhatia3b3e1e12015-01-23 16:21:57 +0000175 if (len(res)!=1):
Tony Mackd8515472015-08-19 11:58:18 -0400176 raise Exception('Could not delete instance %s'%instance.slice.name)