Merge pull request #13 from vishwabmc/master
Enabling updates to /etc/resolve.conf either manually or using DHCP supplied DNS IP
diff --git a/logman.py b/logman.py
new file mode 100755
index 0000000..05ca191
--- /dev/null
+++ b/logman.py
@@ -0,0 +1,266 @@
+#!/usr/bin/env python
+
+from subprocess import call
+import sys
+import subprocess
+import dbus
+import string
+import socket
+import os
+import fcntl
+import time
+import glib
+import gobject
+import dbus.service
+import dbus.mainloop.glib
+
+DBUS_NAME = 'org.openbmc.LogManager'
+ERRL_INTF_NAME = 'org.openbmc.Errl'
+SRVC_INTF_NAME = 'org.openbmc.Service'
+OBJ_NAME_RSYSLOG = '/org/openbmc/LogManager/rsyslog'
+
+'''
+ Object Path > /org/openbmc/LogManager/rsyslog
+ Interface:Method > org.openbmc.Service.Enable dict:string:string
+ Interface:Method > org.openbmc.Service.Disable
+'''
+
+class JournalUtils ():
+ def _isvalidip(self, family, ipaddr):
+ if family == socket.AF_INET:
+ try:
+ socket.inet_pton(socket.AF_INET, ipaddr)
+ except AttributeError: # no inet_pton here, sorry
+ try:
+ socket.inet_aton(ipaddr)
+ except socket.error:
+ return False
+ return ipaddr.count('.') == 3
+ except socket.error: # not a valid address
+ return False
+
+ return True
+
+ elif family == socket.AF_INET6:
+ try:
+ socket.inet_pton(socket.AF_INET6, ipaddr)
+ except socket.error: # not a valid address
+ return False
+ return True
+
+ else: return False
+
+class Rsyslog (dbus.service.Object):
+ def __init__(self, bus, name):
+ self.bus = bus
+ self.name = name
+ dbus.service.Object.__init__(self,bus,name)
+
+ @dbus.service.method(dbus.PROPERTIES_IFACE, "ss", "v")
+ def Get(self, iface, ppty):
+ return self.GetAll(iface)[ppty]
+
+ @dbus.service.method(dbus.PROPERTIES_IFACE, 's', 'a{sv}')
+ def GetAll(self, iface):
+ if iface == ERRL_INTF_NAME:
+ status, remote_ip, remote_port = self.Status()
+ return {'status': status, 'ipaddr': remote_ip, 'port': remote_port }
+ else:
+ raise dbus.exceptions.DBusException('org.openbmc.UnknownInterface',
+ 'This object does not implement the %s interface' % iface)
+
+ @dbus.service.method(SRVC_INTF_NAME, "a{sv}", "x")
+ def Enable (self, argv_dict):
+ remote_ip = ""
+ remote_port = 0
+
+ params = len (argv_dict)
+ if params > 2 : ValueError("Invalid Parameters")
+
+ for property_name in argv_dict:
+ if property_name == "ipaddr":
+ remote_ip = argv_dict [property_name]
+ elif property_name == "port":
+ remote_port = argv_dict [property_name]
+ else:
+ raise ValueError("Invalid Argument: IP Address/Port expected.")
+
+ if not remote_ip:
+ cur_remote = self._GetConfig ('Remote')
+ if not cur_remote:
+ raise ValueError("Invalid Remote Syslog IP Address")
+ else:
+ cur_remote = cur_remote[3:]
+ remote_ip, port_str = cur_remote.split (":")
+ remote_port = int(port_str)
+ if not util._isvalidip (socket.AF_INET, remote_ip): raise ValueError, "Malformed IP Address"
+ if not remote_port : remote_port = 514
+ if remote_port > 65535 : raise ValueError("Invalid Remote Syslog Port")
+
+ remote_addr = remote_ip + ":" + str(remote_port)
+ r = self._ModifyService('Remote', remote_addr)
+
+ cur_options = self._GetConfig ('Options')
+ new_options = self._GetOptions()
+
+ if cur_options != new_options:
+ r = self._ModifyService('Options', new_options)
+ r = self._RestartService ()
+
+ return r
+
+ @dbus.service.method(SRVC_INTF_NAME, "as", "x")
+ def Disable (self, argv_list):
+ params = len (argv_list)
+ if params : ValueError("Invalid Parameters")
+
+ remote = self._GetConfig ('Remote')
+ if not remote : return 0
+
+ r = self._ModifyService('Options', '-C') # FIXME: Restore current options minus the remote.
+ r = self._RestartService ()
+ return r
+
+ def Status (self):
+ remote = self._GetConfig ('Remote')
+ if not remote : return ("Disabled", "0.0.0.0", 0)
+
+ cur_remote = remote[3:]
+ remote_ip, remote_port = cur_remote.split (":")
+
+ options = self._GetConfig ('Options')
+ if not options : return ("Disabled", remote_ip, remote_port)
+
+ if remote in options : return ("Enabled", remote_ip, remote_port)
+
+ return ("Disabled", remote_ip, remote_port)
+
+ def _ModifyService (self, opt, val):
+ if not os.path.isfile(syslog_service_bbx_file):
+ r = call (["cp", syslog_service_lib_file, syslog_service_bbx_file])
+ r = call (["ln", "-s", syslog_service_bbx_file, syslog_service_cfg_file])
+
+ if not os.path.isfile(syslog_service_env_file):
+ env_file = open(syslog_service_env_file, 'w')
+ env_file.write ("OPTIONS=\"-C\"")
+ env_file.close()
+
+ if opt not in OptionKeys: raise ValueError("Invalid Option")
+
+ self._ModifyParam (opt, val)
+
+ return 0
+
+ def _StopService (self):
+ r = call (["systemctl", "stop", "syslog"])
+ r = call (["systemctl", "--no-reload", "kill", "syslog"])
+ return r
+
+ def _StartService (self):
+ r = call (["systemctl", "daemon-reload"])
+ r = call (["systemctl", "start", "syslog"])
+ return r
+
+ def _RestartService (self):
+ r = self._StopService()
+ r = self._StartService()
+ return r
+
+ def _ModifyParam (self, opt, val):
+ env_file = open(syslog_service_env_file, 'r')
+ tmp_file = open(syslog_service_tmp_file, 'w')
+
+ optkey = OptionKeySwitchMap [opt]['key']
+ for line in env_file:
+ if line[0] == '#':
+ tmp_file.write(line)
+ continue
+ curkey = line.strip().split ("=")[0]
+ if curkey != optkey :
+ tmp_file.write(line)
+
+ tmp_file.write(optkey + "=\"" + OptionKeySwitchMap[opt]['switch'] + val + "\"" + "\n")
+
+ env_file.close ()
+ tmp_file.close ()
+
+ r = call (["cp", syslog_service_tmp_file, syslog_service_env_file])
+ return r
+
+ def _GetConfig (self, opt):
+ with open(syslog_service_env_file, "r") as f:
+ for line in f:
+ if line[0] == '#': continue
+ config = line.split ("=")
+ var = config [0]
+ if var == OptionKeySwitchMap[opt]['key']:
+ val = config [1]
+ val = val[1:-2] # FIXME: Why is there a trailing space ???
+ return val
+ return ""
+
+ def _GetOptions(self):
+ cfg = {}
+ i = 0
+
+ for opt in OptionKeys:
+ if opt == 'Options' : continue
+ cfg [i] = self._GetConfig(opt)
+ i+=1
+
+ options = ''
+ j = 0
+ while j<i-1:
+ if cfg[j] : options += cfg [j]
+ j+=1
+
+ return options
+
+def main():
+ dbus.mainloop.glib.DBusGMainLoop(set_as_default=True)
+ bus = dbus.SystemBus()
+ name = dbus.service.BusName(DBUS_NAME, bus)
+
+ global util
+ global rsys
+ global syslog_service_lib_file
+ global syslog_service_bbx_file
+ global syslog_service_cfg_file
+ global syslog_service_env_file
+ global syslog_service_tmp_file
+ global OptionKeys
+ global OptionKeySwitchMap
+
+ OptionKeys = ['Options', 'Outfile', 'Priority', 'Smaller', 'RotateSize', 'RotateNum', 'Remote', 'LocalAndNet', 'DropDup', 'SharedMem', 'ConfFile', 'MarkTime', 'Printk']
+ OptionKeySwitchMap = {
+ 'Options' : { 'switch' : "", 'key' : "OPTIONS" },
+ 'Outfile' : { 'switch' : "-O ", 'key' : "OBMC_SYSLOG_OUTFILE" },
+ 'Priority' : { 'switch' : "-O ", 'key' : "OBMC_SYSLOG_PRIORITY" },
+ 'Smaller' : { 'switch' : "-S ", 'key' : "OBMC_SYSLOG_SMALLER" },
+ 'RotateSize' : { 'switch' : "-s ", 'key' : "OBMC_SYSLOG_ROTATESIZE" },
+ 'RotateNum' : { 'switch' : "-b ", 'key' : "OBMC_SYSLOG_ROTATENUM" },
+ 'Remote' : { 'switch' : "-R ", 'key' : "OBMC_SYSLOG_REMOTE" },
+ 'LocalAndNet' : { 'switch' : "-L ", 'key' : "OBMC_SYSLOG_LOCALNET" },
+ 'DropDup' : { 'switch' : "-D ", 'key' : "OBMC_SYSLOG_DROPDUP" },
+ 'SharedMem' : { 'switch' : "-C ", 'key' : "OBMC_SYSLOG_SHAREDMEM" },
+ 'ConfFile' : { 'switch' : "-f ", 'key' : "OBMC_SYSLOG_CONFFILE" },
+ 'MarkTime' : { 'switch' : "-m ", 'key' : "OBMC_SYSLOG_MARKTIME" },
+ 'Printk' : { 'switch' : "-K ", 'key' : "OBMC_SYSLOG_PRINTK" }
+ }
+
+ syslog_service_lib_file = '/lib/systemd/system/busybox-syslog.service'
+ syslog_service_bbx_file = '/etc/systemd/system/busybox-syslog.service'
+ syslog_service_cfg_file = '/etc/systemd/system/syslog.service'
+ syslog_service_env_file = '/etc/default/busybox-syslog'
+ syslog_service_tmp_file = '/tmp/busybox-syslog.tmp'
+
+ util = JournalUtils ()
+ rsys = Rsyslog (bus, OBJ_NAME_RSYSLOG)
+
+ mainloop = gobject.MainLoop()
+ print("Started")
+ mainloop.run()
+
+if __name__ == '__main__':
+ sys.exit(main())
+