a [f @sddlmZddlZddlZddlZz ddlZWneyFdZYn0ejddkrpddlmZe fZ e Z ddl mZddlZddlZddlmZddlmZmZmZmZmZdd lmZmZmZm Z m!Z!m"Z"m#Z#d d Zddl$Z$dd l$m%Z%m&Z&m'Z'm(Z(m)Z)m*Z*m+Z+m,Z,m-Z-er$dd l$m.Z.ddl/Z/ddl0Z0ddl1Z2ddl3m3Z3ddl4Z4e5Z5ddl6m7Z8ddl6m9Z:nddl;mZemZmZmZmZmZmZmZm#Z#ddl?m&Z&mZm%Z%m Z m!Z!m)Z)m*Z*m+Z+m,Z,m-Z-erdd l?m.Z.ddl@m(Z(m'Z'm"Z"ddlAmBZ/ddl?mCZ$ddlDmBZ0ddl2Z2ddlEm3Z3ddlFmGZ4eHZ5ddl6m:Z:e8Z8zddlmIZImJZJWn6eyGdddeKZJdbddZLddZIYn0zddl mMZNWn$eyGdddeOZNYn0zdd lmPZPWn*ey<ejQejRBdfd!d"ZPYn0dd#lSmTZUeVeUd$r\eUZTn,dd%lSmWZXGd&d'd'eXZWGd(d)d)eUZTzdd*lYmZZZWneyd+d,ZZYn0z ddl[Z[Wn eydd-lm[Z[Yn0ze\Z\Wn(e]ydd.l^m_Z_d/d0Z\Yn0zej`Z`ejaZaWnFebylecpBd1Zdedd2krTd3Zend4Zed5d6Z`d7d8ZaYn0zdd9lfmgZgWnFeydd:lhmiZimjZjddlZekd;Zldd?ZgYn0zdd@lnmoZoWn eydd@lpmoZoYn0ejddAdBkre3jqZqn ddClnmqZqzddDlrmsZsWn`eyddElrmtZtzddFlumvZwWney|dcdHdIZwYn0GdJdKdKetZsYn0zddLlxmyZyWnDeyzddLlzmyZyWneydddMdNZyYn0Yn0zddOlrm{Z{Wney~zddPl|m}Z~Wn ey:ddPlm}Z~Yn0zddQlmZmZmZWneyhYn0GdRdSdSeZ{Yn0zddTlmZmZWnteyekdUejZdVdWZGdXdYdYeZdedZd[ZGd\d]d]eZGd^d_d_eZGd`dadaeOZYn0dS)f)absolute_importN)StringIO)FileType)shutil)urlparse urlunparseurljoinurlsplit urlunsplit) urlretrievequoteunquote url2pathname pathname2urlContentTooShortError splittypecCst|tr|d}t|S)Nutf-8) isinstanceunicodeencode_quote)sr>/usr/lib/python3.9/site-packages/pip/_vendor/distlib/compat.pyrs  r) RequesturlopenURLError HTTPErrorHTTPBasicAuthHandlerHTTPPasswordMgr HTTPHandlerHTTPRedirectHandler build_opener) HTTPSHandler) HTMLParser)ifilter) ifilterfalse) TextIOWrapper)rr r rrr r r) rr rrrr r!r"r#r$)rrr) filterfalse)match_hostnameCertificateErrorc@s eZdZdS)r,N)__name__ __module__ __qualname__rrrrr,`sr,c Csg}|s dS|d}|d|dd}}|d}||krNtdt||sb||kS|dkrv|dn>|d s|d r|t|n|t| d d |D]}|t|qt d d |dtj } | |S)zpMatching according to RFC 6125, section 6.4.3 http://tools.ietf.org/html/rfc6125#section-6.4.3 F.rrN*z,too many wildcards in certificate DNS name: z[^.]+zxn--z\*z[^.]*z\Az\.z\Z)splitcountr,reprlowerappend startswithreescapereplacecompilejoin IGNORECASEmatch) ZdnhostnameZ max_wildcardsZpatspartsZleftmostZ remainderZ wildcardsfragpatrrr_dnsname_matchds*    rCcCs|s tdg}|dd}|D]*\}}|dkr t||r@dS||q |s|ddD]6}|D],\}}|dkrdt||rdS||qdq\t|dkrtd |d tt|fn*t|dkrtd ||d fntd dS)a=Verify that *cert* (in decoded format as returned by SSLSocket.getpeercert()) matches the *hostname*. RFC 2818 and RFC 6125 rules are followed, but IP addresses are not accepted for *hostname*. CertificateError is raised on failure. On success, the function returns nothing. ztempty or no certificate, match_hostname needs a SSL socket or SSL context with either CERT_OPTIONAL or CERT_REQUIREDZsubjectAltNamerZDNSNZsubjectZ commonNamerz&hostname %r doesn't match either of %s, zhostname %r doesn't match %rrz=no appropriate commonName or subjectAltName fields were found) ValueErrorgetrCr6lenr,r<mapr4)Zcertr?ZdnsnamesZsankeyvaluesubrrrr+s2         r+)SimpleNamespacec@seZdZdZddZdS) ContainerzR A generic container for when multiple values need to be returned cKs|j|dSN__dict__update)selfkwargsrrr__init__szContainer.__init__N)r-r.r/__doc__rTrrrrrMsrM)whichc s"dd}tjr&||r"SdS|dur>tjdtj}|sFdS|tj}tj dkrtj |vrt| dtj tjddtj}t fd d |Drg}q‡fd d |D}ng}t }|D]P}tj|}||vr|||D](} tj|| } || |r| SqqdS) aKGiven a command, mode, and a PATH string, return the path which conforms to the given mode on the PATH, or None if there is no such file. `mode` defaults to os.F_OK | os.X_OK. `path` defaults to the result of os.environ.get("PATH"), or can be overridden with a custom search path. cSs&tj|o$t||o$tj| SrN)ospathexistsaccessisdir)fnmoderrr _access_checks zwhich.._access_checkNPATHwin32rPATHEXTc3s |]}|VqdSrN)r5endswith.0extcmdrr zwhich..csg|] }|qSrrrdrgrr rjzwhich..)rWrXdirnameenvironrFdefpathr2pathsepsysplatformcurdirinsertanysetnormcaseaddr<) rhr]rXr^pathextfilesseendirnormdirthefilenamerrgrrVs8         rV)ZipFile __enter__) ZipExtFilec@s$eZdZddZddZddZdS)rcCs|j|jdSrNrO)rRbaserrrrTszZipExtFile.__init__cCs|SrNrrRrrrrszZipExtFile.__enter__cGs |dSrNcloserRexc_inforrr__exit__szZipExtFile.__exit__N)r-r.r/rTrrrrrrrsrc@s$eZdZddZddZddZdS)rcCs|SrNrrrrrr$szZipFile.__enter__cGs |dSrNrrrrrr'szZipFile.__exit__cOs tj|g|Ri|}t|SrN) BaseZipFileopenr)rRargsrSrrrrr+sz ZipFile.openN)r-r.r/rrrrrrrr#sr)python_implementationcCs0dtjvrdStjdkrdStjdr,dSdS)z6Return a string identifying the Python implementation.ZPyPyjavaZJythonZ IronPythonZCPython)rpversionrWr~r7rrrrr2s   r) sysconfig)CallablecCs t|tSrN)rr)objrrrcallableFsrrmbcsstrictsurrogateescapecCs:t|tr|St|tr$|ttStdt|jdSNzexpect bytes or str, not %s) rbytes text_typer _fsencoding _fserrors TypeErrortyper-filenamerrrfsencodeZs   rcCs:t|tr|St|tr$|ttStdt|jdSr) rrrdecoderrrrr-rrrrfsdecodecs   r)detect_encoding)BOM_UTF8lookupzcoding[:=]\s*([-\w.]+)cCsH|dddd}|dks*|dr.dS|dvs@|drDd S|S) z(Imitates get_normal_name in tokenizer.c.N _-rzutf-8-)zlatin-1 iso-8859-1z iso-latin-1)zlatin-1-z iso-8859-1-z iso-latin-1-r)r5r:r7)orig_encencrrr_get_normal_nametsrcsz jjWnty"dYn0dd}d}fdd}fdd}|}|trnd|d d}d }|sz|gfS||}|r||gfS|}|s||gfS||}|r|||gfS|||gfS) a? The detect_encoding() function is used to detect the encoding that should be used to decode a Python source file. It requires one argument, readline, in the same way as the tokenize() generator. It will call readline a maximum of twice, and return the encoding used (as a string) and a list of any lines (left as bytes) it has read in. It detects the encoding from the presence of a utf-8 bom or an encoding cookie as specified in pep-0263. If both a bom and a cookie are present, but disagree, a SyntaxError will be raised. If the encoding cookie is an invalid charset, raise a SyntaxError. Note that if a utf-8 bom is found, 'utf-8-sig' is returned. If no encoding is specified, then the default of 'utf-8' will be returned. NFrcs"zWStyYdS0dS)Nrj) StopIterationr)readlinerr read_or_stops z%detect_encoding..read_or_stopcsz|d}Wn2ty@d}dur4d|}t|Yn0t|}|sTdSt|d}z t|}Wn8tydurd|}n d|}t|Yn0r|j dkrԈdurd}n d}t||d 7}|S) Nrz'invalid or missing encoding declarationz {} for {!r}rzunknown encoding: zunknown encoding for {!r}: {}zencoding problem: utf-8z encoding problem for {!r}: utf-8z-sig) rUnicodeDecodeErrorformat SyntaxError cookie_refindallrr LookupErrorr~)line line_stringmsgmatchesencodingcodec) bom_foundrrr find_cookies8         z$detect_encoding..find_cookieTrz utf-8-sig)__self__r~AttributeErrorr7r)rrdefaultrrfirstsecondr)rrrrrs4    &     r)r9)r)unescape)ChainMap)MutableMapping)recursive_repr...csfdd}|S)zm Decorator to make a repr function return fillvalue for a recursive call csLtfdd}td|_td|_td|_tdi|_|S)Nc sLt|tf}|vrS|z|}W|n |0|SrN)id get_identrwdiscard)rRrIresult) fillvalue repr_running user_functionrrwrappers  z=_recursive_repr..decorating_function..wrapperr.rUr-__annotations__)rugetattrr.rUr-r)rrr)rrrdecorating_functions   z,_recursive_repr..decorating_functionr)rrrrr_recursive_reprs rc@seZdZdZddZddZddZd'd d Zd d Zd dZ ddZ ddZ e ddZ eddZddZeZddZeddZddZdd Zd!d"Zd#d$Zd%d&ZdS)(ra A ChainMap groups multiple dicts (or other mappings) together to create a single, updateable view. The underlying mappings are stored in a list. That list is public and can accessed or updated using the *maps* attribute. There is no other state. Lookups search the underlying mappings successively until a key is found. In contrast, writes, updates, and deletions only operate on the first mapping. cGst|p ig|_dS)zInitialize a ChainMap by setting *maps* to the given mappings. If no mappings are provided, a single empty dictionary is used. N)listmaps)rRrrrrrTszChainMap.__init__cCs t|dSrN)KeyErrorrRrIrrr __missing__szChainMap.__missing__c Cs8|jD]&}z||WSty*Yq0q||SrN)rrr)rRrImappingrrr __getitem__s   zChainMap.__getitem__NcCs||vr||S|SrNrrRrIrrrrrF'sz ChainMap.getcCsttj|jSrN)rGruunionrrrrr__len__*szChainMap.__len__cCsttj|jSrN)iterrurrrrrr__iter__-szChainMap.__iter__cstfdd|jDS)Nc3s|]}|vVqdSrNr)remrIrrri1rjz(ChainMap.__contains__..rtrrrrr __contains__0szChainMap.__contains__cCs t|jSrNrrrrr__bool__3szChainMap.__bool__cCsd|dtt|jS)Nz{0.__class__.__name__}({1})rD)rr<rHr4rrrrr__repr__6szChainMap.__repr__cGs|tj|g|RS)z?Create a ChainMap with a single dict created from the iterable.)dictfromkeys)clsiterablerrrrr;szChainMap.fromkeyscCs&|j|jdg|jddRS)zHNew ChainMap or subclass with a new copy of maps[0] and refs to maps[1:]rrN) __class__rcopyrrrrr@sz ChainMap.copycCs|jig|jRS)z;New ChainMap with a new dict followed by all previous maps.rrrrrr new_childFszChainMap.new_childcCs|j|jddS)zNew ChainMap from maps[1:].rNrrrrrparentsJszChainMap.parentscCs||jd|<dS)Nr)r)rRrIrJrrr __setitem__OszChainMap.__setitem__cCs6z|jd|=Wn ty0td|Yn0dS)Nr(Key not found in the first mapping: {!r})rrrrrrr __delitem__Rs zChainMap.__delitem__cCs0z|jdWSty*tdYn0dS)zPRemove and return an item pair from maps[0]. Raise KeyError is maps[0] is empty.rz#No keys found in the first mapping.N)rpopitemrrrrrrXs zChainMap.popitemcGs@z|jdj|g|RWSty:td|Yn0dS)zWRemove *key* from maps[0] and return its value. Raise KeyError if *key* not in maps[0].rrN)rpoprr)rRrIrrrrr_s z ChainMap.popcCs|jddS)z'Clear maps[0], leaving maps[1:] intact.rN)rclearrrrrrfszChainMap.clear)N)r-r.r/rUrTrrrFrrrrrr classmethodrr__copy__rpropertyrrrrrrrrrrrs.     r)cache_from_sourcecCs"|dur d}|rd}nd}||S)NFcor)rXdebug_overridesuffixrrrrps r) OrderedDict)r)KeysView ValuesView ItemsViewc@seZdZdZddZejfddZejfddZdd Zd d Z d d Z d6ddZ ddZ ddZ ddZddZddZddZddZeZeZefdd Zd7d"d#Zd8d$d%Zd&d'Zd(d)Zed9d*d+Zd,d-Zd.d/Zd0d1Zd2d3Z d4d5Z!d!S):rz)Dictionary that remembers insertion ordercOspt|dkrtdt|z |jWn4tyZg|_}||dg|dd<i|_Yn0|j|i|dS)zInitialize an ordered dictionary. Signature is the same as for regular dictionaries, but keyword arguments are not recommended because their insertion order is arbitrary. rz$expected at most 1 arguments, got %dN)rGr_OrderedDict__rootr_OrderedDict__map_OrderedDict__update)rRrkwdsrootrrrrTs     zOrderedDict.__init__cCsF||vr6|j}|d}|||g|d<|d<|j|<||||dS)z!od.__setitem__(i, y) <==> od[i]=yrrN)rr)rRrIrJZ dict_setitemrlastrrrrs  zOrderedDict.__setitem__cCs0||||j|\}}}||d<||d<dS)z od.__delitem__(y) <==> del od[y]rrN)rr)rRrIZ dict_delitem link_prev link_nextrrrrs zOrderedDict.__delitem__ccs.|j}|d}||ur*|dV|d}qdS)zod.__iter__() <==> iter(od)rrNrrRrZcurrrrrrs  zOrderedDict.__iter__ccs.|j}|d}||ur*|dV|d}qdS)z#od.__reversed__() <==> reversed(od)rrNr r rrr __reversed__s  zOrderedDict.__reversed__cCsbz@|jD]}|dd=q |j}||dg|dd<|jWntyRYn0t|dS)z.od.clear() -> None. Remove all items from od.N)r itervaluesrrrr)rRZnoderrrrrs  zOrderedDict.clearTcCs||s td|j}|r8|d}|d}||d<||d<n |d}|d}||d<||d<|d}|j|=t||}||fS)zod.popitem() -> (k, v), return and remove a (key, value) pair. Pairs are returned in LIFO order if last is true or FIFO order if false. zdictionary is emptyrrr)rrrrr)rRrrlinkr r rIrJrrrrs   zOrderedDict.popitemcCst|S)zod.keys() -> list of keys in od)rrrrrkeysszOrderedDict.keyscsfddDS)z#od.values() -> list of values in odcsg|] }|qSrrrerIrrrrkrjz&OrderedDict.values..rrrrrvaluesszOrderedDict.valuescsfddDS)z.od.items() -> list of (key, value) pairs in odcsg|]}||fqSrrrrrrrkrjz%OrderedDict.items..rrrrritemsszOrderedDict.itemscCst|S)z0od.iterkeys() -> an iterator over the keys in od)rrrrriterkeysszOrderedDict.iterkeysccs|D]}||VqdS)z2od.itervalues -> an iterator over the values in odNrrRkrrrrszOrderedDict.itervaluesccs|D]}|||fVqdS)z=od.iteritems -> an iterator over the (key, value) items in odNrrrrr iteritemsszOrderedDict.iteritemscOst|dkr tdt|fn |s,td|d}d}t|dkrL|d}t|trn|D]}||||<qZn None. Update od from dict/iterable E and F. If E is a dict instance, does: for k in E: od[k] = E[k] If E has a .keys() method, does: for k in E.keys(): od[k] = E[k] Or if E is an iterable of items, does: for k, v in E: od[k] = v In either case, this is followed by: for k, v in F.items(): od[k] = v rz8update() takes at most 2 positional arguments (%d given)z,update() takes at least 1 argument (0 given)rrrrN)rGrrrhasattrrr)rrrRotherrIrJrrrrQ s(       zOrderedDict.updatecCs0||vr||}||=|S||jur,t||S)zod.pop(k[,d]) -> v, remove specified key and return the corresponding value. If key is not found, d is returned if given, otherwise KeyError is raised. )_OrderedDict__markerr)rRrIrrrrrr,s zOrderedDict.popNcCs||vr||S|||<|S)zDod.setdefault(k[,d]) -> od.get(k,d), also set od[k]=d if k not in odrrrrr setdefault9szOrderedDict.setdefaultcCsn|si}t|tf}||vr"dSd||<z6|sFd|jjfW||=Sd|jj|fW||=S||=0dS)zod.__repr__() <==> repr(od)rrz%s()z%s(%r)N)r _get_identrr-r)rRZ _repr_runningZcall_keyrrrr@szOrderedDict.__repr__csXfddD}t}ttD]}||dq(|rLj|f|fSj|ffS)z%Return state information for picklingcsg|]}||gqSrrrerrrrrkPrjz*OrderedDict.__reduce__..N)varsrrrr)rRrZ inst_dictrrrr __reduce__Ns zOrderedDict.__reduce__cCs ||S)z!od.copy() -> a shallow copy of od)rrrrrrXszOrderedDict.copycCs|}|D] }|||<q |S)zOD.fromkeys(S[, v]) -> New ordered dictionary with keys from S and values equal to v (which defaults to None). r)rrrJdrIrrrr\s zOrderedDict.fromkeyscCs6t|tr*t|t|ko(||kSt||S)zod.__eq__(y) <==> od==y. Comparison to another OD is order-sensitive while comparison to a regular mapping is order-insensitive. )rrrGrr__eq__rRrrrrr!gs  zOrderedDict.__eq__cCs ||k SrNrr"rrr__ne__pszOrderedDict.__ne__cCst|S)z@od.viewkeys() -> a set-like object providing a view on od's keys)rrrrrviewkeysuszOrderedDict.viewkeyscCst|S)z an object providing a view on od's values)rrrrr viewvaluesyszOrderedDict.viewvaluescCst|S)zBod.viewitems() -> a set-like object providing a view on od's items)rrrrr viewitems}szOrderedDict.viewitems)T)N)N)N)"r-r.r/rUrTrrrrr rrrrrrrrrQrobjectrrrrrrrrr!r#r$r%r&rrrrrs:         r)BaseConfigurator valid_identz^[a-z_][a-z0-9_]*$cCst|}|std|dS)Nz!Not a valid Python identifier: %rT) IDENTIFIERr>rE)rrrrrr)s  r)c@s"eZdZdZddZdddZdS)ConvertingDictz A converting dictionary wrapper.cCsJt||}|j|}||urF|||<t|tttfvrF||_||_ |SrN) rr configuratorconvertrr+ConvertingListConvertingTupleparentrIrRrIrJrrrrrs   zConvertingDict.__getitem__NcCsLt|||}|j|}||urH|||<t|tttfvrH||_||_ |SrN) rrFr,r-rr+r.r/r0rIrRrIrrJrrrrrFs  zConvertingDict.get)N)r-r.r/rUrrFrrrrr+s r+cCsDt|||}|j|}||ur@t|tttfvr@||_||_ |SrN) rrr,r-rr+r.r/r0rIr2rrrrs  rc@s"eZdZdZddZdddZdS) r.zA converting list wrapper.cCsJt||}|j|}||urF|||<t|tttfvrF||_||_ |SrN) rrr,r-rr+r.r/r0rIr1rrrrs   zConvertingList.__getitem__cCs<t||}|j|}||ur8t|tttfvr8||_|SrN) rrr,r-rr+r.r/r0)rRidxrJrrrrrs   zConvertingList.popN)r3)r-r.r/rUrrrrrrr.s r.c@seZdZdZddZdS)r/zA converting tuple wrapper.cCsBt||}|j|}||ur>t|tttfvr>||_||_ |SrN) tuplerr,r-rr+r.r/r0rIr1rrrrs   zConvertingTuple.__getitem__N)r-r.r/rUrrrrrr/sr/c@seZdZdZedZedZedZedZ edZ ddd Z e e Zd d Zd d ZddZddZddZddZddZdS)r(zQ The configurator base class which defines some useful defaults. z%^(?P[a-z]+)://(?P.*)$z ^\s*(\w+)\s*z^\.\s*(\w+)\s*z^\[\s*(\w+)\s*\]\s*z^\d+$ ext_convert cfg_convert)rfZcfgcCst||_||j_dSrN)r+configr,)rRr8rrrrTs zBaseConfigurator.__init__c Cs|d}|d}z\||}|D]F}|d|7}zt||}Wq$tyh||t||}Yq$0q$|WStytdd\}}td||f}|||_ |_ |Yn0dS)zl Resolve strings to objects using standard import and attribute syntax. r0rrNzCannot resolve %r: %s) r2rimporterrr ImportErrorrprrE __cause__ __traceback__) rRrr~usedfoundrAetbvrrrresolves"       zBaseConfigurator.resolvecCs ||S)z*Default converter for the ext:// protocol.)rBrRrJrrrr6szBaseConfigurator.ext_convertcCs|}|j|}|dur&td|n||d}|j|d}|r|j|}|rn||d}nb|j|}|r|d}|j|s||}n0zt |}||}Wnt y||}Yn0|r||d}qHtd||fqH|S)z*Default converter for the cfg:// protocol.NzUnable to convert %rrzUnable to convert %r at %r) WORD_PATTERNr>rEendr8groups DOT_PATTERN INDEX_PATTERN DIGIT_PATTERNintr)rRrJrestrr r4nrrrr7s4        zBaseConfigurator.cfg_convertcCst|ts$t|tr$t|}||_nt|tsHt|trHt|}||_nzt|tslt|trlt|}||_nVt|tr|j |}|r| }|d}|j |d}|r|d}t||}||}|S)z Convert values to an appropriate type. dicts, lists and tuples are replaced by their converting alternatives. Strings are checked to see if they have a conversion format and are converted if they do. prefixNr)rr+rr,r.rr/r5 string_typesCONVERT_PATTERNr> groupdictvalue_convertersrFr)rRrJrr rMZ converterrrrrr-4s,    zBaseConfigurator.convertcsrd}t|s||}dd}tfddD}|fi|}|rn|D]\}}t|||qX|S)z1Configure an object with a user-supplied factory.z()r0Ncs g|]}t|r||fqSr)r)rr8rrrkWrjz5BaseConfigurator.configure_custom..)rrrBrrsetattr)rRr8rZpropsrSrr~rJrrRrconfigure_customPs   z!BaseConfigurator.configure_customcCst|trt|}|S)z0Utility function which converts lists to tuples.)rrr5rCrrras_tuple^s zBaseConfigurator.as_tupleN)r-r.r/rUr8r;rOrDrGrHrIrQ staticmethod __import__r9rTrBr6r7r-rTrUrrrrr(s"     "r()r)r)N)N)Z __future__rrWr8rpZsslr: version_inforZ basestringrNrrtypesrZ file_typeZ __builtin__builtinsZ ConfigParserZ configparserZ _backportrrr r r r Zurllibr rrrrrrrZurllib2rrrrr r!r"r#r$r%ZhttplibZ xmlrpclibZQueueZqueuer&ZhtmlentitydefsZ raw_input itertoolsr'filterr(r*iostrr) urllib.parseZurllib.requestZ urllib.errorZ http.clientZclientZrequestZ xmlrpc.clientZ html.parserZ html.entitiesZentitiesinputr+r,rErCrLrMr'rVF_OKX_OKzipfilerrrrZBaseZipExtFilerqrrr NameErrorcollections.abcrrrrgetfilesystemencodingrrtokenizercodecsrrr;rrZhtmlr9Zcgir collectionsrrreprlibrrimportlib.utilrimprthreadrrZ dummy_threadZ_abcollrrrrZlogging.configr(r)Ir*r+rrr.r5r/rrrrs(       $,     (0        2+A              [   b w