#!/usr/bin/python # # Copyright 2014 Cumulus Networks, Inc. All rights reserved. # Author: Roopa Prabhu, roopa@cumulusnetworks.com # try: from ipaddr import IPNetwork from sets import Set from ifupdown.iface import * from ifupdownaddons.modulebase import moduleBase from ifupdownaddons.iproute2 import iproute2 import os import glob import logging import signal import subprocess import re except ImportError, e: raise ImportError (str(e) + "- required module not found") class vrrpd(moduleBase): """ ifupdown2 addon module to configure vrrpd attributes """ _modinfo = {'mhelp' : 'ethtool configuration module for interfaces', 'attrs': { 'vrrp-id' : {'help' : 'vrrp instance id', 'example' : ['vrrp-id 1']}, 'vrrp-priority' : {'help': 'set vrrp priority', 'example' : ['vrrp-priority 20']}, 'vrrp-virtual-ip' : {'help': 'set vrrp virtual ip', 'example' : ['vrrp-virtual-ip 10.0.1.254']}}} def __init__(self, *args, **kargs): moduleBase.__init__(self, *args, **kargs) self.ipcmd = None def _check_if_process_is_running(self, cmdname, cmdline): targetpids = [] pidstr = '' try: pidstr = subprocess.check_output(['/bin/pidof', '%s' %cmdname]).strip('\n') except: pass if not pidstr: return [] pids = pidstr.split() if not pids: return targetpids for pid in pids: tmpcmdline = cmdline.replace(' ', '') try: pcmdline = self.read_file_oneline('/proc/%s/cmdline' %pid) pcmdline = re.sub(r'\\(.)', r'\1', pcmdline) self.logger.info('(%s)' %(pcmdline)) self.logger.info('(%s)' %(tmpcmdline)) self.logger.info('(%d) (%d)' %(len(pcmdline), len(tmpcmdline))) if pcmdline and pcmdline == tmpcmdline: targetpids.append(pid) except: pass return targetpids def _up(self, ifaceobj): """ up vrrpd -n -D -i $IFACE -v 1 -p 20 10.0.1.254 up ifplugd -i $IFACE -b -f -u0 -d1 -I -p -q """ if (not self.DRYRUN and not os.path.exists('/sys/class/net/%s' %ifaceobj.name)): return cmd = '' attrval = ifaceobj.get_attr_value_first('vrrp-id') if attrval: cmd += ' -v %s' %attrval else: return attrval = ifaceobj.get_attr_value_first('vrrp-priority') if attrval: cmd += ' -p %s' %attrval else: self.logger.warn('%s: incomplete vrrp parameters ' %ifaceobj.name, '(priority not found)') attrval = ifaceobj.get_attr_value_first('vrrp-virtual-ip') if attrval: cmd += ' %s' %attrval else: self.logger.warn('%s: incomplete vrrp arguments ' %ifaceobj.name, '(virtual ip not found)') return cmd = '/usr/sbin/vrrpd -n -D -i %s %s' %(ifaceobj.name, cmd) self.exec_command(cmd) cmd = '/usr/sbin/ifplugd -i %s -b -f -u0 -d1 -I -p -q' %ifaceobj.name if self._check_if_process_is_running('/usr/sbin/ifplugd', cmd): self.logger.info('%s: ifplugd already running' %ifaceobj.name) return self.exec_command(cmd) def _kill_pid_from_file(self, pidfilename): if os.path.exists(pidfilename): pid = self.read_file_oneline(pidfilename) if os.path.exists('/proc/%s' %pid): os.kill(int(pid), signal.SIGTERM) def _down(self, ifaceobj): """ down ifplugd -k -i $IFACE down kill $(cat /var/run/vrrpd_$IFACE_*.pid) """ attrval = ifaceobj.get_attr_value_first('vrrp-id') if not attrval: return try: self.exec_command('/usr/sbin/ifplugd -k -i %s' %ifaceobj.name) except Exception, e: self.logger.debug('%s: ifplugd down error (%s)' %(ifaceobj.name, str(e))) pass for pidfile in glob.glob('/var/run/vrrpd_%s_*.pid' %ifaceobj.name): try: self._kill_pid_from_file(pidfile) except Exception, e: self.logger.debug('%s: vrrpd down error (%s)' %(ifaceobj.name, str(e))) pass def _query_check(self, ifaceobj, ifaceobjcurr): # XXX return _run_ops = {'post-up' : _up, 'pre-down' : _down} def get_ops(self): """ returns list of ops supported by this module """ return self._run_ops.keys() def run(self, ifaceobj, operation, query_ifaceobj=None, **extra_args): """ run ethtool configuration on the interface object passed as argument Args: **ifaceobj** (object): iface object **operation** (str): any of 'post-up', 'query-checkcurr', 'query-running' Kwargs: **query_ifaceobj** (object): query check ifaceobject. This is only valid when op is 'query-checkcurr'. It is an object same as ifaceobj, but contains running attribute values and its config status. The modules can use it to return queried running state of interfaces. status is success if the running state is same as user required state in ifaceobj. error otherwise. """ op_handler = self._run_ops.get(operation) if not op_handler: return if operation == 'query-checkcurr': op_handler(self, ifaceobj, query_ifaceobj) else: op_handler(self, ifaceobj)