blob: 7e0259ee32e46303455d1b73121a1f1032838871 [file] [log] [blame]
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +00001####
2# Copyright 2000 by Timothy O'Malley <timo@alum.mit.edu>
Tim Peters88869f92001-01-14 23:36:06 +00003#
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +00004# All Rights Reserved
Tim Peters88869f92001-01-14 23:36:06 +00005#
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +00006# Permission to use, copy, modify, and distribute this software
7# and its documentation for any purpose and without fee is hereby
8# granted, provided that the above copyright notice appear in all
9# copies and that both that copyright notice and this permission
10# notice appear in supporting documentation, and that the name of
11# Timothy O'Malley not be used in advertising or publicity
12# pertaining to distribution of the software without specific, written
Tim Peters88869f92001-01-14 23:36:06 +000013# prior permission.
14#
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000015# Timothy O'Malley DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS
16# SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
17# AND FITNESS, IN NO EVENT SHALL Timothy O'Malley BE LIABLE FOR
18# ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
19# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
20# WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
21# ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
Tim Peters88869f92001-01-14 23:36:06 +000022# PERFORMANCE OF THIS SOFTWARE.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000023#
24####
Tim Peters88869f92001-01-14 23:36:06 +000025#
26# Id: Cookie.py,v 2.29 2000/08/23 05:28:49 timo Exp
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000027# by Timothy O'Malley <timo@alum.mit.edu>
28#
29# Cookie.py is a Python module for the handling of HTTP
30# cookies as a Python dictionary. See RFC 2109 for more
31# information on cookies.
32#
33# The original idea to treat Cookies as a dictionary came from
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +000034# Dave Mitchell (davem@magnet.com) in 1995, when he released the
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000035# first version of nscookie.py.
36#
37####
38
Guido van Rossum58b6f5b2001-04-06 19:39:11 +000039r"""
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000040Here's a sample session to show how to use this module.
41At the moment, this is the only documentation.
42
43The Basics
44----------
45
Georg Brandl76e155a2010-07-31 21:04:00 +000046Importing is easy...
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000047
Georg Brandl24420152008-05-26 16:32:26 +000048 >>> from http import cookies
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000049
Georg Brandl61013952008-05-28 15:56:30 +000050Most of the time you start by creating a cookie.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000051
Georg Brandl24420152008-05-26 16:32:26 +000052 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000053
54Once you've created your Cookie, you can add values just as if it were
55a dictionary.
56
Georg Brandl61013952008-05-28 15:56:30 +000057 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000058 >>> C["fig"] = "newton"
59 >>> C["sugar"] = "wafer"
Georg Brandl532efab2005-08-24 22:34:21 +000060 >>> C.output()
61 'Set-Cookie: fig=newton\r\nSet-Cookie: sugar=wafer'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000062
63Notice that the printable representation of a Cookie is the
64appropriate format for a Set-Cookie: header. This is the
65default behavior. You can change the header and printed
Walter Dörwaldf0dfc7a2003-10-20 14:01:56 +000066attributes by using the .output() function
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000067
Georg Brandl61013952008-05-28 15:56:30 +000068 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000069 >>> C["rocky"] = "road"
70 >>> C["rocky"]["path"] = "/cookie"
Guido van Rossumfff80df2007-02-09 20:33:44 +000071 >>> print(C.output(header="Cookie:"))
Georg Brandl532efab2005-08-24 22:34:21 +000072 Cookie: rocky=road; Path=/cookie
Guido van Rossumfff80df2007-02-09 20:33:44 +000073 >>> print(C.output(attrs=[], header="Cookie:"))
Georg Brandl532efab2005-08-24 22:34:21 +000074 Cookie: rocky=road
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000075
76The load() method of a Cookie extracts cookies from a string. In a
77CGI script, you would use this method to extract the cookies from the
78HTTP_COOKIE environment variable.
79
Georg Brandl61013952008-05-28 15:56:30 +000080 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000081 >>> C.load("chips=ahoy; vienna=finger")
Georg Brandl532efab2005-08-24 22:34:21 +000082 >>> C.output()
83 'Set-Cookie: chips=ahoy\r\nSet-Cookie: vienna=finger'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000084
85The load() method is darn-tootin smart about identifying cookies
86within a string. Escaped quotation marks, nested semicolons, and other
87such trickeries do not confuse it.
88
Georg Brandl61013952008-05-28 15:56:30 +000089 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000090 >>> C.load('keebler="E=everybody; L=\\"Loves\\"; fudge=\\012;";')
Guido van Rossumfff80df2007-02-09 20:33:44 +000091 >>> print(C)
Georg Brandl532efab2005-08-24 22:34:21 +000092 Set-Cookie: keebler="E=everybody; L=\"Loves\"; fudge=\012;"
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000093
94Each element of the Cookie also supports all of the RFC 2109
95Cookie attributes. Here's an example which sets the Path
96attribute.
97
Georg Brandl61013952008-05-28 15:56:30 +000098 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +000099 >>> C["oreo"] = "doublestuff"
100 >>> C["oreo"]["path"] = "/"
Guido van Rossumfff80df2007-02-09 20:33:44 +0000101 >>> print(C)
Georg Brandl532efab2005-08-24 22:34:21 +0000102 Set-Cookie: oreo=doublestuff; Path=/
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000103
104Each dictionary element has a 'value' attribute, which gives you
Tim Peters88869f92001-01-14 23:36:06 +0000105back the value associated with the key.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000106
Georg Brandl61013952008-05-28 15:56:30 +0000107 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000108 >>> C["twix"] = "none for you"
109 >>> C["twix"].value
110 'none for you'
111
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000112The SimpleCookie expects that all values should be standard strings.
113Just to be sure, SimpleCookie invokes the str() builtin to convert
114the value to a string, when the values are set dictionary-style.
115
Georg Brandl24420152008-05-26 16:32:26 +0000116 >>> C = cookies.SimpleCookie()
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000117 >>> C["number"] = 7
118 >>> C["string"] = "seven"
119 >>> C["number"].value
120 '7'
121 >>> C["string"].value
122 'seven'
Georg Brandl532efab2005-08-24 22:34:21 +0000123 >>> C.output()
124 'Set-Cookie: number=7\r\nSet-Cookie: string=seven'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000125
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000126Finis.
Georg Brandl76e155a2010-07-31 21:04:00 +0000127"""
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000128
129#
130# Import our required modules
Tim Peters88869f92001-01-14 23:36:06 +0000131#
Georg Brandl76e155a2010-07-31 21:04:00 +0000132import re
Martin v. Löwis02d893c2001-08-02 07:15:29 +0000133import string
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000134
Georg Brandl61013952008-05-28 15:56:30 +0000135__all__ = ["CookieError", "BaseCookie", "SimpleCookie"]
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000136
Fred Draked451ec12002-04-26 02:29:55 +0000137_nulljoin = ''.join
Georg Brandl532efab2005-08-24 22:34:21 +0000138_semispacejoin = '; '.join
Georg Brandl8246c432005-08-25 07:32:42 +0000139_spacejoin = ' '.join
Fred Draked451ec12002-04-26 02:29:55 +0000140
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000141#
142# Define an exception visible to External modules
143#
144class CookieError(Exception):
145 pass
146
147
148# These quoting routines conform to the RFC2109 specification, which in
149# turn references the character definitions from RFC2068. They provide
150# a two-way quoting algorithm. Any non-text character is translated
151# into a 4 character sequence: a forward-slash followed by the
152# three-digit octal equivalent of the character. Any '\' or '"' is
Martin Panter46f50722016-05-26 05:35:26 +0000153# quoted with a preceding '\' slash.
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200154# Because of the way browsers really handle cookies (as opposed to what
155# the RFC says) we also encode "," and ";".
Tim Peters88869f92001-01-14 23:36:06 +0000156#
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000157# These are taken from RFC2068 and RFC2109.
158# _LegalChars is the list of chars which don't require "'s
159# _Translator hash-table for fast quoting
160#
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200161_LegalChars = string.ascii_letters + string.digits + "!#$%&'*+-.^_`|~:"
162_UnescapedChars = _LegalChars + ' ()/<=>?@[]{}'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000163
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200164_Translator = {n: '\\%03o' % n
165 for n in set(range(256)) - set(map(ord, _UnescapedChars))}
166_Translator.update({
167 ord('"'): '\\"',
168 ord('\\'): '\\\\',
169})
R. David Murraye05ca2a2010-12-28 18:54:13 +0000170
Anish Shah102d8132016-02-07 05:36:00 +0500171_is_legal_key = re.compile('[%s]+' % re.escape(_LegalChars)).fullmatch
R. David Murraye05ca2a2010-12-28 18:54:13 +0000172
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200173def _quote(str):
Georg Brandl9cf32a12009-09-04 08:28:01 +0000174 r"""Quote a string for use in a cookie header.
175
176 If the string does not need to be double-quoted, then just return the
177 string. Otherwise, surround the string in doublequotes and quote
178 (with a \) special characters.
179 """
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200180 if str is None or _is_legal_key(str):
Fred Drakeff5364a2000-08-24 14:40:35 +0000181 return str
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000182 else:
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200183 return '"' + str.translate(_Translator) + '"'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000184
185
186_OctalPatt = re.compile(r"\\[0-3][0-7][0-7]")
187_QuotePatt = re.compile(r"[\\].")
188
Fred Draked451ec12002-04-26 02:29:55 +0000189def _unquote(str):
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000190 # If there aren't any doublequotes,
191 # then there can't be any special characters. See RFC 2109.
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200192 if str is None or len(str) < 2:
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000193 return str
194 if str[0] != '"' or str[-1] != '"':
195 return str
196
197 # We have to assume that we must decode this string.
198 # Down to work.
199
200 # Remove the "s
201 str = str[1:-1]
Fred Drakeff5364a2000-08-24 14:40:35 +0000202
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000203 # Check for special sequences. Examples:
204 # \012 --> \n
205 # \" --> "
206 #
207 i = 0
208 n = len(str)
209 res = []
210 while 0 <= i < n:
Georg Brandl76e155a2010-07-31 21:04:00 +0000211 o_match = _OctalPatt.search(str, i)
212 q_match = _QuotePatt.search(str, i)
213 if not o_match and not q_match: # Neither matched
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000214 res.append(str[i:])
215 break
216 # else:
217 j = k = -1
Georg Brandl76e155a2010-07-31 21:04:00 +0000218 if o_match:
219 j = o_match.start(0)
220 if q_match:
221 k = q_match.start(0)
222 if q_match and (not o_match or k < j): # QuotePatt matched
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000223 res.append(str[i:k])
224 res.append(str[k+1])
Georg Brandl76e155a2010-07-31 21:04:00 +0000225 i = k + 2
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000226 else: # OctalPatt matched
227 res.append(str[i:j])
Georg Brandl76e155a2010-07-31 21:04:00 +0000228 res.append(chr(int(str[j+1:j+4], 8)))
229 i = j + 4
Fred Draked451ec12002-04-26 02:29:55 +0000230 return _nulljoin(res)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000231
Georg Brandl76e155a2010-07-31 21:04:00 +0000232# The _getdate() routine is used to set the expiration time in the cookie's HTTP
233# header. By default, _getdate() returns the current time in the appropriate
234# "expires" format for a Set-Cookie header. The one optional argument is an
235# offset from now, in seconds. For example, an offset of -3600 means "one hour
236# ago". The offset may be a floating point number.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000237#
238
239_weekdayname = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun']
240
241_monthname = [None,
242 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun',
243 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
244
245def _getdate(future=0, weekdayname=_weekdayname, monthname=_monthname):
246 from time import gmtime, time
247 now = time()
248 year, month, day, hh, mm, ss, wd, y, z = gmtime(now + future)
Senthil Kumaran00c2ec22012-05-20 12:05:16 +0800249 return "%s, %02d %3s %4d %02d:%02d:%02d GMT" % \
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000250 (weekdayname[wd], day, monthname[month], year, hh, mm, ss)
251
252
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000253class Morsel(dict):
Georg Brandl76e155a2010-07-31 21:04:00 +0000254 """A class to hold ONE (key, value) pair.
Georg Brandl9cf32a12009-09-04 08:28:01 +0000255
256 In a cookie, each such pair may have several attributes, so this class is
257 used to keep the attributes associated with the appropriate key,value pair.
258 This class also includes a coded_value attribute, which is used to hold
259 the network representation of the value. This is most useful when Python
260 objects are pickled for network transit.
261 """
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000262 # RFC 2109 lists these attributes as reserved:
263 # path comment domain
264 # max-age secure version
Tim Peters88869f92001-01-14 23:36:06 +0000265 #
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000266 # For historical reasons, these attributes are also reserved:
267 # expires
268 #
Benjamin Peterson35e661c2008-09-06 19:37:35 +0000269 # This is an extension from Microsoft:
270 # httponly
271 #
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000272 # This dictionary provides a mapping from the lowercase
273 # variant on the left to the appropriate traditional
274 # formatting on the right.
Georg Brandl76e155a2010-07-31 21:04:00 +0000275 _reserved = {
276 "expires" : "expires",
277 "path" : "Path",
278 "comment" : "Comment",
279 "domain" : "Domain",
280 "max-age" : "Max-Age",
Benjamin Petersonbd341622015-01-16 20:43:55 -0500281 "secure" : "Secure",
282 "httponly" : "HttpOnly",
Georg Brandl76e155a2010-07-31 21:04:00 +0000283 "version" : "Version",
284 }
Fred Drakeff5364a2000-08-24 14:40:35 +0000285
R David Murraycd0f74b2013-08-25 11:09:02 -0400286 _flags = {'secure', 'httponly'}
287
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000288 def __init__(self):
289 # Set defaults
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200290 self._key = self._value = self._coded_value = None
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000291
292 # Set default attributes
Georg Brandl76e155a2010-07-31 21:04:00 +0000293 for key in self._reserved:
294 dict.__setitem__(self, key, "")
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000295
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200296 @property
297 def key(self):
298 return self._key
299
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200300 @property
301 def value(self):
302 return self._value
303
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200304 @property
305 def coded_value(self):
306 return self._coded_value
307
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000308 def __setitem__(self, K, V):
Fred Draked451ec12002-04-26 02:29:55 +0000309 K = K.lower()
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000310 if not K in self._reserved:
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200311 raise CookieError("Invalid attribute %r" % (K,))
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000312 dict.__setitem__(self, K, V)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000313
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200314 def setdefault(self, key, val=None):
315 key = key.lower()
316 if key not in self._reserved:
317 raise CookieError("Invalid attribute %r" % (key,))
318 return dict.setdefault(self, key, val)
319
320 def __eq__(self, morsel):
321 if not isinstance(morsel, Morsel):
322 return NotImplemented
323 return (dict.__eq__(self, morsel) and
324 self._value == morsel._value and
325 self._key == morsel._key and
326 self._coded_value == morsel._coded_value)
327
328 __ne__ = object.__ne__
329
330 def copy(self):
331 morsel = Morsel()
332 dict.update(morsel, self)
333 morsel.__dict__.update(self.__dict__)
334 return morsel
335
336 def update(self, values):
337 data = {}
338 for key, val in dict(values).items():
339 key = key.lower()
340 if key not in self._reserved:
341 raise CookieError("Invalid attribute %r" % (key,))
342 data[key] = val
343 dict.update(self, data)
344
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000345 def isReservedKey(self, K):
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000346 return K.lower() in self._reserved
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000347
Serhiy Storchakacc283372017-01-13 09:23:15 +0200348 def set(self, key, val, coded_val):
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000349 if key.lower() in self._reserved:
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200350 raise CookieError('Attempt to set a reserved key %r' % (key,))
351 if not _is_legal_key(key):
352 raise CookieError('Illegal key %r' % (key,))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000353
354 # It's a good key, so save it.
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200355 self._key = key
356 self._value = val
357 self._coded_value = coded_val
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000358
Serhiy Storchaka6c325852015-03-18 18:03:40 +0200359 def __getstate__(self):
360 return {
361 'key': self._key,
362 'value': self._value,
363 'coded_value': self._coded_value,
364 }
365
366 def __setstate__(self, state):
367 self._key = state['key']
368 self._value = state['value']
369 self._coded_value = state['coded_value']
370
Georg Brandl76e155a2010-07-31 21:04:00 +0000371 def output(self, attrs=None, header="Set-Cookie:"):
372 return "%s %s" % (header, self.OutputString(attrs))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000373
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +0000374 __str__ = output
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000375
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +0000376 def __repr__(self):
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200377 return '<%s: %s>' % (self.__class__.__name__, self.OutputString())
Fred Drakeff5364a2000-08-24 14:40:35 +0000378
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000379 def js_output(self, attrs=None):
380 # Print javascript
381 return """
Georg Brandl03a33ea2005-06-26 21:02:49 +0000382 <script type="text/javascript">
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000383 <!-- begin hiding
Georg Brandl03a33ea2005-06-26 21:02:49 +0000384 document.cookie = \"%s\";
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000385 // end hiding -->
386 </script>
Georg Brandl76e155a2010-07-31 21:04:00 +0000387 """ % (self.OutputString(attrs).replace('"', r'\"'))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000388
389 def OutputString(self, attrs=None):
390 # Build up our result
391 #
392 result = []
Georg Brandl76e155a2010-07-31 21:04:00 +0000393 append = result.append
Fred Drakeff5364a2000-08-24 14:40:35 +0000394
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000395 # First, the key=value pair
Georg Brandl76e155a2010-07-31 21:04:00 +0000396 append("%s=%s" % (self.key, self.coded_value))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000397
398 # Now add any defined attributes
Fred Drake8152d322000-12-12 23:20:45 +0000399 if attrs is None:
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000400 attrs = self._reserved
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000401 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000402 for key, value in items:
403 if value == "":
404 continue
405 if key not in attrs:
406 continue
407 if key == "expires" and isinstance(value, int):
408 append("%s=%s" % (self._reserved[key], _getdate(value)))
409 elif key == "max-age" and isinstance(value, int):
410 append("%s=%d" % (self._reserved[key], value))
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200411 elif key in self._flags:
412 if value:
413 append(str(self._reserved[key]))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000414 else:
Georg Brandl76e155a2010-07-31 21:04:00 +0000415 append("%s=%s" % (self._reserved[key], value))
Fred Drakeff5364a2000-08-24 14:40:35 +0000416
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000417 # Return the result
Georg Brandl532efab2005-08-24 22:34:21 +0000418 return _semispacejoin(result)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000419
420
421#
422# Pattern for finding cookie
423#
424# This used to be strict parsing based on the RFC2109 and RFC2068
425# specifications. I have since discovered that MSIE 3.0x doesn't
426# follow the character rules outlined in those specs. As a
427# result, the parsing rules here are less strict.
428#
429
Benjamin Peterson9bd476e2015-05-23 10:36:48 -0500430_LegalKeyChars = r"\w\d!#%&'~_`><@,:/\$\*\+\-\.\^\|\)\(\?\}\{\="
R David Murray44b548d2016-09-08 13:59:53 -0400431_LegalValueChars = _LegalKeyChars + r'\[\]'
Georg Brandl76e155a2010-07-31 21:04:00 +0000432_CookiePattern = re.compile(r"""
Antoine Pitrou7d0b8f92014-09-17 00:23:55 +0200433 \s* # Optional whitespace at start of cookie
Georg Brandl76e155a2010-07-31 21:04:00 +0000434 (?P<key> # Start of group 'key'
Benjamin Peterson9bd476e2015-05-23 10:36:48 -0500435 [""" + _LegalKeyChars + r"""]+? # Any word of at least one letter
Georg Brandl76e155a2010-07-31 21:04:00 +0000436 ) # End of group 'key'
R David Murraycd0f74b2013-08-25 11:09:02 -0400437 ( # Optional group: there may not be a value.
438 \s*=\s* # Equal Sign
439 (?P<val> # Start of group 'val'
440 "(?:[^\\"]|\\.)*" # Any doublequoted string
441 | # or
Senthil Kumaranaeeba262012-05-20 16:58:30 +0800442 \w{3},\s[\w\d\s-]{9,11}\s[\d:]{8}\sGMT # Special case for "expires" attr
R David Murraycd0f74b2013-08-25 11:09:02 -0400443 | # or
Benjamin Petersond504f202015-05-23 10:38:48 -0500444 [""" + _LegalValueChars + r"""]* # Any word or empty string
R David Murraycd0f74b2013-08-25 11:09:02 -0400445 ) # End of group 'val'
446 )? # End of optional value group
447 \s* # Any number of spaces.
448 (\s+|;|$) # Ending either at space, semicolon, or EOS.
Serhiy Storchakabd48d272016-09-11 12:50:02 +0300449 """, re.ASCII | re.VERBOSE) # re.ASCII may be removed if safe.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000450
451
Georg Brandl76e155a2010-07-31 21:04:00 +0000452# At long last, here is the cookie class. Using this class is almost just like
453# using a dictionary. See this module's docstring for example usage.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000454#
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000455class BaseCookie(dict):
Georg Brandl9cf32a12009-09-04 08:28:01 +0000456 """A container class for a set of Morsels."""
Fred Drakeff5364a2000-08-24 14:40:35 +0000457
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000458 def value_decode(self, val):
459 """real_value, coded_value = value_decode(STRING)
460 Called prior to setting a cookie's value from the network
461 representation. The VALUE is the value read from HTTP
462 header.
463 Override this function to modify the behavior of cookies.
464 """
465 return val, val
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000466
467 def value_encode(self, val):
468 """real_value, coded_value = value_encode(VALUE)
469 Called prior to setting a cookie's value from the dictionary
470 representation. The VALUE is the value being assigned.
471 Override this function to modify the behavior of cookies.
472 """
473 strval = str(val)
474 return strval, strval
Fred Drakeff5364a2000-08-24 14:40:35 +0000475
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000476 def __init__(self, input=None):
Georg Brandl76e155a2010-07-31 21:04:00 +0000477 if input:
478 self.load(input)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000479
480 def __set(self, key, real_value, coded_value):
481 """Private method for setting a cookie's value"""
482 M = self.get(key, Morsel())
483 M.set(key, real_value, coded_value)
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000484 dict.__setitem__(self, key, M)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000485
486 def __setitem__(self, key, value):
487 """Dictionary style assignment."""
Serhiy Storchaka8cf7c1c2014-11-02 22:18:25 +0200488 if isinstance(value, Morsel):
489 # allow assignment of constructed Morsels (e.g. for pickling)
490 dict.__setitem__(self, key, value)
491 else:
492 rval, cval = self.value_encode(value)
493 self.__set(key, rval, cval)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000494
Georg Brandl532efab2005-08-24 22:34:21 +0000495 def output(self, attrs=None, header="Set-Cookie:", sep="\015\012"):
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000496 """Return a string suitable for HTTP."""
497 result = []
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000498 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000499 for key, value in items:
500 result.append(value.output(attrs, header))
Fred Draked451ec12002-04-26 02:29:55 +0000501 return sep.join(result)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000502
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +0000503 __str__ = output
504
505 def __repr__(self):
Georg Brandl76e155a2010-07-31 21:04:00 +0000506 l = []
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000507 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000508 for key, value in items:
509 l.append('%s=%s' % (key, repr(value.value)))
510 return '<%s: %s>' % (self.__class__.__name__, _spacejoin(l))
Fred Drakeff5364a2000-08-24 14:40:35 +0000511
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000512 def js_output(self, attrs=None):
513 """Return a string suitable for JavaScript."""
514 result = []
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000515 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000516 for key, value in items:
517 result.append(value.js_output(attrs))
Fred Draked451ec12002-04-26 02:29:55 +0000518 return _nulljoin(result)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000519
520 def load(self, rawdata):
521 """Load cookies from a string (presumably HTTP_COOKIE) or
522 from a dictionary. Loading cookies from a dictionary 'd'
523 is equivalent to calling:
524 map(Cookie.__setitem__, d.keys(), d.values())
525 """
Georg Brandl76e155a2010-07-31 21:04:00 +0000526 if isinstance(rawdata, str):
527 self.__parse_string(rawdata)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000528 else:
Benjamin Peterson8719ad52009-09-11 22:24:02 +0000529 # self.update() wouldn't call our custom __setitem__
Georg Brandl76e155a2010-07-31 21:04:00 +0000530 for key, value in rawdata.items():
531 self[key] = value
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000532 return
Fred Drakeff5364a2000-08-24 14:40:35 +0000533
Georg Brandl76e155a2010-07-31 21:04:00 +0000534 def __parse_string(self, str, patt=_CookiePattern):
Antoine Pitroub1e36072014-11-21 01:20:57 +0100535 i = 0 # Our starting point
536 n = len(str) # Length of string
537 parsed_items = [] # Parsed (type, key, value) triples
538 morsel_seen = False # A key=value pair was previously encountered
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000539
Antoine Pitroub1e36072014-11-21 01:20:57 +0100540 TYPE_ATTRIBUTE = 1
541 TYPE_KEYVALUE = 2
542
543 # We first parse the whole cookie string and reject it if it's
544 # syntactically invalid (this helps avoid some classes of injection
545 # attacks).
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000546 while 0 <= i < n:
547 # Start looking for a cookie
Antoine Pitrou7d0b8f92014-09-17 00:23:55 +0200548 match = patt.match(str, i)
Georg Brandl76e155a2010-07-31 21:04:00 +0000549 if not match:
550 # No more cookies
551 break
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000552
Georg Brandl76e155a2010-07-31 21:04:00 +0000553 key, value = match.group("key"), match.group("val")
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000554 i = match.end(0)
555
Georg Brandl76e155a2010-07-31 21:04:00 +0000556 if key[0] == "$":
Antoine Pitroub1e36072014-11-21 01:20:57 +0100557 if not morsel_seen:
558 # We ignore attributes which pertain to the cookie
559 # mechanism as a whole, such as "$Version".
560 # See RFC 2965. (Does anyone care?)
561 continue
562 parsed_items.append((TYPE_ATTRIBUTE, key[1:], value))
Georg Brandl76e155a2010-07-31 21:04:00 +0000563 elif key.lower() in Morsel._reserved:
Antoine Pitroub1e36072014-11-21 01:20:57 +0100564 if not morsel_seen:
565 # Invalid cookie string
566 return
567 if value is None:
568 if key.lower() in Morsel._flags:
569 parsed_items.append((TYPE_ATTRIBUTE, key, True))
R David Murraycd0f74b2013-08-25 11:09:02 -0400570 else:
Antoine Pitroub1e36072014-11-21 01:20:57 +0100571 # Invalid cookie string
572 return
573 else:
574 parsed_items.append((TYPE_ATTRIBUTE, key, _unquote(value)))
R David Murraycd0f74b2013-08-25 11:09:02 -0400575 elif value is not None:
Antoine Pitroub1e36072014-11-21 01:20:57 +0100576 parsed_items.append((TYPE_KEYVALUE, key, self.value_decode(value)))
577 morsel_seen = True
578 else:
579 # Invalid cookie string
580 return
581
582 # The cookie string is valid, apply it.
583 M = None # current morsel
584 for tp, key, value in parsed_items:
585 if tp == TYPE_ATTRIBUTE:
586 assert M is not None
587 M[key] = value
588 else:
589 assert tp == TYPE_KEYVALUE
590 rval, cval = value
Georg Brandl76e155a2010-07-31 21:04:00 +0000591 self.__set(key, rval, cval)
592 M = self[key]
Georg Brandl4eff9f72009-09-04 08:22:00 +0000593
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000594
595class SimpleCookie(BaseCookie):
Georg Brandl9cf32a12009-09-04 08:28:01 +0000596 """
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000597 SimpleCookie supports strings as cookie values. When setting
598 the value using the dictionary assignment notation, SimpleCookie
599 calls the builtin str() to convert the value to a string. Values
600 received from HTTP are kept as strings.
601 """
602 def value_decode(self, val):
Georg Brandl76e155a2010-07-31 21:04:00 +0000603 return _unquote(val), val
604
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000605 def value_encode(self, val):
606 strval = str(val)
Georg Brandl76e155a2010-07-31 21:04:00 +0000607 return strval, _quote(strval)