poezio/src/config.py

281 lines
9.3 KiB
Python
Raw Normal View History

2011-09-06 00:45:53 +00:00
# Copyright 2010-2011 Florent Le Coz <louiz@louiz.org>
2010-01-10 20:14:17 +00:00
#
# This file is part of Poezio.
#
# Poezio is free software: you can redistribute it and/or modify
2011-09-11 15:10:05 +00:00
# it under the terms of the zlib license. See the COPYING file.
2010-01-10 20:14:17 +00:00
2010-05-18 13:29:02 +00:00
"""
Defines the global config instance, used to get or set (and save) values
from/to the config file
"""
DEFSECTION = "Poezio"
from gettext import gettext as _
import sys
import logging
log = logging.getLogger(__name__)
from configparser import RawConfigParser, NoOptionError, NoSectionError
2010-07-08 20:15:02 +00:00
from os import environ, makedirs, path
2010-01-31 05:33:54 +00:00
from shutil import copy2
from args import parse_args
from common import safeJID
2010-01-10 20:14:17 +00:00
2012-12-15 20:51:14 +00:00
2010-01-10 20:14:17 +00:00
class Config(RawConfigParser):
"""
load/save the config to a file
"""
def __init__(self, file_name):
RawConfigParser.__init__(self, None)
# make the options case sensitive
self.optionxform = str
self.read_file(file_name)
def read_file(self, file_name):
self.file_name = file_name
2012-12-15 20:51:14 +00:00
try:
RawConfigParser.read(self, file_name, encoding='utf-8')
except TypeError: # python < 3.2 sucks
RawConfigParser.read(self, file_name)
# Check config integrity and fix it if its wrong
for section in ('bindings', 'var'):
if not self.has_section(section):
self.add_section(section)
2010-01-10 20:14:17 +00:00
def get(self, option, default, section=DEFSECTION):
"""
get a value from the config but return
a default value if it is not found
The type of default defines the type
returned
"""
2010-05-18 13:29:02 +00:00
try:
if type(default) == int:
res = self.getint(option, section)
elif type(default) == float:
res = self.getfloat(option, section)
elif type(default) == bool:
res = self.getboolean(option, section)
else:
res = self.getstr(option, section)
except (NoOptionError, NoSectionError):
return default
return res
def getl(self, option, default, section=DEFSECTION):
"""
get a value and return it lowercase
"""
return self.get(option, default, section).lower()
def get_by_tabname(self, option, default, tabname, fallback=True, fallback_server=True):
"""
Try to get the value for the option. First we look in
a section named `tabname`, if the option is not present
in the section, we search for the global option if fallback is
True. And we return `default` as a fallback as a last resort.
"""
if tabname in self.sections():
if option in self.options(tabname):
# We go the tab-specific option
return self.get(option, default, tabname)
if fallback_server:
return self.get_by_servname(tabname, option, default, fallback)
if fallback:
# We fallback to the global option
return self.get(option, default)
return default
def get_by_servname(self, jid, option, default, fallback=True):
"""
Try to get the value of an option for a server
"""
server = safeJID(jid).server
if server:
server = '@' + server
if server in self.sections() and option in self.options(server):
return self.get(option, default, server)
if fallback:
return self.get(option, default)
return default
def __get(self, option, section=DEFSECTION):
2010-05-18 13:29:02 +00:00
"""
facility for RawConfigParser.get
"""
return RawConfigParser.get(self, section, option)
2010-01-10 20:14:17 +00:00
def getstr(self, option, section=DEFSECTION):
2010-05-18 13:29:02 +00:00
"""
get a value and returns it as a string
"""
return self.__get(option, section)
2010-01-26 17:10:37 +00:00
def getint(self, option, section=DEFSECTION):
2010-05-18 13:29:02 +00:00
"""
get a value and returns it as an int
"""
try:
return int(self.__get(option, section))
except ValueError:
return -1
2010-01-10 20:14:17 +00:00
def getfloat(self, option, section=DEFSECTION):
2010-05-18 13:29:02 +00:00
"""
get a value and returns it as a float
"""
return float(self.__get(option, section))
2010-01-10 20:14:17 +00:00
def getboolean(self, option, section=DEFSECTION):
2010-05-18 13:29:02 +00:00
"""
get a value and returns it as a boolean
"""
return RawConfigParser.getboolean(self, section, option)
2010-01-10 20:14:17 +00:00
def write_in_file(self, section, option, value):
"""
Our own way to save write the value in the file
Just find the right section, and then find the
right option, and edit it.
TODO: make it write also new values in the file, not just what did already
exist
"""
if path.exists(self.file_name):
df = open(self.file_name, 'r', encoding='utf-8')
lines_before = (line.strip() for line in df.readlines())
df.close()
else:
lines_before = []
result_lines = []
we_are_in_the_right_section = False
written = False
section_found = False
for line in lines_before:
if line.startswith('['): # check the section
if we_are_in_the_right_section and not written:
result_lines.append('%s = %s' % (option, value))
2012-03-08 21:38:26 +00:00
written = True
if line == '[%s]' % section:
we_are_in_the_right_section = True
section_found = True
else:
we_are_in_the_right_section = False
if (line.startswith('%s ' % (option,)) or
line.startswith('%s=' % (option,)) or
line.startswith('%s = ' % (option,))) and we_are_in_the_right_section:
line = '%s = %s' % (option, value)
written = True
result_lines.append(line)
if not section_found:
result_lines.append('[%s]' % section)
result_lines.append('%s = %s' % (option, value))
elif not written:
result_lines.append('%s = %s' % (option, value))
try:
df = open(self.file_name, 'w', encoding='utf-8')
for line in result_lines:
df.write('%s\n' % line)
df.close()
except:
success = False
else:
success = True
return success
2010-01-10 20:14:17 +00:00
def set_and_save(self, option, value, section=DEFSECTION):
2010-05-18 13:29:02 +00:00
"""
set the value in the configuration then save it
to the file
"""
# Special case for a 'toggle' value. We take the current value
# and set the opposite. Warning if the no current value exists
# or it is not a bool.
if value == "toggle":
current = self.get(option, "", section)
if current.lower() == "false":
value = "true"
elif current.lower() == "true":
value = "false"
else:
return (_("Could not toggle option: %s. Current value is %s.") % (option, current or _("empty")), 'Warning')
if self.has_section(section):
RawConfigParser.set(self, section, option, value)
else:
self.add_section(section)
2010-12-17 14:48:31 +00:00
RawConfigParser.set(self, section, option, value)
if not self.write_in_file(section, option, value):
return (_('Unable to write in the config file'), 'Error')
return ("%s=%s" % (option, value), 'Info')
def silent_set(self, option, value, section=DEFSECTION):
"""
Set a value, save, and return True on success and False on failure
"""
if self.has_section(section):
RawConfigParser.set(self, section, option, value)
else:
self.add_section(section)
RawConfigParser.set(self, section, option, value)
return self.write_in_file(section, option, value)
2011-09-10 14:41:47 +00:00
def set(self, option, value, section=DEFSECTION):
"""
Set the value of an option temporarily
"""
try:
RawConfigParser.set(self, section, option, value)
except NoSectionError:
pass
def to_dict(self):
"""
Returns a dict of the form {section: {option: value, option: value}, }
"""
res = {}
for section in self.sections():
res[section] = {}
for option in self.options(section):
res[section][option] = self.get(option, "", section)
return res
2011-09-10 14:41:47 +00:00
firstrun = False
2010-05-18 13:29:02 +00:00
# creates the configuration directory if it doesn't exist
# and copy the default config in it
2010-01-31 05:33:54 +00:00
CONFIG_HOME = environ.get("XDG_CONFIG_HOME")
if not CONFIG_HOME:
CONFIG_HOME = path.join(environ.get('HOME'), '.config')
CONFIG_PATH = path.join(CONFIG_HOME, 'poezio')
2010-01-31 14:28:21 +00:00
try:
makedirs(CONFIG_PATH, exist_ok=True)
2010-05-18 13:29:02 +00:00
except OSError:
pass
2010-01-31 05:33:54 +00:00
options = parse_args(CONFIG_PATH)
# Copy a default file if none exists
if not path.isfile(options.filename):
default = path.join(path.dirname(__file__), '../data/default_config.cfg')
other = path.join(path.dirname(__file__), 'default_config.cfg')
if path.isfile(default):
copy2(default, options.filename)
elif path.isfile(other):
copy2(other, options.filename)
2011-09-10 14:41:47 +00:00
firstrun = True
2010-07-08 20:15:02 +00:00
try:
config = Config(options.filename)
except:
import traceback
sys.stderr.write('Poezio was unable to read or parse the config file.\n')
traceback.print_exc(limit=0)
sys.exit(1)