skip to navigation
skip to content

berry 0.3.1

a minimal DSL for building a WSGI app

Berry is a minimal DSL for building a WSGI application.

Basically, you map some routes to functions. Berry takes this mapping and
generates a WSGI app, which you can serve however you like.


Here is an example using the wsgiref server, included in Python's stdlib
and ideal for development use.

import berry
from wsgiref.simple_server import make_server

def index(req):
return "Welcome to the home page."

def hello(req, name):
return "Hello, %s!" % name

# generate a WSGI app
wsgi_app =

# start a WSGI server
make_server('', 8000, wsgi_app).serve_forever()

How to use it

Decorate a function with berry.get(route) or to serve
GET/POST requests that match a route. Routes must be regular expressions.
Your function will be passed a Request object as the first argument.


def home(req):
return 'This is the home page.'

As above, you can map multiple routes to the same function.

Request objects

Useful attributes of Request objects are:

- env: the WSGI environ variable.
- params: parameters passed through both GET and POST.
- path: the path requested, minus the initial '/' and the query string
- query: the query string, if any
- fullpath: the full path requested, including the initial '/' and the
query string
- method: the method (GET or POST)

def login(req):
username = req.params.get('username')
password = req.params.get('password')
# ...

Note: if you have a field like 'a[b]' with value 'c', Berry will parse it
into a dictionary. For example:

<input name="person[name]" value="James"/>
<input name="person[age]" value="20"/>

will result in req.params being:

{'person': {'name': 'James', 'age': '20'}}.


<input name="person[friends][]" value="James"/>
<input name="person[friends][]" value="John"/>

will result in req.params being:

{'person': {'friends': ['James', 'John']}}.

Handling errors

Using the berry.error(code) decorator, you can make custom error pages.


def notfound(req):
return "%s was not found." % req.fullpath

Berry has Redirect, Forbidden, NotFound, and AppError classes, which
are exceptions that inherit berry.HTTPError. Just raise one of them:

if not user.is_logged_in():
raise berry.Forbidden()

To add an exception for a new HTTP status code you can do something like:

class Unauthorized(berry.HTTPError):
status = (401, 'Unauthorized')
content = "<h1>401 Unauthorized</h1>"

Application errors

If you set berry.debug = True, tracebacks will be outputted when there
is an exception in your code. Otherwise they will only be written to


To redirect, raise the berry.Redirect exception:

raise berry.Redirect(url)


To add HTTP headers, use the berry.header decorator:

@berry.header('Content-Type', 'text/plain')
def download_as_txt(req, id):
# ...

By default the Content-Type is set to 'text/html'.

How to install

If you have setuptools just do:

easy_install berry

Otherwise, you can get the latest release from:

Or get the latest development snapshot with git:

git clone git://


python install

If you try Berry, please write me at and let me know what
you think.

Latest changes

- Not logging requests anymore (this should be done by a middleware).

- Berry simply generates a WSGI app for you to serve yourself, instead of
serving it for you.

- The convenience function redirect() was removed.  
File Type Py Version Uploaded on Size
berry-0.3.1.tar.gz (md5) Source 2011-01-22 4KB