~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/util/configobj/configobj.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2011-08-17 18:13:57 UTC
  • mfrom: (5268.7.29 transport-segments)
  • Revision ID: pqm@pqm.ubuntu.com-20110817181357-y5q5eth1hk8bl3om
(jelmer) Allow specifying the colocated branch to use in the branch URL,
 and retrieving the branch name using ControlDir._get_selected_branch.
 (Jelmer Vernooij)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
1
# configobj.py
2
2
# A config file reader/writer that supports nested sections in config files.
3
 
# Copyright (C) 2005-2008 Michael Foord, Nicola Larosa
 
3
# Copyright (C) 2005-2009 Michael Foord, Nicola Larosa
4
4
# E-mail: fuzzyman AT voidspace DOT org DOT uk
5
5
#         nico AT tekNico DOT net
6
6
 
16
16
# http://lists.sourceforge.net/lists/listinfo/configobj-develop
17
17
# Comments, suggestions and bug reports welcome.
18
18
 
 
19
 
19
20
from __future__ import generators
20
21
 
21
22
import sys
22
 
INTP_VER = sys.version_info[:2]
23
 
if INTP_VER < (2, 2):
24
 
    raise RuntimeError("Python v.2.2 or later needed")
 
23
import os
 
24
import re
25
25
 
26
 
import os, re
27
26
compiler = None
28
 
try:
29
 
    import compiler
30
 
except ImportError:
31
 
    # for IronPython
32
 
    pass
33
 
from types import StringTypes
34
 
from warnings import warn
 
27
# Bzr modification: Disabled import of 'compiler' module
 
28
# bzr doesn't use the 'unrepr' feature of configobj, so importing compiler just
 
29
# wastes several milliseconds on every single bzr invocation.
 
30
#   -- Andrew Bennetts, 2008-10-14
 
31
#try:
 
32
#    import compiler
 
33
#except ImportError:
 
34
#    # for IronPython
 
35
#    pass
 
36
 
 
37
 
35
38
try:
36
39
    from codecs import BOM_UTF8, BOM_UTF16, BOM_UTF16_BE, BOM_UTF16_LE
37
40
except ImportError:
97
100
squot = "'%s'"
98
101
dquot = '"%s"'
99
102
noquot = "%s"
100
 
wspace_plus = ' \r\t\n\v\t\'"'
 
103
wspace_plus = ' \r\n\v\t\'"'
101
104
tsquot = '"""%s"""'
102
105
tdquot = "'''%s'''"
103
106
 
111
114
            i += 1
112
115
            yield i, item
113
116
 
114
 
try:
115
 
    True, False
116
 
except NameError:
117
 
    True, False = 1, 0
118
 
 
119
 
 
120
 
__version__ = '4.5.2'
 
117
# Sentinel for use in getattr calls to replace hasattr
 
118
MISSING = object()
 
119
 
 
120
__version__ = '4.6.0'
121
121
 
122
122
__revision__ = '$Id: configobj.py 156 2006-01-31 14:57:08Z fuzzyman $'
123
123
 
180
180
 
181
181
 
182
182
class Builder(object):
183
 
    
 
183
 
184
184
    def build(self, o):
185
185
        m = getattr(self, 'build_' + o.__class__.__name__, None)
186
186
        if m is None:
187
187
            raise UnknownType(o.__class__.__name__)
188
188
        return m(o)
189
 
    
 
189
 
190
190
    def build_List(self, o):
191
191
        return map(self.build, o.getChildren())
192
 
    
 
192
 
193
193
    def build_Const(self, o):
194
194
        return o.value
195
 
    
 
195
 
196
196
    def build_Dict(self, o):
197
197
        d = {}
198
198
        i = iter(map(self.build, o.getChildren()))
199
199
        for el in i:
200
200
            d[el] = i.next()
201
201
        return d
202
 
    
 
202
 
203
203
    def build_Tuple(self, o):
204
204
        return tuple(self.build_List(o))
205
 
    
 
205
 
206
206
    def build_Name(self, o):
207
207
        if o.name == 'None':
208
208
            return None
210
210
            return True
211
211
        if o.name == 'False':
212
212
            return False
213
 
        
 
213
 
214
214
        # An undefined Name
215
215
        raise UnknownType('Undefined Name')
216
 
    
 
216
 
217
217
    def build_Add(self, o):
218
218
        real, imag = map(self.build_Const, o.getChildren())
219
219
        try:
223
223
        if not isinstance(imag, complex) or imag.real != 0.0:
224
224
            raise UnknownType('Add')
225
225
        return real+imag
226
 
    
 
226
 
227
227
    def build_Getattr(self, o):
228
228
        parent = self.build(o.expr)
229
229
        return getattr(parent, o.attrname)
230
 
    
 
230
 
231
231
    def build_UnarySub(self, o):
232
232
        return -self.build_Const(o.getChildren()[0])
233
 
    
 
233
 
234
234
    def build_UnaryAdd(self, o):
235
235
        return self.build_Const(o.getChildren()[0])
236
236
 
253
253
    def __init__(self, message='', line_number=None, line=''):
254
254
        self.line = line
255
255
        self.line_number = line_number
256
 
        self.message = message
257
256
        SyntaxError.__init__(self, message)
258
257
 
259
258
 
354
353
            This is similar to a depth-first-search algorithm.
355
354
            """
356
355
            # Have we been here already?
357
 
            if backtrail.has_key((key, section.name)):
 
356
            if (key, section.name) in backtrail:
358
357
                # Yes - infinite loop detected
359
358
                raise InterpolationLoopError(key)
360
359
            # Place a marker on our backtrail so we won't come back here again
442
441
        (e.g., if we interpolated "$$" and returned "$").
443
442
        """
444
443
        raise NotImplementedError()
445
 
    
 
444
 
446
445
 
447
446
 
448
447
class ConfigParserInterpolation(InterpolationEngine):
487
486
}
488
487
 
489
488
 
 
489
def __newobj__(cls, *args):
 
490
    # Hack for pickle
 
491
    return cls.__new__(cls, *args)
490
492
 
491
493
class Section(dict):
492
494
    """
493
495
    A dictionary-like object that represents a section in a config file.
494
 
    
 
496
 
495
497
    It does string interpolation if the 'interpolation' attribute
496
498
    of the 'main' object is set to True.
497
 
    
 
499
 
498
500
    Interpolation is tried first from this object, then from the 'DEFAULT'
499
501
    section of this object, next from the parent and its 'DEFAULT' section,
500
502
    and so on until the main object is reached.
501
 
    
 
503
 
502
504
    A Section will behave like an ordered dictionary - following the
503
505
    order of the ``scalars`` and ``sections`` attributes.
504
506
    You can use this to change the order of members.
505
 
    
 
507
 
506
508
    Iteration follows the order: scalars, then sections.
507
509
    """
508
510
 
 
511
 
 
512
    def __setstate__(self, state):
 
513
        dict.update(self, state[0])
 
514
        self.__dict__.update(state[1])
 
515
 
 
516
    def __reduce__(self):
 
517
        state = (dict(self), self.__dict__)
 
518
        return (__newobj__, (self.__class__,), state)
 
519
 
 
520
 
509
521
    def __init__(self, parent, depth, main, indict=None, name=None):
510
522
        """
511
523
        * parent is the section above
530
542
        # (rather than just passing to ``dict.__init__``)
531
543
        for entry, value in indict.iteritems():
532
544
            self[entry] = value
533
 
            
534
 
            
 
545
 
 
546
 
535
547
    def _initialise(self):
536
548
        # the sequence of scalar values in this Section
537
549
        self.scalars = []
540
552
        # for comments :-)
541
553
        self.comments = {}
542
554
        self.inline_comments = {}
543
 
        # for the configspec
544
 
        self.configspec = {}
545
 
        self._order = []
546
 
        self._configspec_comments = {}
547
 
        self._configspec_inline_comments = {}
548
 
        self._cs_section_comments = {}
549
 
        self._cs_section_inline_comments = {}
 
555
        # the configspec
 
556
        self.configspec = None
550
557
        # for defaults
551
558
        self.defaults = []
552
559
        self.default_values = {}
578
585
    def __getitem__(self, key):
579
586
        """Fetch the item and do string interpolation."""
580
587
        val = dict.__getitem__(self, key)
581
 
        if self.main.interpolation and isinstance(val, StringTypes):
 
588
        if self.main.interpolation and isinstance(val, basestring):
582
589
            return self._interpolate(key, val)
583
590
        return val
584
591
 
586
593
    def __setitem__(self, key, value, unrepr=False):
587
594
        """
588
595
        Correctly set a value.
589
 
        
 
596
 
590
597
        Making dictionary values Section instances.
591
598
        (We have to special case 'Section' instances - which are also dicts)
592
 
        
 
599
 
593
600
        Keys must be strings.
594
601
        Values need only be strings (or lists of strings) if
595
602
        ``main.stringify`` is set.
596
 
        
597
 
        `unrepr`` must be set when setting a value to a dictionary, without
 
603
 
 
604
        ``unrepr`` must be set when setting a value to a dictionary, without
598
605
        creating a new sub-section.
599
606
        """
600
 
        if not isinstance(key, StringTypes):
 
607
        if not isinstance(key, basestring):
601
608
            raise ValueError('The key "%s" is not a string.' % key)
602
 
        
 
609
 
603
610
        # add the comment
604
 
        if not self.comments.has_key(key):
 
611
        if key not in self.comments:
605
612
            self.comments[key] = []
606
613
            self.inline_comments[key] = ''
607
614
        # remove the entry from defaults
609
616
            self.defaults.remove(key)
610
617
        #
611
618
        if isinstance(value, Section):
612
 
            if not self.has_key(key):
 
619
            if key not in self:
613
620
                self.sections.append(key)
614
621
            dict.__setitem__(self, key, value)
615
622
        elif isinstance(value, dict) and not unrepr:
616
623
            # First create the new depth level,
617
624
            # then create the section
618
 
            if not self.has_key(key):
 
625
            if key not in self:
619
626
                self.sections.append(key)
620
627
            new_depth = self.depth + 1
621
628
            dict.__setitem__(
628
635
                    indict=value,
629
636
                    name=key))
630
637
        else:
631
 
            if not self.has_key(key):
 
638
            if key not in self:
632
639
                self.scalars.append(key)
633
640
            if not self.main.stringify:
634
 
                if isinstance(value, StringTypes):
 
641
                if isinstance(value, basestring):
635
642
                    pass
636
643
                elif isinstance(value, (list, tuple)):
637
644
                    for entry in value:
638
 
                        if not isinstance(entry, StringTypes):
 
645
                        if not isinstance(entry, basestring):
639
646
                            raise TypeError('Value is not a string "%s".' % entry)
640
647
                else:
641
648
                    raise TypeError('Value is not a string "%s".' % value)
683
690
            del self.comments[key]
684
691
            del self.inline_comments[key]
685
692
            self.sections.remove(key)
686
 
        if self.main.interpolation and isinstance(val, StringTypes):
 
693
        if self.main.interpolation and isinstance(val, basestring):
687
694
            return self._interpolate(key, val)
688
695
        return val
689
696
 
703
710
        """
704
711
        A version of clear that also affects scalars/sections
705
712
        Also clears comments and configspec.
706
 
        
 
713
 
707
714
        Leaves other attributes alone :
708
715
            depth/main/parent are not affected
709
716
        """
712
719
        self.sections = []
713
720
        self.comments = {}
714
721
        self.inline_comments = {}
715
 
        self.configspec = {}
 
722
        self.configspec = None
716
723
 
717
724
 
718
725
    def setdefault(self, key, default=None):
770
777
    def dict(self):
771
778
        """
772
779
        Return a deepcopy of self as a dictionary.
773
 
        
 
780
 
774
781
        All members that are ``Section`` instances are recursively turned to
775
782
        ordinary dictionaries - by calling their ``dict`` method.
776
 
        
 
783
 
777
784
        >>> n = a.dict()
778
785
        >>> n == a
779
786
        1
798
805
    def merge(self, indict):
799
806
        """
800
807
        A recursive update - useful for merging config files.
801
 
        
 
808
 
802
809
        >>> a = '''[section1]
803
810
        ...     option1 = True
804
811
        ...     [[subsection]]
812
819
        >>> c2 = ConfigObj(a)
813
820
        >>> c2.merge(c1)
814
821
        >>> c2
815
 
        {'section1': {'option1': 'False', 'subsection': {'more_options': 'False'}}}
 
822
        ConfigObj({'section1': {'option1': 'False', 'subsection': {'more_options': 'False'}}})
816
823
        """
817
824
        for key, val in indict.items():
818
825
            if (key in self and isinstance(self[key], dict) and
819
826
                                isinstance(val, dict)):
820
827
                self[key].merge(val)
821
 
            else:   
 
828
            else:
822
829
                self[key] = val
823
830
 
824
831
 
825
832
    def rename(self, oldkey, newkey):
826
833
        """
827
834
        Change a keyname to another, without changing position in sequence.
828
 
        
 
835
 
829
836
        Implemented so that transformations can be made on keys,
830
837
        as well as on values. (used by encode and decode)
831
 
        
 
838
 
832
839
        Also renames comments.
833
840
        """
834
841
        if oldkey in self.scalars:
856
863
            call_on_sections=False, **keywargs):
857
864
        """
858
865
        Walk every member and call a function on the keyword and value.
859
 
        
 
866
 
860
867
        Return a dictionary of the return values
861
 
        
 
868
 
862
869
        If the function raises an exception, raise the errror
863
870
        unless ``raise_errors=False``, in which case set the return value to
864
871
        ``False``.
865
 
        
 
872
 
866
873
        Any unrecognised keyword arguments you pass to walk, will be pased on
867
874
        to the function you pass in.
868
 
        
 
875
 
869
876
        Note: if ``call_on_sections`` is ``True`` then - on encountering a
870
877
        subsection, *first* the function is called for the *whole* subsection,
871
878
        and then recurses into it's members. This means your function must be
872
879
        able to handle strings, dictionaries and lists. This allows you
873
880
        to change the key of subsections as well as for ordinary members. The
874
881
        return value when called on the whole subsection has to be discarded.
875
 
        
 
882
 
876
883
        See  the encode and decode methods for examples, including functions.
877
 
        
878
 
        .. caution::
879
 
        
 
884
 
 
885
        .. admonition:: caution
 
886
 
880
887
            You can use ``walk`` to transform the names of members of a section
881
888
            but you mustn't add or delete members.
882
 
        
 
889
 
883
890
        >>> config = '''[XXXXsection]
884
891
        ... XXXXkey = XXXXvalue'''.splitlines()
885
892
        >>> cfg = ConfigObj(config)
886
893
        >>> cfg
887
 
        {'XXXXsection': {'XXXXkey': 'XXXXvalue'}}
 
894
        ConfigObj({'XXXXsection': {'XXXXkey': 'XXXXvalue'}})
888
895
        >>> def transform(section, key):
889
896
        ...     val = section[key]
890
897
        ...     newkey = key.replace('XXXX', 'CLIENT1')
897
904
        >>> cfg.walk(transform, call_on_sections=True)
898
905
        {'CLIENT1section': {'CLIENT1key': None}}
899
906
        >>> cfg
900
 
        {'CLIENT1section': {'CLIENT1key': 'CLIENT1value'}}
 
907
        ConfigObj({'CLIENT1section': {'CLIENT1key': 'CLIENT1value'}})
901
908
        """
902
909
        out = {}
903
910
        # scalars first
937
944
        return out
938
945
 
939
946
 
940
 
    def decode(self, encoding):
941
 
        """
942
 
        Decode all strings and values to unicode, using the specified encoding.
943
 
        
944
 
        Works with subsections and list values.
945
 
        
946
 
        Uses the ``walk`` method.
947
 
        
948
 
        Testing ``encode`` and ``decode``.
949
 
        >>> m = ConfigObj(a)
950
 
        >>> m.decode('ascii')
951
 
        >>> def testuni(val):
952
 
        ...     for entry in val:
953
 
        ...         if not isinstance(entry, unicode):
954
 
        ...             print >> sys.stderr, type(entry)
955
 
        ...             raise AssertionError, 'decode failed.'
956
 
        ...         if isinstance(val[entry], dict):
957
 
        ...             testuni(val[entry])
958
 
        ...         elif not isinstance(val[entry], unicode):
959
 
        ...             raise AssertionError, 'decode failed.'
960
 
        >>> testuni(m)
961
 
        >>> m.encode('ascii')
962
 
        >>> a == m
963
 
        1
964
 
        """
965
 
        warn('use of ``decode`` is deprecated.', DeprecationWarning)
966
 
        def decode(section, key, encoding=encoding, warn=True):
967
 
            """ """
968
 
            val = section[key]
969
 
            if isinstance(val, (list, tuple)):
970
 
                newval = []
971
 
                for entry in val:
972
 
                    newval.append(entry.decode(encoding))
973
 
            elif isinstance(val, dict):
974
 
                newval = val
975
 
            else:
976
 
                newval = val.decode(encoding)
977
 
            newkey = key.decode(encoding)
978
 
            section.rename(key, newkey)
979
 
            section[newkey] = newval
980
 
        # using ``call_on_sections`` allows us to modify section names
981
 
        self.walk(decode, call_on_sections=True)
982
 
 
983
 
 
984
 
    def encode(self, encoding):
985
 
        """
986
 
        Encode all strings and values from unicode,
987
 
        using the specified encoding.
988
 
        
989
 
        Works with subsections and list values.
990
 
        Uses the ``walk`` method.
991
 
        """
992
 
        warn('use of ``encode`` is deprecated.', DeprecationWarning)
993
 
        def encode(section, key, encoding=encoding):
994
 
            """ """
995
 
            val = section[key]
996
 
            if isinstance(val, (list, tuple)):
997
 
                newval = []
998
 
                for entry in val:
999
 
                    newval.append(entry.encode(encoding))
1000
 
            elif isinstance(val, dict):
1001
 
                newval = val
1002
 
            else:
1003
 
                newval = val.encode(encoding)
1004
 
            newkey = key.encode(encoding)
1005
 
            section.rename(key, newkey)
1006
 
            section[newkey] = newval
1007
 
        self.walk(encode, call_on_sections=True)
1008
 
 
1009
 
 
1010
 
    def istrue(self, key):
1011
 
        """A deprecated version of ``as_bool``."""
1012
 
        warn('use of ``istrue`` is deprecated. Use ``as_bool`` method '
1013
 
                'instead.', DeprecationWarning)
1014
 
        return self.as_bool(key)
1015
 
 
1016
 
 
1017
947
    def as_bool(self, key):
1018
948
        """
1019
949
        Accepts a key as input. The corresponding value must be a string or
1020
950
        the objects (``True`` or 1) or (``False`` or 0). We allow 0 and 1 to
1021
951
        retain compatibility with Python 2.2.
1022
 
        
1023
 
        If the string is one of  ``True``, ``On``, ``Yes``, or ``1`` it returns 
 
952
 
 
953
        If the string is one of  ``True``, ``On``, ``Yes``, or ``1`` it returns
1024
954
        ``True``.
1025
 
        
1026
 
        If the string is one of  ``False``, ``Off``, ``No``, or ``0`` it returns 
 
955
 
 
956
        If the string is one of  ``False``, ``Off``, ``No``, or ``0`` it returns
1027
957
        ``False``.
1028
 
        
 
958
 
1029
959
        ``as_bool`` is not case sensitive.
1030
 
        
 
960
 
1031
961
        Any other input will raise a ``ValueError``.
1032
 
        
 
962
 
1033
963
        >>> a = ConfigObj()
1034
964
        >>> a['a'] = 'fish'
1035
965
        >>> a.as_bool('a')
1049
979
            return False
1050
980
        else:
1051
981
            try:
1052
 
                if not isinstance(val, StringTypes):
 
982
                if not isinstance(val, basestring):
1053
983
                    # TODO: Why do we raise a KeyError here?
1054
984
                    raise KeyError()
1055
985
                else:
1061
991
    def as_int(self, key):
1062
992
        """
1063
993
        A convenience method which coerces the specified value to an integer.
1064
 
        
 
994
 
1065
995
        If the value is an invalid literal for ``int``, a ``ValueError`` will
1066
996
        be raised.
1067
 
        
 
997
 
1068
998
        >>> a = ConfigObj()
1069
999
        >>> a['a'] = 'fish'
1070
1000
        >>> a.as_int('a')
1071
1001
        Traceback (most recent call last):
1072
 
        ValueError: invalid literal for int(): fish
 
1002
        ValueError: invalid literal for int() with base 10: 'fish'
1073
1003
        >>> a['b'] = '1'
1074
1004
        >>> a.as_int('b')
1075
1005
        1
1076
1006
        >>> a['b'] = '3.2'
1077
1007
        >>> a.as_int('b')
1078
1008
        Traceback (most recent call last):
1079
 
        ValueError: invalid literal for int(): 3.2
 
1009
        ValueError: invalid literal for int() with base 10: '3.2'
1080
1010
        """
1081
1011
        return int(self[key])
1082
1012
 
1084
1014
    def as_float(self, key):
1085
1015
        """
1086
1016
        A convenience method which coerces the specified value to a float.
1087
 
        
 
1017
 
1088
1018
        If the value is an invalid literal for ``float``, a ``ValueError`` will
1089
1019
        be raised.
1090
 
        
 
1020
 
1091
1021
        >>> a = ConfigObj()
1092
1022
        >>> a['a'] = 'fish'
1093
1023
        >>> a.as_float('a')
1103
1033
        return float(self[key])
1104
1034
 
1105
1035
 
 
1036
    def as_list(self, key):
 
1037
        """
 
1038
        A convenience method which fetches the specified value, guaranteeing
 
1039
        that it is a list.
 
1040
 
 
1041
        >>> a = ConfigObj()
 
1042
        >>> a['a'] = 1
 
1043
        >>> a.as_list('a')
 
1044
        [1]
 
1045
        >>> a['a'] = (1,)
 
1046
        >>> a.as_list('a')
 
1047
        [1]
 
1048
        >>> a['a'] = [1]
 
1049
        >>> a.as_list('a')
 
1050
        [1]
 
1051
        """
 
1052
        result = self[key]
 
1053
        if isinstance(result, (tuple, list)):
 
1054
            return list(result)
 
1055
        return [result]
 
1056
 
 
1057
 
1106
1058
    def restore_default(self, key):
1107
1059
        """
1108
1060
        Restore (and return) default value for the specified key.
1109
 
        
 
1061
 
1110
1062
        This method will only work for a ConfigObj that was created
1111
1063
        with a configspec and has been validated.
1112
 
        
 
1064
 
1113
1065
        If there is no default value for this key, ``KeyError`` is raised.
1114
1066
        """
1115
1067
        default = self.default_values[key]
1118
1070
            self.defaults.append(key)
1119
1071
        return default
1120
1072
 
1121
 
    
 
1073
 
1122
1074
    def restore_defaults(self):
1123
1075
        """
1124
1076
        Recursively restore default values to all members
1125
1077
        that have them.
1126
 
        
 
1078
 
1127
1079
        This method will only work for a ConfigObj that was created
1128
1080
        with a configspec and has been validated.
1129
 
        
 
1081
 
1130
1082
        It doesn't delete or modify entries without default values.
1131
1083
        """
1132
1084
        for key in self.default_values:
1133
1085
            self.restore_default(key)
1134
 
            
 
1086
 
1135
1087
        for section in self.sections:
1136
1088
            self[section].restore_defaults()
1137
1089
 
1239
1191
        }
1240
1192
 
1241
1193
 
1242
 
    def __init__(self, infile=None, options=None, **kwargs):
 
1194
    def __init__(self, infile=None, options=None, _inspec=False, **kwargs):
1243
1195
        """
1244
1196
        Parse a config file or create a config file object.
1245
 
        
 
1197
 
1246
1198
        ``ConfigObj(infile=None, options=None, **kwargs)``
1247
1199
        """
 
1200
        self._inspec = _inspec
1248
1201
        # init the superclass
1249
1202
        Section.__init__(self, self, 0, self)
1250
 
        
1251
 
        if infile is None:
1252
 
            infile = []
1253
 
        if options is None:
1254
 
            options = {}
1255
 
        else:
1256
 
            options = dict(options)
1257
 
            
 
1203
 
 
1204
        infile = infile or []
 
1205
        options = dict(options or {})
 
1206
 
1258
1207
        # keyword arguments take precedence over an options dictionary
1259
1208
        options.update(kwargs)
1260
 
        
 
1209
        if _inspec:
 
1210
            options['list_values'] = False
 
1211
 
1261
1212
        defaults = OPTION_DEFAULTS.copy()
1262
1213
        # TODO: check the values too.
1263
1214
        for entry in options:
1264
1215
            if entry not in defaults:
1265
1216
                raise TypeError('Unrecognised option "%s".' % entry)
1266
 
        
 
1217
 
1267
1218
        # Add any explicit options to the defaults
1268
1219
        defaults.update(options)
1269
1220
        self._initialise(defaults)
1270
1221
        configspec = defaults['configspec']
1271
1222
        self._original_configspec = configspec
1272
1223
        self._load(infile, configspec)
1273
 
        
1274
 
        
 
1224
 
 
1225
 
1275
1226
    def _load(self, infile, configspec):
1276
 
        if isinstance(infile, StringTypes):
 
1227
        if isinstance(infile, basestring):
1277
1228
            self.filename = infile
1278
1229
            if os.path.isfile(infile):
1279
1230
                h = open(infile, 'rb')
1291
1242
                    h.write('')
1292
1243
                    h.close()
1293
1244
                infile = []
1294
 
                
 
1245
 
1295
1246
        elif isinstance(infile, (list, tuple)):
1296
1247
            infile = list(infile)
1297
 
            
 
1248
 
1298
1249
        elif isinstance(infile, dict):
1299
1250
            # initialise self
1300
1251
            # the Section class handles creating subsections
1301
1252
            if isinstance(infile, ConfigObj):
1302
1253
                # get a copy of our ConfigObj
1303
1254
                infile = infile.dict()
1304
 
                
 
1255
 
1305
1256
            for entry in infile:
1306
1257
                self[entry] = infile[entry]
1307
1258
            del self._errors
1308
 
            
 
1259
 
1309
1260
            if configspec is not None:
1310
1261
                self._handle_configspec(configspec)
1311
1262
            else:
1312
1263
                self.configspec = None
1313
1264
            return
1314
 
        
1315
 
        elif getattr(infile, 'read', None) is not None:
 
1265
 
 
1266
        elif getattr(infile, 'read', MISSING) is not MISSING:
1316
1267
            # This supports file like objects
1317
1268
            infile = infile.read() or []
1318
1269
            # needs splitting into lines - but needs doing *after* decoding
1319
1270
            # in case it's not an 8 bit encoding
1320
1271
        else:
1321
1272
            raise TypeError('infile must be a filename, file like object, or list of lines.')
1322
 
        
 
1273
 
1323
1274
        if infile:
1324
1275
            # don't do it for the empty ConfigObj
1325
1276
            infile = self._handle_bom(infile)
1337
1288
                break
1338
1289
 
1339
1290
            infile = [line.rstrip('\r\n') for line in infile]
1340
 
            
 
1291
 
1341
1292
        self._parse(infile)
1342
1293
        # if we had any errors, now is the time to raise them
1343
1294
        if self._errors:
1355
1306
            raise error
1356
1307
        # delete private attributes
1357
1308
        del self._errors
1358
 
        
 
1309
 
1359
1310
        if configspec is None:
1360
1311
            self.configspec = None
1361
1312
        else:
1362
1313
            self._handle_configspec(configspec)
1363
 
    
1364
 
    
 
1314
 
 
1315
 
1365
1316
    def _initialise(self, options=None):
1366
1317
        if options is None:
1367
1318
            options = OPTION_DEFAULTS
1368
 
            
 
1319
 
1369
1320
        # initialise a few variables
1370
1321
        self.filename = None
1371
1322
        self._errors = []
1382
1333
        self.newlines = None
1383
1334
        self.write_empty_values = options['write_empty_values']
1384
1335
        self.unrepr = options['unrepr']
1385
 
        
 
1336
 
1386
1337
        self.initial_comment = []
1387
1338
        self.final_comment = []
1388
 
        self.configspec = {}
1389
 
        
 
1339
        self.configspec = None
 
1340
 
 
1341
        if self._inspec:
 
1342
            self.list_values = False
 
1343
 
1390
1344
        # Clear section attributes as well
1391
1345
        Section._initialise(self)
1392
 
        
1393
 
        
 
1346
 
 
1347
 
1394
1348
    def __repr__(self):
1395
 
        return ('ConfigObj({%s})' % 
1396
 
                ', '.join([('%s: %s' % (repr(key), repr(self[key]))) 
 
1349
        return ('ConfigObj({%s})' %
 
1350
                ', '.join([('%s: %s' % (repr(key), repr(self[key])))
1397
1351
                for key in (self.scalars + self.sections)]))
1398
 
    
1399
 
    
 
1352
 
 
1353
 
1400
1354
    def _handle_bom(self, infile):
1401
1355
        """
1402
1356
        Handle any BOM, and decode if necessary.
1403
 
        
 
1357
 
1404
1358
        If an encoding is specified, that *must* be used - but the BOM should
1405
1359
        still be removed (and the BOM attribute set).
1406
 
        
 
1360
 
1407
1361
        (If the encoding is wrongly specified, then a BOM for an alternative
1408
1362
        encoding won't be discovered or removed.)
1409
 
        
 
1363
 
1410
1364
        If an encoding is not specified, UTF8 or UTF16 BOM will be detected and
1411
1365
        removed. The BOM attribute will be set. UTF16 will be decoded to
1412
1366
        unicode.
1413
 
        
 
1367
 
1414
1368
        NOTE: This method must not be called with an empty ``infile``.
1415
 
        
 
1369
 
1416
1370
        Specifying the *wrong* encoding is likely to cause a
1417
1371
        ``UnicodeDecodeError``.
1418
 
        
 
1372
 
1419
1373
        ``infile`` must always be returned as a list of lines, but may be
1420
1374
        passed in as a single string.
1421
1375
        """
1425
1379
            # the encoding specified doesn't have one
1426
1380
            # just decode
1427
1381
            return self._decode(infile, self.encoding)
1428
 
        
 
1382
 
1429
1383
        if isinstance(infile, (list, tuple)):
1430
1384
            line = infile[0]
1431
1385
        else:
1447
1401
                        ##self.BOM = True
1448
1402
                        # Don't need to remove BOM
1449
1403
                        return self._decode(infile, encoding)
1450
 
                    
 
1404
 
1451
1405
                # If we get this far, will *probably* raise a DecodeError
1452
1406
                # As it doesn't appear to start with a BOM
1453
1407
                return self._decode(infile, self.encoding)
1454
 
            
 
1408
 
1455
1409
            # Must be UTF8
1456
1410
            BOM = BOM_SET[enc]
1457
1411
            if not line.startswith(BOM):
1458
1412
                return self._decode(infile, self.encoding)
1459
 
            
 
1413
 
1460
1414
            newline = line[len(BOM):]
1461
 
            
 
1415
 
1462
1416
            # BOM removed
1463
1417
            if isinstance(infile, (list, tuple)):
1464
1418
                infile[0] = newline
1466
1420
                infile = newline
1467
1421
            self.BOM = True
1468
1422
            return self._decode(infile, self.encoding)
1469
 
        
 
1423
 
1470
1424
        # No encoding specified - so we need to check for UTF8/UTF16
1471
1425
        for BOM, (encoding, final_encoding) in BOMS.items():
1472
1426
            if not line.startswith(BOM):
1484
1438
                    else:
1485
1439
                        infile = newline
1486
1440
                    # UTF8 - don't decode
1487
 
                    if isinstance(infile, StringTypes):
 
1441
                    if isinstance(infile, basestring):
1488
1442
                        return infile.splitlines(True)
1489
1443
                    else:
1490
1444
                        return infile
1491
1445
                # UTF16 - have to decode
1492
1446
                return self._decode(infile, encoding)
1493
 
            
 
1447
 
1494
1448
        # No BOM discovered and no encoding specified, just return
1495
 
        if isinstance(infile, StringTypes):
 
1449
        if isinstance(infile, basestring):
1496
1450
            # infile read from a file will be a single string
1497
1451
            return infile.splitlines(True)
1498
1452
        return infile
1509
1463
    def _decode(self, infile, encoding):
1510
1464
        """
1511
1465
        Decode infile to unicode. Using the specified encoding.
1512
 
        
 
1466
 
1513
1467
        if is a string, it also needs converting to a list.
1514
1468
        """
1515
 
        if isinstance(infile, StringTypes):
 
1469
        if isinstance(infile, basestring):
1516
1470
            # can't be unicode
1517
1471
            # NOTE: Could raise a ``UnicodeDecodeError``
1518
1472
            return infile.decode(encoding).splitlines(True)
1539
1493
        Used by ``stringify`` within validate, to turn non-string values
1540
1494
        into strings.
1541
1495
        """
1542
 
        if not isinstance(value, StringTypes):
 
1496
        if not isinstance(value, basestring):
1543
1497
            return str(value)
1544
1498
        else:
1545
1499
            return value
1550
1504
        temp_list_values = self.list_values
1551
1505
        if self.unrepr:
1552
1506
            self.list_values = False
1553
 
            
 
1507
 
1554
1508
        comment_list = []
1555
1509
        done_start = False
1556
1510
        this_section = self
1557
1511
        maxline = len(infile) - 1
1558
1512
        cur_index = -1
1559
1513
        reset_comment = False
1560
 
        
 
1514
 
1561
1515
        while cur_index < maxline:
1562
1516
            if reset_comment:
1563
1517
                comment_list = []
1569
1523
                reset_comment = False
1570
1524
                comment_list.append(line)
1571
1525
                continue
1572
 
            
 
1526
 
1573
1527
            if not done_start:
1574
1528
                # preserve initial comment
1575
1529
                self.initial_comment = comment_list
1576
1530
                comment_list = []
1577
1531
                done_start = True
1578
 
                
 
1532
 
1579
1533
            reset_comment = True
1580
1534
            # first we check if it's a section marker
1581
1535
            mat = self._sectionmarker.match(line)
1589
1543
                    self._handle_error("Cannot compute the section depth at line %s.",
1590
1544
                                       NestingError, infile, cur_index)
1591
1545
                    continue
1592
 
                
 
1546
 
1593
1547
                if cur_depth < this_section.depth:
1594
1548
                    # the new section is dropping back to a previous level
1595
1549
                    try:
1608
1562
                else:
1609
1563
                    self._handle_error("Section too nested at line %s.",
1610
1564
                                       NestingError, infile, cur_index)
1611
 
                    
 
1565
 
1612
1566
                sect_name = self._unquote(sect_name)
1613
 
                if parent.has_key(sect_name):
 
1567
                if sect_name in parent:
1614
1568
                    self._handle_error('Duplicate section name at line %s.',
1615
1569
                                       DuplicateError, infile, cur_index)
1616
1570
                    continue
1617
 
                
 
1571
 
1618
1572
                # create the new section
1619
1573
                this_section = Section(
1620
1574
                    parent,
1688
1642
                            continue
1689
1643
                #
1690
1644
                key = self._unquote(key)
1691
 
                if this_section.has_key(key):
 
1645
                if key in this_section:
1692
1646
                    self._handle_error(
1693
1647
                        'Duplicate keyword name at line %s.',
1694
1648
                        DuplicateError, infile, cur_index)
1717
1671
        """
1718
1672
        Given a section and a depth level, walk back through the sections
1719
1673
        parents to see if the depth level matches a previous section.
1720
 
        
 
1674
 
1721
1675
        Return a reference to the right section,
1722
1676
        or raise a SyntaxError.
1723
1677
        """
1735
1689
    def _handle_error(self, text, ErrorClass, infile, cur_index):
1736
1690
        """
1737
1691
        Handle an error according to the error settings.
1738
 
        
 
1692
 
1739
1693
        Either raise the error or store it.
1740
1694
        The error will have occured at ``cur_index``
1741
1695
        """
1761
1715
    def _quote(self, value, multiline=True):
1762
1716
        """
1763
1717
        Return a safely quoted version of a value.
1764
 
        
 
1718
 
1765
1719
        Raise a ConfigObjError if the value cannot be safely quoted.
1766
1720
        If multiline is ``True`` (default) then use triple quotes
1767
1721
        if necessary.
1768
 
        
1769
 
        Don't quote values that don't need it.
1770
 
        Recursively quote members of a list and return a comma joined list.
1771
 
        Multiline is ``False`` for lists.
1772
 
        Obey list syntax for empty and single member lists.
1773
 
        
 
1722
 
 
1723
        * Don't quote values that don't need it.
 
1724
        * Recursively quote members of a list and return a comma joined list.
 
1725
        * Multiline is ``False`` for lists.
 
1726
        * Obey list syntax for empty and single member lists.
 
1727
 
1774
1728
        If ``list_values=False`` then the value is only quoted if it contains
1775
 
        a ``\n`` (is multiline) or '#'.
1776
 
        
 
1729
        a ``\\n`` (is multiline) or '#'.
 
1730
 
1777
1731
        If ``write_empty_values`` is set, and the value is an empty string, it
1778
1732
        won't be quoted.
1779
1733
        """
1781
1735
            # Only if multiline is set, so that it is used for values not
1782
1736
            # keys, and not values that are part of a list
1783
1737
            return ''
1784
 
        
 
1738
 
1785
1739
        if multiline and isinstance(value, (list, tuple)):
1786
1740
            if not value:
1787
1741
                return ','
1789
1743
                return self._quote(value[0], multiline=False) + ','
1790
1744
            return ', '.join([self._quote(val, multiline=False)
1791
1745
                for val in value])
1792
 
        if not isinstance(value, StringTypes):
 
1746
        if not isinstance(value, basestring):
1793
1747
            if self.stringify:
1794
1748
                value = str(value)
1795
1749
            else:
1797
1751
 
1798
1752
        if not value:
1799
1753
            return '""'
1800
 
        
 
1754
 
1801
1755
        no_lists_no_quotes = not self.list_values and '\n' not in value and '#' not in value
1802
1756
        need_triple = multiline and ((("'" in value) and ('"' in value)) or ('\n' in value ))
1803
1757
        hash_triple_quote = multiline and not need_triple and ("'" in value) and ('"' in value) and ('#' in value)
1804
1758
        check_for_single = (no_lists_no_quotes or not need_triple) and not hash_triple_quote
1805
 
        
 
1759
 
1806
1760
        if check_for_single:
1807
1761
            if not self.list_values:
1808
1762
                # we don't quote if ``list_values=False``
1820
1774
        else:
1821
1775
            # if value has '\n' or "'" *and* '"', it will need triple quotes
1822
1776
            quot = self._get_triple_quote(value)
1823
 
        
 
1777
 
1824
1778
        if quot == noquot and '#' in value and self.list_values:
1825
1779
            quot = self._get_single_quote(value)
1826
 
                
 
1780
 
1827
1781
        return quot % value
1828
 
    
1829
 
    
 
1782
 
 
1783
 
1830
1784
    def _get_single_quote(self, value):
1831
1785
        if ("'" in value) and ('"' in value):
1832
1786
            raise ConfigObjError('Value "%s" cannot be safely quoted.' % value)
1835
1789
        else:
1836
1790
            quot = dquot
1837
1791
        return quot
1838
 
    
1839
 
    
 
1792
 
 
1793
 
1840
1794
    def _get_triple_quote(self, value):
1841
1795
        if (value.find('"""') != -1) and (value.find("'''") != -1):
1842
1796
            raise ConfigObjError('Value "%s" cannot be safely quoted.' % value)
 
1797
        # upstream version (up to version 4.7.2) has the bug with incorrect quoting;
 
1798
        # fixed in our copy based on the suggestion of ConfigObj's author
1843
1799
        if value.find('"""') == -1:
 
1800
            quot = tsquot
 
1801
        else:
1844
1802
            quot = tdquot
1845
 
        else:
1846
 
            quot = tsquot 
1847
1803
        return quot
1848
1804
 
1849
1805
 
1852
1808
        Given a value string, unquote, remove comment,
1853
1809
        handle lists. (including empty and single member lists)
1854
1810
        """
 
1811
        if self._inspec:
 
1812
            # Parsing a configspec so don't handle comments
 
1813
            return (value, '')
1855
1814
        # do we look for lists in values ?
1856
1815
        if not self.list_values:
1857
1816
            mat = self._nolistvalue.match(value)
1930
1889
 
1931
1890
    def _handle_configspec(self, configspec):
1932
1891
        """Parse the configspec."""
1933
 
        # FIXME: Should we check that the configspec was created with the 
 
1892
        # FIXME: Should we check that the configspec was created with the
1934
1893
        #        correct settings ? (i.e. ``list_values=False``)
1935
1894
        if not isinstance(configspec, ConfigObj):
1936
1895
            try:
1937
1896
                configspec = ConfigObj(configspec,
1938
1897
                                       raise_errors=True,
1939
1898
                                       file_error=True,
1940
 
                                       list_values=False)
 
1899
                                       _inspec=True)
1941
1900
            except ConfigObjError, e:
1942
1901
                # FIXME: Should these errors have a reference
1943
1902
                #        to the already parsed ConfigObj ?
1944
1903
                raise ConfigspecError('Parsing configspec failed: %s' % e)
1945
1904
            except IOError, e:
1946
1905
                raise IOError('Reading configspec failed: %s' % e)
1947
 
        
1948
 
        self._set_configspec_value(configspec, self)
1949
 
 
1950
 
 
1951
 
    def _set_configspec_value(self, configspec, section):
1952
 
        """Used to recursively set configspec values."""
1953
 
        if '__many__' in configspec.sections:
1954
 
            section.configspec['__many__'] = configspec['__many__']
1955
 
            if len(configspec.sections) > 1:
1956
 
                # FIXME: can we supply any useful information here ?
1957
 
                raise RepeatSectionError()
1958
 
            
1959
 
        if getattr(configspec, 'initial_comment', None) is not None:
1960
 
            section._configspec_initial_comment = configspec.initial_comment
1961
 
            section._configspec_final_comment = configspec.final_comment
1962
 
            section._configspec_encoding = configspec.encoding
1963
 
            section._configspec_BOM = configspec.BOM
1964
 
            section._configspec_newlines = configspec.newlines
1965
 
            section._configspec_indent_type = configspec.indent_type
1966
 
            
1967
 
        for entry in configspec.scalars:
1968
 
            section._configspec_comments[entry] = configspec.comments[entry]
1969
 
            section._configspec_inline_comments[entry] = configspec.inline_comments[entry]
1970
 
            section.configspec[entry] = configspec[entry]
1971
 
            section._order.append(entry)
1972
 
            
 
1906
 
 
1907
        self.configspec = configspec
 
1908
 
 
1909
 
 
1910
 
 
1911
    def _set_configspec(self, section, copy):
 
1912
        """
 
1913
        Called by validate. Handles setting the configspec on subsections
 
1914
        including sections to be validated by __many__
 
1915
        """
 
1916
        configspec = section.configspec
 
1917
        many = configspec.get('__many__')
 
1918
        if isinstance(many, dict):
 
1919
            for entry in section.sections:
 
1920
                if entry not in configspec:
 
1921
                    section[entry].configspec = many
 
1922
 
1973
1923
        for entry in configspec.sections:
1974
1924
            if entry == '__many__':
1975
1925
                continue
1976
 
            
1977
 
            section._cs_section_comments[entry] = configspec.comments[entry]
1978
 
            section._cs_section_inline_comments[entry] = configspec.inline_comments[entry]
1979
 
            if not section.has_key(entry):
1980
 
                section[entry] = {}
1981
 
            self._set_configspec_value(configspec[entry], section[entry])
1982
 
 
1983
 
 
1984
 
    def _handle_repeat(self, section, configspec):
1985
 
        """Dynamically assign configspec for repeated section."""
1986
 
        try:
1987
 
            section_keys = configspec.sections
1988
 
            scalar_keys = configspec.scalars
1989
 
        except AttributeError:
1990
 
            section_keys = [entry for entry in configspec 
1991
 
                                if isinstance(configspec[entry], dict)]
1992
 
            scalar_keys = [entry for entry in configspec 
1993
 
                                if not isinstance(configspec[entry], dict)]
1994
 
            
1995
 
        if '__many__' in section_keys and len(section_keys) > 1:
1996
 
            # FIXME: can we supply any useful information here ?
1997
 
            raise RepeatSectionError()
1998
 
        
1999
 
        scalars = {}
2000
 
        sections = {}
2001
 
        for entry in scalar_keys:
2002
 
            val = configspec[entry]
2003
 
            scalars[entry] = val
2004
 
        for entry in section_keys:
2005
 
            val = configspec[entry]
2006
 
            if entry == '__many__':
2007
 
                scalars[entry] = val
2008
 
                continue
2009
 
            sections[entry] = val
2010
 
            
2011
 
        section.configspec = scalars
2012
 
        for entry in sections:
2013
 
            if not section.has_key(entry):
2014
 
                section[entry] = {}
2015
 
            self._handle_repeat(section[entry], sections[entry])
 
1926
            if entry not in section:
 
1927
                section[entry] = {}
 
1928
                if copy:
 
1929
                    # copy comments
 
1930
                    section.comments[entry] = configspec.comments.get(entry, [])
 
1931
                    section.inline_comments[entry] = configspec.inline_comments.get(entry, '')
 
1932
 
 
1933
            # Could be a scalar when we expect a section
 
1934
            if isinstance(section[entry], Section):
 
1935
                section[entry].configspec = configspec[entry]
2016
1936
 
2017
1937
 
2018
1938
    def _write_line(self, indent_string, entry, this_entry, comment):
2053
1973
    def write(self, outfile=None, section=None):
2054
1974
        """
2055
1975
        Write the current ConfigObj as a file
2056
 
        
 
1976
 
2057
1977
        tekNico: FIXME: use StringIO instead of real files
2058
 
        
 
1978
 
2059
1979
        >>> filename = a.filename
2060
1980
        >>> a.filename = 'test.ini'
2061
1981
        >>> a.write()
2066
1986
        if self.indent_type is None:
2067
1987
            # this can be true if initialised from a dictionary
2068
1988
            self.indent_type = DEFAULT_INDENT_TYPE
2069
 
            
 
1989
 
2070
1990
        out = []
2071
1991
        cs = self._a_to_u('#')
2072
1992
        csp = self._a_to_u('# ')
2080
2000
                if stripped_line and not stripped_line.startswith(cs):
2081
2001
                    line = csp + line
2082
2002
                out.append(line)
2083
 
                
 
2003
 
2084
2004
        indent_string = self.indent_type * section.depth
2085
2005
        for entry in (section.scalars + section.sections):
2086
2006
            if entry in section.defaults:
2093
2013
                out.append(indent_string + comment_line)
2094
2014
            this_entry = section[entry]
2095
2015
            comment = self._handle_comment(section.inline_comments[entry])
2096
 
            
 
2016
 
2097
2017
            if isinstance(this_entry, dict):
2098
2018
                # a section
2099
2019
                out.append(self._write_marker(
2108
2028
                    entry,
2109
2029
                    this_entry,
2110
2030
                    comment))
2111
 
                
 
2031
 
2112
2032
        if section is self:
2113
2033
            for line in self.final_comment:
2114
2034
                line = self._decode_element(line)
2117
2037
                    line = csp + line
2118
2038
                out.append(line)
2119
2039
            self.interpolation = int_val
2120
 
            
 
2040
 
2121
2041
        if section is not self:
2122
2042
            return out
2123
 
        
 
2043
 
2124
2044
        if (self.filename is None) and (outfile is None):
2125
2045
            # output a list of lines
2126
2046
            # might need to encode
2134
2054
                    out.append('')
2135
2055
                out[0] = BOM_UTF8 + out[0]
2136
2056
            return out
2137
 
        
 
2057
 
2138
2058
        # Turn the list to a string, joined with correct newlines
2139
2059
        newline = self.newlines or os.linesep
2140
2060
        output = self._a_to_u(newline).join(out)
2143
2063
        if self.BOM and ((self.encoding is None) or match_utf8(self.encoding)):
2144
2064
            # Add the UTF8 BOM
2145
2065
            output = BOM_UTF8 + output
2146
 
            
 
2066
 
2147
2067
        if not output.endswith(newline):
2148
2068
            output += newline
2149
2069
        if outfile is not None:
2158
2078
                 section=None):
2159
2079
        """
2160
2080
        Test the ConfigObj against a configspec.
2161
 
        
 
2081
 
2162
2082
        It uses the ``validator`` object from *validate.py*.
2163
 
        
 
2083
 
2164
2084
        To run ``validate`` on the current ConfigObj, call: ::
2165
 
        
 
2085
 
2166
2086
            test = config.validate(validator)
2167
 
        
 
2087
 
2168
2088
        (Normally having previously passed in the configspec when the ConfigObj
2169
2089
        was created - you can dynamically assign a dictionary of checks to the
2170
2090
        ``configspec`` attribute of a section though).
2171
 
        
 
2091
 
2172
2092
        It returns ``True`` if everything passes, or a dictionary of
2173
2093
        pass/fails (True/False). If every member of a subsection passes, it
2174
2094
        will just have the value ``True``. (It also returns ``False`` if all
2175
2095
        members fail).
2176
 
        
 
2096
 
2177
2097
        In addition, it converts the values from strings to their native
2178
2098
        types if their checks pass (and ``stringify`` is set).
2179
 
        
 
2099
 
2180
2100
        If ``preserve_errors`` is ``True`` (``False`` is default) then instead
2181
2101
        of a marking a fail with a ``False``, it will preserve the actual
2182
2102
        exception object. This can contain info about the reason for failure.
2183
2103
        For example the ``VdtValueTooSmallError`` indicates that the value
2184
2104
        supplied was too small. If a value (or section) is missing it will
2185
2105
        still be marked as ``False``.
2186
 
        
 
2106
 
2187
2107
        You must have the validate module to use ``preserve_errors=True``.
2188
 
        
 
2108
 
2189
2109
        You can then use the ``flatten_errors`` function to turn your nested
2190
2110
        results dictionary into a flattened list of failures - useful for
2191
2111
        displaying meaningful error messages.
2198
2118
                # Which makes importing configobj faster
2199
2119
                from validate import VdtMissingValue
2200
2120
                self._vdtMissingValue = VdtMissingValue
 
2121
 
2201
2122
            section = self
2202
 
        #
2203
 
        spec_section = section.configspec
2204
 
        if copy and getattr(section, '_configspec_initial_comment', None) is not None:
2205
 
            section.initial_comment = section._configspec_initial_comment
2206
 
            section.final_comment = section._configspec_final_comment
2207
 
            section.encoding = section._configspec_encoding
2208
 
            section.BOM = section._configspec_BOM
2209
 
            section.newlines = section._configspec_newlines
2210
 
            section.indent_type = section._configspec_indent_type
2211
 
            
2212
 
        if '__many__' in section.configspec:
2213
 
            many = spec_section['__many__']
2214
 
            # dynamically assign the configspecs
2215
 
            # for the sections below
2216
 
            for entry in section.sections:
2217
 
                self._handle_repeat(section[entry], many)
2218
 
        #
2219
 
        out = {}
2220
 
        ret_true = True
2221
 
        ret_false = True
2222
 
        order = [k for k in section._order if k in spec_section]
2223
 
        order += [k for k in spec_section if k not in order]
2224
 
        for entry in order:
2225
 
            if entry == '__many__':
2226
 
                continue
2227
 
            if (not entry in section.scalars) or (entry in section.defaults):
2228
 
                # missing entries
2229
 
                # or entries from defaults
2230
 
                missing = True
2231
 
                val = None
2232
 
                if copy and not entry in section.scalars:
2233
 
                    # copy comments
2234
 
                    section.comments[entry] = (
2235
 
                        section._configspec_comments.get(entry, []))
2236
 
                    section.inline_comments[entry] = (
2237
 
                        section._configspec_inline_comments.get(entry, ''))
2238
 
                #
2239
 
            else:
2240
 
                missing = False
2241
 
                val = section[entry]
 
2123
 
 
2124
            if copy:
 
2125
                section.initial_comment = section.configspec.initial_comment
 
2126
                section.final_comment = section.configspec.final_comment
 
2127
                section.encoding = section.configspec.encoding
 
2128
                section.BOM = section.configspec.BOM
 
2129
                section.newlines = section.configspec.newlines
 
2130
                section.indent_type = section.configspec.indent_type
 
2131
 
 
2132
        #
 
2133
        configspec = section.configspec
 
2134
        self._set_configspec(section, copy)
 
2135
 
 
2136
        def validate_entry(entry, spec, val, missing, ret_true, ret_false):
2242
2137
            try:
2243
 
                check = validator.check(spec_section[entry],
 
2138
                check = validator.check(spec,
2244
2139
                                        val,
2245
2140
                                        missing=missing
2246
2141
                                        )
2253
2148
                    ret_false = False
2254
2149
                ret_true = False
2255
2150
            else:
2256
 
                try: 
 
2151
                try:
2257
2152
                    section.default_values.pop(entry, None)
2258
 
                except AttributeError: 
 
2153
                except AttributeError:
2259
2154
                    # For Python 2.2 compatibility
2260
2155
                    try:
2261
2156
                        del section.default_values[entry]
2262
2157
                    except KeyError:
2263
2158
                        pass
2264
 
                    
2265
 
                if getattr(validator, 'get_default_value', None) is not None:
2266
 
                    try: 
2267
 
                        section.default_values[entry] = validator.get_default_value(spec_section[entry])
2268
 
                    except KeyError:
2269
 
                        # No default
2270
 
                        pass
2271
 
                    
 
2159
 
 
2160
                try:
 
2161
                    section.default_values[entry] = validator.get_default_value(configspec[entry])
 
2162
                except (KeyError, AttributeError):
 
2163
                    # No default or validator has no 'get_default_value' (e.g. SimpleVal)
 
2164
                    pass
 
2165
 
2272
2166
                ret_false = False
2273
2167
                out[entry] = True
2274
2168
                if self.stringify or missing:
2287
2181
                        section[entry] = check
2288
2182
                if not copy and missing and entry not in section.defaults:
2289
2183
                    section.defaults.append(entry)
 
2184
            return ret_true, ret_false
 
2185
 
 
2186
        #
 
2187
        out = {}
 
2188
        ret_true = True
 
2189
        ret_false = True
 
2190
 
 
2191
        unvalidated = [k for k in section.scalars if k not in configspec]
 
2192
        incorrect_sections = [k for k in configspec.sections if k in section.scalars]
 
2193
        incorrect_scalars = [k for k in configspec.scalars if k in section.sections]
 
2194
 
 
2195
        for entry in configspec.scalars:
 
2196
            if entry in ('__many__', '___many___'):
 
2197
                # reserved names
 
2198
                continue
 
2199
 
 
2200
            if (not entry in section.scalars) or (entry in section.defaults):
 
2201
                # missing entries
 
2202
                # or entries from defaults
 
2203
                missing = True
 
2204
                val = None
 
2205
                if copy and not entry in section.scalars:
 
2206
                    # copy comments
 
2207
                    section.comments[entry] = (
 
2208
                        configspec.comments.get(entry, []))
 
2209
                    section.inline_comments[entry] = (
 
2210
                        configspec.inline_comments.get(entry, ''))
 
2211
                #
 
2212
            else:
 
2213
                missing = False
 
2214
                val = section[entry]
 
2215
 
 
2216
            ret_true, ret_false = validate_entry(entry, configspec[entry], val,
 
2217
                                                 missing, ret_true, ret_false)
 
2218
 
 
2219
        many = None
 
2220
        if '__many__' in configspec.scalars:
 
2221
            many = configspec['__many__']
 
2222
        elif '___many___' in configspec.scalars:
 
2223
            many = configspec['___many___']
 
2224
 
 
2225
        if many is not None:
 
2226
            for entry in unvalidated:
 
2227
                val = section[entry]
 
2228
                ret_true, ret_false = validate_entry(entry, many, val, False,
 
2229
                                                     ret_true, ret_false)
 
2230
 
 
2231
        for entry in incorrect_scalars:
 
2232
            ret_true = False
 
2233
            if not preserve_errors:
 
2234
                out[entry] = False
 
2235
            else:
 
2236
                ret_false = False
 
2237
                msg = 'Value %r was provided as a section' % entry
 
2238
                out[entry] = validator.baseErrorClass(msg)
 
2239
        for entry in incorrect_sections:
 
2240
            ret_true = False
 
2241
            if not preserve_errors:
 
2242
                out[entry] = False
 
2243
            else:
 
2244
                ret_false = False
 
2245
                msg = 'Section %r was provided as a single value' % entry
 
2246
                out[entry] = validator.baseErrorClass(msg)
 
2247
 
2290
2248
        # Missing sections will have been created as empty ones when the
2291
2249
        # configspec was read.
2292
2250
        for entry in section.sections:
2293
2251
            # FIXME: this means DEFAULT is not copied in copy mode
2294
2252
            if section is self and entry == 'DEFAULT':
2295
2253
                continue
 
2254
            if section[entry].configspec is None:
 
2255
                continue
2296
2256
            if copy:
2297
 
                section.comments[entry] = section._cs_section_comments[entry]
2298
 
                section.inline_comments[entry] = (
2299
 
                    section._cs_section_inline_comments[entry])
2300
 
            check = self.validate(validator, preserve_errors=preserve_errors,
2301
 
                copy=copy, section=section[entry])
 
2257
                section.comments[entry] = configspec.comments.get(entry, [])
 
2258
                section.inline_comments[entry] = configspec.inline_comments.get(entry, '')
 
2259
            check = self.validate(validator, preserve_errors=preserve_errors, copy=copy, section=section[entry])
2302
2260
            out[entry] = check
2303
2261
            if check == False:
2304
2262
                ret_true = False
2324
2282
        self.configspec = None
2325
2283
        # Just to be sure ;-)
2326
2284
        self._original_configspec = None
2327
 
        
2328
 
        
 
2285
 
 
2286
 
2329
2287
    def reload(self):
2330
2288
        """
2331
2289
        Reload a ConfigObj from file.
2332
 
        
 
2290
 
2333
2291
        This method raises a ``ReloadError`` if the ConfigObj doesn't have
2334
2292
        a filename attribute pointing to a file.
2335
2293
        """
2336
 
        if not isinstance(self.filename, StringTypes):
 
2294
        if not isinstance(self.filename, basestring):
2337
2295
            raise ReloadError()
2338
2296
 
2339
2297
        filename = self.filename
2342
2300
            if entry == 'configspec':
2343
2301
                continue
2344
2302
            current_options[entry] = getattr(self, entry)
2345
 
            
 
2303
 
2346
2304
        configspec = self._original_configspec
2347
2305
        current_options['configspec'] = configspec
2348
 
            
 
2306
 
2349
2307
        self.clear()
2350
2308
        self._initialise(current_options)
2351
2309
        self._load(filename, configspec)
2352
 
        
 
2310
 
2353
2311
 
2354
2312
 
2355
2313
class SimpleVal(object):
2356
2314
    """
2357
2315
    A simple validator.
2358
2316
    Can be used to check that all members expected are present.
2359
 
    
 
2317
 
2360
2318
    To use it, provide a configspec with all your members in (the value given
2361
2319
    will be ignored). Pass an instance of ``SimpleVal`` to the ``validate``
2362
2320
    method of your ``ConfigObj``. ``validate`` will return ``True`` if all
2363
2321
    members are present, or a dictionary with True/False meaning
2364
2322
    present/missing. (Whole missing sections will be replaced with ``False``)
2365
2323
    """
2366
 
    
 
2324
 
2367
2325
    def __init__(self):
2368
2326
        self.baseErrorClass = ConfigObjError
2369
 
    
 
2327
 
2370
2328
    def check(self, check, member, missing=False):
2371
2329
        """A dummy check method, always returns the value unchanged."""
2372
2330
        if missing:
2379
2337
    """
2380
2338
    An example function that will turn a nested dictionary of results
2381
2339
    (as returned by ``ConfigObj.validate``) into a flat list.
2382
 
    
 
2340
 
2383
2341
    ``cfg`` is the ConfigObj instance being checked, ``res`` is the results
2384
2342
    dictionary returned by ``validate``.
2385
 
    
 
2343
 
2386
2344
    (This is a recursive function, so you shouldn't use the ``levels`` or
2387
 
    ``results`` arguments - they are used by the function.
2388
 
    
 
2345
    ``results`` arguments - they are used by the function.)
 
2346
 
2389
2347
    Returns a list of keys that failed. Each member of the list is a tuple :
 
2348
 
2390
2349
    ::
2391
 
    
 
2350
 
2392
2351
        ([list of sections...], key, result)
2393
 
    
 
2352
 
2394
2353
    If ``validate`` was called with ``preserve_errors=False`` (the default)
2395
2354
    then ``result`` will always be ``False``.
2396
2355
 
2397
2356
    *list of sections* is a flattened list of sections that the key was found
2398
2357
    in.
2399
 
    
2400
 
    If the section was missing then key will be ``None``.
2401
 
    
 
2358
 
 
2359
    If the section was missing (or a section was expected and a scalar provided
 
2360
    - or vice-versa) then key will be ``None``.
 
2361
 
2402
2362
    If the value (or section) was missing then ``result`` will be ``False``.
2403
 
    
 
2363
 
2404
2364
    If ``validate`` was called with ``preserve_errors=True`` and a value
2405
2365
    was present, but failed the check, then ``result`` will be the exception
2406
2366
    object returned. You can use this as a string that describes the failure.
2407
 
    
 
2367
 
2408
2368
    For example *The value "3" is of the wrong type*.
2409
 
    
 
2369
 
2410
2370
    >>> import validate
2411
2371
    >>> vtor = validate.Validator()
2412
2372
    >>> my_ini = '''
2476
2436
        results = []
2477
2437
    if res is True:
2478
2438
        return results
2479
 
    if res is False:
2480
 
        results.append((levels[:], None, False))
 
2439
    if res is False or isinstance(res, Exception):
 
2440
        results.append((levels[:], None, res))
2481
2441
        if levels:
2482
2442
            levels.pop()
2483
2443
        return results