| 
									
										
										
										
											2016-07-22 06:43:20 -03:00
										 |  |  | """
 | 
					
						
							|  |  |  | Copyright (C) 2016, Digium, Inc. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This program is free software, distributed under the terms of | 
					
						
							|  |  |  | the GNU General Public License Version 2. | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2020-10-29 11:21:45 -04:00
										 |  |  | import glob | 
					
						
							| 
									
										
										
										
											2015-01-09 22:09:04 +00:00
										 |  |  | import itertools | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | from astdicts import OrderedDict | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | from astdicts import MultiOrderedDict | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | def merge_values(left, right, key): | 
					
						
							|  |  |  |     """Merges values from right into left.""" | 
					
						
							|  |  |  |     if isinstance(left, list): | 
					
						
							|  |  |  |         vals0 = left | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     else:  # assume dictionary | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         vals0 = left[key] if key in left else [] | 
					
						
							|  |  |  |     vals1 = right[key] if key in right else [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return vals0 + [i for i in vals1 if i not in vals0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | class Section(MultiOrderedDict): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     A Section is a MultiOrderedDict itself that maintains a list of | 
					
						
							|  |  |  |     key/value options.  However, in the case of an Asterisk config | 
					
						
							|  |  |  |     file a section may have other defaults sections that is can pull | 
					
						
							|  |  |  |     data from (i.e. templates).  So when an option is looked up by key | 
					
						
							|  |  |  |     it first checks the base section and if not found looks in the | 
					
						
							|  |  |  |     added default sections. If not found at that point then a 'KeyError' | 
					
						
							|  |  |  |     exception is raised. | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     count = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, defaults=None, templates=None): | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         MultiOrderedDict.__init__(self) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         # track an ordered id of sections | 
					
						
							|  |  |  |         Section.count += 1 | 
					
						
							|  |  |  |         self.id = Section.count | 
					
						
							|  |  |  |         self._defaults = [] if defaults is None else defaults | 
					
						
							|  |  |  |         self._templates = [] if templates is None else templates | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def __cmp__(self, other): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Use self.id as means of determining equality | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2018-04-18 09:27:51 +02:00
										 |  |  |         return (self.id > other.id) - (self.id < other.id) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 06:43:20 -03:00
										 |  |  |     def __eq__(self, other): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Use self.id as means of determining equality | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.id == other.id | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 11:21:45 -04:00
										 |  |  |     def __lt__(self, other): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Use self.id as means of determining equality | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.id < other.id | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __gt__(self, other): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Use self.id as means of determining equality | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.id > other.id | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __le__(self, other): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Use self.id as means of determining equality | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.id <= other.id | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __ge__(self, other): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Use self.id as means of determining equality | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.id >= other.id | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     def get(self, key, from_self=True, from_templates=True, | 
					
						
							|  |  |  |             from_defaults=True): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Get the values corresponding to a given key. The parameters to this | 
					
						
							|  |  |  |         function form a hierarchy that determines priority of the search. | 
					
						
							|  |  |  |         from_self takes priority over from_templates, and from_templates takes | 
					
						
							|  |  |  |         priority over from_defaults. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Parameters: | 
					
						
							|  |  |  |         from_self - If True, search within the given section. | 
					
						
							|  |  |  |         from_templates - If True, search in this section's templates. | 
					
						
							|  |  |  |         from_defaults - If True, search within this section's defaults. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         if from_self and key in self: | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |             return MultiOrderedDict.__getitem__(self, key) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         if from_templates: | 
					
						
							|  |  |  |             if self in self._templates: | 
					
						
							|  |  |  |                 return [] | 
					
						
							|  |  |  |             for t in self._templates: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     # fail if not found on the search - doing it this way | 
					
						
							|  |  |  |                     # allows template's templates to be searched. | 
					
						
							|  |  |  |                     return t.get(key, True, from_templates, from_defaults) | 
					
						
							|  |  |  |                 except KeyError: | 
					
						
							|  |  |  |                     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if from_defaults: | 
					
						
							|  |  |  |             for d in self._defaults: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     return d.get(key, True, from_templates, from_defaults) | 
					
						
							|  |  |  |                 except KeyError: | 
					
						
							|  |  |  |                     pass | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         raise KeyError(key) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def __getitem__(self, key): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Get the value for the given key. If it is not found in the 'self' | 
					
						
							|  |  |  |         then check inside templates and defaults before declaring raising | 
					
						
							|  |  |  |         a KeyError exception. | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         return self.get(key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def keys(self, self_only=False): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Get the keys from this section. If self_only is True, then | 
					
						
							|  |  |  |         keys from this section's defaults and templates are not | 
					
						
							|  |  |  |         included in the returned value | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         res = MultiOrderedDict.keys(self) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         if self_only: | 
					
						
							|  |  |  |             return res | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for d in self._templates: | 
					
						
							|  |  |  |             for key in d.keys(): | 
					
						
							|  |  |  |                 if key not in res: | 
					
						
							|  |  |  |                     res.append(key) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         for d in self._defaults: | 
					
						
							|  |  |  |             for key in d.keys(): | 
					
						
							|  |  |  |                 if key not in res: | 
					
						
							|  |  |  |                     res.append(key) | 
					
						
							|  |  |  |         return res | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def add_defaults(self, defaults): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Add a list of defaults to the section. Defaults are | 
					
						
							|  |  |  |         sections such as 'general' | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         defaults.sort() | 
					
						
							|  |  |  |         for i in defaults: | 
					
						
							|  |  |  |             self._defaults.insert(0, i) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_templates(self, templates): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Add a list of templates to the section. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         templates.sort() | 
					
						
							|  |  |  |         for i in templates: | 
					
						
							|  |  |  |             self._templates.insert(0, i) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_merged(self, key): | 
					
						
							|  |  |  |         """Return a list of values for a given key merged from default(s)""" | 
					
						
							|  |  |  |         # first merge key/values from defaults together | 
					
						
							|  |  |  |         merged = [] | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         for i in reversed(self._defaults): | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |             if not merged: | 
					
						
							|  |  |  |                 merged = i | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             merged = merge_values(merged, i, key) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for i in reversed(self._templates): | 
					
						
							|  |  |  |             if not merged: | 
					
						
							|  |  |  |                 merged = i | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             merged = merge_values(merged, i, key) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         # then merge self in | 
					
						
							|  |  |  |         return merge_values(merged, self, key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | COMMENT = ';' | 
					
						
							|  |  |  | COMMENT_START = ';--' | 
					
						
							|  |  |  | COMMENT_END = '--;' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEFAULTSECT = 'general' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | def remove_comment(line, is_comment): | 
					
						
							|  |  |  |     """Remove any commented elements from the line.""" | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     if not line: | 
					
						
							|  |  |  |         return line, is_comment | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if is_comment: | 
					
						
							|  |  |  |         part = line.partition(COMMENT_END) | 
					
						
							|  |  |  |         if part[1]: | 
					
						
							|  |  |  |             # found multi-line comment end check string after it | 
					
						
							|  |  |  |             return remove_comment(part[2], False) | 
					
						
							|  |  |  |         return "", True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     part = line.partition(COMMENT_START) | 
					
						
							| 
									
										
										
										
											2019-03-04 13:36:01 -05:00
										 |  |  |     if part[1] and not part[2].startswith('-'): | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         # found multi-line comment start check string before | 
					
						
							|  |  |  |         # it to make sure there wasn't an eol comment in it | 
					
						
							|  |  |  |         has_comment = part[0].partition(COMMENT) | 
					
						
							|  |  |  |         if has_comment[1]: | 
					
						
							|  |  |  |             # eol comment found return anything before it | 
					
						
							|  |  |  |             return has_comment[0], False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # check string after it to see if the comment ends | 
					
						
							|  |  |  |         line, is_comment = remove_comment(part[2], True) | 
					
						
							|  |  |  |         if is_comment: | 
					
						
							|  |  |  |             # return possible string data before comment | 
					
						
							|  |  |  |             return part[0].strip(), True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # otherwise it was an embedded comment so combine | 
					
						
							|  |  |  |         return ''.join([part[0].strip(), ' ', line]).rstrip(), False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 06:43:20 -03:00
										 |  |  |     # find the first occurence of a comment that is not escaped | 
					
						
							|  |  |  |     match = re.match(r'.*?([^\\];)', line) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if match: | 
					
						
							|  |  |  |          # the end of where the real string is is where the comment starts | 
					
						
							|  |  |  |          line = line[0:(match.end()-1)] | 
					
						
							| 
									
										
										
										
											2016-08-03 14:47:04 +00:00
										 |  |  |     if line.startswith(";"): | 
					
						
							| 
									
										
										
										
											2016-07-29 06:48:32 -03:00
										 |  |  |          # if the line is actually a comment just ignore it all | 
					
						
							|  |  |  |          line = "" | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 06:43:20 -03:00
										 |  |  |     return line.replace("\\", "").strip(), False | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | def try_include(line): | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     Checks to see if the given line is an include.  If so return the | 
					
						
							|  |  |  |     included filename, otherwise None. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 11:21:45 -04:00
										 |  |  |     match = re.match('^#include\s*([^;]+).*$', line) | 
					
						
							|  |  |  |     if match: | 
					
						
							|  |  |  |         trimmed = match.group(1).rstrip() | 
					
						
							|  |  |  |         quoted = re.match('^"([^"]+)"$', trimmed) | 
					
						
							|  |  |  |         if quoted: | 
					
						
							|  |  |  |             return quoted.group(1) | 
					
						
							|  |  |  |         bracketed = re.match('^<([^>]+)>$', trimmed) | 
					
						
							|  |  |  |         if bracketed: | 
					
						
							|  |  |  |             return bracketed.group(1) | 
					
						
							|  |  |  |         return trimmed | 
					
						
							|  |  |  |     return None | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def try_section(line): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     Checks to see if the given line is a section. If so return the section | 
					
						
							|  |  |  |     name, otherwise return 'None'. | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     # leading spaces were stripped when checking for comments | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |     if not line.startswith('['): | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         return None, False, [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     section, delim, templates = line.partition(']') | 
					
						
							|  |  |  |     if not templates: | 
					
						
							|  |  |  |         return section[1:], False, [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # strip out the parens and parse into an array | 
					
						
							|  |  |  |     templates = templates.replace('(', "").replace(')', "").split(',') | 
					
						
							|  |  |  |     # go ahead and remove extra whitespace | 
					
						
							|  |  |  |     templates = [i.strip() for i in templates] | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         templates.remove('!') | 
					
						
							|  |  |  |         return section[1:], True, templates | 
					
						
							|  |  |  |     except: | 
					
						
							|  |  |  |         return section[1:], False, templates | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | def try_option(line): | 
					
						
							|  |  |  |     """Parses the line as an option, returning the key/value pair.""" | 
					
						
							| 
									
										
										
										
											2016-07-22 06:43:20 -03:00
										 |  |  |     data = re.split('=>?', line, 1) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     # should split in two (key/val), but either way use first two elements | 
					
						
							|  |  |  |     return data[0].rstrip(), data[1].lstrip() | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def find_dict(mdicts, key, val): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     Given a list of mult-dicts, return the multi-dict that contains | 
					
						
							|  |  |  |     the given key/value pair. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def found(d): | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         return key in d and val in d[key] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         return [d for d in mdicts if found(d)][0] | 
					
						
							|  |  |  |     except IndexError: | 
					
						
							|  |  |  |         raise LookupError("Dictionary not located for key = %s, value = %s" | 
					
						
							|  |  |  |                           % (key, val)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | def write_dicts(config_file, mdicts): | 
					
						
							|  |  |  |     """Write the contents of the mdicts to the specified config file""" | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     for section, sect_list in mdicts.iteritems(): | 
					
						
							|  |  |  |         # every section contains a list of dictionaries | 
					
						
							|  |  |  |         for sect in sect_list: | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             config_file.write("[%s]\n" % section) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             for key, val_list in sect.iteritems(): | 
					
						
							|  |  |  |                 # every value is also a list | 
					
						
							|  |  |  |                 for v in val_list: | 
					
						
							|  |  |  |                     key_val = key | 
					
						
							|  |  |  |                     if v is not None: | 
					
						
							|  |  |  |                         key_val += " = " + str(v) | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |                         config_file.write("%s\n" % (key_val)) | 
					
						
							|  |  |  |             config_file.write("\n") | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | class MultiOrderedConfigParser: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def __init__(self, parent=None): | 
					
						
							|  |  |  |         self._parent = parent | 
					
						
							|  |  |  |         self._defaults = MultiOrderedDict() | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         self._sections = MultiOrderedDict() | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         self._includes = OrderedDict() | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     def find_value(self, sections, key): | 
					
						
							|  |  |  |         """Given a list of sections, try to find value(s) for the given key.""" | 
					
						
							|  |  |  |         # always start looking in the last one added | 
					
						
							|  |  |  |         sections.sort(reverse=True) | 
					
						
							|  |  |  |         for s in sections: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 # try to find in section and section's templates | 
					
						
							|  |  |  |                 return s.get(key, from_defaults=False) | 
					
						
							|  |  |  |             except KeyError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # wasn't found in sections or a section's templates so check in | 
					
						
							|  |  |  |         # defaults | 
					
						
							|  |  |  |         for s in sections: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 # try to find in section's defaultsects | 
					
						
							|  |  |  |                 return s.get(key, from_self=False, from_templates=False) | 
					
						
							|  |  |  |             except KeyError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         raise KeyError(key) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def defaults(self): | 
					
						
							|  |  |  |         return self._defaults | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def default(self, key): | 
					
						
							|  |  |  |         """Retrieves a list of dictionaries for a default section.""" | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         return self.get_defaults(key) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def add_default(self, key, template_keys=None): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Adds a default section to defaults, returning the | 
					
						
							|  |  |  |         default Section object. | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         if template_keys is None: | 
					
						
							|  |  |  |             template_keys = [] | 
					
						
							|  |  |  |         return self.add_section(key, template_keys, self._defaults) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def sections(self): | 
					
						
							|  |  |  |         return self._sections | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def section(self, key): | 
					
						
							|  |  |  |         """Retrieves a list of dictionaries for a section.""" | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         return self.get_sections(key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_sections(self, key, attr='_sections', searched=None): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Retrieve a list of sections that have values for the given key. | 
					
						
							|  |  |  |         The attr parameter can be used to control what part of the parser | 
					
						
							|  |  |  |         to retrieve values from. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if searched is None: | 
					
						
							|  |  |  |             searched = [] | 
					
						
							|  |  |  |         if self in searched: | 
					
						
							|  |  |  |             return [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sections = getattr(self, attr) | 
					
						
							|  |  |  |         res = sections[key] if key in sections else [] | 
					
						
							|  |  |  |         searched.append(self) | 
					
						
							|  |  |  |         if self._includes: | 
					
						
							| 
									
										
										
										
											2015-01-09 22:09:04 +00:00
										 |  |  |             res.extend(list(itertools.chain(*[ | 
					
						
							|  |  |  |                 incl.get_sections(key, attr, searched) | 
					
						
							|  |  |  |                 for incl in self._includes.itervalues()]))) | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         if self._parent: | 
					
						
							|  |  |  |             res += self._parent.get_sections(key, attr, searched) | 
					
						
							|  |  |  |         return res | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_defaults(self, key): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Retrieve a list of defaults that have values for the given key. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.get_sections(key, '_defaults') | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def add_section(self, key, template_keys=None, mdicts=None): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Create a new section in the configuration. The name of the | 
					
						
							|  |  |  |         new section is the 'key' parameter. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         if template_keys is None: | 
					
						
							|  |  |  |             template_keys = [] | 
					
						
							|  |  |  |         if mdicts is None: | 
					
						
							|  |  |  |             mdicts = self._sections | 
					
						
							|  |  |  |         res = Section() | 
					
						
							|  |  |  |         for t in template_keys: | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             res.add_templates(self.get_defaults(t)) | 
					
						
							|  |  |  |         res.add_defaults(self.get_defaults(DEFAULTSECT)) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         mdicts.insert(0, key, res) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         return res | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def includes(self): | 
					
						
							|  |  |  |         return self._includes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_include(self, filename, parser=None): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Add a new #include file to the configuration. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         if filename in self._includes: | 
					
						
							|  |  |  |             return self._includes[filename] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._includes[filename] = res = \ | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             MultiOrderedConfigParser(self) if parser is None else parser | 
					
						
							|  |  |  |         return res | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get(self, section, key): | 
					
						
							|  |  |  |         """Retrieves the list of values from a section for a key.""" | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             # search for the value in the list of sections | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             return self.find_value(self.section(section), key) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         except KeyError: | 
					
						
							|  |  |  |             pass | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             # section may be a default section so, search | 
					
						
							|  |  |  |             # for the value in the list of defaults | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             return self.find_value(self.default(section), key) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         except KeyError: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             raise LookupError("key %r not found for section %r" | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |                               % (key, section)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     def multi_get(self, section, key_list): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Retrieves the list of values from a section for a list of keys. | 
					
						
							|  |  |  |         This method is intended to be used for equivalent keys. Thus, as soon | 
					
						
							|  |  |  |         as any match is found for any key in the key_list, the match is | 
					
						
							|  |  |  |         returned. This does not concatenate the lookups of all of the keys | 
					
						
							|  |  |  |         together. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         for i in key_list: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 return self.get(section, i) | 
					
						
							|  |  |  |             except LookupError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Making it here means all lookups failed. | 
					
						
							|  |  |  |         raise LookupError("keys %r not found for section %r" % | 
					
						
							|  |  |  |                           (key_list, section)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |     def set(self, section, key, val): | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         """Sets an option in the given section.""" | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         # TODO - set in multiple sections? (for now set in first) | 
					
						
							|  |  |  |         # TODO - set in both sections and defaults? | 
					
						
							|  |  |  |         if section in self._sections: | 
					
						
							|  |  |  |             self.section(section)[0][key] = val | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             self.defaults(section)[0][key] = val | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-09 22:09:04 +00:00
										 |  |  |     def read(self, filename, sect=None): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """Parse configuration information from a file""" | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             with open(filename, 'rt') as config_file: | 
					
						
							| 
									
										
										
										
											2015-01-09 22:09:04 +00:00
										 |  |  |                 self._read(config_file, sect) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         except IOError: | 
					
						
							| 
									
										
										
										
											2018-04-18 09:27:51 +02:00
										 |  |  |             print("Could not open file " + filename + " for reading") | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-09 22:09:04 +00:00
										 |  |  |     def _read(self, config_file, sect): | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |         """Parse configuration information from the config_file""" | 
					
						
							|  |  |  |         is_comment = False  # used for multi-lined comments | 
					
						
							|  |  |  |         for line in config_file: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             line, is_comment = remove_comment(line, is_comment) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |             if not line: | 
					
						
							|  |  |  |                 # line was empty or was a comment | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             include_name = try_include(line) | 
					
						
							|  |  |  |             if include_name: | 
					
						
							| 
									
										
										
										
											2020-10-29 11:21:45 -04:00
										 |  |  |                 for incl in sorted(glob.iglob(include_name)): | 
					
						
							|  |  |  |                     parser = self.add_include(incl) | 
					
						
							|  |  |  |                     parser.read(incl, sect) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             section, is_template, templates = try_section(line) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |             if section: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |                 if section == DEFAULTSECT or is_template: | 
					
						
							|  |  |  |                     sect = self.add_default(section, templates) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |                     sect = self.add_section(section, templates) | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             key, val = try_option(line) | 
					
						
							| 
									
										
										
										
											2015-01-09 22:09:04 +00:00
										 |  |  |             if sect is None: | 
					
						
							|  |  |  |                 raise Exception("Section not defined before assignment") | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |             sect[key] = val | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |     def write(self, config_file): | 
					
						
							|  |  |  |         """Write configuration information out to a file""" | 
					
						
							| 
									
										
										
										
											2013-07-10 22:26:13 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |             for key, val in self._includes.iteritems(): | 
					
						
							|  |  |  |                 val.write(key) | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |                 config_file.write('#include "%s"\n' % key) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |             config_file.write('\n') | 
					
						
							|  |  |  |             write_dicts(config_file, self._defaults) | 
					
						
							|  |  |  |             write_dicts(config_file, self._sections) | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |         except: | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2013-10-31 22:09:47 +00:00
										 |  |  |                 with open(config_file, 'wt') as fp: | 
					
						
							| 
									
										
										
										
											2013-08-23 17:45:05 +00:00
										 |  |  |                     self.write(fp) | 
					
						
							|  |  |  |             except IOError: | 
					
						
							| 
									
										
										
										
											2018-04-18 09:27:51 +02:00
										 |  |  |                 print("Could not open file " + config_file + " for writing") |