Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 1 | """CGI-savvy HTTP Server. |
| 2 | |
| 3 | This module builds on SimpleHTTPServer by implementing GET and POST |
| 4 | requests to cgi-bin scripts. |
| 5 | |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 6 | If the os.fork() function is not present (e.g. on Windows), |
| 7 | os.popen2() is used as a fallback, with slightly altered semantics; if |
| 8 | that function is not present either (e.g. on Macintosh), only Python |
| 9 | scripts are supported, and they are executed by the current process. |
| 10 | |
| 11 | In all cases, the implementation is intentionally naive -- all |
| 12 | requests are executed sychronously. |
| 13 | |
| 14 | SECURITY WARNING: DON'T USE THIS CODE UNLESS YOU ARE INSIDE A FIREWALL |
| 15 | -- it may execute arbitrary Python code or external programs. |
Fred Drake | 40e84db | 1999-10-16 02:07:50 +0000 | [diff] [blame] | 16 | |
Jeremy Hylton | 6414cd8 | 2004-12-22 14:19:09 +0000 | [diff] [blame] | 17 | Note that status code 200 is sent prior to execution of a CGI script, so |
| 18 | scripts cannot send other status codes such as 302 (redirect). |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 19 | """ |
| 20 | |
| 21 | |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 22 | __version__ = "0.4" |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 23 | |
Skip Montanaro | e99d5ea | 2001-01-20 19:54:20 +0000 | [diff] [blame] | 24 | __all__ = ["CGIHTTPRequestHandler"] |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 25 | |
| 26 | import os |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 27 | import sys |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 28 | import urllib |
| 29 | import BaseHTTPServer |
| 30 | import SimpleHTTPServer |
Steve Holden | 8a978f7 | 2003-01-08 18:53:18 +0000 | [diff] [blame] | 31 | import select |
Senthil Kumaran | a9bd0cc | 2010-10-03 18:16:52 +0000 | [diff] [blame] | 32 | import copy |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 33 | |
| 34 | |
| 35 | class CGIHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler): |
| 36 | |
| 37 | """Complete HTTP server with GET, HEAD and POST commands. |
| 38 | |
| 39 | GET and HEAD also support running CGI scripts. |
| 40 | |
| 41 | The POST command is *only* implemented for CGI scripts. |
| 42 | |
| 43 | """ |
| 44 | |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 45 | # Determine platform specifics |
| 46 | have_fork = hasattr(os, 'fork') |
| 47 | have_popen2 = hasattr(os, 'popen2') |
Guido van Rossum | 8cb6540 | 2002-02-01 16:27:59 +0000 | [diff] [blame] | 48 | have_popen3 = hasattr(os, 'popen3') |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 49 | |
Guido van Rossum | 6aefd91 | 2000-09-01 03:27:34 +0000 | [diff] [blame] | 50 | # Make rfile unbuffered -- we need to read one line and then pass |
| 51 | # the rest to a subprocess, so we can't use buffered input. |
| 52 | rbufsize = 0 |
| 53 | |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 54 | def do_POST(self): |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 55 | """Serve a POST request. |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 56 | |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 57 | This is only implemented for CGI scripts. |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 58 | |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 59 | """ |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 60 | |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 61 | if self.is_cgi(): |
| 62 | self.run_cgi() |
| 63 | else: |
| 64 | self.send_error(501, "Can only POST to CGI scripts") |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 65 | |
| 66 | def send_head(self): |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 67 | """Version of send_head that support CGI scripts""" |
| 68 | if self.is_cgi(): |
| 69 | return self.run_cgi() |
| 70 | else: |
| 71 | return SimpleHTTPServer.SimpleHTTPRequestHandler.send_head(self) |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 72 | |
| 73 | def is_cgi(self): |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 74 | """Test whether self.path corresponds to a CGI script. |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 75 | |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 76 | Returns True and updates the cgi_info attribute to the tuple |
| 77 | (dir, rest) if self.path requires running a CGI script. |
| 78 | Returns False otherwise. |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 79 | |
Gregory P. Smith | 4bd7664 | 2009-05-03 20:27:25 +0000 | [diff] [blame] | 80 | If any exception is raised, the caller should assume that |
| 81 | self.path was rejected as invalid and act accordingly. |
| 82 | |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 83 | The default implementation tests whether the normalized url |
| 84 | path begins with one of the strings in self.cgi_directories |
| 85 | (and the next character is a '/' or the end of the string). |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 86 | """ |
Benjamin Peterson | 8d24d77 | 2014-06-14 18:36:29 -0700 | [diff] [blame] | 87 | collapsed_path = _url_collapse_path(urllib.unquote(self.path)) |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 88 | dir_sep = collapsed_path.find('/', 1) |
| 89 | head, tail = collapsed_path[:dir_sep], collapsed_path[dir_sep+1:] |
Senthil Kumaran | fb2e874 | 2012-04-11 03:07:57 +0800 | [diff] [blame] | 90 | if head in self.cgi_directories: |
| 91 | self.cgi_info = head, tail |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 92 | return True |
Tim Peters | bc0e910 | 2002-04-04 22:55:58 +0000 | [diff] [blame] | 93 | return False |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 94 | |
| 95 | cgi_directories = ['/cgi-bin', '/htbin'] |
| 96 | |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 97 | def is_executable(self, path): |
| 98 | """Test whether argument path is an executable file.""" |
| 99 | return executable(path) |
| 100 | |
| 101 | def is_python(self, path): |
| 102 | """Test whether argument path is a Python script.""" |
| 103 | head, tail = os.path.splitext(path) |
| 104 | return tail.lower() in (".py", ".pyw") |
| 105 | |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 106 | def run_cgi(self): |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 107 | """Execute a CGI script.""" |
| 108 | dir, rest = self.cgi_info |
Ned Deily | c893762 | 2014-07-12 22:01:15 -0700 | [diff] [blame] | 109 | path = dir + '/' + rest |
| 110 | i = path.find('/', len(dir)+1) |
Andrew M. Kuchling | b29069d | 2006-12-22 13:25:02 +0000 | [diff] [blame] | 111 | while i >= 0: |
Ned Deily | c893762 | 2014-07-12 22:01:15 -0700 | [diff] [blame] | 112 | nextdir = path[:i] |
| 113 | nextrest = path[i+1:] |
Andrew M. Kuchling | b29069d | 2006-12-22 13:25:02 +0000 | [diff] [blame] | 114 | |
| 115 | scriptdir = self.translate_path(nextdir) |
| 116 | if os.path.isdir(scriptdir): |
| 117 | dir, rest = nextdir, nextrest |
Ned Deily | c893762 | 2014-07-12 22:01:15 -0700 | [diff] [blame] | 118 | i = path.find('/', len(dir)+1) |
Andrew M. Kuchling | b29069d | 2006-12-22 13:25:02 +0000 | [diff] [blame] | 119 | else: |
| 120 | break |
| 121 | |
| 122 | # find an explicit query string, if present. |
Eric S. Raymond | 6b71e74 | 2001-02-09 08:56:30 +0000 | [diff] [blame] | 123 | i = rest.rfind('?') |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 124 | if i >= 0: |
| 125 | rest, query = rest[:i], rest[i+1:] |
| 126 | else: |
| 127 | query = '' |
Andrew M. Kuchling | b29069d | 2006-12-22 13:25:02 +0000 | [diff] [blame] | 128 | |
| 129 | # dissect the part after the directory name into a script name & |
| 130 | # a possible additional path, to be stored in PATH_INFO. |
Eric S. Raymond | 6b71e74 | 2001-02-09 08:56:30 +0000 | [diff] [blame] | 131 | i = rest.find('/') |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 132 | if i >= 0: |
| 133 | script, rest = rest[:i], rest[i:] |
| 134 | else: |
| 135 | script, rest = rest, '' |
Andrew M. Kuchling | b29069d | 2006-12-22 13:25:02 +0000 | [diff] [blame] | 136 | |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 137 | scriptname = dir + '/' + script |
| 138 | scriptfile = self.translate_path(scriptname) |
| 139 | if not os.path.exists(scriptfile): |
Walter Dörwald | 70a6b49 | 2004-02-12 17:35:32 +0000 | [diff] [blame] | 140 | self.send_error(404, "No such CGI script (%r)" % scriptname) |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 141 | return |
| 142 | if not os.path.isfile(scriptfile): |
Tim Peters | 27f4961 | 2004-03-20 21:51:12 +0000 | [diff] [blame] | 143 | self.send_error(403, "CGI script is not a plain file (%r)" % |
Walter Dörwald | 70a6b49 | 2004-02-12 17:35:32 +0000 | [diff] [blame] | 144 | scriptname) |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 145 | return |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 146 | ispy = self.is_python(scriptname) |
| 147 | if not ispy: |
Guido van Rossum | 8cb6540 | 2002-02-01 16:27:59 +0000 | [diff] [blame] | 148 | if not (self.have_fork or self.have_popen2 or self.have_popen3): |
Walter Dörwald | 70a6b49 | 2004-02-12 17:35:32 +0000 | [diff] [blame] | 149 | self.send_error(403, "CGI script is not a Python script (%r)" % |
| 150 | scriptname) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 151 | return |
| 152 | if not self.is_executable(scriptfile): |
Walter Dörwald | 70a6b49 | 2004-02-12 17:35:32 +0000 | [diff] [blame] | 153 | self.send_error(403, "CGI script is not executable (%r)" % |
| 154 | scriptname) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 155 | return |
| 156 | |
| 157 | # Reference: http://hoohoo.ncsa.uiuc.edu/cgi/env.html |
| 158 | # XXX Much of the following could be prepared ahead of time! |
Senthil Kumaran | a9bd0cc | 2010-10-03 18:16:52 +0000 | [diff] [blame] | 159 | env = copy.deepcopy(os.environ) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 160 | env['SERVER_SOFTWARE'] = self.version_string() |
| 161 | env['SERVER_NAME'] = self.server.server_name |
| 162 | env['GATEWAY_INTERFACE'] = 'CGI/1.1' |
| 163 | env['SERVER_PROTOCOL'] = self.protocol_version |
| 164 | env['SERVER_PORT'] = str(self.server.server_port) |
| 165 | env['REQUEST_METHOD'] = self.command |
| 166 | uqrest = urllib.unquote(rest) |
| 167 | env['PATH_INFO'] = uqrest |
| 168 | env['PATH_TRANSLATED'] = self.translate_path(uqrest) |
| 169 | env['SCRIPT_NAME'] = scriptname |
| 170 | if query: |
| 171 | env['QUERY_STRING'] = query |
| 172 | host = self.address_string() |
| 173 | if host != self.client_address[0]: |
| 174 | env['REMOTE_HOST'] = host |
| 175 | env['REMOTE_ADDR'] = self.client_address[0] |
Martin v. Löwis | a28b3e6 | 2004-08-29 16:53:26 +0000 | [diff] [blame] | 176 | authorization = self.headers.getheader("authorization") |
| 177 | if authorization: |
| 178 | authorization = authorization.split() |
| 179 | if len(authorization) == 2: |
| 180 | import base64, binascii |
| 181 | env['AUTH_TYPE'] = authorization[0] |
| 182 | if authorization[0].lower() == "basic": |
| 183 | try: |
| 184 | authorization = base64.decodestring(authorization[1]) |
| 185 | except binascii.Error: |
| 186 | pass |
| 187 | else: |
| 188 | authorization = authorization.split(':') |
| 189 | if len(authorization) == 2: |
| 190 | env['REMOTE_USER'] = authorization[0] |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 191 | # XXX REMOTE_IDENT |
| 192 | if self.headers.typeheader is None: |
| 193 | env['CONTENT_TYPE'] = self.headers.type |
| 194 | else: |
| 195 | env['CONTENT_TYPE'] = self.headers.typeheader |
| 196 | length = self.headers.getheader('content-length') |
| 197 | if length: |
| 198 | env['CONTENT_LENGTH'] = length |
Collin Winter | 83b2bf6 | 2007-03-09 03:15:56 +0000 | [diff] [blame] | 199 | referer = self.headers.getheader('referer') |
| 200 | if referer: |
| 201 | env['HTTP_REFERER'] = referer |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 202 | accept = [] |
| 203 | for line in self.headers.getallmatchingheaders('accept'): |
Eric S. Raymond | 7e642e8 | 2001-02-09 12:10:26 +0000 | [diff] [blame] | 204 | if line[:1] in "\t\n\r ": |
Eric S. Raymond | 6b71e74 | 2001-02-09 08:56:30 +0000 | [diff] [blame] | 205 | accept.append(line.strip()) |
Guido van Rossum | 01fc65d | 1998-05-13 20:13:24 +0000 | [diff] [blame] | 206 | else: |
Eric S. Raymond | 6b71e74 | 2001-02-09 08:56:30 +0000 | [diff] [blame] | 207 | accept = accept + line[7:].split(',') |
| 208 | env['HTTP_ACCEPT'] = ','.join(accept) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 209 | ua = self.headers.getheader('user-agent') |
| 210 | if ua: |
| 211 | env['HTTP_USER_AGENT'] = ua |
| 212 | co = filter(None, self.headers.getheaders('cookie')) |
| 213 | if co: |
Eric S. Raymond | 6b71e74 | 2001-02-09 08:56:30 +0000 | [diff] [blame] | 214 | env['HTTP_COOKIE'] = ', '.join(co) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 215 | # XXX Other HTTP_* headers |
Guido van Rossum | 70ec0b4 | 2004-03-20 22:18:03 +0000 | [diff] [blame] | 216 | # Since we're setting the env in the parent, provide empty |
| 217 | # values to override previously set values |
| 218 | for k in ('QUERY_STRING', 'REMOTE_HOST', 'CONTENT_LENGTH', |
Collin Winter | 83b2bf6 | 2007-03-09 03:15:56 +0000 | [diff] [blame] | 219 | 'HTTP_USER_AGENT', 'HTTP_COOKIE', 'HTTP_REFERER'): |
Guido van Rossum | 70ec0b4 | 2004-03-20 22:18:03 +0000 | [diff] [blame] | 220 | env.setdefault(k, "") |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 221 | |
| 222 | self.send_response(200, "Script output follows") |
| 223 | |
Eric S. Raymond | 6b71e74 | 2001-02-09 08:56:30 +0000 | [diff] [blame] | 224 | decoded_query = query.replace('+', ' ') |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 225 | |
| 226 | if self.have_fork: |
| 227 | # Unix -- fork as we should |
| 228 | args = [script] |
| 229 | if '=' not in decoded_query: |
| 230 | args.append(decoded_query) |
| 231 | nobody = nobody_uid() |
| 232 | self.wfile.flush() # Always flush before forking |
| 233 | pid = os.fork() |
| 234 | if pid != 0: |
| 235 | # Parent |
| 236 | pid, sts = os.waitpid(pid, 0) |
Steve Holden | 8a978f7 | 2003-01-08 18:53:18 +0000 | [diff] [blame] | 237 | # throw away additional data [see bug #427345] |
| 238 | while select.select([self.rfile], [], [], 0)[0]: |
Raymond Hettinger | e2f1837 | 2003-06-29 05:06:56 +0000 | [diff] [blame] | 239 | if not self.rfile.read(1): |
| 240 | break |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 241 | if sts: |
| 242 | self.log_error("CGI script exit status %#x", sts) |
| 243 | return |
| 244 | # Child |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 245 | try: |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 246 | try: |
| 247 | os.setuid(nobody) |
| 248 | except os.error: |
| 249 | pass |
| 250 | os.dup2(self.rfile.fileno(), 0) |
| 251 | os.dup2(self.wfile.fileno(), 1) |
Senthil Kumaran | a9bd0cc | 2010-10-03 18:16:52 +0000 | [diff] [blame] | 252 | os.execve(scriptfile, args, env) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 253 | except: |
| 254 | self.server.handle_error(self.request, self.client_address) |
| 255 | os._exit(127) |
| 256 | |
Senthil Kumaran | 3a145a1 | 2009-11-11 01:34:44 +0000 | [diff] [blame] | 257 | else: |
| 258 | # Non Unix - use subprocess |
| 259 | import subprocess |
| 260 | cmdline = [scriptfile] |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 261 | if self.is_python(scriptfile): |
| 262 | interp = sys.executable |
| 263 | if interp.lower().endswith("w.exe"): |
Guido van Rossum | 0afde13 | 2001-10-26 03:38:46 +0000 | [diff] [blame] | 264 | # On Windows, use python.exe, not pythonw.exe |
| 265 | interp = interp[:-5] + interp[-4:] |
Senthil Kumaran | 3a145a1 | 2009-11-11 01:34:44 +0000 | [diff] [blame] | 266 | cmdline = [interp, '-u'] + cmdline |
| 267 | if '=' not in query: |
| 268 | cmdline.append(query) |
| 269 | |
| 270 | self.log_message("command: %s", subprocess.list2cmdline(cmdline)) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 271 | try: |
| 272 | nbytes = int(length) |
Guido van Rossum | b390315 | 2002-10-17 16:21:35 +0000 | [diff] [blame] | 273 | except (TypeError, ValueError): |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 274 | nbytes = 0 |
Senthil Kumaran | 5dff354 | 2009-11-11 17:22:35 +0000 | [diff] [blame] | 275 | p = subprocess.Popen(cmdline, |
| 276 | stdin = subprocess.PIPE, |
| 277 | stdout = subprocess.PIPE, |
Senthil Kumaran | a9bd0cc | 2010-10-03 18:16:52 +0000 | [diff] [blame] | 278 | stderr = subprocess.PIPE, |
| 279 | env = env |
Senthil Kumaran | 5dff354 | 2009-11-11 17:22:35 +0000 | [diff] [blame] | 280 | ) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 281 | if self.command.lower() == "post" and nbytes > 0: |
| 282 | data = self.rfile.read(nbytes) |
Senthil Kumaran | 3a145a1 | 2009-11-11 01:34:44 +0000 | [diff] [blame] | 283 | else: |
| 284 | data = None |
Steve Holden | 8a978f7 | 2003-01-08 18:53:18 +0000 | [diff] [blame] | 285 | # throw away additional data [see bug #427345] |
| 286 | while select.select([self.rfile._sock], [], [], 0)[0]: |
Raymond Hettinger | e2f1837 | 2003-06-29 05:06:56 +0000 | [diff] [blame] | 287 | if not self.rfile._sock.recv(1): |
| 288 | break |
Senthil Kumaran | 3a145a1 | 2009-11-11 01:34:44 +0000 | [diff] [blame] | 289 | stdout, stderr = p.communicate(data) |
| 290 | self.wfile.write(stdout) |
| 291 | if stderr: |
| 292 | self.log_error('%s', stderr) |
Brian Curtin | 3606f95 | 2010-11-05 15:12:47 +0000 | [diff] [blame] | 293 | p.stderr.close() |
| 294 | p.stdout.close() |
Senthil Kumaran | 3a145a1 | 2009-11-11 01:34:44 +0000 | [diff] [blame] | 295 | status = p.returncode |
| 296 | if status: |
| 297 | self.log_error("CGI script exit status %#x", status) |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 298 | else: |
Guido van Rossum | bcbdc95 | 2001-10-17 06:45:56 +0000 | [diff] [blame] | 299 | self.log_message("CGI script exited OK") |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 300 | |
| 301 | |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 302 | def _url_collapse_path(path): |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 303 | """ |
| 304 | Given a URL path, remove extra '/'s and '.' path elements and collapse |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 305 | any '..' references and returns a colllapsed path. |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 306 | |
| 307 | Implements something akin to RFC-2396 5.2 step 6 to parse relative paths. |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 308 | The utility of this function is limited to is_cgi method and helps |
| 309 | preventing some security attacks. |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 310 | |
| 311 | Returns: A tuple of (head, tail) where tail is everything after the final / |
| 312 | and head is everything before it. Head will always start with a '/' and, |
| 313 | if it contains anything else, never have a trailing '/'. |
| 314 | |
| 315 | Raises: IndexError if too many '..' occur within the path. |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 316 | |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 317 | """ |
| 318 | # Similar to os.path.split(os.path.normpath(path)) but specific to URL |
| 319 | # path semantics rather than local operating system semantics. |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 320 | path_parts = path.split('/') |
| 321 | head_parts = [] |
| 322 | for part in path_parts[:-1]: |
| 323 | if part == '..': |
| 324 | head_parts.pop() # IndexError if more '..' than prior parts |
| 325 | elif part and part != '.': |
| 326 | head_parts.append( part ) |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 327 | if path_parts: |
Senthil Kumaran | fb2e874 | 2012-04-11 03:07:57 +0800 | [diff] [blame] | 328 | tail_part = path_parts.pop() |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 329 | if tail_part: |
| 330 | if tail_part == '..': |
| 331 | head_parts.pop() |
| 332 | tail_part = '' |
| 333 | elif tail_part == '.': |
| 334 | tail_part = '' |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 335 | else: |
| 336 | tail_part = '' |
Senthil Kumaran | 5f7e734 | 2012-04-12 02:23:23 +0800 | [diff] [blame] | 337 | |
| 338 | splitpath = ('/' + '/'.join(head_parts), tail_part) |
| 339 | collapsed_path = "/".join(splitpath) |
| 340 | |
| 341 | return collapsed_path |
Gregory P. Smith | 923ba36 | 2009-04-06 06:33:26 +0000 | [diff] [blame] | 342 | |
| 343 | |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 344 | nobody = None |
| 345 | |
| 346 | def nobody_uid(): |
| 347 | """Internal routine to get nobody's uid""" |
| 348 | global nobody |
| 349 | if nobody: |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 350 | return nobody |
Guido van Rossum | e7d6b0a | 2000-09-19 04:01:01 +0000 | [diff] [blame] | 351 | try: |
| 352 | import pwd |
| 353 | except ImportError: |
| 354 | return -1 |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 355 | try: |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 356 | nobody = pwd.getpwnam('nobody')[2] |
Guido van Rossum | 630b811 | 1999-04-28 12:21:47 +0000 | [diff] [blame] | 357 | except KeyError: |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 358 | nobody = 1 + max(map(lambda x: x[2], pwd.getpwall())) |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 359 | return nobody |
| 360 | |
| 361 | |
| 362 | def executable(path): |
| 363 | """Test for executable file.""" |
| 364 | try: |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 365 | st = os.stat(path) |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 366 | except os.error: |
Guido van Rossum | 8ca162f | 2002-04-07 06:36:23 +0000 | [diff] [blame] | 367 | return False |
Raymond Hettinger | 32200ae | 2002-06-01 19:51:15 +0000 | [diff] [blame] | 368 | return st.st_mode & 0111 != 0 |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 369 | |
| 370 | |
| 371 | def test(HandlerClass = CGIHTTPRequestHandler, |
Guido van Rossum | 45e2fbc | 1998-03-26 21:13:24 +0000 | [diff] [blame] | 372 | ServerClass = BaseHTTPServer.HTTPServer): |
Guido van Rossum | e7e578f | 1995-08-04 04:00:20 +0000 | [diff] [blame] | 373 | SimpleHTTPServer.test(HandlerClass, ServerClass) |
| 374 | |
| 375 | |
| 376 | if __name__ == '__main__': |
| 377 | test() |