Using Python with FastGCI
Plain Old Python with FastCGI
of these instructions are similar to Django on FastCGI. First, obtain shell access and then:
cd ~/www mkdir project cd project wget h p://svn.saddi.com/py-lib/trunk/fcgi.py chmod 755 fcgi.py
FastCGI works based on a .fcgi file and it detects reloads based on timestamps. For this reason and that you probably don't want your Python project to be entirely wri en for FastCGI, you probably want to use .htaccess to route everything through a dispatch.fcgi
file as Django does. First, the < >project/.htaccess
:
RewriteEngine On RewriteBase /project RewriteCond %{REQUEST_URI} !^/project/dispatch.fcgi [NC] RewriteCond %{REQUEST_URI} !^/project/static [NC] RewriteCond %{REQUEST_URI} !^/project/more_static [NC] RewriteCond %{REQUEST_URI} !\.(css|png|jpg|gif)$ [NC] RewriteRule ^(.*)$ dispatch.fcgi/$1 [QSA,NC]
You can add more lines to ignore folders as we did in the example < >static and < >more_static
folder or include more file extensions to ignore. That way those files aren't routed through our dispatcher and will be efficiently served by Apache instead. Note that we are appending URLs to < >dispatch.fcgi, which we can parse later with the < >urlparse library and < >cgi.FieldStorage.
Now, create < >project/dispatch.fcgi with Unix line endings. You will receive puzzling FastCGI errors if you use any other line endings, especially if you are on Windows like me and don't like spending hours on end diagnosing puzzling FastCGI errors.
#!/usr/bin/env python # This must be in Unix line endings import sys from fcgi import WSGIServer import application import cgi class Request(object): pass def app(e, start_response): req = Request() # POST data req.form = cgi.FieldStorage(fp = e['wsgi.input'], environ = e) # GET data req.params = cgi.parse_qs(e['QUERY_STRING']) start_response('200 OK', [('Content-type', 'text/html')]) return [application.start(req)] WSGIServer(app).run()
The script, as above, can employ the environ variables and can call any one of your functions in your application stored in another file such as < >start
in < >application.py
. From that, you can then actually dispatch to the right module to handle the request based on where < >PATH_INFO
is or start your application logic.
Before accessing it and possibly forcing FastCGI to generate a broken cache (see below), type < >python dispatch.fcgi
in your shell. You should see some output although the environ variables aren't set properly from the shell. If you see Python errors, though, this is a quick way to partially debug your application without having to go through FastCGI just yet.
Now navigate to < >example.com/project/
or whereever and you should see output. Remember: < >example.com/project/jobs
maps to < >example.com/project/dispatch.fcgi/jobs
due to < >.htaccess
Tips
- If you want to update your application, simply run < >touch dispatch.fcgi in your shell.
- If your < >dispatch.fcgi is somehow broken, FastCGI may add an error stating that it couldn't get the file to run after three requests and has backed off the reload interval to some ungodly time. (You can see this in the error log from your control panel.) Unfortunately, FastCGI is then unable to detect changes to your file so you need to wait a while before you start fixing the problem. (I can't seem to find a workaround besides renaming < >dispatch.fcgi, but that gets tedious.)
- By the way, that < >fcgi.py will print to output very useful exceptions and tracebacks if your module throws an uncaught exception using the < >cgitb module. This is, however, undesirable for production web sites since it does expose lines of your code to your users. To remedy this, you can monkeypatch the function or just manually edit the < >Server.error method in < >fcgi.py. (Grep for "cgitb.")
- To redirect, < >start_response('302 Found', [('Location', url)]).