Creating Your First Pecan Application¶
Let’s create a small sample project with Pecan.
Note
This guide does not cover the installation of Pecan. If you need instructions for installing Pecan, refer to Installation.
Base Application Template¶
Pecan includes a basic template for starting a new project. From your shell, type:
$ pecan create test_project
This example uses test_project as your project name, but you can replace it with any valid Python package name you like.
Go ahead and change into your newly created project directory.:
$ cd test_project
You’ll want to deploy it in “development mode”, such that it’s
available on sys.path
, yet can still be edited directly from its
source distribution:
$ python setup.py develop
Your new project contain these files:
$ ls
├── MANIFEST.in
├── config.py
├── public
│ ├── css
│ │ └── style.css
│ └── images
├── setup.cfg
├── setup.py
└── test_project
├── __init__.py
├── app.py
├── controllers
│ ├── __init__.py
│ └── root.py
├── model
│ └── __init__.py
├── templates
│ ├── error.html
│ ├── index.html
│ └── layout.html
└── tests
├── __init__.py
├── config.py
├── test_functional.py
└── test_units.py
The number of files and directories may vary based on the version of Pecan, but the above structure should give you an idea of what to expect.
Let’s review the files created by the template.
- public
All your static files (like CSS, Javascript, and images) live here. Pecan comes with a simple file server that serves these static files as you develop.
Pecan application structure generally follows the MVC pattern. The
directories under test_project
encompass your models, controllers
and templates.
- test_project/controllers
The container directory for your controller files.
- test_project/templates
All your templates go in here.
- test_project/model
Container for your model files.
Finally, a directory to house unit and integration tests:
- test_project/tests
All of the tests for your application.
The test_project/app.py
file controls how the Pecan application will be
created. This file must contain a setup_app()
function which returns the
WSGI application object. Generally you will not need to modify the app.py
file provided by the base application template unless you need to customize
your app in a way that cannot be accomplished using config. See
Python-Based Configuration below.
To avoid unneeded dependencies and to remain as flexible as possible,
Pecan doesn’t impose any database or ORM (Object Relational
Mapper). If your project will interact with a database, you can add
code to model/__init__.py
to load database bindings from your
configuration file and define tables and ORM definitions.
Running the Application¶
The base project template creates the configuration file with the
basic settings you need to run your Pecan application in
config.py
. This file includes the host and port to run the server
on, the location where your controllers and templates are stored on
disk, and the name of the directory containing any static files.
If you just run pecan serve, passing config.py
as the
configuration file, it will bring up the development server and serve
the app:
$ pecan serve config.py
Starting server in PID 000.
serving on 0.0.0.0:8080, view at http://127.0.0.1:8080
The location for the configuration file and the argument itself are very flexible - you can pass an absolute or relative path to the file.
Python-Based Configuration¶
For ease of use, Pecan configuration files are pure Python–they’re even saved
as .py
files.
This is how your default (generated) configuration file should look:
# Server Specific Configurations
server = {
'port': '8080',
'host': '0.0.0.0'
}
# Pecan Application Configurations
app = {
'root': '${package}.controllers.root.RootController',
'modules': ['${package}'],
'static_root': '%(confdir)s/public',
'template_path': '%(confdir)s/${package}/templates',
'debug': True,
'errors': {
'404': '/error/404',
'__force_dict__': True
}
}
logging = {
'loggers': {
'root' : {'level': 'INFO', 'handlers': ['console']},
'${package}': {'level': 'DEBUG', 'handlers': ['console']}
},
'handlers': {
'console': {
'level': 'DEBUG',
'class': 'logging.StreamHandler',
'formatter': 'simple'
}
},
'formatters': {
'simple': {
'format': ('%(asctime)s %(levelname)-5.5s [%(name)s]'
'[%(threadName)s] %(message)s')
}
}
}
# Custom Configurations must be in Python dictionary format::
#
# foo = {'bar':'baz'}
#
# All configurations are accessible at::
# pecan.conf
You can also add your own configuration as Python dictionaries.
There’s a lot to cover here, so we’ll come back to configuration files in a later chapter (Configuring Pecan Applications).
The Application Root¶
The Root Controller is the entry point for your application. You
can think of it as being analogous to your application’s root URL path
(in our case, http://localhost:8080/
).
This is how it looks in the project template
(test_project.controllers.root.RootController
):
from pecan import expose
from webob.exc import status_map
class RootController(object):
@expose(generic=True, template='index.html')
def index(self):
return dict()
@index.when(method='POST')
def index_post(self, q):
redirect('https://pecan.readthedocs.io/en/latest/search.html?q=%s' % q)
@expose('error.html')
def error(self, status):
try:
status = int(status)
except ValueError:
status = 0
message = getattr(status_map.get(status), 'explanation', '')
return dict(status=status, message=message)
You can specify additional classes and methods if you need to do so, but for now, let’s examine the sample project, controller by controller:
@expose(generic=True, template='index.html')
def index(self):
return dict()
The index()
method is marked as publicly available via the
expose()
decorator (which in turn uses the
index.html
template) at the root of the application
(http://127.0.0.1:8080/), so any HTTP GET
that hits the root of your
application (/
) will be routed to this method.
Notice that the index()
method returns a Python dictionary. This dictionary
is used as a namespace to render the specified template (index.html
) into
HTML, and is the primary mechanism by which data is passed from controller to
template.
@index.when(method='POST')
def index_post(self, q):
redirect('https://pecan.readthedocs.io/en/latest/search.html?q=%s' % q)
The index_post()
method receives one HTTP POST
argument (q
). Because
the argument method
to @index.when()
has been set to 'POST'
, any
HTTP POST
to the application root (in the example project, a form
submission) will be routed to this method.
@expose('error.html')
def error(self, status):
try:
status = int(status)
except ValueError:
status = 0
message = getattr(status_map.get(status), 'explanation', '')
return dict(status=status, message=message)
Finally, we have the error()
method, which allows the application to display
custom pages for certain HTTP errors (404
, etc…).
Running the Tests For Your Application¶
Your application comes with a few example tests that you can run, replace, and add to. To run them:
$ python setup.py test -q
running test
running egg_info
writing requirements to sam.egg-info/requires.txt
writing sam.egg-info/PKG-INFO
writing top-level names to sam.egg-info/top_level.txt
writing dependency_links to sam.egg-info/dependency_links.txt
reading manifest file 'sam.egg-info/SOURCES.txt'
reading manifest template 'MANIFEST.in'
writing manifest file 'sam.egg-info/SOURCES.txt'
running build_ext
....
----------------------------------------------------------------------
Ran 4 tests in 0.009s
OK
The tests themselves can be found in the tests
module in your project.
Deploying to a Web Server¶
Ready to deploy your new Pecan app? Take a look at Deploying Pecan in Production.