blob: dbddd6cb8c2a343fe30b47491528c2f6178f5c2c [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
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200141def _warn_deprecated_setter(setter):
142 import warnings
143 msg = ('The .%s setter is deprecated. The attribute will be read-only in '
144 'future releases. Please use the set() method instead.' % setter)
145 warnings.warn(msg, DeprecationWarning, stacklevel=3)
146
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000147#
148# Define an exception visible to External modules
149#
150class CookieError(Exception):
151 pass
152
153
154# These quoting routines conform to the RFC2109 specification, which in
155# turn references the character definitions from RFC2068. They provide
156# a two-way quoting algorithm. Any non-text character is translated
157# into a 4 character sequence: a forward-slash followed by the
158# three-digit octal equivalent of the character. Any '\' or '"' is
159# quoted with a preceeding '\' slash.
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200160# Because of the way browsers really handle cookies (as opposed to what
161# the RFC says) we also encode "," and ";".
Tim Peters88869f92001-01-14 23:36:06 +0000162#
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000163# These are taken from RFC2068 and RFC2109.
164# _LegalChars is the list of chars which don't require "'s
165# _Translator hash-table for fast quoting
166#
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200167_LegalChars = string.ascii_letters + string.digits + "!#$%&'*+-.^_`|~:"
168_UnescapedChars = _LegalChars + ' ()/<=>?@[]{}'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000169
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200170_Translator = {n: '\\%03o' % n
171 for n in set(range(256)) - set(map(ord, _UnescapedChars))}
172_Translator.update({
173 ord('"'): '\\"',
174 ord('\\'): '\\\\',
175})
R. David Murraye05ca2a2010-12-28 18:54:13 +0000176
Anish Shah102d8132016-02-07 05:36:00 +0500177_is_legal_key = re.compile('[%s]+' % re.escape(_LegalChars)).fullmatch
R. David Murraye05ca2a2010-12-28 18:54:13 +0000178
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200179def _quote(str):
Georg Brandl9cf32a12009-09-04 08:28:01 +0000180 r"""Quote a string for use in a cookie header.
181
182 If the string does not need to be double-quoted, then just return the
183 string. Otherwise, surround the string in doublequotes and quote
184 (with a \) special characters.
185 """
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200186 if str is None or _is_legal_key(str):
Fred Drakeff5364a2000-08-24 14:40:35 +0000187 return str
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000188 else:
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200189 return '"' + str.translate(_Translator) + '"'
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000190
191
192_OctalPatt = re.compile(r"\\[0-3][0-7][0-7]")
193_QuotePatt = re.compile(r"[\\].")
194
Fred Draked451ec12002-04-26 02:29:55 +0000195def _unquote(str):
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000196 # If there aren't any doublequotes,
197 # then there can't be any special characters. See RFC 2109.
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200198 if str is None or len(str) < 2:
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000199 return str
200 if str[0] != '"' or str[-1] != '"':
201 return str
202
203 # We have to assume that we must decode this string.
204 # Down to work.
205
206 # Remove the "s
207 str = str[1:-1]
Fred Drakeff5364a2000-08-24 14:40:35 +0000208
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000209 # Check for special sequences. Examples:
210 # \012 --> \n
211 # \" --> "
212 #
213 i = 0
214 n = len(str)
215 res = []
216 while 0 <= i < n:
Georg Brandl76e155a2010-07-31 21:04:00 +0000217 o_match = _OctalPatt.search(str, i)
218 q_match = _QuotePatt.search(str, i)
219 if not o_match and not q_match: # Neither matched
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000220 res.append(str[i:])
221 break
222 # else:
223 j = k = -1
Georg Brandl76e155a2010-07-31 21:04:00 +0000224 if o_match:
225 j = o_match.start(0)
226 if q_match:
227 k = q_match.start(0)
228 if q_match and (not o_match or k < j): # QuotePatt matched
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000229 res.append(str[i:k])
230 res.append(str[k+1])
Georg Brandl76e155a2010-07-31 21:04:00 +0000231 i = k + 2
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000232 else: # OctalPatt matched
233 res.append(str[i:j])
Georg Brandl76e155a2010-07-31 21:04:00 +0000234 res.append(chr(int(str[j+1:j+4], 8)))
235 i = j + 4
Fred Draked451ec12002-04-26 02:29:55 +0000236 return _nulljoin(res)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000237
Georg Brandl76e155a2010-07-31 21:04:00 +0000238# The _getdate() routine is used to set the expiration time in the cookie's HTTP
239# header. By default, _getdate() returns the current time in the appropriate
240# "expires" format for a Set-Cookie header. The one optional argument is an
241# offset from now, in seconds. For example, an offset of -3600 means "one hour
242# ago". The offset may be a floating point number.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000243#
244
245_weekdayname = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun']
246
247_monthname = [None,
248 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun',
249 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
250
251def _getdate(future=0, weekdayname=_weekdayname, monthname=_monthname):
252 from time import gmtime, time
253 now = time()
254 year, month, day, hh, mm, ss, wd, y, z = gmtime(now + future)
Senthil Kumaran00c2ec22012-05-20 12:05:16 +0800255 return "%s, %02d %3s %4d %02d:%02d:%02d GMT" % \
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000256 (weekdayname[wd], day, monthname[month], year, hh, mm, ss)
257
258
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000259class Morsel(dict):
Georg Brandl76e155a2010-07-31 21:04:00 +0000260 """A class to hold ONE (key, value) pair.
Georg Brandl9cf32a12009-09-04 08:28:01 +0000261
262 In a cookie, each such pair may have several attributes, so this class is
263 used to keep the attributes associated with the appropriate key,value pair.
264 This class also includes a coded_value attribute, which is used to hold
265 the network representation of the value. This is most useful when Python
266 objects are pickled for network transit.
267 """
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000268 # RFC 2109 lists these attributes as reserved:
269 # path comment domain
270 # max-age secure version
Tim Peters88869f92001-01-14 23:36:06 +0000271 #
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000272 # For historical reasons, these attributes are also reserved:
273 # expires
274 #
Benjamin Peterson35e661c2008-09-06 19:37:35 +0000275 # This is an extension from Microsoft:
276 # httponly
277 #
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000278 # This dictionary provides a mapping from the lowercase
279 # variant on the left to the appropriate traditional
280 # formatting on the right.
Georg Brandl76e155a2010-07-31 21:04:00 +0000281 _reserved = {
282 "expires" : "expires",
283 "path" : "Path",
284 "comment" : "Comment",
285 "domain" : "Domain",
286 "max-age" : "Max-Age",
Benjamin Petersonbd341622015-01-16 20:43:55 -0500287 "secure" : "Secure",
288 "httponly" : "HttpOnly",
Georg Brandl76e155a2010-07-31 21:04:00 +0000289 "version" : "Version",
290 }
Fred Drakeff5364a2000-08-24 14:40:35 +0000291
R David Murraycd0f74b2013-08-25 11:09:02 -0400292 _flags = {'secure', 'httponly'}
293
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000294 def __init__(self):
295 # Set defaults
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200296 self._key = self._value = self._coded_value = None
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000297
298 # Set default attributes
Georg Brandl76e155a2010-07-31 21:04:00 +0000299 for key in self._reserved:
300 dict.__setitem__(self, key, "")
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000301
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200302 @property
303 def key(self):
304 return self._key
305
306 @key.setter
307 def key(self, key):
308 _warn_deprecated_setter('key')
309 self._key = key
310
311 @property
312 def value(self):
313 return self._value
314
315 @value.setter
316 def value(self, value):
317 _warn_deprecated_setter('value')
318 self._value = value
319
320 @property
321 def coded_value(self):
322 return self._coded_value
323
324 @coded_value.setter
325 def coded_value(self, coded_value):
326 _warn_deprecated_setter('coded_value')
327 self._coded_value = coded_value
328
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000329 def __setitem__(self, K, V):
Fred Draked451ec12002-04-26 02:29:55 +0000330 K = K.lower()
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000331 if not K in self._reserved:
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200332 raise CookieError("Invalid attribute %r" % (K,))
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000333 dict.__setitem__(self, K, V)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000334
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200335 def setdefault(self, key, val=None):
336 key = key.lower()
337 if key not in self._reserved:
338 raise CookieError("Invalid attribute %r" % (key,))
339 return dict.setdefault(self, key, val)
340
341 def __eq__(self, morsel):
342 if not isinstance(morsel, Morsel):
343 return NotImplemented
344 return (dict.__eq__(self, morsel) and
345 self._value == morsel._value and
346 self._key == morsel._key and
347 self._coded_value == morsel._coded_value)
348
349 __ne__ = object.__ne__
350
351 def copy(self):
352 morsel = Morsel()
353 dict.update(morsel, self)
354 morsel.__dict__.update(self.__dict__)
355 return morsel
356
357 def update(self, values):
358 data = {}
359 for key, val in dict(values).items():
360 key = key.lower()
361 if key not in self._reserved:
362 raise CookieError("Invalid attribute %r" % (key,))
363 data[key] = val
364 dict.update(self, data)
365
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000366 def isReservedKey(self, K):
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000367 return K.lower() in self._reserved
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000368
R David Murray1813c172015-03-29 17:09:21 -0400369 def set(self, key, val, coded_val, LegalChars=_LegalChars):
370 if LegalChars != _LegalChars:
371 import warnings
372 warnings.warn(
373 'LegalChars parameter is deprecated, ignored and will '
374 'be removed in future versions.', DeprecationWarning,
375 stacklevel=2)
376
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000377 if key.lower() in self._reserved:
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200378 raise CookieError('Attempt to set a reserved key %r' % (key,))
379 if not _is_legal_key(key):
380 raise CookieError('Illegal key %r' % (key,))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000381
382 # It's a good key, so save it.
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200383 self._key = key
384 self._value = val
385 self._coded_value = coded_val
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000386
Serhiy Storchaka6c325852015-03-18 18:03:40 +0200387 def __getstate__(self):
388 return {
389 'key': self._key,
390 'value': self._value,
391 'coded_value': self._coded_value,
392 }
393
394 def __setstate__(self, state):
395 self._key = state['key']
396 self._value = state['value']
397 self._coded_value = state['coded_value']
398
Georg Brandl76e155a2010-07-31 21:04:00 +0000399 def output(self, attrs=None, header="Set-Cookie:"):
400 return "%s %s" % (header, self.OutputString(attrs))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000401
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +0000402 __str__ = output
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000403
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +0000404 def __repr__(self):
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200405 return '<%s: %s>' % (self.__class__.__name__, self.OutputString())
Fred Drakeff5364a2000-08-24 14:40:35 +0000406
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000407 def js_output(self, attrs=None):
408 # Print javascript
409 return """
Georg Brandl03a33ea2005-06-26 21:02:49 +0000410 <script type="text/javascript">
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000411 <!-- begin hiding
Georg Brandl03a33ea2005-06-26 21:02:49 +0000412 document.cookie = \"%s\";
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000413 // end hiding -->
414 </script>
Georg Brandl76e155a2010-07-31 21:04:00 +0000415 """ % (self.OutputString(attrs).replace('"', r'\"'))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000416
417 def OutputString(self, attrs=None):
418 # Build up our result
419 #
420 result = []
Georg Brandl76e155a2010-07-31 21:04:00 +0000421 append = result.append
Fred Drakeff5364a2000-08-24 14:40:35 +0000422
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000423 # First, the key=value pair
Georg Brandl76e155a2010-07-31 21:04:00 +0000424 append("%s=%s" % (self.key, self.coded_value))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000425
426 # Now add any defined attributes
Fred Drake8152d322000-12-12 23:20:45 +0000427 if attrs is None:
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000428 attrs = self._reserved
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000429 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000430 for key, value in items:
431 if value == "":
432 continue
433 if key not in attrs:
434 continue
435 if key == "expires" and isinstance(value, int):
436 append("%s=%s" % (self._reserved[key], _getdate(value)))
437 elif key == "max-age" and isinstance(value, int):
438 append("%s=%d" % (self._reserved[key], value))
Serhiy Storchaka9c1a9b22015-03-18 10:59:57 +0200439 elif key in self._flags:
440 if value:
441 append(str(self._reserved[key]))
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000442 else:
Georg Brandl76e155a2010-07-31 21:04:00 +0000443 append("%s=%s" % (self._reserved[key], value))
Fred Drakeff5364a2000-08-24 14:40:35 +0000444
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000445 # Return the result
Georg Brandl532efab2005-08-24 22:34:21 +0000446 return _semispacejoin(result)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000447
448
449#
450# Pattern for finding cookie
451#
452# This used to be strict parsing based on the RFC2109 and RFC2068
453# specifications. I have since discovered that MSIE 3.0x doesn't
454# follow the character rules outlined in those specs. As a
455# result, the parsing rules here are less strict.
456#
457
Benjamin Peterson9bd476e2015-05-23 10:36:48 -0500458_LegalKeyChars = r"\w\d!#%&'~_`><@,:/\$\*\+\-\.\^\|\)\(\?\}\{\="
459_LegalValueChars = _LegalKeyChars + '\[\]'
Georg Brandl76e155a2010-07-31 21:04:00 +0000460_CookiePattern = re.compile(r"""
461 (?x) # This is a verbose pattern
Antoine Pitrou7d0b8f92014-09-17 00:23:55 +0200462 \s* # Optional whitespace at start of cookie
Georg Brandl76e155a2010-07-31 21:04:00 +0000463 (?P<key> # Start of group 'key'
Benjamin Peterson9bd476e2015-05-23 10:36:48 -0500464 [""" + _LegalKeyChars + r"""]+? # Any word of at least one letter
Georg Brandl76e155a2010-07-31 21:04:00 +0000465 ) # End of group 'key'
R David Murraycd0f74b2013-08-25 11:09:02 -0400466 ( # Optional group: there may not be a value.
467 \s*=\s* # Equal Sign
468 (?P<val> # Start of group 'val'
469 "(?:[^\\"]|\\.)*" # Any doublequoted string
470 | # or
Senthil Kumaranaeeba262012-05-20 16:58:30 +0800471 \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 -0400472 | # or
Benjamin Petersond504f202015-05-23 10:38:48 -0500473 [""" + _LegalValueChars + r"""]* # Any word or empty string
R David Murraycd0f74b2013-08-25 11:09:02 -0400474 ) # End of group 'val'
475 )? # End of optional value group
476 \s* # Any number of spaces.
477 (\s+|;|$) # Ending either at space, semicolon, or EOS.
Georg Brandl76e155a2010-07-31 21:04:00 +0000478 """, re.ASCII) # May be removed if safe.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000479
480
Georg Brandl76e155a2010-07-31 21:04:00 +0000481# At long last, here is the cookie class. Using this class is almost just like
482# using a dictionary. See this module's docstring for example usage.
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000483#
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000484class BaseCookie(dict):
Georg Brandl9cf32a12009-09-04 08:28:01 +0000485 """A container class for a set of Morsels."""
Fred Drakeff5364a2000-08-24 14:40:35 +0000486
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000487 def value_decode(self, val):
488 """real_value, coded_value = value_decode(STRING)
489 Called prior to setting a cookie's value from the network
490 representation. The VALUE is the value read from HTTP
491 header.
492 Override this function to modify the behavior of cookies.
493 """
494 return val, val
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000495
496 def value_encode(self, val):
497 """real_value, coded_value = value_encode(VALUE)
498 Called prior to setting a cookie's value from the dictionary
499 representation. The VALUE is the value being assigned.
500 Override this function to modify the behavior of cookies.
501 """
502 strval = str(val)
503 return strval, strval
Fred Drakeff5364a2000-08-24 14:40:35 +0000504
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000505 def __init__(self, input=None):
Georg Brandl76e155a2010-07-31 21:04:00 +0000506 if input:
507 self.load(input)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000508
509 def __set(self, key, real_value, coded_value):
510 """Private method for setting a cookie's value"""
511 M = self.get(key, Morsel())
512 M.set(key, real_value, coded_value)
Raymond Hettinger0a2963c2002-06-26 15:19:01 +0000513 dict.__setitem__(self, key, M)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000514
515 def __setitem__(self, key, value):
516 """Dictionary style assignment."""
Serhiy Storchaka8cf7c1c2014-11-02 22:18:25 +0200517 if isinstance(value, Morsel):
518 # allow assignment of constructed Morsels (e.g. for pickling)
519 dict.__setitem__(self, key, value)
520 else:
521 rval, cval = self.value_encode(value)
522 self.__set(key, rval, cval)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000523
Georg Brandl532efab2005-08-24 22:34:21 +0000524 def output(self, attrs=None, header="Set-Cookie:", sep="\015\012"):
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000525 """Return a string suitable for HTTP."""
526 result = []
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000527 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000528 for key, value in items:
529 result.append(value.output(attrs, header))
Fred Draked451ec12002-04-26 02:29:55 +0000530 return sep.join(result)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000531
Andrew M. Kuchling0b29b112000-08-24 11:52:33 +0000532 __str__ = output
533
534 def __repr__(self):
Georg Brandl76e155a2010-07-31 21:04:00 +0000535 l = []
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000536 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000537 for key, value in items:
538 l.append('%s=%s' % (key, repr(value.value)))
539 return '<%s: %s>' % (self.__class__.__name__, _spacejoin(l))
Fred Drakeff5364a2000-08-24 14:40:35 +0000540
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000541 def js_output(self, attrs=None):
542 """Return a string suitable for JavaScript."""
543 result = []
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000544 items = sorted(self.items())
Georg Brandl76e155a2010-07-31 21:04:00 +0000545 for key, value in items:
546 result.append(value.js_output(attrs))
Fred Draked451ec12002-04-26 02:29:55 +0000547 return _nulljoin(result)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000548
549 def load(self, rawdata):
550 """Load cookies from a string (presumably HTTP_COOKIE) or
551 from a dictionary. Loading cookies from a dictionary 'd'
552 is equivalent to calling:
553 map(Cookie.__setitem__, d.keys(), d.values())
554 """
Georg Brandl76e155a2010-07-31 21:04:00 +0000555 if isinstance(rawdata, str):
556 self.__parse_string(rawdata)
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000557 else:
Benjamin Peterson8719ad52009-09-11 22:24:02 +0000558 # self.update() wouldn't call our custom __setitem__
Georg Brandl76e155a2010-07-31 21:04:00 +0000559 for key, value in rawdata.items():
560 self[key] = value
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000561 return
Fred Drakeff5364a2000-08-24 14:40:35 +0000562
Georg Brandl76e155a2010-07-31 21:04:00 +0000563 def __parse_string(self, str, patt=_CookiePattern):
Antoine Pitroub1e36072014-11-21 01:20:57 +0100564 i = 0 # Our starting point
565 n = len(str) # Length of string
566 parsed_items = [] # Parsed (type, key, value) triples
567 morsel_seen = False # A key=value pair was previously encountered
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000568
Antoine Pitroub1e36072014-11-21 01:20:57 +0100569 TYPE_ATTRIBUTE = 1
570 TYPE_KEYVALUE = 2
571
572 # We first parse the whole cookie string and reject it if it's
573 # syntactically invalid (this helps avoid some classes of injection
574 # attacks).
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000575 while 0 <= i < n:
576 # Start looking for a cookie
Antoine Pitrou7d0b8f92014-09-17 00:23:55 +0200577 match = patt.match(str, i)
Georg Brandl76e155a2010-07-31 21:04:00 +0000578 if not match:
579 # No more cookies
580 break
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000581
Georg Brandl76e155a2010-07-31 21:04:00 +0000582 key, value = match.group("key"), match.group("val")
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000583 i = match.end(0)
584
Georg Brandl76e155a2010-07-31 21:04:00 +0000585 if key[0] == "$":
Antoine Pitroub1e36072014-11-21 01:20:57 +0100586 if not morsel_seen:
587 # We ignore attributes which pertain to the cookie
588 # mechanism as a whole, such as "$Version".
589 # See RFC 2965. (Does anyone care?)
590 continue
591 parsed_items.append((TYPE_ATTRIBUTE, key[1:], value))
Georg Brandl76e155a2010-07-31 21:04:00 +0000592 elif key.lower() in Morsel._reserved:
Antoine Pitroub1e36072014-11-21 01:20:57 +0100593 if not morsel_seen:
594 # Invalid cookie string
595 return
596 if value is None:
597 if key.lower() in Morsel._flags:
598 parsed_items.append((TYPE_ATTRIBUTE, key, True))
R David Murraycd0f74b2013-08-25 11:09:02 -0400599 else:
Antoine Pitroub1e36072014-11-21 01:20:57 +0100600 # Invalid cookie string
601 return
602 else:
603 parsed_items.append((TYPE_ATTRIBUTE, key, _unquote(value)))
R David Murraycd0f74b2013-08-25 11:09:02 -0400604 elif value is not None:
Antoine Pitroub1e36072014-11-21 01:20:57 +0100605 parsed_items.append((TYPE_KEYVALUE, key, self.value_decode(value)))
606 morsel_seen = True
607 else:
608 # Invalid cookie string
609 return
610
611 # The cookie string is valid, apply it.
612 M = None # current morsel
613 for tp, key, value in parsed_items:
614 if tp == TYPE_ATTRIBUTE:
615 assert M is not None
616 M[key] = value
617 else:
618 assert tp == TYPE_KEYVALUE
619 rval, cval = value
Georg Brandl76e155a2010-07-31 21:04:00 +0000620 self.__set(key, rval, cval)
621 M = self[key]
Georg Brandl4eff9f72009-09-04 08:22:00 +0000622
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000623
624class SimpleCookie(BaseCookie):
Georg Brandl9cf32a12009-09-04 08:28:01 +0000625 """
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000626 SimpleCookie supports strings as cookie values. When setting
627 the value using the dictionary assignment notation, SimpleCookie
628 calls the builtin str() to convert the value to a string. Values
629 received from HTTP are kept as strings.
630 """
631 def value_decode(self, val):
Georg Brandl76e155a2010-07-31 21:04:00 +0000632 return _unquote(val), val
633
Andrew M. Kuchling52ea8722000-08-19 13:01:19 +0000634 def value_encode(self, val):
635 strval = str(val)
Georg Brandl76e155a2010-07-31 21:04:00 +0000636 return strval, _quote(strval)