73
73
74
74
75
75
class MetaParserBuilder (abc .ABCMeta ): # noqa: B024
76
- """Utility class wrapping base-class methods into decorators that assure read-only properties."""
76
+ """Utility class wrapping base-class methods into decorators that assure read-only
77
+ properties."""
77
78
78
79
def __new__ (cls , name : str , bases : Tuple , clsdict : Dict [str , Any ]) -> "MetaParserBuilder" :
79
80
"""Equip all base-class methods with a needs_values decorator, and all non-const
80
- methods with a set_dirty_and_flush_changes decorator in addition to that.
81
+ methods with a :func:`set_dirty_and_flush_changes` decorator in addition to
82
+ that.
81
83
"""
82
84
kmm = "_mutating_methods_"
83
85
if kmm in clsdict :
@@ -102,7 +104,8 @@ def __new__(cls, name: str, bases: Tuple, clsdict: Dict[str, Any]) -> "MetaParse
102
104
103
105
104
106
def needs_values (func : Callable [..., _T ]) -> Callable [..., _T ]:
105
- """Return a method for ensuring we read values (on demand) before we try to access them."""
107
+ """Return a method for ensuring we read values (on demand) before we try to access
108
+ them."""
106
109
107
110
@wraps (func )
108
111
def assure_data_present (self : "GitConfigParser" , * args : Any , ** kwargs : Any ) -> _T :
@@ -116,7 +119,8 @@ def assure_data_present(self: "GitConfigParser", *args: Any, **kwargs: Any) -> _
116
119
def set_dirty_and_flush_changes (non_const_func : Callable [..., _T ]) -> Callable [..., _T ]:
117
120
"""Return a method that checks whether given non constant function may be called.
118
121
119
- If so, the instance will be set dirty. Additionally, we flush the changes right to disk.
122
+ If so, the instance will be set dirty. Additionally, we flush the changes right to
123
+ disk.
120
124
"""
121
125
122
126
def flush_changes (self : "GitConfigParser" , * args : Any , ** kwargs : Any ) -> _T :
@@ -136,7 +140,8 @@ class SectionConstraint(Generic[T_ConfigParser]):
136
140
137
141
It supports all ConfigParser methods that operate on an option.
138
142
139
- :note: If used as a context manager, will release the wrapped ConfigParser.
143
+ :note:
144
+ If used as a context manager, will release the wrapped ConfigParser.
140
145
"""
141
146
142
147
__slots__ = ("_config" , "_section_name" )
@@ -171,8 +176,8 @@ def __getattr__(self, attr: str) -> Any:
171
176
return super ().__getattribute__ (attr )
172
177
173
178
def _call_config (self , method : str , * args : Any , ** kwargs : Any ) -> Any :
174
- """Call the configuration at the given method which must take a section name
175
- as first argument."""
179
+ """Call the configuration at the given method which must take a section name as
180
+ first argument."""
176
181
return getattr (self ._config , method )(self ._section_name , * args , ** kwargs )
177
182
178
183
@property
@@ -254,7 +259,8 @@ def get_config_path(config_level: Lit_config_levels) -> str:
254
259
elif config_level == "repository" :
255
260
raise ValueError ("No repo to get repository configuration from. Use Repo._get_config_path" )
256
261
else :
257
- # Should not reach here. Will raise ValueError if does. Static typing will warn missing elifs
262
+ # Should not reach here. Will raise ValueError if does. Static typing will warn
263
+ # about missing elifs.
258
264
assert_never ( # type: ignore[unreachable]
259
265
config_level ,
260
266
ValueError (f"Invalid configuration level: { config_level !r} " ),
@@ -264,14 +270,15 @@ def get_config_path(config_level: Lit_config_levels) -> str:
264
270
class GitConfigParser (cp .RawConfigParser , metaclass = MetaParserBuilder ):
265
271
"""Implements specifics required to read git style configuration files.
266
272
267
- This variation behaves much like the git.config command such that the configuration
268
- will be read on demand based on the filepath given during initialization.
273
+ This variation behaves much like the ``git config`` command, such that the
274
+ configuration will be read on demand based on the filepath given during
275
+ initialization.
269
276
270
277
The changes will automatically be written once the instance goes out of scope, but
271
278
can be triggered manually as well.
272
279
273
- The configuration file will be locked if you intend to change values preventing other
274
- instances to write concurrently.
280
+ The configuration file will be locked if you intend to change values preventing
281
+ other instances to write concurrently.
275
282
276
283
:note:
277
284
The config is case-sensitive even when queried, hence section and option names
@@ -301,7 +308,8 @@ class GitConfigParser(cp.RawConfigParser, metaclass=MetaParserBuilder):
301
308
del optvalueonly_source
302
309
303
310
_mutating_methods_ = ("add_section" , "remove_section" , "remove_option" , "set" )
304
- """List of RawConfigParser methods able to change the instance."""
311
+ """Names of :class:`~configparser.RawConfigParser` methods able to change the
312
+ instance."""
305
313
306
314
def __init__ (
307
315
self ,
@@ -311,8 +319,8 @@ def __init__(
311
319
config_level : Union [Lit_config_levels , None ] = None ,
312
320
repo : Union ["Repo" , None ] = None ,
313
321
) -> None :
314
- """Initialize a configuration reader to read the given file_or_files and to
315
- possibly allow changes to it by setting read_only False.
322
+ """Initialize a configuration reader to read the given ` file_or_files` and to
323
+ possibly allow changes to it by setting ` read_only` False.
316
324
317
325
:param file_or_files:
318
326
A file path or file object, or a sequence of possibly more than one of them.
@@ -385,7 +393,7 @@ def _acquire_lock(self) -> None:
385
393
# END read-only check
386
394
387
395
def __del__ (self ) -> None :
388
- """Write pending changes if required and release locks"""
396
+ """Write pending changes if required and release locks. """
389
397
# NOTE: Only consistent in Python 2.
390
398
self .release ()
391
399
@@ -397,10 +405,12 @@ def __exit__(self, *args: Any) -> None:
397
405
self .release ()
398
406
399
407
def release (self ) -> None :
400
- """Flush changes and release the configuration write lock. This instance must not be used anymore afterwards.
408
+ """Flush changes and release the configuration write lock. This instance must
409
+ not be used anymore afterwards.
401
410
402
- In Python 3, it's required to explicitly release locks and flush changes, as __del__ is not called
403
- deterministically anymore."""
411
+ In Python 3, it's required to explicitly release locks and flush changes, as
412
+ :meth:`__del__` is not called deterministically anymore.
413
+ """
404
414
# Checking for the lock here makes sure we do not raise during write()
405
415
# in case an invalid parser was created who could not get a lock.
406
416
if self .read_only or (self ._lock and not self ._lock ._has_lock ()):
@@ -424,8 +434,9 @@ def optionxform(self, optionstr: str) -> str:
424
434
return optionstr
425
435
426
436
def _read (self , fp : Union [BufferedReader , IO [bytes ]], fpname : str ) -> None :
427
- """Originally a direct copy of the Python 2.4 version of RawConfigParser._read,
428
- to ensure it uses ordered dicts.
437
+ """Originally a direct copy of the Python 2.4 version of
438
+ :meth:`RawConfigParser._read <configparser.RawConfigParser._read>`, to ensure it
439
+ uses ordered dicts.
429
440
430
441
The ordering bug was fixed in Python 2.4, and dict itself keeps ordering since
431
442
Python 3.7. This has some other changes, especially that it ignores initial
@@ -525,7 +536,8 @@ def _has_includes(self) -> Union[bool, int]:
525
536
def _included_paths (self ) -> List [Tuple [str , str ]]:
526
537
"""List all paths that must be included to configuration.
527
538
528
- :return: The list of paths, where each path is a tuple of ``(option, value)``.
539
+ :return:
540
+ The list of paths, where each path is a tuple of ``(option, value)``.
529
541
"""
530
542
paths = []
531
543
@@ -577,8 +589,11 @@ def read(self) -> None: # type: ignore[override]
577
589
This will ignore files that cannot be read, possibly leaving an empty
578
590
configuration.
579
591
580
- :return: Nothing
581
- :raise IOError: If a file cannot be handled
592
+ :return:
593
+ Nothing
594
+
595
+ :raise IOError:
596
+ If a file cannot be handled
582
597
"""
583
598
if self ._is_initialized :
584
599
return
@@ -591,7 +606,7 @@ def read(self) -> None: # type: ignore[override]
591
606
elif not isinstance (self ._file_or_files , (tuple , list , Sequence )):
592
607
# Could merge with above isinstance once runtime type known.
593
608
files_to_read = [self ._file_or_files ]
594
- else : # for lists or tuples
609
+ else : # For lists or tuples.
595
610
files_to_read = list (self ._file_or_files )
596
611
# END ensure we have a copy of the paths to handle
597
612
@@ -603,7 +618,8 @@ def read(self) -> None: # type: ignore[override]
603
618
604
619
if hasattr (file_path , "seek" ):
605
620
# Must be a file-object.
606
- file_path = cast (IO [bytes ], file_path ) # TODO: Replace with assert to narrow type, once sure.
621
+ # TODO: Replace cast with assert to narrow type, once sure.
622
+ file_path = cast (IO [bytes ], file_path )
607
623
self ._read (file_path , file_path .name )
608
624
else :
609
625
# Assume a path if it is not a file-object.
@@ -615,8 +631,8 @@ def read(self) -> None: # type: ignore[override]
615
631
except IOError :
616
632
continue
617
633
618
- # Read includes and append those that we didn't handle yet.
619
- # We expect all paths to be normalized and absolute (and will ensure that is the case).
634
+ # Read includes and append those that we didn't handle yet. We expect all
635
+ # paths to be normalized and absolute (and will ensure that is the case).
620
636
if self ._has_includes ():
621
637
for _ , include_path in self ._included_paths ():
622
638
if include_path .startswith ("~" ):
@@ -695,8 +711,9 @@ def items_all(self, section_name: str) -> List[Tuple[str, List[str]]]:
695
711
def write (self ) -> None :
696
712
"""Write changes to our file, if there are changes at all.
697
713
698
- :raise IOError: If this is a read-only writer instance or if we could not obtain
699
- a file lock"""
714
+ :raise IOError:
715
+ If this is a read-only writer instance or if we could not obtain a file lock
716
+ """
700
717
self ._assure_writable ("write" )
701
718
if not self ._dirty :
702
719
return
@@ -740,7 +757,7 @@ def _assure_writable(self, method_name: str) -> None:
740
757
raise IOError ("Cannot execute non-constant method %s.%s" % (self , method_name ))
741
758
742
759
def add_section (self , section : str ) -> None :
743
- """Assures added options will stay in order"""
760
+ """Assures added options will stay in order. """
744
761
return super ().add_section (section )
745
762
746
763
@property
@@ -757,16 +774,18 @@ def get_value(
757
774
) -> Union [int , float , str , bool ]:
758
775
"""Get an option's value.
759
776
760
- If multiple values are specified for this option in the section, the
761
- last one specified is returned.
777
+ If multiple values are specified for this option in the section, the last one
778
+ specified is returned.
762
779
763
780
:param default:
764
- If not None, the given default value will be returned in case
765
- the option did not exist
781
+ If not None, the given default value will be returned in case the option did
782
+ not exist
766
783
767
- :return: a properly typed value, either int, float or string
784
+ :return:
785
+ A properly typed value, either int, float or string
768
786
769
- :raise TypeError: in case the value could not be understood
787
+ :raise TypeError:
788
+ In case the value could not be understood.
770
789
Otherwise the exceptions known to the ConfigParser will be raised.
771
790
"""
772
791
try :
@@ -790,12 +809,14 @@ def get_values(
790
809
returned.
791
810
792
811
:param default:
793
- If not None, a list containing the given default value will be
794
- returned in case the option did not exist
812
+ If not None, a list containing the given default value will be returned in
813
+ case the option did not exist.
795
814
796
- :return: a list of properly typed values, either int, float or string
815
+ :return:
816
+ A list of properly typed values, either int, float or string
797
817
798
- :raise TypeError: in case the value could not be understood
818
+ :raise TypeError:
819
+ In case the value could not be understood.
799
820
Otherwise the exceptions known to the ConfigParser will be raised.
800
821
"""
801
822
try :
@@ -849,11 +870,17 @@ def set_value(self, section: str, option: str, value: Union[str, bytes, int, flo
849
870
This will create the section if required, and will not throw as opposed to the
850
871
default ConfigParser 'set' method.
851
872
852
- :param section: Name of the section in which the option resides or should reside
853
- :param option: Name of the options whose value to set
854
- :param value: Value to set the option to. It must be a string or convertible to
855
- a string.
856
- :return: This instance
873
+ :param section:
874
+ Name of the section in which the option resides or should reside.
875
+
876
+ :param option:
877
+ Name of the options whose value to set.
878
+
879
+ :param value:
880
+ Value to set the option to. It must be a string or convertible to a string.
881
+
882
+ :return:
883
+ This instance
857
884
"""
858
885
if not self .has_section (section ):
859
886
self .add_section (section )
@@ -865,15 +892,22 @@ def set_value(self, section: str, option: str, value: Union[str, bytes, int, flo
865
892
def add_value (self , section : str , option : str , value : Union [str , bytes , int , float , bool ]) -> "GitConfigParser" :
866
893
"""Add a value for the given option in section.
867
894
868
- This will create the section if required, and will not throw as opposed to the default
869
- ConfigParser 'set' method. The value becomes the new value of the option as returned
870
- by 'get_value', and appends to the list of values returned by 'get_values`'.
895
+ This will create the section if required, and will not throw as opposed to the
896
+ default ConfigParser 'set' method. The value becomes the new value of the option
897
+ as returned by 'get_value', and appends to the list of values returned by
898
+ 'get_values'.
871
899
872
- :param section: Name of the section in which the option resides or should reside
873
- :param option: Name of the option
874
- :param value: Value to add to option. It must be a string or convertible
875
- to a string
876
- :return: This instance
900
+ :param section:
901
+ Name of the section in which the option resides or should reside.
902
+
903
+ :param option:
904
+ Name of the option.
905
+
906
+ :param value:
907
+ Value to add to option. It must be a string or convertible to a string.
908
+
909
+ :return:
910
+ This instance
877
911
"""
878
912
if not self .has_section (section ):
879
913
self .add_section (section )
@@ -883,8 +917,12 @@ def add_value(self, section: str, option: str, value: Union[str, bytes, int, flo
883
917
def rename_section (self , section : str , new_name : str ) -> "GitConfigParser" :
884
918
"""Rename the given section to new_name.
885
919
886
- :raise ValueError: If ``section`` doesn't exist
887
- :raise ValueError: If a section with ``new_name`` does already exist
920
+ :raise ValueError:
921
+ If:
922
+
923
+ * ``section`` doesn't exist.
924
+ * A section with ``new_name`` does already exist.
925
+
888
926
:return: This instance
889
927
"""
890
928
if not self .has_section (section ):
@@ -898,6 +936,7 @@ def rename_section(self, section: str, new_name: str) -> "GitConfigParser":
898
936
new_section .setall (k , vs )
899
937
# END for each value to copy
900
938
901
- # This call writes back the changes, which is why we don't have the respective decorator.
939
+ # This call writes back the changes, which is why we don't have the respective
940
+ # decorator.
902
941
self .remove_section (section )
903
942
return self
0 commit comments