Loading...

Knowledge Base

Use Python with FastGCI

The first part of these instructions are similar to Django on FastCGI. First, log into SSH (shell) and then:

cd ~/www
mkdir project
cd project
wget http://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 written for FastCGI, you probably want to use .htaccess to route everything through adispatch.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)]).

Did you find this article helpful?

 
* Your feedback is too short

Loading...