2007-04-17 16:48:32 +08:00
|
|
|
"""A collection of string constants.
|
1997-12-30 03:26:28 +08:00
|
|
|
|
|
|
|
Public module variables:
|
|
|
|
|
2008-11-22 16:31:09 +08:00
|
|
|
whitespace -- a string containing all ASCII whitespace
|
|
|
|
ascii_lowercase -- a string containing all ASCII lowercase letters
|
|
|
|
ascii_uppercase -- a string containing all ASCII uppercase letters
|
|
|
|
ascii_letters -- a string containing all ASCII letters
|
|
|
|
digits -- a string containing all ASCII decimal digits
|
|
|
|
hexdigits -- a string containing all ASCII hexadecimal digits
|
|
|
|
octdigits -- a string containing all ASCII octal digits
|
|
|
|
punctuation -- a string containing all ASCII punctuation characters
|
|
|
|
printable -- a string containing all ASCII characters considered printable
|
1997-12-30 03:26:28 +08:00
|
|
|
|
|
|
|
"""
|
|
|
|
|
2016-06-05 03:35:05 +08:00
|
|
|
__all__ = ["ascii_letters", "ascii_lowercase", "ascii_uppercase", "capwords",
|
|
|
|
"digits", "hexdigits", "octdigits", "printable", "punctuation",
|
|
|
|
"whitespace", "Formatter", "Template"]
|
|
|
|
|
2010-10-14 15:04:07 +08:00
|
|
|
import _string
|
|
|
|
|
1990-10-14 03:23:40 +08:00
|
|
|
# Some strings for ctype-style character classification
|
1993-07-29 17:37:38 +08:00
|
|
|
whitespace = ' \t\n\r\v\f'
|
2007-08-14 17:23:10 +08:00
|
|
|
ascii_lowercase = 'abcdefghijklmnopqrstuvwxyz'
|
|
|
|
ascii_uppercase = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
|
2001-07-21 02:38:26 +08:00
|
|
|
ascii_letters = ascii_lowercase + ascii_uppercase
|
1990-10-14 03:23:40 +08:00
|
|
|
digits = '0123456789'
|
|
|
|
hexdigits = digits + 'abcdef' + 'ABCDEF'
|
|
|
|
octdigits = '01234567'
|
2016-09-09 01:59:53 +08:00
|
|
|
punctuation = r"""!"#$%&'()*+,-./:;<=>?@[\]^_`{|}~"""
|
2007-08-14 17:23:10 +08:00
|
|
|
printable = digits + ascii_letters + punctuation + whitespace
|
1990-10-14 03:23:40 +08:00
|
|
|
|
2004-08-25 10:22:30 +08:00
|
|
|
# Functions which aren't available as string methods.
|
|
|
|
|
|
|
|
# Capitalize the words in a string, e.g. " aBc dEf " -> "Abc Def".
|
|
|
|
def capwords(s, sep=None):
|
2009-09-26 20:33:22 +08:00
|
|
|
"""capwords(s [,sep]) -> string
|
2004-08-25 10:22:30 +08:00
|
|
|
|
|
|
|
Split the argument into words using split, capitalize each
|
|
|
|
word using capitalize, and join the capitalized words using
|
2009-09-26 20:33:22 +08:00
|
|
|
join. If the optional second argument sep is absent or None,
|
|
|
|
runs of whitespace characters are replaced by a single space
|
|
|
|
and leading and trailing whitespace are removed, otherwise
|
|
|
|
sep is used to split and join the words.
|
2004-08-25 10:22:30 +08:00
|
|
|
|
|
|
|
"""
|
2021-09-17 03:49:38 +08:00
|
|
|
return (sep or ' ').join(map(str.capitalize, s.split(sep)))
|
2004-08-25 10:22:30 +08:00
|
|
|
|
|
|
|
|
2004-08-26 08:21:13 +08:00
|
|
|
####################################################################
|
2004-08-25 10:22:30 +08:00
|
|
|
import re as _re
|
2016-06-05 03:35:05 +08:00
|
|
|
from collections import ChainMap as _ChainMap
|
2004-09-13 22:35:04 +08:00
|
|
|
|
2019-06-01 16:00:15 +08:00
|
|
|
_sentinel_dict = {}
|
|
|
|
|
2019-10-21 14:36:21 +08:00
|
|
|
class Template:
|
2004-08-25 10:22:30 +08:00
|
|
|
"""A string class for supporting $-substitutions."""
|
2004-09-10 11:08:08 +08:00
|
|
|
|
2004-09-18 08:06:34 +08:00
|
|
|
delimiter = '$'
|
2017-11-21 23:28:13 +08:00
|
|
|
# r'[a-z]' matches to non-ASCII letters when used with IGNORECASE, but
|
|
|
|
# without the ASCII flag. We can't add re.ASCII to flags because of
|
|
|
|
# backward compatibility. So we use the ?a local flag and [a-z] pattern.
|
2017-10-13 15:02:23 +08:00
|
|
|
# See https://bugs.python.org/issue31672
|
2018-01-05 01:20:11 +08:00
|
|
|
idpattern = r'(?a:[_a-z][_a-z0-9]*)'
|
2017-09-05 04:32:10 +08:00
|
|
|
braceidpattern = None
|
2010-07-30 01:16:10 +08:00
|
|
|
flags = _re.IGNORECASE
|
2004-09-10 11:08:08 +08:00
|
|
|
|
2019-10-21 14:36:21 +08:00
|
|
|
def __init_subclass__(cls):
|
|
|
|
super().__init_subclass__()
|
|
|
|
if 'pattern' in cls.__dict__:
|
|
|
|
pattern = cls.pattern
|
|
|
|
else:
|
|
|
|
delim = _re.escape(cls.delimiter)
|
|
|
|
id = cls.idpattern
|
|
|
|
bid = cls.braceidpattern or cls.idpattern
|
|
|
|
pattern = fr"""
|
|
|
|
{delim}(?:
|
|
|
|
(?P<escaped>{delim}) | # Escape sequence of two delimiters
|
|
|
|
(?P<named>{id}) | # delimiter and a Python identifier
|
|
|
|
{{(?P<braced>{bid})}} | # delimiter and a braced identifier
|
|
|
|
(?P<invalid>) # Other ill-formed delimiter exprs
|
|
|
|
)
|
|
|
|
"""
|
|
|
|
cls.pattern = _re.compile(pattern, cls.flags | _re.VERBOSE)
|
|
|
|
|
2004-09-10 11:08:08 +08:00
|
|
|
def __init__(self, template):
|
|
|
|
self.template = template
|
2004-08-25 10:22:30 +08:00
|
|
|
|
|
|
|
# Search for $$, $identifier, ${identifier}, and any bare $'s
|
2004-09-10 11:08:08 +08:00
|
|
|
|
2004-09-14 04:52:50 +08:00
|
|
|
def _invalid(self, mo):
|
|
|
|
i = mo.start('invalid')
|
2011-09-28 22:37:55 +08:00
|
|
|
lines = self.template[:i].splitlines(keepends=True)
|
2004-09-10 11:08:08 +08:00
|
|
|
if not lines:
|
|
|
|
colno = 1
|
|
|
|
lineno = 1
|
|
|
|
else:
|
|
|
|
colno = i - len(''.join(lines[:-1]))
|
|
|
|
lineno = len(lines)
|
|
|
|
raise ValueError('Invalid placeholder in string: line %d, col %d' %
|
|
|
|
(lineno, colno))
|
|
|
|
|
2019-06-01 16:00:15 +08:00
|
|
|
def substitute(self, mapping=_sentinel_dict, /, **kws):
|
|
|
|
if mapping is _sentinel_dict:
|
2004-09-13 23:25:15 +08:00
|
|
|
mapping = kws
|
2004-09-13 22:35:04 +08:00
|
|
|
elif kws:
|
2019-06-01 16:00:15 +08:00
|
|
|
mapping = _ChainMap(kws, mapping)
|
2004-09-13 22:35:04 +08:00
|
|
|
# Helper function for .sub()
|
2004-08-25 10:22:30 +08:00
|
|
|
def convert(mo):
|
2004-09-14 04:52:50 +08:00
|
|
|
# Check the most common path first.
|
|
|
|
named = mo.group('named') or mo.group('braced')
|
|
|
|
if named is not None:
|
2015-05-29 01:45:29 +08:00
|
|
|
return str(mapping[named])
|
2004-08-26 08:21:13 +08:00
|
|
|
if mo.group('escaped') is not None:
|
2004-09-18 08:06:34 +08:00
|
|
|
return self.delimiter
|
2004-09-14 04:52:50 +08:00
|
|
|
if mo.group('invalid') is not None:
|
|
|
|
self._invalid(mo)
|
2004-10-18 00:27:18 +08:00
|
|
|
raise ValueError('Unrecognized named group in pattern',
|
|
|
|
self.pattern)
|
2004-09-10 11:08:08 +08:00
|
|
|
return self.pattern.sub(convert, self.template)
|
2004-08-25 10:22:30 +08:00
|
|
|
|
2019-06-01 16:00:15 +08:00
|
|
|
def safe_substitute(self, mapping=_sentinel_dict, /, **kws):
|
|
|
|
if mapping is _sentinel_dict:
|
2004-09-13 23:25:15 +08:00
|
|
|
mapping = kws
|
2004-09-13 22:35:04 +08:00
|
|
|
elif kws:
|
2019-06-01 16:00:15 +08:00
|
|
|
mapping = _ChainMap(kws, mapping)
|
2004-09-13 22:35:04 +08:00
|
|
|
# Helper function for .sub()
|
2004-08-25 10:22:30 +08:00
|
|
|
def convert(mo):
|
2010-09-19 07:34:07 +08:00
|
|
|
named = mo.group('named') or mo.group('braced')
|
2004-08-25 10:22:30 +08:00
|
|
|
if named is not None:
|
|
|
|
try:
|
2015-05-29 01:45:29 +08:00
|
|
|
return str(mapping[named])
|
2004-08-25 10:22:30 +08:00
|
|
|
except KeyError:
|
2010-09-19 07:34:07 +08:00
|
|
|
return mo.group()
|
2004-09-14 04:52:50 +08:00
|
|
|
if mo.group('escaped') is not None:
|
2004-09-18 08:06:34 +08:00
|
|
|
return self.delimiter
|
2004-09-14 04:52:50 +08:00
|
|
|
if mo.group('invalid') is not None:
|
2010-09-19 07:34:07 +08:00
|
|
|
return mo.group()
|
2004-10-18 00:27:18 +08:00
|
|
|
raise ValueError('Unrecognized named group in pattern',
|
|
|
|
self.pattern)
|
2004-09-10 11:08:08 +08:00
|
|
|
return self.pattern.sub(convert, self.template)
|
2007-08-25 10:26:07 +08:00
|
|
|
|
2022-01-12 03:15:42 +08:00
|
|
|
def is_valid(self):
|
|
|
|
for mo in self.pattern.finditer(self.template):
|
|
|
|
if mo.group('invalid') is not None:
|
|
|
|
return False
|
|
|
|
if (mo.group('named') is None
|
|
|
|
and mo.group('braced') is None
|
|
|
|
and mo.group('escaped') is None):
|
|
|
|
# If all the groups are None, there must be
|
|
|
|
# another group we're not expecting
|
|
|
|
raise ValueError('Unrecognized named group in pattern',
|
|
|
|
self.pattern)
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_identifiers(self):
|
|
|
|
ids = []
|
|
|
|
for mo in self.pattern.finditer(self.template):
|
|
|
|
named = mo.group('named') or mo.group('braced')
|
|
|
|
if named is not None and named not in ids:
|
|
|
|
# add a named group only the first time it appears
|
|
|
|
ids.append(named)
|
|
|
|
elif (named is None
|
|
|
|
and mo.group('invalid') is None
|
|
|
|
and mo.group('escaped') is None):
|
|
|
|
# If all the groups are None, there must be
|
|
|
|
# another group we're not expecting
|
|
|
|
raise ValueError('Unrecognized named group in pattern',
|
|
|
|
self.pattern)
|
|
|
|
return ids
|
|
|
|
|
2019-10-21 14:36:21 +08:00
|
|
|
# Initialize Template.pattern. __init_subclass__() is automatically called
|
|
|
|
# only for subclasses, not for the Template class itself.
|
|
|
|
Template.__init_subclass__()
|
2007-08-25 10:26:07 +08:00
|
|
|
|
|
|
|
|
|
|
|
########################################################################
|
|
|
|
# the Formatter class
|
|
|
|
# see PEP 3101 for details and purpose of this class
|
|
|
|
|
Merged revisions 67154,67157-67159,67175-67176,67189,67224-67227,67234 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
r67154 | hirokazu.yamamoto | 2008-11-07 21:46:17 -0600 (Fri, 07 Nov 2008) | 1 line
Issue #4071: ntpath.abspath returned an empty string for long unicode path.
........
r67157 | georg.brandl | 2008-11-08 05:47:44 -0600 (Sat, 08 Nov 2008) | 2 lines
Don't use "HOWTO" as the title for all howto .tex files.
........
r67158 | georg.brandl | 2008-11-08 05:48:20 -0600 (Sat, 08 Nov 2008) | 2 lines
Update "Documenting" a bit. Concentrate on Python-specifics.
........
r67159 | georg.brandl | 2008-11-08 06:52:25 -0600 (Sat, 08 Nov 2008) | 2 lines
Fix warning.
........
r67175 | benjamin.peterson | 2008-11-08 19:44:32 -0600 (Sat, 08 Nov 2008) | 1 line
update link
........
r67176 | benjamin.peterson | 2008-11-08 19:52:32 -0600 (Sat, 08 Nov 2008) | 1 line
fix comment
........
r67189 | benjamin.peterson | 2008-11-11 15:56:06 -0600 (Tue, 11 Nov 2008) | 1 line
use correct name
........
r67224 | georg.brandl | 2008-11-15 02:10:04 -0600 (Sat, 15 Nov 2008) | 2 lines
#4324: fix getlocale() argument.
........
r67225 | brett.cannon | 2008-11-15 16:33:25 -0600 (Sat, 15 Nov 2008) | 1 line
Clarify the docs for the 'strict' argument to httplib.HTTPConnection.
........
r67226 | brett.cannon | 2008-11-15 16:40:44 -0600 (Sat, 15 Nov 2008) | 4 lines
The docs for httplib.HTTPConnection.putheader() have claimed for quite a while
that their could be an arbitrary number of values passed in. Turns out the code
did not match that. The code now matches the docs.
........
r67227 | georg.brandl | 2008-11-16 02:00:17 -0600 (Sun, 16 Nov 2008) | 2 lines
#4316: fix configure.in markup problem.
........
r67234 | benjamin.peterson | 2008-11-16 11:54:55 -0600 (Sun, 16 Nov 2008) | 1 line
run autoconf
........
2008-11-17 02:33:53 +08:00
|
|
|
# The hard parts are reused from the C implementation. They're exposed as "_"
|
2010-09-07 04:27:55 +08:00
|
|
|
# prefixed methods of str.
|
2007-08-25 10:26:07 +08:00
|
|
|
|
2010-10-14 15:04:07 +08:00
|
|
|
# The overall parser is implemented in _string.formatter_parser.
|
|
|
|
# The field name parser is implemented in _string.formatter_field_name_split
|
2007-08-25 10:26:07 +08:00
|
|
|
|
|
|
|
class Formatter:
|
2019-06-01 16:00:15 +08:00
|
|
|
def format(self, format_string, /, *args, **kwargs):
|
2007-08-25 10:26:07 +08:00
|
|
|
return self.vformat(format_string, args, kwargs)
|
|
|
|
|
|
|
|
def vformat(self, format_string, args, kwargs):
|
2007-08-31 10:26:31 +08:00
|
|
|
used_args = set()
|
2015-09-29 22:27:38 +08:00
|
|
|
result, _ = self._vformat(format_string, args, kwargs, used_args, 2)
|
2007-09-05 07:04:22 +08:00
|
|
|
self.check_unused_args(used_args, args, kwargs)
|
|
|
|
return result
|
|
|
|
|
2014-04-15 04:43:50 +08:00
|
|
|
def _vformat(self, format_string, args, kwargs, used_args, recursion_depth,
|
|
|
|
auto_arg_index=0):
|
2007-09-05 07:04:22 +08:00
|
|
|
if recursion_depth < 0:
|
|
|
|
raise ValueError('Max string recursion exceeded')
|
2007-08-25 10:26:07 +08:00
|
|
|
result = []
|
2007-08-28 19:15:20 +08:00
|
|
|
for literal_text, field_name, format_spec, conversion in \
|
|
|
|
self.parse(format_string):
|
Modified parsing of format strings, so that we always return
a tuple (literal, field_name, format_spec, conversion).
literal will always be a string, but might be of zero length.
field_name will be None if there is no markup text
format_spec will be a (possibly zero length) string if
field_name is non-None
conversion will be a one character string, or None
This makes the Formatter class, and especially it's parse()
method, easier to understand.
Suggestion was by Jim Jewett, inspired by the "tail" of an
elementtree node.
Also, fixed a reference leak in fieldnameiter_next.
2007-08-29 11:22:59 +08:00
|
|
|
|
|
|
|
# output the literal text
|
|
|
|
if literal_text:
|
|
|
|
result.append(literal_text)
|
|
|
|
|
|
|
|
# if there's a field, output it
|
|
|
|
if field_name is not None:
|
2007-08-28 19:15:20 +08:00
|
|
|
# this is some markup, find the object and do
|
|
|
|
# the formatting
|
2007-08-27 06:27:13 +08:00
|
|
|
|
2014-04-15 04:43:50 +08:00
|
|
|
# handle arg indexing when empty field_names are given.
|
|
|
|
if field_name == '':
|
|
|
|
if auto_arg_index is False:
|
|
|
|
raise ValueError('cannot switch from manual field '
|
|
|
|
'specification to automatic field '
|
|
|
|
'numbering')
|
|
|
|
field_name = str(auto_arg_index)
|
|
|
|
auto_arg_index += 1
|
|
|
|
elif field_name.isdigit():
|
|
|
|
if auto_arg_index:
|
|
|
|
raise ValueError('cannot switch from manual field '
|
|
|
|
'specification to automatic field '
|
|
|
|
'numbering')
|
|
|
|
# disable auto arg incrementing, if it gets
|
|
|
|
# used later on, then an exception will be raised
|
|
|
|
auto_arg_index = False
|
|
|
|
|
2007-08-28 19:15:20 +08:00
|
|
|
# given the field_name, find the object it references
|
2007-08-31 10:26:31 +08:00
|
|
|
# and the argument it came from
|
2007-09-02 23:33:26 +08:00
|
|
|
obj, arg_used = self.get_field(field_name, args, kwargs)
|
2007-08-31 10:26:31 +08:00
|
|
|
used_args.add(arg_used)
|
2007-08-27 06:27:13 +08:00
|
|
|
|
|
|
|
# do any conversion on the resulting object
|
2007-08-28 19:15:20 +08:00
|
|
|
obj = self.convert_field(obj, conversion)
|
2007-08-27 06:27:13 +08:00
|
|
|
|
2007-09-05 07:04:22 +08:00
|
|
|
# expand the format spec, if needed
|
2015-09-29 22:27:38 +08:00
|
|
|
format_spec, auto_arg_index = self._vformat(
|
|
|
|
format_spec, args, kwargs,
|
|
|
|
used_args, recursion_depth-1,
|
|
|
|
auto_arg_index=auto_arg_index)
|
2007-09-05 07:04:22 +08:00
|
|
|
|
2007-08-27 06:27:13 +08:00
|
|
|
# format the object and append to the result
|
|
|
|
result.append(self.format_field(obj, format_spec))
|
Modified parsing of format strings, so that we always return
a tuple (literal, field_name, format_spec, conversion).
literal will always be a string, but might be of zero length.
field_name will be None if there is no markup text
format_spec will be a (possibly zero length) string if
field_name is non-None
conversion will be a one character string, or None
This makes the Formatter class, and especially it's parse()
method, easier to understand.
Suggestion was by Jim Jewett, inspired by the "tail" of an
elementtree node.
Also, fixed a reference leak in fieldnameiter_next.
2007-08-29 11:22:59 +08:00
|
|
|
|
2015-09-29 22:27:38 +08:00
|
|
|
return ''.join(result), auto_arg_index
|
2007-08-25 10:26:07 +08:00
|
|
|
|
2007-08-28 19:15:20 +08:00
|
|
|
|
2007-08-25 10:26:07 +08:00
|
|
|
def get_value(self, key, args, kwargs):
|
2007-08-27 06:27:13 +08:00
|
|
|
if isinstance(key, int):
|
|
|
|
return args[key]
|
|
|
|
else:
|
|
|
|
return kwargs[key]
|
2007-08-25 10:26:07 +08:00
|
|
|
|
2007-08-28 19:15:20 +08:00
|
|
|
|
2007-08-25 10:26:07 +08:00
|
|
|
def check_unused_args(self, used_args, args, kwargs):
|
|
|
|
pass
|
|
|
|
|
2007-08-28 19:15:20 +08:00
|
|
|
|
2007-08-25 10:26:07 +08:00
|
|
|
def format_field(self, value, format_spec):
|
2007-08-27 06:27:13 +08:00
|
|
|
return format(value, format_spec)
|
2007-08-28 19:15:20 +08:00
|
|
|
|
|
|
|
|
|
|
|
def convert_field(self, value, conversion):
|
|
|
|
# do any conversion on the resulting object
|
2012-08-20 05:26:34 +08:00
|
|
|
if conversion is None:
|
|
|
|
return value
|
2007-08-28 19:15:20 +08:00
|
|
|
elif conversion == 's':
|
|
|
|
return str(value)
|
2012-08-20 05:26:34 +08:00
|
|
|
elif conversion == 'r':
|
|
|
|
return repr(value)
|
|
|
|
elif conversion == 'a':
|
|
|
|
return ascii(value)
|
2010-09-07 04:27:55 +08:00
|
|
|
raise ValueError("Unknown conversion specifier {0!s}".format(conversion))
|
2007-08-28 19:15:20 +08:00
|
|
|
|
|
|
|
|
|
|
|
# returns an iterable that contains tuples of the form:
|
|
|
|
# (literal_text, field_name, format_spec, conversion)
|
Modified parsing of format strings, so that we always return
a tuple (literal, field_name, format_spec, conversion).
literal will always be a string, but might be of zero length.
field_name will be None if there is no markup text
format_spec will be a (possibly zero length) string if
field_name is non-None
conversion will be a one character string, or None
This makes the Formatter class, and especially it's parse()
method, easier to understand.
Suggestion was by Jim Jewett, inspired by the "tail" of an
elementtree node.
Also, fixed a reference leak in fieldnameiter_next.
2007-08-29 11:22:59 +08:00
|
|
|
# literal_text can be zero length
|
|
|
|
# field_name can be None, in which case there's no
|
|
|
|
# object to format and output
|
|
|
|
# if field_name is not None, it is looked up, formatted
|
|
|
|
# with format_spec and conversion and then used
|
2007-08-28 19:15:20 +08:00
|
|
|
def parse(self, format_string):
|
2010-10-14 15:04:07 +08:00
|
|
|
return _string.formatter_parser(format_string)
|
2007-08-28 19:15:20 +08:00
|
|
|
|
|
|
|
|
|
|
|
# given a field_name, find the object it references.
|
|
|
|
# field_name: the field being looked up, e.g. "0.name"
|
|
|
|
# or "lookup[3]"
|
|
|
|
# used_args: a set of which args have been used
|
|
|
|
# args, kwargs: as passed in to vformat
|
2007-09-02 23:33:26 +08:00
|
|
|
def get_field(self, field_name, args, kwargs):
|
2010-10-14 15:04:07 +08:00
|
|
|
first, rest = _string.formatter_field_name_split(field_name)
|
2007-08-28 19:15:20 +08:00
|
|
|
|
|
|
|
obj = self.get_value(first, args, kwargs)
|
|
|
|
|
|
|
|
# loop through the rest of the field_name, doing
|
|
|
|
# getattr or getitem as needed
|
|
|
|
for is_attr, i in rest:
|
|
|
|
if is_attr:
|
|
|
|
obj = getattr(obj, i)
|
|
|
|
else:
|
|
|
|
obj = obj[i]
|
|
|
|
|
2007-08-31 10:26:31 +08:00
|
|
|
return obj, first
|