9d8ff1bacf
svn: r13347
709 lines
25 KiB
Python
709 lines
25 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2005-2007 Donald N. Allingham
|
|
# Copyright (C) 2008-2009 Gary Burton
|
|
# Copyright (C) 2009 Doug Blank <doug.blank@gmail.com>
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
|
|
# $Id$
|
|
|
|
"""
|
|
This package implements access to GRAMPS configuration.
|
|
"""
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# System imports
|
|
#
|
|
#---------------------------------------------------------------
|
|
import os
|
|
import time
|
|
import ConfigParser
|
|
import errno
|
|
from gettext import gettext as _
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Gramps imports
|
|
#
|
|
#---------------------------------------------------------------
|
|
import const
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Constants
|
|
#
|
|
#---------------------------------------------------------------
|
|
INIFILE = os.path.join(const.HOME_DIR, "gramps32.ini")
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Local functions
|
|
#
|
|
#---------------------------------------------------------------
|
|
def eval_item(setting):
|
|
"""
|
|
Given a value from an ini file, return it in proper type.
|
|
May be recursively called, in the case of nested structures.
|
|
"""
|
|
setting = setting.strip()
|
|
value = None
|
|
if setting.startswith("'") and setting.endswith("'"):
|
|
value = setting[1:-1]
|
|
elif setting.startswith("[") and setting.endswith("]"):
|
|
list_data = setting[1:-1]
|
|
value = [eval_item(item) for item in list_data.split(",")]
|
|
elif setting == "True":
|
|
value = True
|
|
elif setting == "False":
|
|
value = False
|
|
elif "." in setting:
|
|
value = float(setting)
|
|
else:
|
|
value = int(setting)
|
|
return value
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Classes
|
|
#
|
|
#---------------------------------------------------------------
|
|
class ConfigManager(object):
|
|
"""
|
|
Class to construct the singleton CONFIGMAN where all
|
|
settings are stored.
|
|
"""
|
|
def __init__(self, filename = None):
|
|
"""
|
|
Configure manager constructor takes an optional filename.
|
|
|
|
The data dictionary stores the settings:
|
|
|
|
self.data[section][setting] = value
|
|
|
|
The value has a type that matches the default. It is an error
|
|
to attempt to set the setting to a different type. To change
|
|
the type, you must re-register the setting, and re-set the
|
|
value.
|
|
|
|
The default values are given in Python code and stored here
|
|
on start-up:
|
|
|
|
self.default[section][setting] = default_value
|
|
|
|
Callbacks are stored as callables here:
|
|
|
|
self.callbacks[section][setting] = (id, func)
|
|
|
|
The default filename (usually the one you are reading from)
|
|
is stored as self.filename. However, you can save to another
|
|
filename using self.save(otherfilename).
|
|
"""
|
|
self._cb_id = 0 # callback id counter
|
|
self.filename = filename
|
|
self.callbacks = {}
|
|
self.default = {}
|
|
self.data = {}
|
|
self.reset()
|
|
|
|
def reset(self, section=None):
|
|
"""
|
|
Resets all settings values to their defaults.
|
|
"""
|
|
if section is None:
|
|
self.data = {}
|
|
self.data.update(self.default)
|
|
else:
|
|
self.data[section] = {}
|
|
self.data[section].update(self.default[section])
|
|
# Callbacks are still connected
|
|
|
|
def get_sections(self):
|
|
"""
|
|
Return all section names.
|
|
"""
|
|
return self.data.keys()
|
|
|
|
def get_section_settings(self, section):
|
|
"""
|
|
Return all section setting names.
|
|
"""
|
|
return self.data[section].keys()
|
|
|
|
def load(self, filename=None, oldstyle=False):
|
|
"""
|
|
Loads an .ini into self.data.
|
|
"""
|
|
if filename is None:
|
|
filename = self.filename
|
|
if filename and os.path.exists(filename):
|
|
parser = ConfigParser.ConfigParser()
|
|
parser.read(filename)
|
|
for sec in parser.sections():
|
|
name = sec.lower()
|
|
if name not in self.data:
|
|
# Add the setting from file
|
|
# These might be old settings, or third-party settings
|
|
self.data[name] = {}
|
|
for opt in parser.options(sec):
|
|
setting = parser.get(sec, opt).strip()
|
|
if oldstyle:
|
|
####################### Upgrade from oldstyle < 3.2
|
|
# if we know this setting, convert type
|
|
key = "%s.%s" % (name, opt)
|
|
if self.has_default(key):
|
|
vtype = type(self.get_default(key))
|
|
if vtype == bool:
|
|
value = setting in ["1", "True"]
|
|
elif vtype == list:
|
|
print "WARNING: ignoring old key '%s'" % key
|
|
continue # there were no lists in oldstyle
|
|
else:
|
|
value = vtype(setting)
|
|
else:
|
|
# else, ignore it
|
|
print "WARNING: ignoring old key '%s'" % key
|
|
continue # with next setting
|
|
####################### End upgrade code
|
|
else:
|
|
value = eval_item(setting)
|
|
####################### Now, let's test and set:
|
|
if type(value) == type(self.default[name][opt.lower()]):
|
|
self.data[name][opt.lower()] = value
|
|
else:
|
|
print ("WARNING: ignoring key with wrong type '%s.%s'"
|
|
% (name, opt.lower()))
|
|
|
|
def save(self, filename = None):
|
|
"""
|
|
Saves the current section/settings to an .ini file. Optional filename
|
|
will override the default filename to save to, if given.
|
|
"""
|
|
if filename is None:
|
|
filename = self.filename
|
|
if filename:
|
|
try:
|
|
head = os.path.split( filename )[0]
|
|
os.makedirs( head )
|
|
except OSError, exp:
|
|
if exp.errno != errno.EEXIST:
|
|
raise
|
|
key_file = open(filename, "w")
|
|
key_file.write(";; Gramps key file\n")
|
|
key_file.write((";; Automatically created at %s" %
|
|
time.strftime("%Y/%m/%d %H:%M:%S")) + "\n\n")
|
|
sections = sorted(self.data)
|
|
for section in sections:
|
|
key_file.write(("[%s]\n") % section)
|
|
keys = sorted(self.data[section])
|
|
for key in keys:
|
|
value = self.data[section][key]
|
|
if isinstance(value, long):
|
|
value = int(value)
|
|
key_file.write(("%s=%s\n")% (key, repr(value)))
|
|
key_file.write("\n")
|
|
key_file.close()
|
|
# else, no filename given; nothing to save so do nothing quietly
|
|
|
|
def get(self, key):
|
|
"""
|
|
Get the setting's value. raise an error if an invalid section.setting.
|
|
Key is a sting in the "section.setting" format.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
raise AttributeError("Invalid config section.setting name: '%s'" %
|
|
key)
|
|
if section not in self.data:
|
|
raise AttributeError("No such config section name: '%s'" % section)
|
|
if setting not in self.data[section]:
|
|
raise AttributeError("No such config setting name: '%s.%s'" %
|
|
(section, setting))
|
|
return self.data[section][setting]
|
|
|
|
def is_set(self, key):
|
|
"""
|
|
Does the setting exist? Returns True if does, False otherwise.
|
|
Key is a sting in the "section.setting" format.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
return False
|
|
if section not in self.data:
|
|
return False
|
|
if setting not in self.data[section]:
|
|
return False
|
|
return True
|
|
|
|
def has_default(self, key):
|
|
"""
|
|
Does the setting have a default value? Returns True if it does,
|
|
False otherwise. Key is a sting in the "section.setting" format.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
return False
|
|
if section not in self.default:
|
|
return False
|
|
if setting not in self.default[section]:
|
|
return False
|
|
return True
|
|
|
|
def get_default(self, key):
|
|
"""
|
|
Get the setting's default value. Raises an error if invalid key is
|
|
give. Key is a sting in the "section.setting" format.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
raise AttributeError("Invalid config section.setting name: '%s'" %
|
|
key)
|
|
if section not in self.default:
|
|
raise AttributeError("No such config section name: '%s'" % section)
|
|
if setting not in self.default[section]:
|
|
raise AttributeError("No such config setting name: '%s.%s'" %
|
|
(section, setting))
|
|
return self.default[section][setting]
|
|
|
|
def register(self, key, default):
|
|
"""
|
|
Register a section.setting, and set the default.
|
|
Will overwrite any previously set default, and set setting if not one.
|
|
The default value deterimines the type of the setting.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
raise AttributeError("Invalid config section.setting name: '%s'" %
|
|
key)
|
|
if section not in self.data:
|
|
self.data[section] = {}
|
|
if section not in self.default:
|
|
self.default[section] = {}
|
|
if section not in self.callbacks:
|
|
self.callbacks[section] = {}
|
|
if setting not in self.callbacks[section]:
|
|
self.callbacks[section][setting] = []
|
|
# Add the default value to settings, if not exist:
|
|
if setting not in self.data[section]:
|
|
self.data[section][setting] = default
|
|
# Set the default, regardless:
|
|
self.default[section][setting] = default
|
|
|
|
def connect(self, key, func):
|
|
"""
|
|
Connect a callback func that gets called when key is changed.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
raise AttributeError("Invalid config section.setting name: '%s'" %
|
|
key)
|
|
if section not in self.data:
|
|
raise AttributeError("No such config section name: '%s'" % section)
|
|
if setting not in self.data[section]:
|
|
raise AttributeError("No such config setting name: '%s.%s'" %
|
|
(section, setting))
|
|
self._cb_id += 1
|
|
self.callbacks[section][setting].append((self._cb_id, func))
|
|
return self._cb_id
|
|
|
|
def disconnect(self, callback_id):
|
|
"""
|
|
"""
|
|
for section in self.callbacks:
|
|
for setting in self.callbacks[section]:
|
|
for (cbid, func) in self.callbacks[section][setting]:
|
|
if callback_id == cbid:
|
|
self.callbacks[section][setting].remove((cbid, func))
|
|
|
|
def set(self, key, value):
|
|
"""
|
|
Set the setting's value. There are only two ways to get into
|
|
the data dictionary: via the load() method that reads a file,
|
|
or from this method.
|
|
"""
|
|
if "." in key:
|
|
section, setting = key.split(".", 1)
|
|
else:
|
|
raise AttributeError("Invalid config section.setting name: '%s'" %
|
|
key)
|
|
if section not in self.data:
|
|
raise AttributeError("No such config section name: '%s'" % section)
|
|
if setting not in self.data[section]:
|
|
raise AttributeError("No such config setting name: '%s.%s'" %
|
|
(section, setting))
|
|
# Check value to see if right type:
|
|
if type(value) == long:
|
|
value = int(value)
|
|
if type(value) == unicode:
|
|
value = str(value)
|
|
if self.has_default(key):
|
|
if type(self.get_default(key)) != type(value):
|
|
raise AttributeError("attempting to set '%s' to wrong type "
|
|
"'%s'; should be '%s'" %
|
|
(key, type(value),
|
|
type(self.get_default(key))))
|
|
if (setting in self.data[section] and
|
|
self.data[section][setting] == value):
|
|
# Do nothing if existed and is the same
|
|
pass
|
|
else:
|
|
# Set the value:
|
|
self.data[section][setting] = value
|
|
# Only call callback if the value changed!
|
|
if (section in self.callbacks and
|
|
setting in self.callbacks[section]):
|
|
for (cbid, func) in self.callbacks[section][setting]:
|
|
# Call all callbacks for this key:
|
|
func(self, 0, str(self.data[section][setting]), None)
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Convience functions to call ConfigManager instance methods
|
|
#
|
|
#---------------------------------------------------------------
|
|
|
|
def register(key, value):
|
|
""" Module shortcut to register key, value """
|
|
CONFIGMAN.register(key, value)
|
|
|
|
def get(key):
|
|
""" Module shortcut to get value from key """
|
|
return CONFIGMAN.get(key)
|
|
|
|
def get_default(key):
|
|
""" Module shortcut to get default from key """
|
|
return CONFIGMAN.get_default(key)
|
|
|
|
def set(key, value):
|
|
""" Module shortcut to set value from key """
|
|
CONFIGMAN.set(key, value)
|
|
|
|
def save(filename=None):
|
|
""" Module shortcut to save config file """
|
|
CONFIGMAN.save(filename)
|
|
|
|
def connect(key, func):
|
|
""" Module shortcut to callbacks """
|
|
return CONFIGMAN.connect(key, func)
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Register the system-wide settings in a singleton config manager
|
|
#
|
|
#---------------------------------------------------------------
|
|
|
|
CONFIGMAN = ConfigManager(INIFILE)
|
|
|
|
register('behavior.addmedia-image-dir', '')
|
|
register('behavior.addmedia-relative-path', False)
|
|
register('behavior.autoload', False)
|
|
register('behavior.avg-generation-gap', 20)
|
|
register('behavior.betawarn', False)
|
|
register('behavior.database-path', os.path.join( const.HOME_DIR, 'grampsdb'))
|
|
register('behavior.date-about-range', 10)
|
|
register('behavior.date-after-range', 10)
|
|
register('behavior.date-before-range', 10)
|
|
register('behavior.generation-depth', 15)
|
|
register('behavior.max-age-prob-alive', 110)
|
|
register('behavior.max-sib-age-diff', 20)
|
|
register('behavior.min-generation-years', 13)
|
|
register('behavior.owner-warn', False)
|
|
register('behavior.pop-plugin-status', False)
|
|
register('behavior.recent-export-type', 1)
|
|
register('behavior.spellcheck', False)
|
|
register('behavior.startup', 0)
|
|
register('behavior.surname-guessing', 0)
|
|
register('behavior.use-tips', False)
|
|
register('behavior.welcome', 100)
|
|
|
|
register('export.no-private', True)
|
|
register('export.no-unlinked', True)
|
|
register('export.restrict-living', True)
|
|
|
|
register('geoview.latitude', "0.0")
|
|
register('geoview.lock', False)
|
|
register('geoview.longitude', "0.0")
|
|
register('geoview.map', "person")
|
|
register('geoview.stylesheet', "")
|
|
register('geoview.zoom', 0)
|
|
|
|
register('htmlview.start-url', "http://gramps-project.org")
|
|
|
|
register('interface.address-height', 450)
|
|
register('interface.address-width', 650)
|
|
register('interface.attribute-height', 350)
|
|
register('interface.attribute-width', 600)
|
|
register('interface.child-ref-height', 450)
|
|
register('interface.child-ref-width', 600)
|
|
register('interface.clipboard-height', 300)
|
|
register('interface.clipboard-width', 300)
|
|
register('interface.dont-ask', False)
|
|
register('interface.data-views',
|
|
['GrampletView', 'PersonView', 'RelationshipView',
|
|
'FamilyListView', 'PedigreeView', 'EventView', 'SourceView',
|
|
'PlaceView', 'MediaView', 'RepositoryView', 'NoteView'])
|
|
register('interface.event-height', 450)
|
|
register('interface.event-ref-height', 450)
|
|
register('interface.event-ref-width', 600)
|
|
register('interface.event-sel-height', 450)
|
|
register('interface.event-sel-width', 600)
|
|
register('interface.event-width', 600)
|
|
register('interface.family-height', 500)
|
|
register('interface.family-sel-height', 450)
|
|
register('interface.family-sel-width', 600)
|
|
register('interface.family-width', 700)
|
|
register('interface.filter', False)
|
|
register('interface.fullscreen', False)
|
|
register('interface.height', 500)
|
|
register('interface.lds-height', 450)
|
|
register('interface.lds-width', 600)
|
|
register('interface.location-height', 250)
|
|
register('interface.location-width', 600)
|
|
register('interface.mapservice', 'OpenStreetMap')
|
|
register('interface.media-height', 450)
|
|
register('interface.media-ref-height', 450)
|
|
register('interface.media-ref-width', 600)
|
|
register('interface.media-sel-height', 450)
|
|
register('interface.media-sel-width', 600)
|
|
register('interface.media-width', 650)
|
|
register('interface.name-height', 350)
|
|
register('interface.name-width', 600)
|
|
register('interface.note-height', 500)
|
|
register('interface.note-sel-height', 450)
|
|
register('interface.note-sel-width', 600)
|
|
register('interface.note-width', 700)
|
|
register('interface.patro-title', 0)
|
|
register('interface.pedview-layout', 0)
|
|
register('interface.pedview-show-images', True)
|
|
register('interface.pedview-show-marriage', False)
|
|
register('interface.pedview-tree-size', 0)
|
|
register('interface.person-height', 550)
|
|
register('interface.person-ref-height', 350)
|
|
register('interface.person-ref-width', 600)
|
|
register('interface.person-sel-height', 450)
|
|
register('interface.person-sel-width', 600)
|
|
register('interface.person-width', 750)
|
|
register('interface.place-height', 450)
|
|
register('interface.place-sel-height', 450)
|
|
register('interface.place-sel-width', 600)
|
|
register('interface.place-width', 650)
|
|
register('interface.prefix-suffix', 0)
|
|
register('interface.releditbtn', False)
|
|
register('interface.repo-height', 450)
|
|
register('interface.repo-ref-height', 450)
|
|
register('interface.repo-ref-width', 600)
|
|
register('interface.repo-sel-height', 450)
|
|
register('interface.repo-sel-width', 600)
|
|
register('interface.repo-width', 650)
|
|
register('interface.sidebar-text', True)
|
|
register('interface.size-checked', False)
|
|
register('interface.source-height', 450)
|
|
register('interface.source-ref-height', 450)
|
|
register('interface.source-ref-width', 600)
|
|
register('interface.source-sel-height', 450)
|
|
register('interface.source-sel-width', 600)
|
|
register('interface.source-width', 600)
|
|
register('interface.statusbar', 1)
|
|
register('interface.toolbar-on', True)
|
|
register('interface.url-height', 150)
|
|
register('interface.url-width', 600)
|
|
register('interface.view', True)
|
|
register('interface.width', 775)
|
|
|
|
register('paths.recent-export-dir', '')
|
|
register('paths.recent-file', '')
|
|
register('paths.recent-import-dir', '')
|
|
register('paths.report-directory', const.USER_HOME)
|
|
register('paths.website-directory', const.USER_HOME)
|
|
|
|
register('preferences.complete-color', '#008b00')
|
|
register('preferences.custom-marker-color', '#8b008b')
|
|
register('preferences.date-format', 0)
|
|
register('preferences.default-source', False)
|
|
register('preferences.eprefix', 'E%04d')
|
|
register('preferences.family-details', True)
|
|
register('preferences.family-siblings', True)
|
|
register('preferences.family-warn', True)
|
|
register('preferences.fprefix', 'F%04d')
|
|
register('preferences.geoview', False)
|
|
register('preferences.googlemap', True)
|
|
register('preferences.hide-ep-msg', False)
|
|
register('preferences.invalid-date-format', "<b>%s</b>")
|
|
register('preferences.iprefix', 'I%04d')
|
|
register('preferences.last-view', 0)
|
|
register('preferences.microsoft', False)
|
|
register('preferences.name-format', 1)
|
|
register('preferences.no-given-text', "[%s]" % _("Missing Given Name"))
|
|
register('preferences.no-record-text', "[%s]" % _("Missing Record"))
|
|
register('preferences.no-surname-text', "[%s]" % _("Missing Surname"))
|
|
register('preferences.nprefix', 'N%04d')
|
|
register('preferences.online-maps', False)
|
|
register('preferences.openlayers', False)
|
|
register('preferences.oprefix', 'O%04d')
|
|
register('preferences.paper-metric', 0)
|
|
register('preferences.paper-preference', 'Letter')
|
|
register('preferences.pprefix', 'P%04d')
|
|
register('preferences.private-given-text', "[%s]" % _("Living"))
|
|
register('preferences.private-record-text', "[%s]" % _("Private Record"))
|
|
register('preferences.private-surname-text', "[%s]" % _("Living"))
|
|
register('preferences.relation-display-theme', "CLASSIC")
|
|
register('preferences.relation-shade', True)
|
|
register('preferences.rprefix', 'R%04d')
|
|
register('preferences.sprefix', 'S%04d')
|
|
register('preferences.todo-color', '#ff0000')
|
|
register('preferences.use-last-view', True)
|
|
register('preferences.yahoo', False)
|
|
register('preferences.microsoft', False)
|
|
|
|
register('researcher.researcher-addr', '')
|
|
register('researcher.researcher-city', '')
|
|
register('researcher.researcher-country', '')
|
|
register('researcher.researcher-email', '')
|
|
register('researcher.researcher-name', '')
|
|
register('researcher.researcher-phone', '')
|
|
register('researcher.researcher-postal', '')
|
|
register('researcher.researcher-state', '')
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Now, load the settings from the config file
|
|
#
|
|
#---------------------------------------------------------------
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Upgrade Conversions go here.
|
|
#
|
|
#---------------------------------------------------------------
|
|
|
|
# If we have not already upgraded to this version,
|
|
# we can tell by seeing if there is a key file for this version:
|
|
if not os.path.exists(CONFIGMAN.filename):
|
|
# If not, let's read old if there:
|
|
if os.path.exists(os.path.join(const.HOME_DIR, "keys.ini")):
|
|
# read it in old style:
|
|
print "Importing old key file 'keys.ini'..."
|
|
CONFIGMAN.load(os.path.join(const.HOME_DIR, "keys.ini"),
|
|
oldstyle=True)
|
|
print "Done importing old key file 'keys.ini'"
|
|
# other version upgrades here...
|
|
|
|
#---------------------------------------------------------------
|
|
#
|
|
# Now, load the settings from the config file, if one
|
|
#
|
|
#---------------------------------------------------------------
|
|
CONFIGMAN.load()
|
|
|
|
if __name__ == "__main__":
|
|
CM = ConfigManager("./temp.ini")
|
|
CM.register("section.setting1", 1) # int
|
|
CM.register("section.setting2", 3.1415) # float
|
|
CM.register("section.setting3", "String") # string
|
|
CM.register("section.setting4", False) # boolean
|
|
|
|
assert CM.get("section.setting1") == 1
|
|
assert CM.get("section.setting2") == 3.1415
|
|
assert CM.get("section.setting3") == "String"
|
|
assert CM.get("section.setting4") == False
|
|
|
|
CM.set("section.setting1", 2)
|
|
CM.set("section.setting2", 8.6)
|
|
CM.set("section.setting3", "Another String")
|
|
CM.set("section.setting4", True)
|
|
|
|
assert CM.get("section.setting1") == 2
|
|
assert CM.get("section.setting2") == 8.6
|
|
assert CM.get("section.setting3") == "Another String"
|
|
assert CM.get("section.setting4") == True
|
|
|
|
try:
|
|
CM.set("section.setting1", 2.8)
|
|
except AttributeError:
|
|
pass
|
|
else:
|
|
raise AssertionError
|
|
|
|
try:
|
|
CM.set("section.setting2", 2)
|
|
except AttributeError:
|
|
pass
|
|
else:
|
|
raise AssertionError
|
|
|
|
try:
|
|
CM.set("section.setting3", 6)
|
|
except AttributeError:
|
|
pass
|
|
else:
|
|
raise AssertionError
|
|
|
|
try:
|
|
CM.set("section.setting4", 1)
|
|
except AttributeError:
|
|
pass
|
|
else:
|
|
raise AssertionError
|
|
|
|
assert CM.get("section.setting1") == 2
|
|
assert CM.get("section.setting2") == 8.6
|
|
assert CM.get("section.setting3") == "Another String"
|
|
assert CM.get("section.setting4") == True
|
|
|
|
# Try to set one that doesn't exist:
|
|
try:
|
|
CM.set("section.setting5", 1)
|
|
except AttributeError:
|
|
pass
|
|
else:
|
|
raise AssertionError
|
|
|
|
CM.save()
|
|
|
|
CM.reset()
|
|
|
|
assert CM.get("section.setting1") == 1
|
|
assert CM.get("section.setting2") == 3.1415
|
|
assert CM.get("section.setting3") == "String"
|
|
assert CM.get("section.setting4") == False
|
|
|
|
x = None
|
|
def callback(*args):
|
|
# args: self, 0, str(setting), None
|
|
global x
|
|
x = args[2]
|
|
|
|
cbid = CM.connect("section.setting1", callback)
|
|
assert x == None
|
|
|
|
CM.set("section.setting1", 1024)
|
|
assert x == "1024"
|
|
|
|
CM.disconnect(cbid)
|
|
|
|
CM.set("section.setting1", -1)
|
|
assert x == "1024"
|
|
|
|
CM.save("./test2.ini")
|