"""A collection of string constants.Public module variables:whitespace -- a string containing all ASCII whitespaceascii_lowercase -- a string containing all ASCII lowercase lettersascii_uppercase -- a string containing all ASCII uppercase lettersascii_letters -- a string containing all ASCII lettersdigits -- a string containing all ASCII decimal digitshexdigits -- a string containing all ASCII hexadecimal digitsoctdigits -- a string containing all ASCII octal digitspunctuation -- a string containing all ASCII punctuation charactersprintable -- a string containing all ASCII characters considered printable"""__all__=["ascii_letters","ascii_lowercase","ascii_uppercase","capwords","digits","hexdigits","octdigits","printable","punctuation","whitespace","Formatter","Template"]import_string# Some strings for ctype-style character classificationwhitespace=' \t\n\r\v\f'ascii_lowercase='abcdefghijklmnopqrstuvwxyz'ascii_uppercase='ABCDEFGHIJKLMNOPQRSTUVWXYZ'ascii_letters=ascii_lowercase+ascii_uppercasedigits='0123456789'hexdigits=digits+'abcdef'+'ABCDEF'octdigits='01234567'punctuation=r"""!"#$%&'()*+,-./:;<=>?@[\]^_`{|}~"""printable=digits+ascii_letters+punctuation+whitespace# Functions which aren't available as string methods.# Capitalize the words in a string, e.g. " aBc dEf " -> "Abc Def".defcapwords(s,sep=None):"""capwords(s [,sep]) -> string Split the argument into words using split, capitalize each word using capitalize, and join the capitalized words using 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. """return(sepor' ').join(map(str.capitalize,s.split(sep)))####################################################################importreas_refromcollectionsimportChainMapas_ChainMap_sentinel_dict={}classTemplate:"""A string class for supporting $-substitutions."""delimiter='$'# 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.# See https://bugs.python.org/issue31672idpattern=r'(?a:[_a-z][_a-z0-9]*)'braceidpattern=Noneflags=_re.IGNORECASEdef__init_subclass__(cls):super().__init_subclass__()if'pattern'incls.__dict__:pattern=cls.patternelse:delim=_re.escape(cls.delimiter)id=cls.idpatternbid=cls.braceidpatternorcls.idpatternpattern=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)def__init__(self,template):self.template=template# Search for $$, $identifier, ${identifier}, and any bare $'sdef_invalid(self,mo):i=mo.start('invalid')lines=self.template[:i].splitlines(keepends=True)ifnotlines:colno=1lineno=1else:colno=i-len(''.join(lines[:-1]))lineno=len(lines)raiseValueError('Invalid placeholder in string: line %d, col %d'%(lineno,colno))defsubstitute(self,mapping=_sentinel_dict,/,**kws):ifmappingis_sentinel_dict:mapping=kwselifkws:mapping=_ChainMap(kws,mapping)# Helper function for .sub()defconvert(mo):# Check the most common path first.named=mo.group('named')ormo.group('braced')ifnamedisnotNone:returnstr(mapping[named])ifmo.group('escaped')isnotNone:returnself.delimiterifmo.group('invalid')isnotNone:self._invalid(mo)raiseValueError('Unrecognized named group in pattern',self.pattern)returnself.pattern.sub(convert,self.template)defsafe_substitute(self,mapping=_sentinel_dict,/,**kws):ifmappingis_sentinel_dict:mapping=kwselifkws:mapping=_ChainMap(kws,mapping)# Helper function for .sub()defconvert(mo):named=mo.group('named')ormo.group('braced')ifnamedisnotNone:try:returnstr(mapping[named])exceptKeyError:returnmo.group()ifmo.group('escaped')isnotNone:returnself.delimiterifmo.group('invalid')isnotNone:returnmo.group()raiseValueError('Unrecognized named group in pattern',self.pattern)returnself.pattern.sub(convert,self.template)defis_valid(self):formoinself.pattern.finditer(self.template):ifmo.group('invalid')isnotNone:returnFalseif(mo.group('named')isNoneandmo.group('braced')isNoneandmo.group('escaped')isNone):# If all the groups are None, there must be# another group we're not expectingraiseValueError('Unrecognized named group in pattern',self.pattern)returnTruedefget_identifiers(self):ids=[]formoinself.pattern.finditer(self.template):named=mo.group('named')ormo.group('braced')ifnamedisnotNoneandnamednotinids:# add a named group only the first time it appearsids.append(named)elif(namedisNoneandmo.group('invalid')isNoneandmo.group('escaped')isNone):# If all the groups are None, there must be# another group we're not expectingraiseValueError('Unrecognized named group in pattern',self.pattern)returnids# Initialize Template.pattern. __init_subclass__() is automatically called# only for subclasses, not for the Template class itself.Template.__init_subclass__()######################################################################### the Formatter class# see PEP 3101 for details and purpose of this class# The hard parts are reused from the C implementation. They're exposed as "_"# prefixed methods of str.# The overall parser is implemented in _string.formatter_parser.# The field name parser is implemented in _string.formatter_field_name_splitclassFormatter:defformat(self,format_string,/,*args,**kwargs):returnself.vformat(format_string,args,kwargs)defvformat(self,format_string,args,kwargs):used_args=set()result,_=self._vformat(format_string,args,kwargs,used_args,2)self.check_unused_args(used_args,args,kwargs)returnresultdef_vformat(self,format_string,args,kwargs,used_args,recursion_depth,auto_arg_index=0):ifrecursion_depth<0:raiseValueError('Max string recursion exceeded')result=[]forliteral_text,field_name,format_spec,conversionin \
self.parse(format_string):# output the literal textifliteral_text:result.append(literal_text)# if there's a field, output itiffield_nameisnotNone:# this is some markup, find the object and do# the formatting# handle arg indexing when empty field_names are given.iffield_name=='':ifauto_arg_indexisFalse:raiseValueError('cannot switch from manual field ''specification to automatic field ''numbering')field_name=str(auto_arg_index)auto_arg_index+=1eliffield_name.isdigit():ifauto_arg_index:raiseValueError('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 raisedauto_arg_index=False# given the field_name, find the object it references# and the argument it came fromobj,arg_used=self.get_field(field_name,args,kwargs)used_args.add(arg_used)# do any conversion on the resulting objectobj=self.convert_field(obj,conversion)# expand the format spec, if neededformat_spec,auto_arg_index=self._vformat(format_spec,args,kwargs,used_args,recursion_depth-1,auto_arg_index=auto_arg_index)# format the object and append to the resultresult.append(self.format_field(obj,format_spec))return''.join(result),auto_arg_indexdefget_value(self,key,args,kwargs):ifisinstance(key,int):returnargs[key]else:returnkwargs[key]defcheck_unused_args(self,used_args,args,kwargs):passdefformat_field(self,value,format_spec):returnformat(value,format_spec)defconvert_field(self,value,conversion):# do any conversion on the resulting objectifconversionisNone:returnvalueelifconversion=='s':returnstr(value)elifconversion=='r':returnrepr(value)elifconversion=='a':returnascii(value)raiseValueError("Unknown conversion specifier {0!s}".format(conversion))# returns an iterable that contains tuples of the form:# (literal_text, field_name, format_spec, conversion)# 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 useddefparse(self,format_string):return_string.formatter_parser(format_string)# 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 vformatdefget_field(self,field_name,args,kwargs):first,rest=_string.formatter_field_name_split(field_name)obj=self.get_value(first,args,kwargs)# loop through the rest of the field_name, doing# getattr or getitem as neededforis_attr,iinrest:ifis_attr:obj=getattr(obj,i)else:obj=obj[i]returnobj,first