mirror of
				https://github.com/openvswitch/ovs
				synced 2025-10-23 14:57:06 +00:00 
			
		
		
		
	Starting in XenServer 5.6.0, a "vswitch_controller" key is available to
store the controller's IP address in the "pool" table of XAPI.  Older
versions must still use the "vSwitchController" key in "other_config".
Based on commits 37fee7 and 0ebd737 from the xs5.7 branch written by
Ian Campbell.
		
	
		
			
				
	
	
		
			345 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (c) Citrix Systems 2008. All rights reserved.
 | |
| # xsconsole is proprietary software.
 | |
| #
 | |
| # Xen, the Xen logo, XenCenter, XenMotion are trademarks or registered
 | |
| # trademarks of Citrix Systems, Inc., in the United States and other
 | |
| # countries.
 | |
| 
 | |
| # Copyright (c) 2009, 2010 Nicira Networks.
 | |
| 
 | |
| from XSConsoleLog import *
 | |
| 
 | |
| import os
 | |
| import socket
 | |
| import subprocess
 | |
| 
 | |
| vsctl="/usr/bin/ovs-vsctl"
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     raise Exception("This script is a plugin for xsconsole and cannot run independently")
 | |
| 
 | |
| from XSConsoleStandard import *
 | |
| 
 | |
| class VSwitchService:
 | |
|     service = {}
 | |
| 
 | |
|     def __init__(self, name, processname=None):
 | |
|         self.name = name
 | |
|         self.processname = processname
 | |
|         if self.processname == None:
 | |
|             self.processname = name
 | |
| 
 | |
|     def version(self):
 | |
|         try:
 | |
|             output = ShellPipe(["service", self.name, "version"]).Stdout()
 | |
|         except StandardError, e:
 | |
|             XSLogError("vswitch version retrieval error: " + str(e))
 | |
|             return "<unknown>"
 | |
|         for line in output:
 | |
|             if self.processname in line:
 | |
|                 return line.split()[-1]
 | |
|         return "<unknown>"
 | |
| 
 | |
|     def status(self):
 | |
|         try:
 | |
|             output = ShellPipe(["service", self.name, "status"]).Stdout()
 | |
|         except StandardError, e:
 | |
|             XSLogError("vswitch status retrieval error: " + str(e))
 | |
|             return "<unknown>"
 | |
|         if len(output) == 0:
 | |
|             return "<unknown>"
 | |
|         for line in output:
 | |
|             if self.processname not in line:
 | |
|                 continue
 | |
|             elif "running" in line:
 | |
|                 return "Running"
 | |
|             elif "stop" in line:
 | |
|                 return "Stopped"
 | |
|             else:
 | |
|                 return "<unknown>"
 | |
|         return "<unknown>"
 | |
| 
 | |
|     def restart(self):
 | |
|         try:
 | |
|             ShellPipe(["service", self.name, "restart"]).Call()
 | |
|         except StandardError, e:
 | |
|             XSLogError("vswitch restart error: " + str(e))
 | |
| 
 | |
|     @classmethod
 | |
|     def Inst(cls, name, processname=None):
 | |
|         key = name
 | |
|         if processname != None:
 | |
|             key = key + "-" + processname
 | |
|         if name not in cls.service:
 | |
|             cls.service[key] = VSwitchService(name, processname)
 | |
|         return cls.service[key]
 | |
| 
 | |
| class VSwitchConfig:
 | |
| 
 | |
|     @staticmethod
 | |
|     def Get(action):
 | |
|         try:
 | |
|             arg = [vsctl, "-vANY:console:emer"] + action.split()
 | |
|             output = ShellPipe(arg).Stdout()
 | |
|         except StandardError, e:
 | |
|             XSLogError("config retrieval error: " + str(e))
 | |
|             return "<unknown>"
 | |
| 
 | |
|         if len(output) == 0:
 | |
|             output = ""
 | |
|         else:
 | |
|             output = output[0].strip()
 | |
|         return output
 | |
| 
 | |
| 
 | |
| class VSwitchControllerDialogue(Dialogue):
 | |
|     def __init__(self):
 | |
|         Dialogue.__init__(self)
 | |
|         data=Data.Inst()
 | |
| 
 | |
|         self.hostsInPool = 0
 | |
|         self.hostsUpdated = 0
 | |
|         self.xs_version = data.host.software_version.product_version('')
 | |
|         pool = data.GetPoolForThisHost()
 | |
|         if pool is not None:
 | |
|             if self.xs_version == "5.5.0":
 | |
|                 self.controller = pool.get("other_config", {}).get("vSwitchController", "")
 | |
|             else:
 | |
|                 self.controller = pool.get("vswitch_controller", "")
 | |
|         else:
 | |
|             self.controller = ""
 | |
| 
 | |
|         choiceDefs = [
 | |
|             ChoiceDef(Lang("Set pool-wide controller"),
 | |
|                       lambda: self.getController()),
 | |
|             ChoiceDef(Lang("Delete pool-wide controller"),
 | |
|                       lambda: self.deleteController()),
 | |
|             ChoiceDef(Lang("Resync server controller config"),
 | |
|                       lambda: self.syncController()),
 | |
| #             ChoiceDef(Lang("Restart ovs-vswitchd"),
 | |
| #                       lambda: self.restartService("vswitch")),
 | |
| #             ChoiceDef(Lang("Restart ovs-brcompatd"),
 | |
| #                       lambda: self.restartService("vswitch-brcompatd"))
 | |
|             ]
 | |
|         self.menu = Menu(self, None, Lang("Configure Open vSwitch"), choiceDefs)
 | |
| 
 | |
|         self.ChangeState("INITIAL")
 | |
| 
 | |
|     def BuildPane(self):
 | |
|         pane = self.NewPane(DialoguePane(self.parent))
 | |
|         pane.TitleSet(Lang("Configure Open vSwitch"))
 | |
|         pane.AddBox()
 | |
| 
 | |
|     def ChangeState(self, inState):
 | |
|         self.state = inState
 | |
|         self.BuildPane()
 | |
|         self.UpdateFields()
 | |
| 
 | |
|     def UpdateFields(self):
 | |
|         self.Pane().ResetPosition()
 | |
|         getattr(self, "UpdateFields" + self.state)() # Dispatch method named 'UpdateFields'+self.state
 | |
| 
 | |
|     def UpdateFieldsINITIAL(self):
 | |
|         pane = self.Pane()
 | |
|         pane.AddTitleField(Lang("Select an action"))
 | |
|         pane.AddMenuField(self.menu)
 | |
|         pane.AddKeyHelpField( { Lang("<Enter>") : Lang("OK"), Lang("<Esc>") : Lang("Cancel") } )
 | |
| 
 | |
|     def UpdateFieldsGETCONTROLLER(self):
 | |
|         pane = self.Pane()
 | |
|         pane.ResetFields()
 | |
| 
 | |
|         pane.AddTitleField(Lang("Enter IP address of controller"))
 | |
|         pane.AddInputField(Lang("Address", 16), self.controller, "address")
 | |
|         pane.AddKeyHelpField( { Lang("<Enter>") : Lang("OK"), Lang("<Esc>") : Lang("Exit") } )
 | |
|         if pane.CurrentInput() is None:
 | |
|             pane.InputIndexSet(0)
 | |
| 
 | |
|     def HandleKey(self, inKey):
 | |
|         handled = False
 | |
|         if hasattr(self, "HandleKey" + self.state):
 | |
|             handled = getattr(self, "HandleKey" + self.state)(inKey)
 | |
|         if not handled and inKey == 'KEY_ESCAPE':
 | |
|             Layout.Inst().PopDialogue()
 | |
|             handled = True
 | |
|         return handled
 | |
| 
 | |
|     def HandleKeyINITIAL(self, inKey):
 | |
|         return self.menu.HandleKey(inKey)
 | |
| 
 | |
|     def HandleKeyGETCONTROLLER(self, inKey):
 | |
|         pane = self.Pane()
 | |
|         if pane.CurrentInput() is None:
 | |
|             pane.InputIndexSet(0)
 | |
|         if inKey == 'KEY_ENTER':
 | |
|             inputValues = pane.GetFieldValues()
 | |
|             self.controller = inputValues['address']
 | |
|             Layout.Inst().PopDialogue()
 | |
| 
 | |
|             # Make sure the controller is specified as a valid dotted quad
 | |
|             try:
 | |
|                 socket.inet_aton(self.controller)
 | |
|             except socket.error:
 | |
|                 Layout.Inst().PushDialogue(InfoDialogue(Lang("Please enter in dotted quad format")))
 | |
|                 return True
 | |
| 
 | |
|             Layout.Inst().TransientBanner(Lang("Setting controller..."))
 | |
|             try:
 | |
|                 self.SetController(self.controller)
 | |
|                 Layout.Inst().PushDialogue(InfoDialogue(Lang("Setting controller successful")))
 | |
|             except Exception, e:
 | |
|                 Layout.Inst().PushDialogue(InfoDialogue(Lang("Setting controller failed")))
 | |
| 
 | |
|             self.ChangeState("INITIAL")
 | |
|             return True
 | |
|         else:
 | |
|             return pane.CurrentInput().HandleKey(inKey)
 | |
| 
 | |
|     def restartService(self, name):
 | |
|         s = VSwitchService.Inst(name)
 | |
|         s.restart()
 | |
|         Layout.Inst().PopDialogue()
 | |
| 
 | |
|     def getController(self):
 | |
|         self.ChangeState("GETCONTROLLER")
 | |
|         self.Pane().InputIndexSet(0)
 | |
| 
 | |
|     def deleteController(self):
 | |
|         self.controller = ""
 | |
|         Layout.Inst().PopDialogue()
 | |
|         Layout.Inst().TransientBanner(Lang("Deleting controller..."))
 | |
|         try:
 | |
|             self.SetController(None)
 | |
|             Layout.Inst().PushDialogue(InfoDialogue(Lang("Controller deletion successful")))
 | |
|         except Exception, e:
 | |
|             Layout.Inst().PushDialogue(InfoDialogue(Lang("Controller deletion failed")))
 | |
| 
 | |
|     def syncController(self):
 | |
|         Layout.Inst().PopDialogue()
 | |
|         Layout.Inst().TransientBanner(Lang("Resyncing controller setting..."))
 | |
|         try:
 | |
|             Task.Sync(lambda s: self._updateThisServer(s))
 | |
|             Layout.Inst().PushDialogue(InfoDialogue(Lang("Resyncing controller config successful")))
 | |
|         except Exception, e:
 | |
|             Layout.Inst().PushDialogue(InfoDialogue(Lang("Resyncing controller config failed")))
 | |
| 
 | |
|     def SetController(self, ip):
 | |
|         self.hostsInPool = 0
 | |
|         self.hostsUpdated = 0
 | |
|         Task.Sync(lambda s: self._modifyPoolConfig(s, ip or ""))
 | |
|         # Should be done asynchronously, maybe with an external script?
 | |
|         Task.Sync(lambda s: self._updateActiveServers(s))
 | |
| 
 | |
|     def _modifyPoolConfig(self, session, value):
 | |
|         """Modify pool configuration.
 | |
| 
 | |
|         If value == "" then delete configuration, otherwise set to value.
 | |
|         """
 | |
|         pools = session.xenapi.pool.get_all()
 | |
|         # We assume there is only ever one pool...
 | |
|         if len(pools) == 0:
 | |
|             XSLogFatal(Lang("No pool found for host."))
 | |
|             return
 | |
|         if len(pools) > 1:
 | |
|             XSLogFatal(Lang("More than one pool for host."))
 | |
|             return
 | |
|         if self.xs_version == "5.5.0":
 | |
|             key = "vSwitchController"
 | |
|             session.xenapi.pool.remove_from_other_config(pools[0], key)
 | |
|             if value != None:
 | |
|                 session.xenapi.pool.add_to_other_config(pools[0], key, value)
 | |
|         else:
 | |
|             session.xenapi.pool.set_vswitch_controller(value)
 | |
|         Data.Inst().Update()
 | |
| 
 | |
|     def _updateActiveServers(self, session):
 | |
|         hosts = session.xenapi.host.get_all()
 | |
|         self.hostsUpdated = 0
 | |
|         self.hostsInPool = len(hosts)
 | |
|         self.UpdateFields()
 | |
|         for host in hosts:
 | |
|             Layout.Inst().TransientBanner("Updating host %d out of %d" 
 | |
|                     % (self.hostsUpdated + 1, self.hostsInPool))
 | |
|             session.xenapi.host.call_plugin(host, "openvswitch-cfg-update", "update", {})
 | |
|             self.hostsUpdated = self.hostsUpdated + 1
 | |
| 
 | |
|     def _updateThisServer(self, session):
 | |
|         data = Data.Inst()
 | |
|         host = data.host.opaqueref()
 | |
|         session.xenapi.host.call_plugin(host, "openvswitch-cfg-update", "update", {})
 | |
| 
 | |
| 
 | |
| class XSFeatureVSwitch:
 | |
| 
 | |
|     @classmethod
 | |
|     def StatusUpdateHandler(cls, inPane):
 | |
|         data = Data.Inst()
 | |
|         xs_version = data.host.software_version.product_version('')
 | |
| 
 | |
|         inPane.AddTitleField(Lang("Open vSwitch"))
 | |
| 
 | |
|         inPane.NewLine()
 | |
| 
 | |
|         inPane.AddStatusField(Lang("Version", 20),
 | |
|                               VSwitchService.Inst("openvswitch", "ovs-vswitchd").version())
 | |
| 
 | |
|         inPane.NewLine()
 | |
| 
 | |
|         pool = data.GetPoolForThisHost()
 | |
|         if pool is not None:
 | |
|             if (xs_version == "5.5.0"):
 | |
|                 dbController = pool.get("other_config", {}).get("vSwitchController", "")
 | |
|             else:
 | |
|                 dbController = pool.get("vswitch_controller", "")
 | |
|         else:
 | |
|             dbController = ""
 | |
| 
 | |
|         if dbController == "":
 | |
|             dbController = Lang("<None>")
 | |
|         inPane.AddStatusField(Lang("Controller (config)", 20), dbController)
 | |
|         controller = VSwitchConfig.Get("get Open_vSwitch . managers")
 | |
|         controller = controller.strip('[]"')
 | |
| 
 | |
|         if controller == "":
 | |
|             controller = Lang("<None>")
 | |
|         elif controller[0:4] == "ssl:":
 | |
|             controller = controller.split(':')[1]
 | |
|         inPane.AddStatusField(Lang("Controller (in-use)", 20), controller)
 | |
| 
 | |
|         inPane.NewLine()
 | |
|         inPane.AddStatusField(Lang("ovs-vswitchd status", 20),
 | |
|                               VSwitchService.Inst("openvswitch", "ovs-vswitchd").status())
 | |
|         inPane.AddStatusField(Lang("ovsdb-server status", 20),
 | |
|                               VSwitchService.Inst("openvswitch", "ovsdb-server").status())
 | |
| 
 | |
|         # Only XenServer 5.5.0 runs ovs-brcompatd
 | |
|         if (xs_version == "5.5.0"):
 | |
|             inPane.AddStatusField(Lang("ovs-brcompatd status", 20),
 | |
|                    VSwitchService.Inst("openvswitch", "ovs-brcompatd").status())
 | |
| 
 | |
|         inPane.AddKeyHelpField( {
 | |
|             Lang("<Enter>") : Lang("Reconfigure"),
 | |
|             Lang("<F5>") : Lang("Refresh")
 | |
|         })
 | |
| 
 | |
|     @classmethod
 | |
|     def ActivateHandler(cls):
 | |
|         DialogueUtils.AuthenticatedOnly(lambda: Layout.Inst().PushDialogue(VSwitchControllerDialogue()))
 | |
| 
 | |
|     def Register(self):
 | |
|         Importer.RegisterNamedPlugIn(
 | |
|             self,
 | |
|             'VSwitch', # Key of this plugin for replacement, etc.
 | |
|             {
 | |
|                 'menuname' : 'MENU_NETWORK',
 | |
|                 'menupriority' : 800,
 | |
|                 'menutext' : Lang('Open vSwitch') ,
 | |
|                 'statusupdatehandler' : self.StatusUpdateHandler,
 | |
|                 'activatehandler' : self.ActivateHandler
 | |
|             }
 | |
|         )
 | |
| 
 | |
| # Register this plugin when module is imported, IFF vswitchd is running
 | |
| if os.path.exists('/var/run/openvswitch/ovs-vswitchd.pid'):
 | |
|     XSFeatureVSwitch().Register()
 |