B °-_÷Xã@s¶dZddgZddlZddlmZddlmZddlmZddl m Z e  d ¡Z e  d ¡Z e  d ¡Ze  d ¡Ze  d ¡Zd ZdZeƒZGdd„deƒZGdd„dƒZGdd„deƒZdS)aFeedParser - An email feed parser. The feed parser implements an interface for incrementally parsing an email message, line by line. This has advantages for certain applications, such as those reading email messages off a socket. FeedParser.feed() is the primary interface for pushing new data into the parser. It returns when there's nothing more it can do with the available data. When you have no more data to push into the parser, call .close(). This completes the parsing and returns the root message object. The other advantage of this parser is that it will never raise a parsing exception. Instead, when it finds something unexpected, it adds a 'defect' to the current message. Defects are just instances that live on the message object's .defects attribute. Ú FeedParserÚBytesFeedParseréN)Úerrors)Úcompat32)Údeque)ÚStringIOz \r\n|\r|\nz (\r\n|\r|\n)z(\r\n|\r|\n)\Zz%^(From |[\041-\071\073-\176]*:|[\t ])ÚÚ c@s`eZdZdZdd„Zdd„Zdd„Zdd „Zd d „Zd d „Z dd„Z dd„Z dd„Z dd„Z dS)ÚBufferedSubFileakA file-ish object that can have new data loaded into it. You can also push and pop line-matching predicates onto a stack. When the current predicate matches the current line, a false EOF response (i.e. empty string) is returned instead. This lets the parser adhere to a simple abstraction -- it parses until EOF closes the current message. cCs$tdd|_tƒ|_g|_d|_dS)Nr)ÚnewlineF)rÚ_partialrÚ_linesÚ _eofstackÚ_closed)Úself©rú&/usr/lib/python3.7/email/feedparser.pyÚ__init__5s zBufferedSubFile.__init__cCs|j |¡dS)N)rÚappend)rZpredrrrÚpush_eof_matcher@sz BufferedSubFile.push_eof_matchercCs |j ¡S)N)rÚpop)rrrrÚpop_eof_matcherCszBufferedSubFile.pop_eof_matchercCs<|j d¡| |j ¡¡|j d¡|j ¡d|_dS)NrT)r ÚseekÚ pushlinesÚ readlinesÚtruncater)rrrrÚcloseFs    zBufferedSubFile.closecCsN|js|jrdStS|j ¡}x*t|jƒD]}||ƒr*|j |¡dSq*W|S)Nr)r rÚ NeedMoreDataÚpopleftÚreversedrÚ appendleft)rÚlineZateofrrrÚreadlineNs  zBufferedSubFile.readlinecCs|tk s t‚|j |¡dS)N)rÚAssertionErrorr r )rr!rrrÚ unreadline`s zBufferedSubFile.unreadlinecCsx|j |¡d|kr d|kr dS|j d¡|j ¡}|j d¡|j ¡|d d¡sj|j | ¡¡| |¡dS)z$Push some new data into this object.r ú Nréÿÿÿÿ)r ÚwriterrrÚendswithrr)rÚdataÚpartsrrrÚpushes     zBufferedSubFile.pushcCs|j |¡dS)N)r Úextend)rÚlinesrrrrzszBufferedSubFile.pushlinescCs|S)Nr)rrrrÚ__iter__}szBufferedSubFile.__iter__cCs| ¡}|dkrt‚|S)Nr)r"Ú StopIteration)rr!rrrÚ__next__€szBufferedSubFile.__next__N)Ú__name__Ú __module__Ú __qualname__Ú__doc__rrrrr"r$r+rr.r0rrrrr -s r c@s`eZdZdZdedœdd„Zdd„Zdd „Zd d „Zd d „Z dd„Z dd„Z dd„Z dd„Z dS)rzA feed-style parser of email.N)ÚpolicycCsž||_d|_|dkr<|jdkr2ddlm}||_qn|j|_n2||_y||jdWntk rld|_YnXtƒ|_g|_ |  ¡j |_ d|_ d|_d|_dS)a_factory is called with no arguments to create a new message obj The policy keyword specifies a policy object that controls a number of aspects of the parser's operation. The default policy maintains backward compatibility. FNr)ÚMessage)r5T)r5Ú_old_style_factoryZmessage_factoryZ email.messager6Ú_factoryÚ TypeErrorr Ú_inputÚ _msgstackÚ _parsegenr0Ú_parseÚ_curÚ_lastÚ _headersonly)rr8r5r6rrrr‹s$     zFeedParser.__init__cCs d|_dS)NT)r@)rrrrÚ_set_headersonlyªszFeedParser._set_headersonlycCs|j |¡| ¡dS)zPush more data into the parser.N)r:r+Ú _call_parse)rr)rrrÚfeed­s zFeedParser.feedcCs&y | ¡Wntk r YnXdS)N)r=r/)rrrrrB²s zFeedParser._call_parsecCsR|j ¡| ¡| ¡}|jr$t‚| ¡dkrN| ¡sNt  ¡}|j   ||¡|S)zÚget_content_typeZset_default_typer;Zattachrr?)rÚmsgrrrÚ _new_messageÅs   zFeedParser._new_messagecCs(|j ¡}|jr|jd|_nd|_|S)Nr&)r;rr>)rÚretvalrrrrEÒs  zFeedParser._pop_messageccs@| ¡g}xb|jD]X}|tkr(tVqt |¡sbt |¡s`t ¡}|j  |j |¡|j  |¡P|  |¡qW|  |¡|jrÎg}x2|j ¡}|tkr tVq†|dkrªP|  |¡q†W|j  t |¡¡dS|j  ¡dkr–x²|j tj¡x$| ¡D]}|tkrtVqøPqøW| ¡}|j ¡x&|j ¡}|tkrFtVq(Pq(Wx&|j ¡}|tkrntVqPPqPW|dkr‚P|j  |¡qàWdS|j  ¡dkrÜx(| ¡D]}|tkrÈtVq°Pq°W| ¡dS|j  ¡dkrö|j  ¡}|dkrbt ¡}|j  |j |¡g}x.|jD]$}|tkrz4)(?P--)?(?P[ \t]*)(?P\r\n|\r|\n)?$TFÚendÚlinesepr&r)/rLr:rÚheaderREÚmatchÚNLCRErZ MissingHeaderBodySeparatorDefectr5rGr>r$rÚ_parse_headersr@r"Z set_payloadÚ EMPTYSTRINGÚjoinrJrr<rErrFZ get_boundaryZNoBoundaryInMultipartDefectÚgetÚlowerZ-InvalidMultipartContentTransferEncodingDefectÚreÚcompileÚescapeÚgroupÚ NLCRE_eolÚsearchÚlenÚpreambler?ÚepilogueZ_payloadÚ isinstanceÚstrr#ZStartBoundaryNotFoundDefectZCloseBoundaryNotFoundDefectÚ NLCRE_bol)rZheadersr!rIr-rMrKÚboundaryZ separatorZ boundaryreZcapturing_preamblerarQZclose_boundary_seenÚmoZlastlineZeolmorbrPZpayloadÚ firstlineZbolmorrrr<Úsh                                                 zFeedParser._parsegenc Csnd}g}xDt|ƒD]6\}}|ddkrV|sJt |¡}|j |j|¡q| |¡q|rx|jj|j |¡Ždg}}|  d¡rú|dkrÀt   |¡}|r°|dt |  d¡ƒ …}|j |¡qn:|t |ƒdkrà|j |¡dSt |¡}|jj |¡q| d¡}|dkr(t d¡}|jj |¡q|dks:tdƒ‚|d|…}|g}qW|rj|jj|j |¡ŽdS) Nrrz zFrom éú:zMissing header name.z3_parse_headers fed line with no : and no leading WS)Ú enumeraterZ#FirstHeaderLineIsContinuationDefectr5rGr>rZset_rawZheader_source_parseÚ startswithr^r_r`r]Z set_unixfromr:r$ZMisplacedEnvelopeHeaderDefectZdefectsÚfindZInvalidHeaderDefectr#) rr-Z lastheaderZ lastvalueÚlinenor!rIrgÚirrrrU×sH              zFeedParser._parse_headers)N)r1r2r3r4rrrArCrBrrLrEr<rUrrrrrˆs  ~cs eZdZdZ‡fdd„Z‡ZS)rz(Like FeedParser, but feed accepts bytes.cstƒ | dd¡¡dS)NÚasciiÚsurrogateescape)ÚsuperrCÚdecode)rr))Ú __class__rrrCszBytesFeedParser.feed)r1r2r3r4rCÚ __classcell__rr)rtrrs)r4Ú__all__rZZemailrZemail._policybaserÚ collectionsrÚiorr[rTrer^Z NLCRE_crackrRrVÚNLÚobjectrr rrrrrrÚs(         [