Re dZddlZddlZddlZddlZddlmZmZmZm Z m Z m Z m Z ddl mZddlmZddlmZddlmZddlmZdd lmZmZmZmZdd lmZmZdd lm Z dd l!m"Z"dd l#m$Z$ddl%m&Z&ddl'm(Z(ddl)m*Z*ddl+m,Z,ddl-m.Z.ddl/m0Z0ddl1m2Z2ddl3m4Z4ddl5m6Z6ddl7m8Z8ddl9m:Z:ddl;mZ>gdZ?e0e@ZAe e de eBeCffZDe eBeBeBee eBeDfZE d5de$de eBeBeBfd eFd!eFfd"ZGGd#d$ZHd%ee d&e4d'eCd!ee fd(ZIGd)d*ZJGd+d,ZKGd-d.ZLGd/d0ZMd1eCd2eCd!eBfd3ZNd1eCd2eCd!e eCfd4ZOdS)6z!Routines related to PyPI, indexesN) FrozenSetIterableListOptionalSetTupleUnion) specifiers)Tag)canonicalize_name) _BaseVersion)parse)BestVersionAlreadyInstalledDistributionNotFoundInvalidWheelFilenameUnsupportedWheel) LinkCollector parse_links)InstallationCandidate) FormatControl)Link) SearchScope)SelectionPreferences) TargetPython)Wheel)InstallRequirement) getLogger)WHEEL_EXTENSION)Hashes) indent_log) build_netloc)check_requires_python)SUPPORTED_EXTENSIONS) url_to_path)rBestCandidateResult PackageFinderFlink version_infoignore_requires_pythonreturnc t|j|}|spdtt|}|s$t d||j|dSt d||j|n6#tj $r$t d|j|YnwxYwdS)aa Return whether the given Python version is compatible with a link's "Requires-Python" value. :param version_info: A 3-tuple of ints representing the Python major-minor-micro version to check. :param ignore_requires_python: Whether to ignore the "Requires-Python" value if the given Python version isn't compatible. )r).z4Link requires a different Python (%s not in: %r): %sFzBIgnoring failed Requires-Python check (%s not in: %r) for link: %sz2Ignoring invalid Requires-Python (%r) for link: %sT) r"requires_pythonjoinmapstrloggerverbosedebugr InvalidSpecifier)r(r)r* is_compatibleversions /builddir/build/BUILDROOT/alt-python311-pip-21.3.1-3.el8.x86_64/opt/alt/python311/lib/python3.11/site-packages/pip/_internal/index/package_finder.py_check_link_requires_pythonr93s-  %    hhs3 5566G) J(  u LLT$    %  &    @        2 4sB 0B>=B>ceZdZdZejdZ ddededeede de d e e d dfd Z d e d ee e effd ZdS) LinkEvaluatorzD Responsible for evaluating links for a particular project. z-py([123]\.?[0-9]?)$N project_namecanonical_nameformats target_python allow_yankedr*r+cb|d}||_||_||_||_||_||_dS)a :param project_name: The user supplied package name. :param canonical_name: The canonical package name. :param formats: The formats allowed for this package. Should be a set with 'binary' or 'source' or both in it. :param target_python: The target Python interpreter to use when evaluating link compatibility. This is used, for example, to check wheel compatibility, as well as when checking the Python version, e.g. the Python version embedded in a link filename (or egg fragment) and against an HTML link's optional PEP 503 "data-requires-python" attribute. :param allow_yanked: Whether files marked as yanked (in the sense of PEP 592) are permitted to be candidates for install. :param ignore_requires_python: Whether to ignore incompatible PEP 503 "data-requires-python" values in HTML links. Defaults to False. NF) _allow_yanked_canonical_name_ignore_requires_python_formats_target_pythonr<)selfr<r=r>r?r@r*s r8__init__zLinkEvaluator.__init__nsE4 " )%* ")-'=$ +(r(cd}|jr|js|jpd}dd|fS|jr|j}|j}nP|\}}|sdS|t vrdd|fSd|jvr)|tkrd |j }d|fSd |j vr|d krd S|tkr t|j }n#t$rYd SwxYwt|j|jkrd |j }d|fS|j}||s@|}d d|}d|fS|j}d|jvr|tkrd|j }d|fS|st1||j}|sd|j }d|fS|j|} | rC|d| }| d} | |jjkrdSt=||jj|j } | sdStB"d||d|fS)aG Determine whether a link is a candidate for installation. :return: A tuple (is_candidate, result), where `result` is (1) a version string if `is_candidate` is True, and (2) if `is_candidate` is False, an optional string to log the reason the link fails to qualify. Nz Fzyanked for reason: )Fz not a filezunsupported archive format: binaryzNo binaries permitted for {}macosx10z.zip)Fz macosx10 one)Fzinvalid wheel filenamezwrong project name (not {})z^none of the wheel's tags ({}) are compatible (run pip debug --verbose to show compatible tags), sourcezNo sources permitted for zMissing project version for )FzPython version is incorrect)r)r*)FNzFound link %s, version: %sT)# is_yankedrB yanked_reason egg_fragmentextsplitextr#rErformatr<pathrfilenamerr namerCrFget_tags supportedget_formatted_file_tagsr/r7_extract_version_from_fragment_py_version_researchstartgroup py_versionr9py_version_inforDr2r4) rGr(r7reasonegg_inforSwheelsupported_tags file_tagsmatchrasupports_pythons r8 evaluate_linkzLinkEvaluator.evaluate_links  > ;$"4 ;'9>F999: :  $ ((H(CC MMOOMHc -,,...CcCCDDt},,1G1G7>>t?PQQv&TY&&3&==..o%%=!$-00EE+===<<<=$UZ00D4HHH:AA$BSTTF!6?*!%!4!=!=!?!?~66 +!& = = ? ?ILLRF IIi00MM "6?*- 4= ( (SO-C-CD1BDDF6? " 4$G #GD4EGGF6? "#**733  >o o.GQJT0;;;==5 ,<#'#?     !!= 14AAAgs2C CCN)__name__ __module__ __qualname____doc__recompiler]r1rrboolrrHrrrjr'rIr8r;r;bs RZ 788N26#)#)#)#)3 #) $ #)  #)!)#) #)#)#)#)JU$U5x}1D+EUUUUUUrIr; candidateshashesr<c |s8tdt||t|Sg}g}d}|D]X}|j}|jsn2||r|dz }n||C||Y|r|}nt|}t|t|krd} n@dt|d d|D} td t|||j |t||z | |S) a Filter out candidates whose hashes aren't allowed, and return a new list of candidates. If at least one candidate has an allowed hash, then all candidates with either an allowed hash or no hash specified are returned. Otherwise, the given candidates are returned. Including the candidates with no hash specified when there is a match allows a warning to be logged if there is a more preferred candidate with no hash specified. Returning all candidates in the case of no matches lets pip report the hash of the candidate that would otherwise have been installed (e.g. permitting the user to more easily update their requirements file with the desired hash). zJGiven no hashes to check %s links for project %r: discarding no candidatesr)rtrOzdiscarding no candidateszdiscarding {} non-matches: {}z c3>K|]}t|jVdSrk)r1r().0 candidates r8 z*filter_unallowed_hashes..$s,II IN++IIIIIIrIzPChecked %s links for project %r against %s hashes (%s matches, %s no digest): %s) r2r4lenlistr(has_hashis_hash_allowedappendrUr/ digest_count) rsrtr<matches_or_no_digest non_matches match_countrxr(filtereddiscard_messages r8filter_unallowed_hashesrs(    '  OO     JKK / / ~}    ! ! ! 0 0  1 KK   y ) ) ) ##I....$' ## 8}}J''4<CC    KKII[III I I    LL ) J !!K/    OrIc,eZdZdZ ddededdfdZdS) CandidatePreferenceszk Encapsulates some of the preferences for filtering and sorting InstallationCandidate objects. F prefer_binaryallow_all_prereleasesr+Nc"||_||_dS)zR :param allow_all_prereleases: Whether to allow all pre-releases. N)rr)rGrrs r8rHzCandidatePreferences.__init__<s&;"*rI)FF)rlrmrnrorrrHr'rIr8rr5s[$&+ + + + $ +  + + + + + +rIrc~eZdZdZdeedeedeeddfdZdeefdZ deefd Z dS) r%zA collection of candidates, returned by `PackageFinder.find_best_candidate`. This class is only intended to be instantiated by CandidateEvaluator's `compute_best_candidate()` method. rsapplicable_candidatesbest_candidater+Nct|t|ksJ||rJn||vsJ||_||_||_dS)a :param candidates: A sequence of all available candidates found. :param applicable_candidates: The applicable candidates. :param best_candidate: The most preferred candidate found, or None if no applicable candidates were found. N)set_applicable_candidates _candidatesrrGrsrrs r8rHzBestCandidateResult.__init__Osi())S__<<<<  !, , , , ,!%:::::&;#%,rIc*t|jS)zIterate through all candidates.)iterrrGs r8iter_allzBestCandidateResult.iter_allgsD$%%%rIc*t|jS)z*Iterate through the applicable candidates.)rrrs r8iter_applicablez#BestCandidateResult.iter_applicableksD/000rI) rlrmrnrorrrrHrrrr'rIr8r%r%Hs -./- $$9:-!!67 -  ----0&(#89&&&&1*?!@111111rIr%cJeZdZdZe ddedeedededee j d ee d dfd Z dded e ede j deded ee d dfd Zde ed e efdZded efdZde ed eefdZde ed efdZdS)CandidateEvaluatorzm Responsible for filtering and sorting candidates for installation based on what tags are valid. NFr<r?rr specifierrtr+c|t}|tj}|}|||||||S)aCreate a CandidateEvaluator object. :param target_python: The target Python interpreter to use when checking compatibility. If None (the default), a TargetPython object will be constructed from the running Python. :param specifier: An optional object implementing `filter` (e.g. `packaging.specifiers.SpecifierSet`) to filter applicable versions. :param hashes: An optional collection of allowed hashes. N)r<rfrrrrt)rr SpecifierSetrY)clsr<r?rrrrtrfs r8createzCandidateEvaluator.createwsc(  (NNM  "/11I&//11s%)'"7     rIrfc||_||_||_||_||_||_dt |D|_dS)z :param supported_tags: The PEP 425 tags supported by the target Python in order of preference (most preferred first). ci|]\}}|| Sr'r')rwidxtags r8 z/CandidateEvaluator.__init__..s+' ' ' !cC' ' ' rIN)_allow_all_prereleases_hashes_prefer_binary _project_name _specifier_supported_tags enumerate_wheel_tag_preferences)rGr<rfrrrrts r8rHzCandidateEvaluator.__init__sa'<# +)#-' ' %.~%>%>' ' ' ###rIrsc|jpd}|j}d|d|D|Dfd|D}t||j|j}t ||jS)zM Return the applicable candidates from a list of candidates. Nc,h|]}t|Sr')r1)rwvs r8 z?CandidateEvaluator.get_applicable_candidates..s.     FF    rIc3>K|]}t|jVdSrkr1r7rwcs r8ryz?CandidateEvaluator.get_applicable_candidates..s*44AQY444444rI) prereleasesc@g|]}t|jv|Sr'r)rwrversionss r8 z@CandidateEvaluator.get_applicable_candidates..s+ U U Uq#ai..H:T:T:T:T:TrI)rsrtr<key)rrfilterrrrsorted _sort_key)rGrsallow_prereleasesrrfiltered_applicable_candidatesrs @r8get_applicable_candidatesz,CandidateEvaluator.get_applicable_candidatess!7?4O    %%54444-&       !V U U UJ U U U)@,<+* * * & 4$.IIIIrIrxct|j}t|}d}d}|j}|jrt |j} |||j }n5#t$r(td |jwxYw|j rd}|j Ktjd|j } | } t!| d| df}n| }t!||j} dt!|jz} | | ||j||fS)a) Function to pass as the `key` argument to a call to sorted() to sort InstallationCandidates by preference. Returns a tuple such that tuples sorting as greater using Python's default comparison operator are more preferred. The preference is as follows: First and foremost, candidates with allowed (matching) hashes are always preferred over candidates without matching hashes. This is because e.g. if the only candidate with an allowed hash is yanked, we still want to use that candidate. Second, excepting hash considerations, candidates that have been yanked (in the sense of PEP 592) are always less preferred than candidates that haven't been yanked. Then: If not finding wheels, they are sorted by version only. If finding wheels, then the sort order is by version, then: 1. existing installs 2. wheels ordered via Wheel.support_index_min(self._supported_tags) 3. source archives If prefer_binary was set, then all wheels are sorted above sources. Note: it was considered to embed this logic into the Link comparison operators, but then different sdist links with the same version, would have to be considered equal r'rzB{} is not a supported wheel for this platform. It can't be sorted.rONz ^(\d+)(.*)$)rrzr(is_wheelrrWfind_most_preferred_tagr ValueErrorrrUr build_tagrprhgroupsintr}rrPr7) rGrx valid_tags support_numrbinary_preferencer(reprirhbuild_tag_groupshas_allowed_hash yank_values r8rzCandidateEvaluator._sort_keysa<) *oo  ~ = !$-((E 11"D$?    &''-ven'='=  " &$%!*AA#(<<>>  !1!!4557G7JK .Ct33DLAABB#dn---          s A2B c:|sdSt||j}|S)zy Return the best candidate per the instance's sort order, or None if no candidate is acceptable. Nr)maxr)rGrsrs r8sort_best_candidatez&CandidateEvaluator.sort_best_candidates+ 4ZT^<<<rIcz||}||}t|||S)zF Compute and return a `BestCandidateResult` instance. )rr)rrr%rs r8compute_best_candidatez)CandidateEvaluator.compute_best_candidate*sL!% > >z J J112GHH" "7)    rI)NFFNN)FFN)rlrmrnro classmethodr1rrrrr BaseSpecifierrrrr rHrrCandidateSortingKeyrrr%rr'rIr8rrps 15#&+8<#'! ! !  -!  ! $ ! J45 !  !  ! ! ! [! P$&+#'   S  +    $         4"J./"J # $"J"J"J"JHB #8B =PB B B B H ./  ' (     ./        rIrceZdZdZ d-dedededeedee deed dfd Z e d.ded e deed dfd Z ed efd Zed efdZejded dfdZed eefdZed eefdZed eefdZed efdZd/dZed efdZd/dZded efdZdeed eefdZ deded dfdZ!deded ee"fd Z#dedeed ee"fd!Z$d"eded ee"fd#Z%e&j'd$ded ee"fd%Z( d0ded&ee)j*d'ee+d e,fd(Z-e&j'd$ d0ded&ee)j*d'ee+d e.fd)Z/d*e0d+ed ee"fd,Z1dS)1r&zThis finds packages. This is meant to match easy_install's technique for looking for packages, by reading pages and looking for appropriate links. Nlink_collectorr?r@format_controlcandidate_prefsr*r+c|t}|p'ttt}||_||_||_||_||_||_t|_ dS)a This constructor is primarily meant to be used by the create() class method and from tests. :param format_control: A FormatControl object, used to control the selection of source packages / binary packages when consulting the index and links. :param candidate_prefs: Options to use when creating a CandidateEvaluator object. N) rrrrB_candidate_prefsrD_link_collectorrFr _logged_links)rGrr?r@rrr*s r8rHzPackageFinder.__init__Csu&  "244O'F=+F+F) /'=$-+,),rIselection_prefsc|t}t|j|j}|||||j|j|jS)afCreate a PackageFinder. :param selection_prefs: The candidate selection preferences, as a SelectionPreferences object. :param target_python: The target Python interpreter to use when checking compatibility. If None (the default), a TargetPython object will be constructed from the running Python. N)rr)rrr?r@rr*)rrrrr@rr*)rrrr?rs r8rzPackageFinder.createjsi  (NNM.)7"1"G    s+)'(5*9#2#I     rIc|jSrk)rFrs r8r?zPackageFinder.target_pythons ""rIc|jjSrkr search_scopers r8rzPackageFinder.search_scopes#00rIrc||j_dSrkr)rGrs r8rzPackageFinder.search_scopes,8)))rIc|jjSrk)r find_linksrs r8rzPackageFinder.find_linkss#..rIc|jjSrk)r index_urlsrs r8rzPackageFinder.index_urlss ++rIc#HK|jjjD] }t|VdSrk)rsessionpip_trusted_originsr!)rG host_ports r8 trusted_hostszPackageFinder.trusted_hostss<-5I + +I * * * * * + +rIc|jjSrkrrrs r8rz#PackageFinder.allow_all_prereleasess$::rIcd|j_dSNTrrs r8set_allow_all_prereleasesz'PackageFinder.set_allow_all_prereleasess6:333rIc|jjSrkrrrs r8rzPackageFinder.prefer_binarys$22rIcd|j_dSrrrs r8set_prefer_binaryzPackageFinder.set_prefer_binarys.2+++rIr<ct|}|j|}t||||j|j|jS)N)r<r=r>r?r@r*)r rget_allowed_formatsr;rFrBrD)rGr<r=r>s r8make_link_evaluatorz!PackageFinder.make_link_evaluatorsW*<88%99.II%)-+#'#?     rIlinkscgg}}t}|D]M}||vrG|||jr||8||N||zS)z Returns elements of links in order, non-egg links first, egg links second, while eliminating duplicates )raddrRr~)rGreggsno_eggsseenr(s r8 _sort_linkszPackageFinder._sort_linkss| Bg%% ) )D4$)KK%%%%NN4(((~rIr(rcc||jvr8td|||j|dSdS)NzSkipping link: %s: %s)rr2r4r)rGr(rcs r8_log_skipped_linkzPackageFinder._log_skipped_linksN t) ) ) LL0&$ ? ? ?   " "4 ( ( ( ( ( * )rIlink_evaluatorc||\}}|s|r|||dSt|j||S)z If the link is a candidate for install, convert it to an InstallationCandidate and return it. Otherwise, return None. )rcN)rXr(r7)rjrrr<)rGrr( is_candidateresults r8get_install_candidatez#PackageFinder.get_install_candidatesm .;;DAA f  <&&tF&;;;4$,    rIcg}||D]/}|||}|||0|S)zU Convert links that are candidates to InstallationCandidate objects. )rr r~)rGrrrsr(rxs r8evaluate_linkszPackageFinder.evaluate_linkss\  $$U++ - -D22>4HHI$!!),,,rI project_urlc*td||j|}|gSt t |}t 5|||}dddn #1swxYwY|S)Nz-Fetching project page and analyzing links: %s)r)r2r4r fetch_pager{rr r )rGr r html_page page_links package_linkss r8process_project_urlz!PackageFinder.process_project_urls  ;    (33K@@  I+i0011 \\   // 0M                s$BB B )maxsizecl||}|j|tj|j|}t jd|D}t|}t jd|D}| |t|d}t tjr<|r:d|D}tdd |||zS) aFind all available InstallationCandidate for project_name This checks index_urls and find_links. All versions found are returned as an InstallationCandidate list. See LinkEvaluator.evaluate_link() for details on which files are accepted. )r)r<candidates_from_pagec3LK|]}|D]}||V dSrk)page_candidatesrwsourcesrNs r8ryz4PackageFinder.find_all_candidates..sW; ; !; ; !  " " $ $"!!!! ; ; rIc3LK|]}|D]}||V dSrk) file_linksrs r8ryz4PackageFinder.find_all_candidates..'sW6 6 !6 6 !     "!!!! 6 6 rIT)reversec@g|]}t|jjSr')r$r(urlrs r8rz5PackageFinder.find_all_candidates..3s$FFF[,,FFFrIzLocal files found: %srM)rrcollect_sources functoolspartialr itertoolschain from_iterabler{r rr2 isEnabledForloggingDEBUGr4r/) rGr<rcollected_sourcespage_candidates_itr file_links_itfile_candidatespathss r8find_all_candidatesz!PackageFinder.find_all_candidates s^11,?? 0@@%!*!2(-"""A  '_::; ; ,; ; ;    122!556 6 ,6 6 6   --  =$ / / /     w} - - D/ DFFoFFFE LL0$))E2B2B C C C00rIrrtcn|j}t||j|j|j||S)z*Create a CandidateEvaluator object to use.)r<r?rrrrt)rrrrFrr)rGr<rrtrs r8make_candidate_evaluatorz&PackageFinder.make_candidate_evaluator9sD/!((%-)7"1"G )   rIc||}||||}||S)aFind matches for the given project and specifier. :param specifier: An optional object implementing `filter` (e.g. `packaging.specifiers.SpecifierSet`) to filter applicable versions. :return: A `BestCandidateResult` instance. )r<rrt)r-r/r)rGr<rrtrscandidate_evaluators r8find_best_candidatez!PackageFinder.find_best_candidateJsQ--l;; ";;%<   #99*EEErIrequpgradecd|d}||j|j|}|j}d}|jt |jj}dttdtfd}|[|Yt d||| td |d}|r| |j|krd }|sC|A|rtd |n!td ||jdS|r>td |||t$td|j|||S)zTry to find a Link matching req Expects req, an InstallRequirement and upgrade, a boolean Returns a InstallationCandidate if found, Raises DistributionNotFound or BestVersionAlreadyInstalled otherwise F)trust_internet)rrtN cand_iterr+cldtd|DtpdS)NrMc6h|]}t|jSr'rrs r8rzKPackageFinder.find_requirement.._format_versions..~s ;;;AQY;;;rIrnone)r/r parse_version)r7s r8_format_versionsz8PackageFinder.find_requirement.._format_versionsvsJ  ;;;;;)  rIzNCould not find a version that satisfies the requirement %s (from versions: %s)z%No matching distribution found for {}TzLExisting installed version (%s) is most up-to-date and satisfies requirementzUExisting installed version (%s) satisfies requirement (most up-to-date version is %s)z=Installed version (%s) is most up-to-date (past versions: %s)z)Using version %s (newest of versions: %s))rtr2rXrr satisfied_byr;r7rrr1r2criticalrrrUr4rr) rGr3r4rtbest_candidate_resultrinstalled_versionr<best_installeds r8find_requirementzPackageFinder.find_requirementas511 $ 8 8 Hm!9! !  /=48   ' -c.>.F G G  1F(G C      $)? OO&  !6!?!?!A!ABB    '7>>sCC   "  "n&<@Q&Q&Q!N ,8  ,%  6%"*  4  . LLO!  !6!F!F!H!HII    . - 7  "  2BBDD E E   rI)NNNrk)r+N)NN)2rlrmrnrorrrrrrrrHrrrpropertyr?rrsetterrr1rrrrrrrrr;rrrrrr r rr  lru_cacher-r rrrr/r%r2rrBr'rIr8r&r&<s37:>15!.!.%!.$!. !. !/ !. ""67 !.!)!. !.!.!.!.N 15   % .  -      [ >#|###X#1k111X1999999/DI///X/,DI,,,X,+x}+++X+;t;;;X;;;;;3t333X33333        $DJ )d)C)D)))) + 37 ' (    & + 4 # $*Y&&&+1+1=R8S+1+1+1'&+1`9=#'    J45        "Y&&&9=#' FFFJ45F F  FFF'&F,T%T04T ' (TTTTTTrIr&fragmentr=ct|D]+\}}|dkr t|d||kr|cS,t|d|)aFind the separator's index based on the package's canonical name. :param fragment: A + filename "fragment" (stem) or egg fragment. :param canonical_name: The package's canonical name. This function is needed since the canonicalized name does not necessarily have the same length as the egg info's name part. An example:: >>> fragment = 'foo__bar-1.0' >>> canonical_name = 'foo-bar' >>> _find_name_version_sep(fragment, canonical_name) 8 -Nz does not match )rr r)rFr=irs r8_find_name_version_seprJsp$(##1 88  Xbqb\ * *n < <HHH = BB.BB C CCrIcn t||dz}n#t$rYdSwxYw||d}|sdS|S)zParse the version string from a + filename "fragment" (stem) or egg fragment. :param fragment: The string to parse. E.g. foo-2.1 :param canonical_name: The canonicalized name of the package this belongs to. rON)rJr)rFr= version_startr7s r8r\r\s_.xHH1L tt}~~&G t Ns  $$)F)Pror r"r&rptypingrrrrrrr pip._vendor.packagingr pip._vendor.packaging.tagsr pip._vendor.packaging.utilsr pip._vendor.packaging.versionr rr;pip._internal.exceptionsrrrrpip._internal.index.collectorrrpip._internal.models.candidater#pip._internal.models.format_controlrpip._internal.models.linkr!pip._internal.models.search_scoper$pip._internal.models.selection_prefsr"pip._internal.models.target_pythonrpip._internal.models.wheelrpip._internal.reqrpip._internal.utils._logrpip._internal.utils.filetypesrpip._internal.utils.hashesrpip._internal.utils.loggingr pip._internal.utils.miscr!pip._internal.utils.packagingr"pip._internal.utils.unpackingr#pip._internal.utils.urlsr$__all__rlr2rr1BuildTagrrrr9r;rrr%rr&rJr\r'rIr8rfsx''  IIIIIIIIIIIIIIIIII,,,,,,******999999666666@@@@@@ EDDDDDDD@@@@@@======******999999EEEEEE;;;;;;,,,,,,000000......999999------222222111111??????>>>>>>000000 C C C 8   rE#s(O+ ,Cc<#PQ $),, ,S# &,!, ,,,,^FFFFFFFFRG*+G GG  GGGGT++++++++&%1%1%1%1%1%1%1%1PI I I I I I I I Xyyyyyyyyx DSD#D#DDDD4S#(SV-rI