| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | #!/usr/bin/python | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  | # Copyright (c) 2009, 2010, 2011, 2012 Nicira Networks | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | # | 
					
						
							|  |  |  | # Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  | # you may not use this file except in compliance with the License. | 
					
						
							|  |  |  | # You may obtain a copy of the License at: | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #     http://www.apache.org/licenses/LICENSE-2.0 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  | # distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  | # See the License for the specific language governing permissions and | 
					
						
							|  |  |  | # limitations under the License. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # A daemon to monitor the external_ids columns of the Bridge and | 
					
						
							| 
									
										
										
										
											2011-02-11 12:35:37 -08:00
										 |  |  | # Interface OVSDB tables for changes that require interrogating XAPI. | 
					
						
							|  |  |  | # Its responsibilities include: | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #   - Set the "bridge-id" key in the Bridge table. | 
					
						
							|  |  |  | #   - Set the "iface-id" key in the Interface table. | 
					
						
							|  |  |  | #   - Set the fail-mode on internal bridges. | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  | import argparse | 
					
						
							| 
									
										
										
										
											2010-09-10 14:20:49 -07:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  | import signal | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2010-09-10 14:20:49 -07:00
										 |  |  | import time | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | import XenAPI | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | import ovs.dirs | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | from ovs.db import error | 
					
						
							|  |  |  | from ovs.db import types | 
					
						
							|  |  |  | import ovs.daemon | 
					
						
							|  |  |  | import ovs.db.idl | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  | vlog = ovs.vlog.Vlog("ovs-xapi-sync") | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | session = None | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  | force_run = False | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | # Set up a session to interact with XAPI. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # On system start-up, OVS comes up before XAPI, so we can't log into the | 
					
						
							|  |  |  | # session until later.  Try to do this on-demand, since we won't | 
					
						
							|  |  |  | # actually do anything interesting until XAPI is up. | 
					
						
							|  |  |  | def init_session(): | 
					
						
							|  |  |  |     global session | 
					
						
							|  |  |  |     if session is not None: | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         session = XenAPI.xapi_local() | 
					
						
							|  |  |  |         session.xenapi.login_with_password("", "") | 
					
						
							| 
									
										
										
										
											2011-09-26 13:07:29 -07:00
										 |  |  |     except XenAPI.Failure, e: | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |         session = None | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |         vlog.warn("Couldn't login to XAPI (%s)" % e) | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  | def get_network_by_bridge(br_name): | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |     if not init_session(): | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |         vlog.warn("Failed to get bridge id %s because" | 
					
						
							| 
									
										
										
										
											2010-09-22 15:38:17 -07:00
										 |  |  |                 " XAPI session could not be initialized" % br_name) | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  |         return None | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for n in session.xenapi.network.get_all(): | 
					
						
							|  |  |  |         rec = session.xenapi.network.get_record(n) | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  |         if rec['bridge'] == br_name: | 
					
						
							|  |  |  |             return rec | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  | # By default, the "bridge-id" external id in the Bridge table is the | 
					
						
							|  |  |  | # same as "xs-network-uuids".  This may be overridden by defining a | 
					
						
							|  |  |  | # "nicira-bridge-id" key in the "other_config" field of the network | 
					
						
							|  |  |  | # record of XAPI.  If nicira-bridge-id is undefined returns default. | 
					
						
							|  |  |  | # On error returns None. | 
					
						
							|  |  |  | def get_bridge_id(br_name, default=None): | 
					
						
							|  |  |  |     rec = get_network_by_bridge(br_name) | 
					
						
							|  |  |  |     if rec: | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |         return rec['other_config'].get('nicira-bridge-id', default) | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  |     return None | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-21 18:03:07 -07:00
										 |  |  | # By default, the "iface-id" external id in the Interface table is the | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | # same as "xs-vif-uuid".  This may be overridden by defining a | 
					
						
							|  |  |  | # "nicira-iface-id" key in the "other_config" field of the VIF | 
					
						
							|  |  |  | # record of XAPI. | 
					
						
							| 
									
										
										
										
											2010-10-25 17:26:44 -07:00
										 |  |  | def get_iface_id(if_name, xs_vif_uuid): | 
					
						
							| 
									
										
										
										
											2011-06-16 15:02:50 -07:00
										 |  |  |     if not if_name.startswith("vif") and not if_name.startswith("tap"): | 
					
						
							| 
									
										
										
										
											2010-10-25 17:26:44 -07:00
										 |  |  |         # Treat whatever was passed into 'xs_vif_uuid' as a default | 
					
						
							|  |  |  |         # value for non-VIFs. | 
					
						
							|  |  |  |         return xs_vif_uuid | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if not init_session(): | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |         vlog.warn("Failed to get interface id %s because" | 
					
						
							| 
									
										
										
										
											2010-09-22 15:38:17 -07:00
										 |  |  |                 " XAPI session could not be initialized" % if_name) | 
					
						
							| 
									
										
										
										
											2010-10-25 17:26:44 -07:00
										 |  |  |         return xs_vif_uuid | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-25 17:26:44 -07:00
										 |  |  |     try: | 
					
						
							|  |  |  |         vif = session.xenapi.VIF.get_by_uuid(xs_vif_uuid) | 
					
						
							|  |  |  |         rec = session.xenapi.VIF.get_record(vif) | 
					
						
							|  |  |  |         return rec['other_config'].get('nicira-iface-id', xs_vif_uuid) | 
					
						
							|  |  |  |     except XenAPI.Failure: | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |         vlog.warn("Could not find XAPI entry for VIF %s" % if_name) | 
					
						
							| 
									
										
										
										
											2010-10-25 17:26:44 -07:00
										 |  |  |         return xs_vif_uuid | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 10:20:00 -08:00
										 |  |  | # By default, the "vm-id" external id in the Interface table is the | 
					
						
							|  |  |  | # same as "xs-vm-uuid".  This may be overridden by defining a | 
					
						
							|  |  |  | # "nicira-vm-id" key in the "other_config" field of the VM | 
					
						
							|  |  |  | # record of XAPI. | 
					
						
							|  |  |  | def get_vm_id(if_name, xs_vm_uuid): | 
					
						
							|  |  |  |     if not if_name.startswith("vif") and not if_name.startswith("tap"): | 
					
						
							|  |  |  |         # Treat whatever was passed into 'xs_vm_uuid' as a default | 
					
						
							|  |  |  |         # value for non-VIFs. | 
					
						
							|  |  |  |         return xs_vm_uuid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not init_session(): | 
					
						
							|  |  |  |         vlog.warn("Failed to get vm id for interface id %s because" | 
					
						
							|  |  |  |                 " XAPI session could not be initialized" % if_name) | 
					
						
							|  |  |  |         return xs_vm_uuid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         vm = session.xenapi.VM.get_by_uuid(xs_vm_uuid) | 
					
						
							|  |  |  |         rec = session.xenapi.VM.get_record(vm) | 
					
						
							|  |  |  |         return rec['other_config'].get('nicira-vm-id', xs_vm_uuid) | 
					
						
							|  |  |  |     except XenAPI.Failure: | 
					
						
							|  |  |  |         vlog.warn("Could not find XAPI entry for VIF %s" % if_name) | 
					
						
							|  |  |  |         return xs_vm_uuid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def set_or_delete(d, key, value): | 
					
						
							|  |  |  |     if value is None: | 
					
						
							|  |  |  |         if key in d: | 
					
						
							|  |  |  |             del d[key] | 
					
						
							|  |  |  |             return True | 
					
						
							| 
									
										
										
										
											2011-06-22 13:42:52 -07:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |         if d.get(key) != value: | 
					
						
							|  |  |  |             d[key] = value | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |     return False | 
					
						
							| 
									
										
										
										
											2011-06-22 13:42:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def set_external_id(row, key, value): | 
					
						
							|  |  |  |     external_ids = row.external_ids | 
					
						
							|  |  |  |     if set_or_delete(external_ids, key, value): | 
					
						
							|  |  |  |         row.external_ids = external_ids | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-11 12:35:37 -08:00
										 |  |  | # XenServer does not call interface-reconfigure on internal networks, | 
					
						
							|  |  |  | # which is where the fail-mode would normally be set. | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def update_fail_mode(row): | 
					
						
							|  |  |  |     rec = get_network_by_bridge(row.name) | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  |     if not rec: | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fail_mode = rec['other_config'].get('vswitch-controller-fail-mode') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not fail_mode: | 
					
						
							|  |  |  |         pools = session.xenapi.pool.get_all() | 
					
						
							|  |  |  |         if len(pools) == 1: | 
					
						
							|  |  |  |             prec = session.xenapi.pool.get_record(pools[0]) | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  |             fail_mode = prec['other_config'].get( | 
					
						
							|  |  |  |                     'vswitch-controller-fail-mode') | 
					
						
							| 
									
										
										
										
											2011-02-08 18:55:17 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if fail_mode not in ['standalone', 'secure']: | 
					
						
							|  |  |  |         fail_mode = 'standalone' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |     if row.fail_mode != fail_mode: | 
					
						
							|  |  |  |         row.fail_mode = fail_mode | 
					
						
							| 
									
										
										
										
											2011-02-28 18:26:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def update_in_band_mgmt(row): | 
					
						
							|  |  |  |     rec = get_network_by_bridge(row.name) | 
					
						
							| 
									
										
										
										
											2011-02-28 18:26:04 -08:00
										 |  |  |     if not rec: | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dib = rec['other_config'].get('vswitch-disable-in-band') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |     other_config = row.other_config | 
					
						
							|  |  |  |     if dib and dib not in ['true', 'false']: | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |         vlog.warn('"%s" isn\'t a valid setting for ' | 
					
						
							|  |  |  |                 "other_config:disable-in-band on %s" % (dib, row.name)) | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |     elif set_or_delete(other_config, 'disable-in-band', dib): | 
					
						
							|  |  |  |         row.other_config = other_config | 
					
						
							| 
									
										
										
										
											2010-09-21 23:57:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def keep_table_columns(schema, table_name, columns): | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |     table = schema.tables.get(table_name) | 
					
						
							|  |  |  |     if not table: | 
					
						
							|  |  |  |         raise error.Error("schema has no %s table" % table_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     new_columns = {} | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |     for column_name in columns: | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |         column = table.columns.get(column_name) | 
					
						
							|  |  |  |         if not column: | 
					
						
							|  |  |  |             raise error.Error("%s table schema lacks %s column" | 
					
						
							|  |  |  |                               % (table_name, column_name)) | 
					
						
							|  |  |  |         new_columns[column_name] = column | 
					
						
							|  |  |  |     table.columns = new_columns | 
					
						
							|  |  |  |     return table | 
					
						
							| 
									
										
										
										
											2010-09-21 18:03:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def prune_schema(schema): | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |     new_tables = {} | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |     new_tables["Bridge"] = keep_table_columns( | 
					
						
							|  |  |  |         schema, "Bridge", ("name", "external_ids", "other_config", | 
					
						
							|  |  |  |                            "fail_mode")) | 
					
						
							|  |  |  |     new_tables["Interface"] = keep_table_columns( | 
					
						
							|  |  |  |         schema, "Interface", ("name", "external_ids")) | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |     schema.tables = new_tables | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | def handler(signum, _): | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  |     global force_run | 
					
						
							|  |  |  |     if (signum == signal.SIGHUP): | 
					
						
							|  |  |  |         force_run = True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  | def main(): | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  |     global force_run | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  |     parser = argparse.ArgumentParser() | 
					
						
							|  |  |  |     parser.add_argument("database", metavar="DATABASE", | 
					
						
							|  |  |  |             help="A socket on which ovsdb-server is listening.") | 
					
						
							| 
									
										
										
										
											2012-03-06 11:16:38 -08:00
										 |  |  |     parser.add_argument("--root-prefix", metavar="DIR", default='', | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  |                         help="Use DIR as alternate root directory" | 
					
						
							|  |  |  |                         " (for testing).") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |     ovs.vlog.add_args(parser) | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  |     ovs.daemon.add_args(parser) | 
					
						
							|  |  |  |     args = parser.parse_args() | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |     ovs.vlog.handle_args(args) | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  |     ovs.daemon.handle_args(args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     remote = args.database | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |     schema_file = "%s/vswitch.ovsschema" % ovs.dirs.PKGDATADIR | 
					
						
							|  |  |  |     schema = ovs.db.schema.DbSchema.from_json(ovs.json.from_file(schema_file)) | 
					
						
							|  |  |  |     prune_schema(schema) | 
					
						
							|  |  |  |     idl = ovs.db.idl.Idl(remote, schema) | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ovs.daemon.daemonize() | 
					
						
							| 
									
										
										
										
											2010-09-10 14:20:49 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # This daemon is usually started before XAPI, but to complete our | 
					
						
							|  |  |  |     # tasks, we need it.  Wait here until it's up. | 
					
						
							| 
									
										
										
										
											2012-03-06 11:16:38 -08:00
										 |  |  |     cookie_file = args.root_prefix + "/var/run/xapi_init_complete.cookie" | 
					
						
							| 
									
										
										
										
											2011-09-26 13:07:29 -07:00
										 |  |  |     while not os.path.exists(cookie_file): | 
					
						
							| 
									
										
										
										
											2010-09-10 14:20:49 -07:00
										 |  |  |         time.sleep(1) | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     signal.signal(signal.SIGHUP, handler) | 
					
						
							| 
									
										
										
										
											2010-09-21 18:03:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 15:24:32 -08:00
										 |  |  |     bridges = {}                # Map from bridge name to nicira-bridge-id | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  |     iface_ids = {}              # Map from xs-vif-uuid to iface-id | 
					
						
							| 
									
										
										
										
											2012-03-05 10:20:00 -08:00
										 |  |  |     vm_ids = {}                 # Map from xs-vm-uuid to vm-id | 
					
						
							| 
									
										
										
										
											2012-03-08 10:49:47 -08:00
										 |  |  |     seqno = idl.change_seqno    # Sequence number when we last processed the db | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |     while True: | 
					
						
							| 
									
										
										
										
											2012-03-08 10:49:47 -08:00
										 |  |  |         idl.run() | 
					
						
							|  |  |  |         if not force_run and seqno == idl.change_seqno: | 
					
						
							| 
									
										
										
										
											2010-08-26 09:38:52 -07:00
										 |  |  |             poller = ovs.poller.Poller() | 
					
						
							|  |  |  |             idl.wait(poller) | 
					
						
							|  |  |  |             poller.block() | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |             continue | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if force_run: | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |             vlog.info("Forced to re-run as the result of a SIGHUP") | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  |             bridges = {} | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  |             iface_ids = {} | 
					
						
							| 
									
										
										
										
											2012-03-05 10:20:00 -08:00
										 |  |  |             vm_ids = {} | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  |             force_run = False | 
					
						
							| 
									
										
										
										
											2012-03-08 10:49:47 -08:00
										 |  |  |         seqno = idl.change_seqno | 
					
						
							| 
									
										
										
										
											2010-09-17 15:07:02 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |         txn = ovs.db.idl.Transaction(idl) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  |         new_bridges = {} | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |         for row in idl.tables["Bridge"].rows.itervalues(): | 
					
						
							| 
									
										
										
										
											2012-03-02 15:24:32 -08:00
										 |  |  |             if row.name in bridges: | 
					
						
							|  |  |  |                 nbd = bridges[row.name] | 
					
						
							|  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |                 # New bridge. | 
					
						
							|  |  |  |                 update_fail_mode(row) | 
					
						
							|  |  |  |                 update_in_band_mgmt(row) | 
					
						
							| 
									
										
										
										
											2012-03-02 15:24:32 -08:00
										 |  |  |                 nbd = get_bridge_id(row.name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             bridge_id = nbd | 
					
						
							|  |  |  |             if bridge_id is None: | 
					
						
							|  |  |  |                 bridge_id = row.external_ids.get("xs-network-uuids") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if bridge_id is not None: | 
					
						
							|  |  |  |                 set_external_id(row, "bridge-id", bridge_id.split(";")[0]) | 
					
						
							| 
									
										
										
										
											2012-02-29 18:41:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 15:24:32 -08:00
										 |  |  |             new_bridges[row.name] = nbd | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |         bridges = new_bridges | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         iface_by_name = {} | 
					
						
							|  |  |  |         for row in idl.tables["Interface"].rows.itervalues(): | 
					
						
							|  |  |  |             iface_by_name[row.name] = row | 
					
						
							| 
									
										
										
										
											2010-09-21 18:03:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  |         new_iface_ids = {} | 
					
						
							| 
									
										
										
										
											2012-03-05 10:20:00 -08:00
										 |  |  |         new_vm_ids = {} | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |         for row in idl.tables["Interface"].rows.itervalues(): | 
					
						
							|  |  |  |             # Match up paired vif and tap devices. | 
					
						
							|  |  |  |             if row.name.startswith("vif"): | 
					
						
							|  |  |  |                 vif = row | 
					
						
							|  |  |  |                 tap = iface_by_name.get("tap%s" % row.name[3:]) | 
					
						
							|  |  |  |             elif row.name.startswith("tap"): | 
					
						
							|  |  |  |                 tap = row | 
					
						
							|  |  |  |                 vif = iface_by_name.get("vif%s" % row.name[3:]) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 tap = vif = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Several tap external-ids need to be copied from the vif. | 
					
						
							|  |  |  |             if row == tap and vif: | 
					
						
							|  |  |  |                 keys = ["attached-mac", | 
					
						
							|  |  |  |                         "xs-network-uuid", | 
					
						
							|  |  |  |                         "xs-vif-uuid", | 
					
						
							|  |  |  |                         "xs-vm-uuid"] | 
					
						
							|  |  |  |                 for k in keys: | 
					
						
							|  |  |  |                     set_external_id(row, k, vif.external_ids.get(k)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  |             # Map from xs-vif-uuid to iface-id. | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |             # | 
					
						
							|  |  |  |             # (A tap's xs-vif-uuid comes from its vif.  That falls out | 
					
						
							|  |  |  |             # naturally from the copy loop above.) | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  |             xvu = row.external_ids.get("xs-vif-uuid") | 
					
						
							|  |  |  |             if xvu: | 
					
						
							|  |  |  |                 iface_id = (new_iface_ids.get(xvu) | 
					
						
							|  |  |  |                             or iface_ids.get(xvu) | 
					
						
							|  |  |  |                             or get_iface_id(row.name, xvu)) | 
					
						
							|  |  |  |                 new_iface_ids[xvu] = iface_id | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 # No xs-vif-uuid therefore no iface-id. | 
					
						
							|  |  |  |                 iface_id = None | 
					
						
							|  |  |  |             set_external_id(row, "iface-id", iface_id) | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 10:20:00 -08:00
										 |  |  |             # Map from xs-vm-uuid to vm-id. | 
					
						
							|  |  |  |             xvmu = row.external_ids.get("xs-vm-uuid") | 
					
						
							|  |  |  |             if xvmu: | 
					
						
							|  |  |  |                 vm_id = (new_vm_ids.get(xvmu) | 
					
						
							|  |  |  |                          or vm_ids.get(xvmu) | 
					
						
							|  |  |  |                          or get_vm_id(row.name, xvmu)) | 
					
						
							|  |  |  |                 new_vm_ids[xvmu] = vm_id | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 vm_id = None | 
					
						
							|  |  |  |             set_external_id(row, "vm-id", vm_id) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  |             # When there's a vif and a tap, the tap is active (used for | 
					
						
							|  |  |  |             # traffic).  When there's just a vif, the vif is active. | 
					
						
							|  |  |  |             # | 
					
						
							|  |  |  |             # A tap on its own shouldn't happen, and we don't know | 
					
						
							|  |  |  |             # anything about other kinds of devices, so we don't use | 
					
						
							|  |  |  |             # an iface-status for those devices at all. | 
					
						
							|  |  |  |             if vif and tap: | 
					
						
							|  |  |  |                 set_external_id(tap, "iface-status", "active") | 
					
						
							|  |  |  |                 set_external_id(vif, "iface-status", "inactive") | 
					
						
							|  |  |  |             elif vif: | 
					
						
							|  |  |  |                 set_external_id(vif, "iface-status", "active") | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 set_external_id(row, "iface-status", None) | 
					
						
							| 
									
										
										
										
											2012-02-01 17:18:52 -08:00
										 |  |  |         iface_ids = new_iface_ids | 
					
						
							| 
									
										
										
										
											2012-03-05 10:20:00 -08:00
										 |  |  |         vm_ids = new_vm_ids | 
					
						
							| 
									
										
										
										
											2011-09-21 10:43:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         txn.commit_block() | 
					
						
							| 
									
										
										
										
											2010-09-21 18:03:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-23 17:18:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-24 14:50:06 -07:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     try: | 
					
						
							| 
									
										
										
										
											2011-09-26 16:02:26 -07:00
										 |  |  |         main() | 
					
						
							| 
									
										
										
										
											2010-09-22 15:38:17 -07:00
										 |  |  |     except SystemExit: | 
					
						
							|  |  |  |         # Let system.exit() calls complete normally | 
					
						
							|  |  |  |         raise | 
					
						
							|  |  |  |     except: | 
					
						
							| 
									
										
										
										
											2011-09-25 15:40:10 -07:00
										 |  |  |         vlog.exception("traceback") | 
					
						
							| 
									
										
										
										
											2010-09-22 12:40:39 -07:00
										 |  |  |         sys.exit(ovs.daemon.RESTART_EXIT_CODE) |