| Sapan Bhatia | 24836f1 | 2013-08-27 10:16:05 -0400 | [diff] [blame] | 1 | import os |
| 2 | import base64 |
| Sapan Bhatia | 1e5d344 | 2015-01-23 16:26:46 +0000 | [diff] [blame] | 3 | import socket |
| Tony Mack | ae7f30c | 2013-09-25 12:46:50 -0400 | [diff] [blame] | 4 | from django.db.models import F, Q |
| Scott Baker | 76a840e | 2015-02-11 21:38:09 -0800 | [diff] [blame] | 5 | from xos.config import Config |
| Scott Baker | 6861c1f | 2015-03-09 16:24:45 -0700 | [diff] [blame] | 6 | from xos.settings import RESTAPI_HOSTNAME, RESTAPI_PORT |
| Sapan Bhatia | 04c94ad | 2013-09-02 18:00:28 -0400 | [diff] [blame] | 7 | from observer.openstacksyncstep import OpenStackSyncStep |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 8 | from core.models.instance import Instance |
| Tony Mack | a7dbd42 | 2015-01-05 22:48:11 -0500 | [diff] [blame] | 9 | from core.models.slice import Slice, SlicePrivilege, ControllerSlice |
| 10 | from core.models.network import Network, NetworkSlice, ControllerNetwork |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 11 | from observer.ansible import * |
| Sapan Bhatia | e6376de | 2015-05-13 15:51:03 +0200 | [diff] [blame] | 12 | from observer.syncstep import * |
| Tony Mack | 08f8288 | 2015-03-29 08:32:21 -0400 | [diff] [blame] | 13 | from util.logger import observer_logger as logger |
| Sapan Bhatia | 24836f1 | 2013-08-27 10:16:05 -0400 | [diff] [blame] | 14 | |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 15 | def escape(s): |
| 16 | s = s.replace('\n',r'\n').replace('"',r'\"') |
| 17 | return s |
| Sapan | d1a582d | 2014-12-03 00:01:23 -0500 | [diff] [blame] | 18 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 19 | class SyncInstances(OpenStackSyncStep): |
| 20 | provides=[Instance] |
| Tony Mack | dacfb98 | 2013-09-24 21:57:16 -0400 | [diff] [blame] | 21 | requested_interval=0 |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 22 | observes=Instance |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 23 | playbook='sync_instances.yaml' |
| Tony Mack | 66646d5 | 2013-09-24 21:47:12 -0400 | [diff] [blame] | 24 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 25 | def get_userdata(self, instance, pubkeys): |
| 26 | 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 Bavier | 03d2560 | 2015-04-16 13:41:50 -0400 | [diff] [blame] | 27 | userdata += 'ssh_authorized_keys:\n' |
| Andy Bavier | fd0669c | 2015-03-18 15:59:19 -0400 | [diff] [blame] | 28 | for key in pubkeys: |
| 29 | userdata += ' - %s\n' % key |
| Scott Baker | db0a110 | 2014-08-08 09:57:28 -0700 | [diff] [blame] | 30 | return userdata |
| 31 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 32 | def map_sync_inputs(self, instance): |
| 33 | inputs = {} |
| 34 | metadata_update = {} |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 35 | if (instance.numberCores): |
| 36 | metadata_update["cpu_cores"] = str(instance.numberCores) |
| Tony Mack | 6fd3ef8 | 2013-10-08 22:17:54 -0400 | [diff] [blame] | 37 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 38 | for tag in instance.slice.tags.all(): |
| Tony Mack | 6fd3ef8 | 2013-10-08 22:17:54 -0400 | [diff] [blame] | 39 | if tag.name.startswith("sysctl-"): |
| 40 | metadata_update[tag.name] = tag.value |
| 41 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 42 | slice_memberships = SlicePrivilege.objects.filter(slice=instance.slice) |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 43 | pubkeys = set([sm.user.public_key for sm in slice_memberships if sm.user.public_key]) |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 44 | if instance.creator.public_key: |
| 45 | pubkeys.add(instance.creator.public_key) |
| Tony Mack | ed9925a | 2014-04-14 22:15:02 -0400 | [diff] [blame] | 46 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 47 | if instance.slice.creator.public_key: |
| 48 | pubkeys.add(instance.slice.creator.public_key) |
| Scott Baker | 9c7a1e1 | 2014-05-30 14:42:42 -0700 | [diff] [blame] | 49 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 50 | if instance.slice.service and instance.slice.service.public_key: |
| 51 | pubkeys.add(instance.slice.service.public_key) |
| Scott Baker | 6894474 | 2015-04-30 14:30:56 -0700 | [diff] [blame] | 52 | |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 53 | nics = [] |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 54 | networks = [ns.network for ns in NetworkSlice.objects.filter(slice=instance.slice)] |
| Tony Mack | a7dbd42 | 2015-01-05 22:48:11 -0500 | [diff] [blame] | 55 | controller_networks = ControllerNetwork.objects.filter(network__in=networks, |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 56 | controller=instance.node.site_deployment.controller) |
| Scott Baker | db0a110 | 2014-08-08 09:57:28 -0700 | [diff] [blame] | 57 | |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 58 | for controller_network in controller_networks: |
| Sapan Bhatia | 57f1692 | 2015-10-14 19:40:10 +0200 | [diff] [blame] | 59 | |
| 60 | # Lenient exception - causes slow backoff |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 61 | if controller_network.network.template.visibility == 'private' and \ |
| Scott Baker | 3c61f7a | 2015-04-30 14:56:46 -0700 | [diff] [blame] | 62 | controller_network.network.template.translation == 'none': |
| 63 | if not controller_network.net_id: |
| Sapan Bhatia | 57f1692 | 2015-10-14 19:40:10 +0200 | [diff] [blame] | 64 | raise DeferredException("Private Network %s has no id; Try again later" % controller_network.network.name) |
| Scott Baker | 3c61f7a | 2015-04-30 14:56:46 -0700 | [diff] [blame] | 65 | nics.append(controller_network.net_id) |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 66 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 67 | # now include network template |
| Scott Baker | 81fa17f | 2015-01-03 12:03:38 -0800 | [diff] [blame] | 68 | network_templates = [network.template.shared_network_name for network in networks \ |
| 69 | if network.template.shared_network_name] |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 70 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 71 | #driver = self.driver.client_driver(caller=instance.creator, tenant=instance.slice.name, controller=instance.controllerNetwork) |
| 72 | driver = self.driver.admin_driver(tenant='admin', controller=instance.node.site_deployment.controller) |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 73 | nets = driver.shell.quantum.list_networks()['networks'] |
| 74 | for net in nets: |
| 75 | if net['name'] in network_templates: |
| 76 | nics.append(net['id']) |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 77 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 78 | if (not nics): |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 79 | for net in nets: |
| 80 | if net['name']=='public': |
| 81 | nics.append(net['id']) |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 82 | |
| Sapan Bhatia | ff35796 | 2015-05-13 18:49:55 +0200 | [diff] [blame] | 83 | image_name = None |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 84 | controller_images = instance.image.controllerimages.filter(controller=instance.node.site_deployment.controller) |
| Scott Baker | 6894474 | 2015-04-30 14:30:56 -0700 | [diff] [blame] | 85 | if controller_images: |
| Sapan Bhatia | ff35796 | 2015-05-13 18:49:55 +0200 | [diff] [blame] | 86 | image_name = controller_images[0].image.name |
| 87 | logger.info("using image from ControllerImage object: " + str(image_name)) |
| Scott Baker | 6894474 | 2015-04-30 14:30:56 -0700 | [diff] [blame] | 88 | |
| Sapan Bhatia | ff35796 | 2015-05-13 18:49:55 +0200 | [diff] [blame] | 89 | if image_name is None: |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 90 | controller_driver = self.driver.admin_driver(controller=instance.node.site_deployment.controller) |
| Sapan Bhatia | 84ed9d6 | 2015-03-16 12:55:23 -0400 | [diff] [blame] | 91 | images = controller_driver.shell.glanceclient.images.list() |
| 92 | for image in images: |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 93 | if image.name == instance.image.name or not image_name: |
| Sapan Bhatia | ff35796 | 2015-05-13 18:49:55 +0200 | [diff] [blame] | 94 | image_name = image.name |
| 95 | logger.info("using image from glance: " + str(image_name)) |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 96 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 97 | try: |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 98 | legacy = Config().observer_legacy |
| 99 | except: |
| 100 | legacy = False |
| 101 | |
| 102 | if (legacy): |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 103 | host_filter = instance.node.name.split('.',1)[0] |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 104 | else: |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 105 | host_filter = instance.node.name.strip() |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 106 | |
| 107 | availability_zone_filter = 'nova:%s'%host_filter |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 108 | instance_name = '%s-%d'%(instance.slice.name,instance.id) |
| Sapan Bhatia | af143dc | 2015-09-16 19:52:40 +0200 | [diff] [blame] | 109 | self.instance_name = instance_name |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 110 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 111 | userData = self.get_userdata(instance, pubkeys) |
| 112 | if instance.userData: |
| Sapan Bhatia | 8610549 | 2015-10-11 21:11:47 +0200 | [diff] [blame] | 113 | userData += instance.userData |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 114 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 115 | controller = instance.node.site_deployment.controller |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 116 | fields = {'endpoint':controller.auth_url, |
| Tony Mack | 09a2f07 | 2015-09-14 00:53:39 +0000 | [diff] [blame] | 117 | 'endpoint_v3': controller.auth_url_v3, |
| 118 | 'domain': controller.domain, |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 119 | '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 Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 126 | 'availability_zone': availability_zone_filter, |
| Sapan Bhatia | ff35796 | 2015-05-13 18:49:55 +0200 | [diff] [blame] | 127 | 'image_name':image_name, |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 128 | 'flavor_name':instance.flavor.name, |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 129 | 'nics':nics, |
| 130 | 'meta':metadata_update, |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 131 | 'user_data':r'%s'%escape(userData)} |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 132 | return fields |
| Sapan Bhatia | f3d2132 | 2014-12-22 11:50:47 -0500 | [diff] [blame] | 133 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 134 | |
| 135 | def map_sync_outputs(self, instance, res): |
| 136 | instance_id = res[0]['info']['OS-EXT-SRV-ATTR:instance_name'] |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 137 | instance_uuid = res[0]['id'] |
| Sapan Bhatia | cb5c24a | 2015-01-20 20:59:24 +0000 | [diff] [blame] | 138 | |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 139 | try: |
| Andy Bavier | 87e7a92 | 2015-04-17 14:40:57 -0400 | [diff] [blame] | 140 | hostname = res[0]['info']['OS-EXT-SRV-ATTR:hypervisor_hostname'] |
| Sapan Bhatia | b0464ba | 2015-01-23 16:21:57 +0000 | [diff] [blame] | 141 | ip = socket.gethostbyname(hostname) |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 142 | instance.ip = ip |
| Sapan Bhatia | b0464ba | 2015-01-23 16:21:57 +0000 | [diff] [blame] | 143 | except: |
| 144 | pass |
| Sapan Bhatia | 475c597 | 2014-11-05 10:32:41 -0500 | [diff] [blame] | 145 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 146 | instance.instance_id = instance_id |
| 147 | instance.instance_uuid = instance_uuid |
| Sapan Bhatia | af143dc | 2015-09-16 19:52:40 +0200 | [diff] [blame] | 148 | instance.instance_name = self.instance_name |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 149 | instance.save() |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 150 | |
| 151 | |
| 152 | def map_delete_inputs(self, instance): |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 153 | controller_register = json.loads(instance.node.site_deployment.controller.backend_register) |
| Sapan Bhatia | 9028c9a | 2015-05-09 18:14:40 +0200 | [diff] [blame] | 154 | |
| 155 | if (controller_register.get('disabled',False)): |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 156 | raise InnocuousException('Controller %s is disabled'%instance.node.site_deployment.controller.name) |
| Sapan Bhatia | 9028c9a | 2015-05-09 18:14:40 +0200 | [diff] [blame] | 157 | |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 158 | instance_name = '%s-%d'%(instance.slice.name,instance.id) |
| 159 | controller = instance.node.site_deployment.controller |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 160 | input = {'endpoint':controller.auth_url, |
| Tony Mack | 3de59e3 | 2015-08-19 11:58:18 -0400 | [diff] [blame] | 161 | 'admin_user': instance.creator.email, |
| 162 | 'admin_password': instance.creator.remote_password, |
| 163 | 'admin_tenant': instance.slice.name, |
| 164 | 'tenant': instance.slice.name, |
| 165 | 'tenant_description': instance.slice.description, |
| 166 | 'name':instance_name, |
| 167 | 'ansible_tag':instance_name, |
| Sapan Bhatia | b0464ba | 2015-01-23 16:21:57 +0000 | [diff] [blame] | 168 | 'delete': True} |
| Sapan Bhatia | 321b70e | 2015-08-19 12:20:47 -0400 | [diff] [blame] | 169 | return input |