mirror of https://github.com/patacrep/patacrep.git
Engine for LaTeX songbooks
http://www.patacrep.com
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
88 lines
2.5 KiB
88 lines
2.5 KiB
"""Some utility functions"""
|
|
|
|
import contextlib
|
|
import logging
|
|
|
|
from collections import UserDict
|
|
|
|
class DictOfDict(UserDict):
|
|
"""Dictionary, with a recursive :meth:`update` method.
|
|
|
|
By "recursive", we mean: if `self.update(other)` is called, and for some
|
|
key both `self[key]` and `other[key]` are dictionary, then `self[key]` is
|
|
not replaced by `other[key]`, but instead is updated. This is done
|
|
recursively (that is, `self[foo][bar][baz]` is updated with
|
|
`other[foo][bar][baz]`, if the corresponding objects are dictionaries).
|
|
|
|
>>> ordinal = DictOfDict({
|
|
... "francais": {
|
|
... 1: "premier",
|
|
... 2: "deuxieme",
|
|
... },
|
|
... "english": {
|
|
... 1: "first",
|
|
... },
|
|
... })
|
|
>>> ordinal.update({
|
|
... "francais": {
|
|
... 2: "second",
|
|
... 3: "troisieme",
|
|
... },
|
|
... "espanol": {
|
|
... 1: "primero",
|
|
... },
|
|
... })
|
|
>>> ordinal == {
|
|
... "francais": {
|
|
... 1: "premier",
|
|
... 2: "second",
|
|
... 3: "troisieme",
|
|
... },
|
|
... "english": {
|
|
... 1: "first",
|
|
... },
|
|
... "espanol": {
|
|
... 1: "primero",
|
|
... },
|
|
... }
|
|
True
|
|
"""
|
|
|
|
def update(self, other):
|
|
# pylint: disable=arguments-differ
|
|
self._update(self, other)
|
|
|
|
@staticmethod
|
|
def _update(left, right):
|
|
"""Equivalent to `left.update(right)`, with recursive update."""
|
|
for key in right:
|
|
if key not in left:
|
|
left[key] = right[key]
|
|
elif isinstance(left[key], dict) and isinstance(right[key], dict):
|
|
DictOfDict._update(left[key], right[key])
|
|
else:
|
|
left[key] = right[key]
|
|
|
|
def yesno(string):
|
|
"""Interpret string argument as a boolean.
|
|
|
|
May raise `ValueError` if argument cannot be interpreted.
|
|
"""
|
|
yes_strings = ["y", "yes", "true", "1"]
|
|
no_strings = ["n", "no", "false", "0"]
|
|
if string.lower() in yes_strings:
|
|
return True
|
|
if string.lower() in no_strings:
|
|
return False
|
|
raise ValueError("'{}' is supposed to be one of {}.".format(
|
|
string,
|
|
", ".join(["'{}'".format(string) for string in yes_strings + no_strings]),
|
|
))
|
|
|
|
@contextlib.contextmanager
|
|
def logging_reduced(module_name):
|
|
logger = logging.getLogger(module_name)
|
|
previous_loglevel = logger.getEffectiveLevel()
|
|
logger.setLevel(logging.CRITICAL)
|
|
yield
|
|
logger.setLevel(previous_loglevel)
|
|
|