| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | #### | 
					
						
							|  |  |  | # Copyright 2000 by Timothy O'Malley <timo@alum.mit.edu> | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | #                All Rights Reserved | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # Permission to use, copy, modify, and distribute this software | 
					
						
							|  |  |  | # and its documentation for any purpose and without fee is hereby | 
					
						
							|  |  |  | # granted, provided that the above copyright notice appear in all | 
					
						
							|  |  |  | # copies and that both that copyright notice and this permission | 
					
						
							|  |  |  | # notice appear in supporting documentation, and that the name of | 
					
						
							|  |  |  | # Timothy O'Malley  not be used in advertising or publicity | 
					
						
							|  |  |  | # pertaining to distribution of the software without specific, written | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # prior permission. | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # Timothy O'Malley DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS | 
					
						
							|  |  |  | # SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY | 
					
						
							|  |  |  | # AND FITNESS, IN NO EVENT SHALL Timothy O'Malley BE LIABLE FOR | 
					
						
							|  |  |  | # ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | 
					
						
							|  |  |  | # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, | 
					
						
							|  |  |  | # WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS | 
					
						
							|  |  |  | # ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # PERFORMANCE OF THIS SOFTWARE. | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # | 
					
						
							|  |  |  | #### | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # | 
					
						
							|  |  |  | # Id: Cookie.py,v 2.29 2000/08/23 05:28:49 timo Exp | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | #   by Timothy O'Malley <timo@alum.mit.edu> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #  Cookie.py is a Python module for the handling of HTTP | 
					
						
							|  |  |  | #  cookies as a Python dictionary.  See RFC 2109 for more | 
					
						
							|  |  |  | #  information on cookies. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #  The original idea to treat Cookies as a dictionary came from | 
					
						
							| 
									
										
										
										
											2000-08-24 11:52:33 +00:00
										 |  |  | #  Dave Mitchell (davem@magnet.com) in 1995, when he released the | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | #  first version of nscookie.py. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:39:11 +00:00
										 |  |  | r"""
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | Here's a sample session to show how to use this module. | 
					
						
							|  |  |  | At the moment, this is the only documentation. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The Basics | 
					
						
							|  |  |  | ---------- | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  | Importing is easy... | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-26 16:32:26 +00:00
										 |  |  |    >>> from http import cookies | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  | Most of the time you start by creating a cookie. | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-26 16:32:26 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Once you've created your Cookie, you can add values just as if it were | 
					
						
							|  |  |  | a dictionary. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C["fig"] = "newton" | 
					
						
							|  |  |  |    >>> C["sugar"] = "wafer" | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    >>> C.output() | 
					
						
							|  |  |  |    'Set-Cookie: fig=newton\r\nSet-Cookie: sugar=wafer' | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Notice that the printable representation of a Cookie is the | 
					
						
							|  |  |  | appropriate format for a Set-Cookie: header.  This is the | 
					
						
							|  |  |  | default behavior.  You can change the header and printed | 
					
						
							| 
									
										
										
										
											2003-10-20 14:01:56 +00:00
										 |  |  | attributes by using the .output() function | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C["rocky"] = "road" | 
					
						
							|  |  |  |    >>> C["rocky"]["path"] = "/cookie" | 
					
						
							| 
									
										
										
										
											2007-02-09 20:33:44 +00:00
										 |  |  |    >>> print(C.output(header="Cookie:")) | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    Cookie: rocky=road; Path=/cookie | 
					
						
							| 
									
										
										
										
											2007-02-09 20:33:44 +00:00
										 |  |  |    >>> print(C.output(attrs=[], header="Cookie:")) | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    Cookie: rocky=road | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | The load() method of a Cookie extracts cookies from a string.  In a | 
					
						
							|  |  |  | CGI script, you would use this method to extract the cookies from the | 
					
						
							|  |  |  | HTTP_COOKIE environment variable. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C.load("chips=ahoy; vienna=finger") | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    >>> C.output() | 
					
						
							|  |  |  |    'Set-Cookie: chips=ahoy\r\nSet-Cookie: vienna=finger' | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | The load() method is darn-tootin smart about identifying cookies | 
					
						
							|  |  |  | within a string.  Escaped quotation marks, nested semicolons, and other | 
					
						
							|  |  |  | such trickeries do not confuse it. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C.load('keebler="E=everybody; L=\\"Loves\\"; fudge=\\012;";') | 
					
						
							| 
									
										
										
										
											2007-02-09 20:33:44 +00:00
										 |  |  |    >>> print(C) | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    Set-Cookie: keebler="E=everybody; L=\"Loves\"; fudge=\012;" | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Each element of the Cookie also supports all of the RFC 2109 | 
					
						
							|  |  |  | Cookie attributes.  Here's an example which sets the Path | 
					
						
							|  |  |  | attribute. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C["oreo"] = "doublestuff" | 
					
						
							|  |  |  |    >>> C["oreo"]["path"] = "/" | 
					
						
							| 
									
										
										
										
											2007-02-09 20:33:44 +00:00
										 |  |  |    >>> print(C) | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    Set-Cookie: oreo=doublestuff; Path=/ | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Each dictionary element has a 'value' attribute, which gives you | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | back the value associated with the key. | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C["twix"] = "none for you" | 
					
						
							|  |  |  |    >>> C["twix"].value | 
					
						
							|  |  |  |    'none for you' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The SimpleCookie expects that all values should be standard strings. | 
					
						
							|  |  |  | Just to be sure, SimpleCookie invokes the str() builtin to convert | 
					
						
							|  |  |  | the value to a string, when the values are set dictionary-style. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-26 16:32:26 +00:00
										 |  |  |    >>> C = cookies.SimpleCookie() | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |    >>> C["number"] = 7 | 
					
						
							|  |  |  |    >>> C["string"] = "seven" | 
					
						
							|  |  |  |    >>> C["number"].value | 
					
						
							|  |  |  |    '7' | 
					
						
							|  |  |  |    >>> C["string"].value | 
					
						
							|  |  |  |    'seven' | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |    >>> C.output() | 
					
						
							|  |  |  |    'Set-Cookie: number=7\r\nSet-Cookie: string=seven' | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Finis. | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  | """
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Import our required modules | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2001-08-02 07:15:29 +00:00
										 |  |  | import string | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-28 15:56:30 +00:00
										 |  |  | __all__ = ["CookieError", "BaseCookie", "SimpleCookie"] | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  | _nulljoin = ''.join | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  | _semispacejoin = '; '.join | 
					
						
							| 
									
										
										
										
											2005-08-25 07:32:42 +00:00
										 |  |  | _spacejoin = ' '.join | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # | 
					
						
							|  |  |  | # Define an exception visible to External modules | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | class CookieError(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # These quoting routines conform to the RFC2109 specification, which in | 
					
						
							|  |  |  | # turn references the character definitions from RFC2068.  They provide | 
					
						
							|  |  |  | # a two-way quoting algorithm.  Any non-text character is translated | 
					
						
							|  |  |  | # into a 4 character sequence: a forward-slash followed by the | 
					
						
							|  |  |  | # three-digit octal equivalent of the character.  Any '\' or '"' is | 
					
						
							|  |  |  | # quoted with a preceeding '\' slash. | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # These are taken from RFC2068 and RFC2109. | 
					
						
							|  |  |  | #       _LegalChars       is the list of chars which don't require "'s | 
					
						
							|  |  |  | #       _Translator       hash-table for fast quoting | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2012-04-22 09:19:04 +08:00
										 |  |  | _LegalChars       = string.ascii_letters + string.digits + "!#$%&'*+-.^_`|~:" | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | _Translator       = { | 
					
						
							|  |  |  |     '\000' : '\\000',  '\001' : '\\001',  '\002' : '\\002', | 
					
						
							|  |  |  |     '\003' : '\\003',  '\004' : '\\004',  '\005' : '\\005', | 
					
						
							|  |  |  |     '\006' : '\\006',  '\007' : '\\007',  '\010' : '\\010', | 
					
						
							|  |  |  |     '\011' : '\\011',  '\012' : '\\012',  '\013' : '\\013', | 
					
						
							|  |  |  |     '\014' : '\\014',  '\015' : '\\015',  '\016' : '\\016', | 
					
						
							|  |  |  |     '\017' : '\\017',  '\020' : '\\020',  '\021' : '\\021', | 
					
						
							|  |  |  |     '\022' : '\\022',  '\023' : '\\023',  '\024' : '\\024', | 
					
						
							|  |  |  |     '\025' : '\\025',  '\026' : '\\026',  '\027' : '\\027', | 
					
						
							|  |  |  |     '\030' : '\\030',  '\031' : '\\031',  '\032' : '\\032', | 
					
						
							|  |  |  |     '\033' : '\\033',  '\034' : '\\034',  '\035' : '\\035', | 
					
						
							|  |  |  |     '\036' : '\\036',  '\037' : '\\037', | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-28 18:54:13 +00:00
										 |  |  |     # Because of the way browsers really handle cookies (as opposed | 
					
						
							|  |  |  |     # to what the RFC says) we also encode , and ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ',' : '\\054', ';' : '\\073', | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     '"' : '\\"',       '\\' : '\\\\', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     '\177' : '\\177',  '\200' : '\\200',  '\201' : '\\201', | 
					
						
							|  |  |  |     '\202' : '\\202',  '\203' : '\\203',  '\204' : '\\204', | 
					
						
							|  |  |  |     '\205' : '\\205',  '\206' : '\\206',  '\207' : '\\207', | 
					
						
							|  |  |  |     '\210' : '\\210',  '\211' : '\\211',  '\212' : '\\212', | 
					
						
							|  |  |  |     '\213' : '\\213',  '\214' : '\\214',  '\215' : '\\215', | 
					
						
							|  |  |  |     '\216' : '\\216',  '\217' : '\\217',  '\220' : '\\220', | 
					
						
							|  |  |  |     '\221' : '\\221',  '\222' : '\\222',  '\223' : '\\223', | 
					
						
							|  |  |  |     '\224' : '\\224',  '\225' : '\\225',  '\226' : '\\226', | 
					
						
							|  |  |  |     '\227' : '\\227',  '\230' : '\\230',  '\231' : '\\231', | 
					
						
							|  |  |  |     '\232' : '\\232',  '\233' : '\\233',  '\234' : '\\234', | 
					
						
							|  |  |  |     '\235' : '\\235',  '\236' : '\\236',  '\237' : '\\237', | 
					
						
							|  |  |  |     '\240' : '\\240',  '\241' : '\\241',  '\242' : '\\242', | 
					
						
							|  |  |  |     '\243' : '\\243',  '\244' : '\\244',  '\245' : '\\245', | 
					
						
							|  |  |  |     '\246' : '\\246',  '\247' : '\\247',  '\250' : '\\250', | 
					
						
							|  |  |  |     '\251' : '\\251',  '\252' : '\\252',  '\253' : '\\253', | 
					
						
							|  |  |  |     '\254' : '\\254',  '\255' : '\\255',  '\256' : '\\256', | 
					
						
							|  |  |  |     '\257' : '\\257',  '\260' : '\\260',  '\261' : '\\261', | 
					
						
							|  |  |  |     '\262' : '\\262',  '\263' : '\\263',  '\264' : '\\264', | 
					
						
							|  |  |  |     '\265' : '\\265',  '\266' : '\\266',  '\267' : '\\267', | 
					
						
							|  |  |  |     '\270' : '\\270',  '\271' : '\\271',  '\272' : '\\272', | 
					
						
							|  |  |  |     '\273' : '\\273',  '\274' : '\\274',  '\275' : '\\275', | 
					
						
							|  |  |  |     '\276' : '\\276',  '\277' : '\\277',  '\300' : '\\300', | 
					
						
							|  |  |  |     '\301' : '\\301',  '\302' : '\\302',  '\303' : '\\303', | 
					
						
							|  |  |  |     '\304' : '\\304',  '\305' : '\\305',  '\306' : '\\306', | 
					
						
							|  |  |  |     '\307' : '\\307',  '\310' : '\\310',  '\311' : '\\311', | 
					
						
							|  |  |  |     '\312' : '\\312',  '\313' : '\\313',  '\314' : '\\314', | 
					
						
							|  |  |  |     '\315' : '\\315',  '\316' : '\\316',  '\317' : '\\317', | 
					
						
							|  |  |  |     '\320' : '\\320',  '\321' : '\\321',  '\322' : '\\322', | 
					
						
							|  |  |  |     '\323' : '\\323',  '\324' : '\\324',  '\325' : '\\325', | 
					
						
							|  |  |  |     '\326' : '\\326',  '\327' : '\\327',  '\330' : '\\330', | 
					
						
							|  |  |  |     '\331' : '\\331',  '\332' : '\\332',  '\333' : '\\333', | 
					
						
							|  |  |  |     '\334' : '\\334',  '\335' : '\\335',  '\336' : '\\336', | 
					
						
							|  |  |  |     '\337' : '\\337',  '\340' : '\\340',  '\341' : '\\341', | 
					
						
							|  |  |  |     '\342' : '\\342',  '\343' : '\\343',  '\344' : '\\344', | 
					
						
							|  |  |  |     '\345' : '\\345',  '\346' : '\\346',  '\347' : '\\347', | 
					
						
							|  |  |  |     '\350' : '\\350',  '\351' : '\\351',  '\352' : '\\352', | 
					
						
							|  |  |  |     '\353' : '\\353',  '\354' : '\\354',  '\355' : '\\355', | 
					
						
							|  |  |  |     '\356' : '\\356',  '\357' : '\\357',  '\360' : '\\360', | 
					
						
							|  |  |  |     '\361' : '\\361',  '\362' : '\\362',  '\363' : '\\363', | 
					
						
							|  |  |  |     '\364' : '\\364',  '\365' : '\\365',  '\366' : '\\366', | 
					
						
							|  |  |  |     '\367' : '\\367',  '\370' : '\\370',  '\371' : '\\371', | 
					
						
							|  |  |  |     '\372' : '\\372',  '\373' : '\\373',  '\374' : '\\374', | 
					
						
							|  |  |  |     '\375' : '\\375',  '\376' : '\\376',  '\377' : '\\377' | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-08 15:33:46 +00:00
										 |  |  | def _quote(str, LegalChars=_LegalChars): | 
					
						
							| 
									
										
										
										
											2009-09-04 08:28:01 +00:00
										 |  |  |     r"""Quote a string for use in a cookie header.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If the string does not need to be double-quoted, then just return the | 
					
						
							|  |  |  |     string.  Otherwise, surround the string in doublequotes and quote | 
					
						
							|  |  |  |     (with a \) special characters. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2007-07-03 16:46:40 +00:00
										 |  |  |     if all(c in LegalChars for c in str): | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  |         return str | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2010-12-04 10:39:14 +00:00
										 |  |  |         return '"' + _nulljoin(_Translator.get(s, s) for s in str) + '"' | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _OctalPatt = re.compile(r"\\[0-3][0-7][0-7]") | 
					
						
							|  |  |  | _QuotePatt = re.compile(r"[\\].") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  | def _unquote(str): | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     # If there aren't any doublequotes, | 
					
						
							|  |  |  |     # then there can't be any special characters.  See RFC 2109. | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     if len(str) < 2: | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         return str | 
					
						
							|  |  |  |     if str[0] != '"' or str[-1] != '"': | 
					
						
							|  |  |  |         return str | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # We have to assume that we must decode this string. | 
					
						
							|  |  |  |     # Down to work. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Remove the "s | 
					
						
							|  |  |  |     str = str[1:-1] | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     # Check for special sequences.  Examples: | 
					
						
							|  |  |  |     #    \012 --> \n | 
					
						
							|  |  |  |     #    \"   --> " | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     i = 0 | 
					
						
							|  |  |  |     n = len(str) | 
					
						
							|  |  |  |     res = [] | 
					
						
							|  |  |  |     while 0 <= i < n: | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         o_match = _OctalPatt.search(str, i) | 
					
						
							|  |  |  |         q_match = _QuotePatt.search(str, i) | 
					
						
							|  |  |  |         if not o_match and not q_match:              # Neither matched | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             res.append(str[i:]) | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  |         # else: | 
					
						
							|  |  |  |         j = k = -1 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         if o_match: | 
					
						
							|  |  |  |             j = o_match.start(0) | 
					
						
							|  |  |  |         if q_match: | 
					
						
							|  |  |  |             k = q_match.start(0) | 
					
						
							|  |  |  |         if q_match and (not o_match or k < j):     # QuotePatt matched | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             res.append(str[i:k]) | 
					
						
							|  |  |  |             res.append(str[k+1]) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             i = k + 2 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         else:                                      # OctalPatt matched | 
					
						
							|  |  |  |             res.append(str[i:j]) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             res.append(chr(int(str[j+1:j+4], 8))) | 
					
						
							|  |  |  |             i = j + 4 | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  |     return _nulljoin(res) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  | # The _getdate() routine is used to set the expiration time in the cookie's HTTP | 
					
						
							|  |  |  | # header.  By default, _getdate() returns the current time in the appropriate | 
					
						
							|  |  |  | # "expires" format for a Set-Cookie header.  The one optional argument is an | 
					
						
							|  |  |  | # offset from now, in seconds.  For example, an offset of -3600 means "one hour | 
					
						
							|  |  |  | # ago".  The offset may be a floating point number. | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _weekdayname = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _monthname = [None, | 
					
						
							|  |  |  |               'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', | 
					
						
							|  |  |  |               'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _getdate(future=0, weekdayname=_weekdayname, monthname=_monthname): | 
					
						
							|  |  |  |     from time import gmtime, time | 
					
						
							|  |  |  |     now = time() | 
					
						
							|  |  |  |     year, month, day, hh, mm, ss, wd, y, z = gmtime(now + future) | 
					
						
							| 
									
										
										
										
											2012-05-20 12:05:16 +08:00
										 |  |  |     return "%s, %02d %3s %4d %02d:%02d:%02d GMT" % \ | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |            (weekdayname[wd], day, monthname[month], year, hh, mm, ss) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  | class Morsel(dict): | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     """A class to hold ONE (key, value) pair.
 | 
					
						
							| 
									
										
										
										
											2009-09-04 08:28:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     In a cookie, each such pair may have several attributes, so this class is | 
					
						
							|  |  |  |     used to keep the attributes associated with the appropriate key,value pair. | 
					
						
							|  |  |  |     This class also includes a coded_value attribute, which is used to hold | 
					
						
							|  |  |  |     the network representation of the value.  This is most useful when Python | 
					
						
							|  |  |  |     objects are pickled for network transit. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     # RFC 2109 lists these attributes as reserved: | 
					
						
							|  |  |  |     #   path       comment         domain | 
					
						
							|  |  |  |     #   max-age    secure      version | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     # | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     # For historical reasons, these attributes are also reserved: | 
					
						
							|  |  |  |     #   expires | 
					
						
							|  |  |  |     # | 
					
						
							| 
									
										
										
										
											2008-09-06 19:37:35 +00:00
										 |  |  |     # This is an extension from Microsoft: | 
					
						
							|  |  |  |     #   httponly | 
					
						
							|  |  |  |     # | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     # This dictionary provides a mapping from the lowercase | 
					
						
							|  |  |  |     # variant on the left to the appropriate traditional | 
					
						
							|  |  |  |     # formatting on the right. | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     _reserved = { | 
					
						
							|  |  |  |         "expires"  : "expires", | 
					
						
							|  |  |  |         "path"     : "Path", | 
					
						
							|  |  |  |         "comment"  : "Comment", | 
					
						
							|  |  |  |         "domain"   : "Domain", | 
					
						
							|  |  |  |         "max-age"  : "Max-Age", | 
					
						
							|  |  |  |         "secure"   : "secure", | 
					
						
							|  |  |  |         "httponly" : "httponly", | 
					
						
							|  |  |  |         "version"  : "Version", | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-25 11:09:02 -04:00
										 |  |  |     _flags = {'secure', 'httponly'} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     def __init__(self): | 
					
						
							|  |  |  |         # Set defaults | 
					
						
							|  |  |  |         self.key = self.value = self.coded_value = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Set default attributes | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         for key in self._reserved: | 
					
						
							|  |  |  |             dict.__setitem__(self, key, "") | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __setitem__(self, K, V): | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  |         K = K.lower() | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  |         if not K in self._reserved: | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             raise CookieError("Invalid Attribute %s" % K) | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  |         dict.__setitem__(self, K, V) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def isReservedKey(self, K): | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  |         return K.lower() in self._reserved | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-08 15:33:46 +00:00
										 |  |  |     def set(self, key, val, coded_val, LegalChars=_LegalChars): | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         # First we verify that the key isn't a reserved word | 
					
						
							|  |  |  |         # Second we make sure it only contains legal characters | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  |         if key.lower() in self._reserved: | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             raise CookieError("Attempt to set a reserved key: %s" % key) | 
					
						
							| 
									
										
										
										
											2007-07-03 16:46:40 +00:00
										 |  |  |         if any(c not in LegalChars for c in key): | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             raise CookieError("Illegal key value: %s" % key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # It's a good key, so save it. | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         self.key = key | 
					
						
							|  |  |  |         self.value = val | 
					
						
							|  |  |  |         self.coded_value = coded_val | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     def output(self, attrs=None, header="Set-Cookie:"): | 
					
						
							|  |  |  |         return "%s %s" % (header, self.OutputString(attrs)) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-24 11:52:33 +00:00
										 |  |  |     __str__ = output | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-24 11:52:33 +00:00
										 |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return '<%s: %s=%s>' % (self.__class__.__name__, | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |                                 self.key, repr(self.value)) | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     def js_output(self, attrs=None): | 
					
						
							|  |  |  |         # Print javascript | 
					
						
							|  |  |  |         return """
 | 
					
						
							| 
									
										
										
										
											2005-06-26 21:02:49 +00:00
										 |  |  |         <script type="text/javascript"> | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         <!-- begin hiding | 
					
						
							| 
									
										
										
										
											2005-06-26 21:02:49 +00:00
										 |  |  |         document.cookie = \"%s\"; | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         // end hiding --> | 
					
						
							|  |  |  |         </script> | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         """ % (self.OutputString(attrs).replace('"', r'\"'))
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def OutputString(self, attrs=None): | 
					
						
							|  |  |  |         # Build up our result | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         result = [] | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         append = result.append | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         # First, the key=value pair | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         append("%s=%s" % (self.key, self.coded_value)) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Now add any defined attributes | 
					
						
							| 
									
										
										
										
											2000-12-12 23:20:45 +00:00
										 |  |  |         if attrs is None: | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  |             attrs = self._reserved | 
					
						
							| 
									
										
										
										
											2007-02-11 06:12:03 +00:00
										 |  |  |         items = sorted(self.items()) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         for key, value in items: | 
					
						
							|  |  |  |             if value == "": | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             if key not in attrs: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             if key == "expires" and isinstance(value, int): | 
					
						
							|  |  |  |                 append("%s=%s" % (self._reserved[key], _getdate(value))) | 
					
						
							|  |  |  |             elif key == "max-age" and isinstance(value, int): | 
					
						
							|  |  |  |                 append("%s=%d" % (self._reserved[key], value)) | 
					
						
							|  |  |  |             elif key == "secure": | 
					
						
							|  |  |  |                 append(str(self._reserved[key])) | 
					
						
							|  |  |  |             elif key == "httponly": | 
					
						
							|  |  |  |                 append(str(self._reserved[key])) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |                 append("%s=%s" % (self._reserved[key], value)) | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         # Return the result | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |         return _semispacejoin(result) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Pattern for finding cookie | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This used to be strict parsing based on the RFC2109 and RFC2068 | 
					
						
							|  |  |  | # specifications.  I have since discovered that MSIE 3.0x doesn't | 
					
						
							|  |  |  | # follow the character rules outlined in those specs.  As a | 
					
						
							|  |  |  | # result, the parsing rules here are less strict. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-02-20 22:11:24 +00:00
										 |  |  | _LegalCharsPatt  = r"[\w\d!#%&'~_`><@,:/\$\*\+\-\.\^\|\)\(\?\}\{\=]" | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  | _CookiePattern = re.compile(r"""
 | 
					
						
							|  |  |  |     (?x)                           # This is a verbose pattern | 
					
						
							| 
									
										
										
										
											2014-09-17 00:23:55 +02:00
										 |  |  |     \s*                            # Optional whitespace at start of cookie | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     (?P<key>                       # Start of group 'key' | 
					
						
							|  |  |  |     """ + _LegalCharsPatt + r"""+?   # Any word of at least one letter | 
					
						
							|  |  |  |     )                              # End of group 'key' | 
					
						
							| 
									
										
										
										
											2013-08-25 11:09:02 -04:00
										 |  |  |     (                              # Optional group: there may not be a value. | 
					
						
							|  |  |  |     \s*=\s*                          # Equal Sign | 
					
						
							|  |  |  |     (?P<val>                         # Start of group 'val' | 
					
						
							|  |  |  |     "(?:[^\\"]|\\.)*"                  # Any doublequoted string | 
					
						
							|  |  |  |     |                                  # or | 
					
						
							| 
									
										
										
										
											2012-05-20 16:58:30 +08:00
										 |  |  |     \w{3},\s[\w\d\s-]{9,11}\s[\d:]{8}\sGMT  # Special case for "expires" attr | 
					
						
							| 
									
										
										
										
											2013-08-25 11:09:02 -04:00
										 |  |  |     |                                  # or | 
					
						
							|  |  |  |     """ + _LegalCharsPatt + r"""*      # Any word or empty string | 
					
						
							|  |  |  |     )                                # End of group 'val' | 
					
						
							|  |  |  |     )?                             # End of optional value group | 
					
						
							|  |  |  |     \s*                            # Any number of spaces. | 
					
						
							|  |  |  |     (\s+|;|$)                      # Ending either at space, semicolon, or EOS. | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     """, re.ASCII)                 # May be removed if safe.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # At long last, here is the cookie class.  Using this class is almost just like | 
					
						
							|  |  |  | # using a dictionary.  See this module's docstring for example usage. | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  | class BaseCookie(dict): | 
					
						
							| 
									
										
										
										
											2009-09-04 08:28:01 +00:00
										 |  |  |     """A container class for a set of Morsels.""" | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     def value_decode(self, val): | 
					
						
							|  |  |  |         """real_value, coded_value = value_decode(STRING)
 | 
					
						
							|  |  |  |         Called prior to setting a cookie's value from the network | 
					
						
							|  |  |  |         representation.  The VALUE is the value read from HTTP | 
					
						
							|  |  |  |         header. | 
					
						
							|  |  |  |         Override this function to modify the behavior of cookies. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return val, val | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def value_encode(self, val): | 
					
						
							|  |  |  |         """real_value, coded_value = value_encode(VALUE)
 | 
					
						
							|  |  |  |         Called prior to setting a cookie's value from the dictionary | 
					
						
							|  |  |  |         representation.  The VALUE is the value being assigned. | 
					
						
							|  |  |  |         Override this function to modify the behavior of cookies. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         strval = str(val) | 
					
						
							|  |  |  |         return strval, strval | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     def __init__(self, input=None): | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         if input: | 
					
						
							|  |  |  |             self.load(input) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __set(self, key, real_value, coded_value): | 
					
						
							|  |  |  |         """Private method for setting a cookie's value""" | 
					
						
							|  |  |  |         M = self.get(key, Morsel()) | 
					
						
							|  |  |  |         M.set(key, real_value, coded_value) | 
					
						
							| 
									
										
										
										
											2002-06-26 15:19:01 +00:00
										 |  |  |         dict.__setitem__(self, key, M) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __setitem__(self, key, value): | 
					
						
							|  |  |  |         """Dictionary style assignment.""" | 
					
						
							| 
									
										
										
										
											2014-11-02 22:18:25 +02:00
										 |  |  |         if isinstance(value, Morsel): | 
					
						
							|  |  |  |             # allow assignment of constructed Morsels (e.g. for pickling) | 
					
						
							|  |  |  |             dict.__setitem__(self, key, value) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             rval, cval = self.value_encode(value) | 
					
						
							|  |  |  |             self.__set(key, rval, cval) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-08-24 22:34:21 +00:00
										 |  |  |     def output(self, attrs=None, header="Set-Cookie:", sep="\015\012"): | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         """Return a string suitable for HTTP.""" | 
					
						
							|  |  |  |         result = [] | 
					
						
							| 
									
										
										
										
											2007-02-11 06:12:03 +00:00
										 |  |  |         items = sorted(self.items()) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         for key, value in items: | 
					
						
							|  |  |  |             result.append(value.output(attrs, header)) | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  |         return sep.join(result) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-24 11:52:33 +00:00
										 |  |  |     __str__ = output | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         l = [] | 
					
						
							| 
									
										
										
										
											2007-02-11 06:12:03 +00:00
										 |  |  |         items = sorted(self.items()) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         for key, value in items: | 
					
						
							|  |  |  |             l.append('%s=%s' % (key, repr(value.value))) | 
					
						
							|  |  |  |         return '<%s: %s>' % (self.__class__.__name__, _spacejoin(l)) | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     def js_output(self, attrs=None): | 
					
						
							|  |  |  |         """Return a string suitable for JavaScript.""" | 
					
						
							|  |  |  |         result = [] | 
					
						
							| 
									
										
										
										
											2007-02-11 06:12:03 +00:00
										 |  |  |         items = sorted(self.items()) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         for key, value in items: | 
					
						
							|  |  |  |             result.append(value.js_output(attrs)) | 
					
						
							| 
									
										
										
										
											2002-04-26 02:29:55 +00:00
										 |  |  |         return _nulljoin(result) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def load(self, rawdata): | 
					
						
							|  |  |  |         """Load cookies from a string (presumably HTTP_COOKIE) or
 | 
					
						
							|  |  |  |         from a dictionary.  Loading cookies from a dictionary 'd' | 
					
						
							|  |  |  |         is equivalent to calling: | 
					
						
							|  |  |  |             map(Cookie.__setitem__, d.keys(), d.values()) | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         if isinstance(rawdata, str): | 
					
						
							|  |  |  |             self.__parse_string(rawdata) | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
											  
											
												Merged revisions 74277,74321,74323,74326,74355,74465,74467,74488,74492,74513,74531,74549,74553,74625,74632,74643-74644,74647,74652,74666,74671,74727,74739 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r74277 | sean.reifschneider | 2009-08-01 18:54:55 -0500 (Sat, 01 Aug 2009) | 3 lines
  - Issue #6624: yArg_ParseTuple with "s" format when parsing argument with
    NUL: Bogus TypeError detail string.
........
  r74321 | guilherme.polo | 2009-08-05 11:51:41 -0500 (Wed, 05 Aug 2009) | 1 line
  Easier reference to find (at least while svn continues being used).
........
  r74323 | guilherme.polo | 2009-08-05 18:48:26 -0500 (Wed, 05 Aug 2009) | 1 line
  Typo.
........
  r74326 | jesse.noller | 2009-08-05 21:05:56 -0500 (Wed, 05 Aug 2009) | 1 line
  Fix issue 4660: spurious task_done errors in multiprocessing, remove doc note for from_address
........
  r74355 | gregory.p.smith | 2009-08-12 12:02:37 -0500 (Wed, 12 Aug 2009) | 2 lines
  comment typo fix
........
  r74465 | vinay.sajip | 2009-08-15 18:23:12 -0500 (Sat, 15 Aug 2009) | 1 line
  Added section on logging to one file from multiple processes.
........
  r74467 | vinay.sajip | 2009-08-15 18:34:47 -0500 (Sat, 15 Aug 2009) | 1 line
  Refined section on logging to one file from multiple processes.
........
  r74488 | vinay.sajip | 2009-08-17 08:14:37 -0500 (Mon, 17 Aug 2009) | 1 line
  Further refined section on logging to one file from multiple processes.
........
  r74492 | r.david.murray | 2009-08-17 14:26:49 -0500 (Mon, 17 Aug 2009) | 2 lines
  Issue 6685: 'toupper' -> 'upper' in cgi doc example explanation.
........
  r74513 | skip.montanaro | 2009-08-18 09:37:52 -0500 (Tue, 18 Aug 2009) | 1 line
  missing module ref (issue6723)
........
  r74531 | vinay.sajip | 2009-08-20 17:04:32 -0500 (Thu, 20 Aug 2009) | 1 line
  Added section on exceptions raised during logging.
........
  r74549 | benjamin.peterson | 2009-08-24 12:42:36 -0500 (Mon, 24 Aug 2009) | 1 line
  fix pdf building by teaching latex the right encoding package
........
  r74553 | r.david.murray | 2009-08-26 20:04:59 -0500 (Wed, 26 Aug 2009) | 2 lines
  Remove leftover text from end of sentence.
........
  r74625 | benjamin.peterson | 2009-09-01 17:27:57 -0500 (Tue, 01 Sep 2009) | 1 line
  remove the check that classmethod's argument is a callable
........
  r74632 | georg.brandl | 2009-09-03 02:27:26 -0500 (Thu, 03 Sep 2009) | 1 line
  #6828: fix wrongly highlighted blocks.
........
  r74643 | georg.brandl | 2009-09-04 01:59:20 -0500 (Fri, 04 Sep 2009) | 2 lines
  Issue #2666: Handle BROWSER environment variable properly for unknown browser names in the webbrowser module.
........
  r74644 | georg.brandl | 2009-09-04 02:55:14 -0500 (Fri, 04 Sep 2009) | 1 line
  #5047: remove Monterey support from configure.
........
  r74647 | georg.brandl | 2009-09-04 03:17:04 -0500 (Fri, 04 Sep 2009) | 2 lines
  Issue #5275: In Cookie's Cookie.load(), properly handle non-string arguments as documented.
........
  r74652 | georg.brandl | 2009-09-04 06:25:37 -0500 (Fri, 04 Sep 2009) | 1 line
  #6756: add some info about the "acct" parameter.
........
  r74666 | georg.brandl | 2009-09-05 04:04:09 -0500 (Sat, 05 Sep 2009) | 1 line
  #6841: remove duplicated word.
........
  r74671 | georg.brandl | 2009-09-05 11:47:17 -0500 (Sat, 05 Sep 2009) | 1 line
  #6843: add link from filterwarnings to where the meaning of the arguments is covered.
........
  r74727 | benjamin.peterson | 2009-09-08 18:04:22 -0500 (Tue, 08 Sep 2009) | 1 line
  #6865 fix ref counting in initialization of pwd module
........
  r74739 | georg.brandl | 2009-09-11 02:55:20 -0500 (Fri, 11 Sep 2009) | 1 line
  Move function back to its section.
........
											
										 
											2009-09-11 22:24:02 +00:00
										 |  |  |             # self.update() wouldn't call our custom __setitem__ | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             for key, value in rawdata.items(): | 
					
						
							|  |  |  |                 self[key] = value | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         return | 
					
						
							| 
									
										
										
										
											2000-08-24 14:40:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |     def __parse_string(self, str, patt=_CookiePattern): | 
					
						
							| 
									
										
										
										
											2014-11-21 01:20:57 +01:00
										 |  |  |         i = 0                 # Our starting point | 
					
						
							|  |  |  |         n = len(str)          # Length of string | 
					
						
							|  |  |  |         parsed_items = []     # Parsed (type, key, value) triples | 
					
						
							|  |  |  |         morsel_seen = False   # A key=value pair was previously encountered | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         TYPE_ATTRIBUTE = 1 | 
					
						
							|  |  |  |         TYPE_KEYVALUE = 2 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-21 01:20:57 +01:00
										 |  |  |         # We first parse the whole cookie string and reject it if it's | 
					
						
							|  |  |  |         # syntactically invalid (this helps avoid some classes of injection | 
					
						
							|  |  |  |         # attacks). | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |         while 0 <= i < n: | 
					
						
							|  |  |  |             # Start looking for a cookie | 
					
						
							| 
									
										
										
										
											2014-09-17 00:23:55 +02:00
										 |  |  |             match = patt.match(str, i) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             if not match: | 
					
						
							|  |  |  |                 # No more cookies | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             key, value = match.group("key"), match.group("val") | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |             i = match.end(0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             if key[0] == "$": | 
					
						
							| 
									
										
										
										
											2014-11-21 01:20:57 +01:00
										 |  |  |                 if not morsel_seen: | 
					
						
							|  |  |  |                     # We ignore attributes which pertain to the cookie | 
					
						
							|  |  |  |                     # mechanism as a whole, such as "$Version". | 
					
						
							|  |  |  |                     # See RFC 2965. (Does anyone care?) | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 parsed_items.append((TYPE_ATTRIBUTE, key[1:], value)) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |             elif key.lower() in Morsel._reserved: | 
					
						
							| 
									
										
										
										
											2014-11-21 01:20:57 +01:00
										 |  |  |                 if not morsel_seen: | 
					
						
							|  |  |  |                     # Invalid cookie string | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 if value is None: | 
					
						
							|  |  |  |                     if key.lower() in Morsel._flags: | 
					
						
							|  |  |  |                         parsed_items.append((TYPE_ATTRIBUTE, key, True)) | 
					
						
							| 
									
										
										
										
											2013-08-25 11:09:02 -04:00
										 |  |  |                     else: | 
					
						
							| 
									
										
										
										
											2014-11-21 01:20:57 +01:00
										 |  |  |                         # Invalid cookie string | 
					
						
							|  |  |  |                         return | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     parsed_items.append((TYPE_ATTRIBUTE, key, _unquote(value))) | 
					
						
							| 
									
										
										
										
											2013-08-25 11:09:02 -04:00
										 |  |  |             elif value is not None: | 
					
						
							| 
									
										
										
										
											2014-11-21 01:20:57 +01:00
										 |  |  |                 parsed_items.append((TYPE_KEYVALUE, key, self.value_decode(value))) | 
					
						
							|  |  |  |                 morsel_seen = True | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 # Invalid cookie string | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The cookie string is valid, apply it. | 
					
						
							|  |  |  |         M = None         # current morsel | 
					
						
							|  |  |  |         for tp, key, value in parsed_items: | 
					
						
							|  |  |  |             if tp == TYPE_ATTRIBUTE: | 
					
						
							|  |  |  |                 assert M is not None | 
					
						
							|  |  |  |                 M[key] = value | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 assert tp == TYPE_KEYVALUE | 
					
						
							|  |  |  |                 rval, cval = value | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |                 self.__set(key, rval, cval) | 
					
						
							|  |  |  |                 M = self[key] | 
					
						
							| 
									
										
										
										
											2009-09-04 08:22:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class SimpleCookie(BaseCookie): | 
					
						
							| 
									
										
										
										
											2009-09-04 08:28:01 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     SimpleCookie supports strings as cookie values.  When setting | 
					
						
							|  |  |  |     the value using the dictionary assignment notation, SimpleCookie | 
					
						
							|  |  |  |     calls the builtin str() to convert the value to a string.  Values | 
					
						
							|  |  |  |     received from HTTP are kept as strings. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     def value_decode(self, val): | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         return _unquote(val), val | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-19 13:01:19 +00:00
										 |  |  |     def value_encode(self, val): | 
					
						
							|  |  |  |         strval = str(val) | 
					
						
							| 
									
										
										
										
											2010-07-31 21:04:00 +00:00
										 |  |  |         return strval, _quote(strval) |