blob: f93fa555c5ea5d00d0cec0b4fa7255d9d88bea67 [file] [log] [blame]
Barry Warsaw408b6d32002-07-30 23:27:12 +00001from test.test_support import verify, verbose
Jeremy Hyltond9827c42000-08-03 22:11:43 +00002import cgi
3import os
4import sys
Thomas Wouters00ee7ba2006-08-21 19:07:27 +00005import tempfile
6from StringIO import StringIO
Jeremy Hyltond9827c42000-08-03 22:11:43 +00007
8class HackedSysModule:
9 # The regression test will have real values in sys.argv, which
Fred Drake004d5e62000-10-23 17:22:08 +000010 # will completely confuse the test of the cgi module
Jeremy Hyltond9827c42000-08-03 22:11:43 +000011 argv = []
12 stdin = sys.stdin
13
14cgi.sys = HackedSysModule()
15
16try:
17 from cStringIO import StringIO
18except ImportError:
19 from StringIO import StringIO
20
21class ComparableException:
22 def __init__(self, err):
23 self.err = err
24
25 def __str__(self):
26 return str(self.err)
27
Guido van Rossum47b9ff62006-08-24 00:41:19 +000028 def __eq__(self, anExc):
Jeremy Hyltond9827c42000-08-03 22:11:43 +000029 if not isinstance(anExc, Exception):
Guido van Rossum47b9ff62006-08-24 00:41:19 +000030 return NotImplemented
31 return (self.err.__class__ == anExc.__class__ and
32 self.err.args == anExc.args)
Jeremy Hyltond9827c42000-08-03 22:11:43 +000033
34 def __getattr__(self, attr):
Guido van Rossum846d6db2001-01-17 15:08:37 +000035 return getattr(self.err, attr)
Jeremy Hyltond9827c42000-08-03 22:11:43 +000036
37def do_test(buf, method):
38 env = {}
39 if method == "GET":
40 fp = None
41 env['REQUEST_METHOD'] = 'GET'
42 env['QUERY_STRING'] = buf
43 elif method == "POST":
44 fp = StringIO(buf)
45 env['REQUEST_METHOD'] = 'POST'
46 env['CONTENT_TYPE'] = 'application/x-www-form-urlencoded'
47 env['CONTENT_LENGTH'] = str(len(buf))
48 else:
49 raise ValueError, "unknown method: %s" % method
50 try:
51 return cgi.parse(fp, env, strict_parsing=1)
52 except StandardError, err:
53 return ComparableException(err)
54
55# A list of test cases. Each test case is a a two-tuple that contains
56# a string with the query and a dictionary with the expected result.
Fred Drake004d5e62000-10-23 17:22:08 +000057
Neil Schemenauer66edb622004-07-19 15:38:11 +000058parse_qsl_test_cases = [
59 ("", []),
60 ("&", []),
61 ("&&", []),
62 ("=", [('', '')]),
63 ("=a", [('', 'a')]),
64 ("a", [('a', '')]),
65 ("a=", [('a', '')]),
66 ("a=", [('a', '')]),
67 ("&a=b", [('a', 'b')]),
68 ("a=a+b&b=b+c", [('a', 'a b'), ('b', 'b c')]),
69 ("a=1&a=2", [('a', '1'), ('a', '2')]),
70]
71
72parse_strict_test_cases = [
Jeremy Hyltond9827c42000-08-03 22:11:43 +000073 ("", ValueError("bad query field: ''")),
74 ("&", ValueError("bad query field: ''")),
75 ("&&", ValueError("bad query field: ''")),
Jeremy Hyltonafde7e22000-09-15 20:06:57 +000076 (";", ValueError("bad query field: ''")),
77 (";&;", ValueError("bad query field: ''")),
Jeremy Hyltond9827c42000-08-03 22:11:43 +000078 # Should the next few really be valid?
79 ("=", {}),
80 ("=&=", {}),
Jeremy Hyltonafde7e22000-09-15 20:06:57 +000081 ("=;=", {}),
Jeremy Hyltond9827c42000-08-03 22:11:43 +000082 # This rest seem to make sense
83 ("=a", {'': ['a']}),
84 ("&=a", ValueError("bad query field: ''")),
85 ("=a&", ValueError("bad query field: ''")),
86 ("=&a", ValueError("bad query field: 'a'")),
87 ("b=a", {'b': ['a']}),
88 ("b+=a", {'b ': ['a']}),
89 ("a=b=a", {'a': ['b=a']}),
90 ("a=+b=a", {'a': [' b=a']}),
91 ("&b=a", ValueError("bad query field: ''")),
92 ("b&=a", ValueError("bad query field: 'b'")),
93 ("a=a+b&b=b+c", {'a': ['a b'], 'b': ['b c']}),
94 ("a=a+b&a=b+a", {'a': ['a b', 'b a']}),
95 ("x=1&y=2.0&z=2-3.%2b0", {'x': ['1'], 'y': ['2.0'], 'z': ['2-3.+0']}),
Jeremy Hyltonafde7e22000-09-15 20:06:57 +000096 ("x=1;y=2.0&z=2-3.%2b0", {'x': ['1'], 'y': ['2.0'], 'z': ['2-3.+0']}),
97 ("x=1;y=2.0;z=2-3.%2b0", {'x': ['1'], 'y': ['2.0'], 'z': ['2-3.+0']}),
Jeremy Hyltond9827c42000-08-03 22:11:43 +000098 ("Hbc5161168c542333633315dee1182227:key_store_seqid=400006&cuyer=r&view=bustomer&order_id=0bb2e248638833d48cb7fed300000f1b&expire=964546263&lobale=en-US&kid=130003.300038&ss=env",
99 {'Hbc5161168c542333633315dee1182227:key_store_seqid': ['400006'],
100 'cuyer': ['r'],
101 'expire': ['964546263'],
102 'kid': ['130003.300038'],
103 'lobale': ['en-US'],
104 'order_id': ['0bb2e248638833d48cb7fed300000f1b'],
105 'ss': ['env'],
106 'view': ['bustomer'],
107 }),
Fred Drake004d5e62000-10-23 17:22:08 +0000108
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000109 ("group_id=5470&set=custom&_assigned_to=31392&_status=1&_category=100&SUBMIT=Browse",
110 {'SUBMIT': ['Browse'],
111 '_assigned_to': ['31392'],
112 '_category': ['100'],
113 '_status': ['1'],
114 'group_id': ['5470'],
115 'set': ['custom'],
116 })
117 ]
118
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000119def norm(seq):
120 if isinstance(seq, list):
121 seq.sort(key=repr)
122 return seq
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000123
124def first_elts(list):
125 return map(lambda x:x[0], list)
126
127def first_second_elts(list):
128 return map(lambda p:(p[0], p[1][0]), list)
129
130def main():
Neil Schemenauer66edb622004-07-19 15:38:11 +0000131 for orig, expect in parse_qsl_test_cases:
132 result = cgi.parse_qsl(orig, keep_blank_values=True)
133 print repr(orig), '=>', result
134 verify(result == expect, "Error parsing %s" % repr(orig))
135
136 for orig, expect in parse_strict_test_cases:
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000137 # Test basic parsing
138 print repr(orig)
139 d = do_test(orig, "GET")
Marc-André Lemburg36619082001-01-17 19:11:13 +0000140 verify(d == expect, "Error parsing %s" % repr(orig))
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000141 d = do_test(orig, "POST")
Marc-André Lemburg36619082001-01-17 19:11:13 +0000142 verify(d == expect, "Error parsing %s" % repr(orig))
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000143
Moshe Zadkaa1a4b592000-08-25 21:47:56 +0000144 env = {'QUERY_STRING': orig}
145 fcd = cgi.FormContentDict(env)
146 sd = cgi.SvFormContentDict(env)
147 fs = cgi.FieldStorage(environ=env)
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000148 if type(expect) == type({}):
149 # test dict interface
Marc-André Lemburg36619082001-01-17 19:11:13 +0000150 verify(len(expect) == len(fcd))
151 verify(norm(expect.keys()) == norm(fcd.keys()))
152 verify(norm(expect.values()) == norm(fcd.values()))
153 verify(norm(expect.items()) == norm(fcd.items()))
154 verify(fcd.get("nonexistent field", "default") == "default")
155 verify(len(sd) == len(fs))
156 verify(norm(sd.keys()) == norm(fs.keys()))
157 verify(fs.getvalue("nonexistent field", "default") == "default")
Moshe Zadkaa1a4b592000-08-25 21:47:56 +0000158 # test individual fields
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000159 for key in expect.keys():
160 expect_val = expect[key]
Guido van Rossume2b70bc2006-08-18 22:13:04 +0000161 verify(key in fcd)
Marc-André Lemburg36619082001-01-17 19:11:13 +0000162 verify(norm(fcd[key]) == norm(expect[key]))
163 verify(fcd.get(key, "default") == fcd[key])
Guido van Rossume2b70bc2006-08-18 22:13:04 +0000164 verify(key in fs)
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000165 if len(expect_val) > 1:
166 single_value = 0
167 else:
168 single_value = 1
169 try:
170 val = sd[key]
171 except IndexError:
Marc-André Lemburg36619082001-01-17 19:11:13 +0000172 verify(not single_value)
173 verify(fs.getvalue(key) == expect_val)
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000174 else:
Marc-André Lemburg36619082001-01-17 19:11:13 +0000175 verify(single_value)
176 verify(val == expect_val[0])
177 verify(fs.getvalue(key) == expect_val[0])
178 verify(norm(sd.getlist(key)) == norm(expect_val))
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000179 if single_value:
Marc-André Lemburg36619082001-01-17 19:11:13 +0000180 verify(norm(sd.values()) == \
181 first_elts(norm(expect.values())))
182 verify(norm(sd.items()) == \
183 first_second_elts(norm(expect.items())))
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000184
185 # Test the weird FormContentDict classes
186 env = {'QUERY_STRING': "x=1&y=2.0&z=2-3.%2b0&1=1abc"}
187 expect = {'x': 1, 'y': 2.0, 'z': '2-3.+0', '1': '1abc'}
188 d = cgi.InterpFormContentDict(env)
189 for k, v in expect.items():
Marc-André Lemburg36619082001-01-17 19:11:13 +0000190 verify(d[k] == v)
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000191 for k, v in d.items():
Marc-André Lemburg36619082001-01-17 19:11:13 +0000192 verify(expect[k] == v)
193 verify(norm(expect.values()) == norm(d.values()))
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000194
195 print "Testing log"
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000196 cgi.log("Testing")
197 cgi.logfp = sys.stdout
198 cgi.initlog("%s", "Testing initlog 1")
199 cgi.log("%s", "Testing log 2")
200 if os.path.exists("/dev/null"):
201 cgi.logfp = None
202 cgi.logfile = "/dev/null"
203 cgi.initlog("%s", "Testing log 3")
204 cgi.log("Testing log 4")
205
Thomas Wouters00ee7ba2006-08-21 19:07:27 +0000206 print "Test FieldStorage methods that use readline"
207 # FieldStorage uses readline, which has the capacity to read all
208 # contents of the input file into memory; we use readline's size argument
209 # to prevent that for files that do not contain any newlines in
210 # non-GET/HEAD requests
211 class TestReadlineFile:
212 def __init__(self, file):
213 self.file = file
214 self.numcalls = 0
215
216 def readline(self, size=None):
217 self.numcalls += 1
218 if size:
219 return self.file.readline(size)
220 else:
221 return self.file.readline()
222
223 def __getattr__(self, name):
224 file = self.__dict__['file']
225 a = getattr(file, name)
226 if not isinstance(a, int):
227 setattr(self, name, a)
228 return a
229
230 f = TestReadlineFile(tempfile.TemporaryFile())
231 f.write('x' * 256 * 1024)
232 f.seek(0)
233 env = {'REQUEST_METHOD':'PUT'}
234 fs = cgi.FieldStorage(fp=f, environ=env)
235 # if we're not chunking properly, readline is only called twice
236 # (by read_binary); if we are chunking properly, it will be called 5 times
237 # as long as the chunksize is 1 << 16.
238 verify(f.numcalls > 2)
239
240 print "Test basic FieldStorage multipart parsing"
241 env = {'REQUEST_METHOD':'POST', 'CONTENT_TYPE':'multipart/form-data; boundary=---------------------------721837373350705526688164684', 'CONTENT_LENGTH':'558'}
242 postdata = """-----------------------------721837373350705526688164684
243Content-Disposition: form-data; name="id"
244
2451234
246-----------------------------721837373350705526688164684
247Content-Disposition: form-data; name="title"
248
249
250-----------------------------721837373350705526688164684
251Content-Disposition: form-data; name="file"; filename="test.txt"
252Content-Type: text/plain
253
254Testing 123.
255
256-----------------------------721837373350705526688164684
257Content-Disposition: form-data; name="submit"
258
259 Add\x20
260-----------------------------721837373350705526688164684--
261"""
262 fs = cgi.FieldStorage(fp=StringIO(postdata), environ=env)
263 verify(len(fs.list) == 4)
264 expect = [{'name':'id', 'filename':None, 'value':'1234'},
265 {'name':'title', 'filename':None, 'value':''},
266 {'name':'file', 'filename':'test.txt','value':'Testing 123.\n'},
267 {'name':'submit', 'filename':None, 'value':' Add '}]
268 for x in range(len(fs.list)):
269 for k, exp in expect[x].items():
270 got = getattr(fs.list[x], k)
271 verify(got == exp)
272
Jeremy Hyltond9827c42000-08-03 22:11:43 +0000273main()