2012-10-31 03:53:56 +08:00
|
|
|
|
=============================================
|
|
|
|
|
Advanced tutorial: How to write reusable apps
|
|
|
|
|
=============================================
|
|
|
|
|
|
2013-03-04 19:05:11 +08:00
|
|
|
|
This advanced tutorial begins where :doc:`Tutorial 6 </intro/tutorial06>`
|
|
|
|
|
left off. We'll be turning our Web-poll into a standalone Python package
|
|
|
|
|
you can reuse in new projects and share with other people.
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
2013-03-04 19:05:11 +08:00
|
|
|
|
If you haven't recently completed Tutorials 1–6, we encourage you to review
|
2012-10-31 03:53:56 +08:00
|
|
|
|
these so that your example project matches the one described below.
|
|
|
|
|
|
|
|
|
|
Reusability matters
|
|
|
|
|
===================
|
|
|
|
|
|
|
|
|
|
It's a lot of work to design, build, test and maintain a web application. Many
|
|
|
|
|
Python and Django projects share common problems. Wouldn't it be great if we
|
|
|
|
|
could save some of this repeated work?
|
|
|
|
|
|
|
|
|
|
Reusability is the way of life in Python. `The Python Package Index (PyPI)
|
|
|
|
|
<http://guide.python-distribute.org/contributing.html#pypi-info>`_ has a vast
|
|
|
|
|
range of packages you can use in your own Python programs. Check out `Django
|
|
|
|
|
Packages <http://www.djangopackages.com>`_ for existing reusable apps you could
|
|
|
|
|
incorporate in your project. Django itself is also just a Python package. This
|
|
|
|
|
means that you can take existing Python packages or Django apps and compose
|
|
|
|
|
them into your own web project. You only need to write the parts that make
|
|
|
|
|
your project unique.
|
|
|
|
|
|
|
|
|
|
Let's say you were starting a new project that needed a polls app like the one
|
|
|
|
|
we've been working on. How do you make this app reusable? Luckily, you're well
|
|
|
|
|
on the way already. In :doc:`Tutorial 3 </intro/tutorial03>`, we saw how we
|
|
|
|
|
could decouple polls from the project-level URLconf using an ``include``.
|
|
|
|
|
In this tutorial, we'll take further steps to make the app easy to use in new
|
|
|
|
|
projects and ready to publish for others to install and use.
|
|
|
|
|
|
|
|
|
|
.. admonition:: Package? App?
|
|
|
|
|
|
|
|
|
|
A Python `package <http://docs.python.org/tutorial/modules.html#packages>`_
|
|
|
|
|
provides a way of grouping related Python code for easy reuse. A package
|
|
|
|
|
contains one or more files of Python code (also known as "modules").
|
|
|
|
|
|
|
|
|
|
A package can be imported with ``import foo.bar`` or ``from foo import
|
|
|
|
|
bar``. For a directory (like ``polls``) to form a package, it must contain
|
|
|
|
|
a special file ``__init__.py``, even if this file is empty.
|
|
|
|
|
|
|
|
|
|
A Django *app* is just a Python package that is specifically intended for
|
|
|
|
|
use in a Django project. An app may also use common Django conventions,
|
|
|
|
|
such as having a ``models.py`` file.
|
|
|
|
|
|
|
|
|
|
Later on we use the term *packaging* to describe the process of making a
|
|
|
|
|
Python package easy for others to install. It can be a little confusing, we
|
|
|
|
|
know.
|
|
|
|
|
|
2013-02-07 18:51:25 +08:00
|
|
|
|
Your project and your reusable app
|
|
|
|
|
==================================
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
After the previous tutorials, our project should look like this::
|
|
|
|
|
|
|
|
|
|
mysite/
|
|
|
|
|
manage.py
|
|
|
|
|
mysite/
|
|
|
|
|
__init__.py
|
|
|
|
|
settings.py
|
|
|
|
|
urls.py
|
|
|
|
|
wsgi.py
|
|
|
|
|
polls/
|
|
|
|
|
admin.py
|
|
|
|
|
__init__.py
|
|
|
|
|
models.py
|
2013-03-04 19:05:11 +08:00
|
|
|
|
static/
|
2013-06-28 15:43:14 +08:00
|
|
|
|
polls/
|
2013-04-21 21:52:02 +08:00
|
|
|
|
images/
|
|
|
|
|
background.gif
|
|
|
|
|
style.css
|
2012-10-31 03:53:56 +08:00
|
|
|
|
templates/
|
|
|
|
|
polls/
|
|
|
|
|
detail.html
|
|
|
|
|
index.html
|
|
|
|
|
results.html
|
2013-04-21 21:52:02 +08:00
|
|
|
|
tests.py
|
2012-10-31 03:53:56 +08:00
|
|
|
|
urls.py
|
|
|
|
|
views.py
|
2013-02-23 22:19:32 +08:00
|
|
|
|
templates/
|
2013-02-07 18:51:25 +08:00
|
|
|
|
admin/
|
|
|
|
|
base_site.html
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
2013-02-23 22:19:32 +08:00
|
|
|
|
You created ``mysite/templates`` in :doc:`Tutorial 2 </intro/tutorial02>`,
|
2013-02-07 18:51:25 +08:00
|
|
|
|
and ``polls/templates`` in :doc:`Tutorial 3 </intro/tutorial03>`. Now perhaps
|
|
|
|
|
it is clearer why we chose to have separate template directories for the
|
|
|
|
|
project and application: everything that is part of the polls application is in
|
|
|
|
|
``polls``. It makes the application self-contained and easier to drop into a
|
|
|
|
|
new project.
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
The ``polls`` directory could now be copied into a new Django project and
|
|
|
|
|
immediately reused. It's not quite ready to be published though. For that, we
|
|
|
|
|
need to package the app to make it easy for others to install.
|
|
|
|
|
|
|
|
|
|
.. _installing-reusable-apps-prerequisites:
|
|
|
|
|
|
|
|
|
|
Installing some prerequisites
|
|
|
|
|
=============================
|
|
|
|
|
|
|
|
|
|
The current state of Python packaging is a bit muddled with various tools. For
|
|
|
|
|
this tutorial, we're going to use distribute_ to build our package. It's a
|
|
|
|
|
community-maintained fork of the older ``setuptools`` project. We'll also be
|
|
|
|
|
using `pip`_ to uninstall it after we're finished. You should install these
|
|
|
|
|
two packages now. If you need help, you can refer to :ref:`how to install
|
|
|
|
|
Django with pip<installing-official-release>`. You can install ``distribute``
|
|
|
|
|
the same way.
|
|
|
|
|
|
|
|
|
|
.. _distribute: http://pypi.python.org/pypi/distribute
|
|
|
|
|
.. _pip: http://pypi.python.org/pypi/pip
|
|
|
|
|
|
|
|
|
|
Packaging your app
|
|
|
|
|
==================
|
|
|
|
|
|
|
|
|
|
Python *packaging* refers to preparing your app in a specific format that can
|
|
|
|
|
be easily installed and used. Django itself is packaged very much like
|
|
|
|
|
this. For a small app like polls, this process isn't too difficult.
|
|
|
|
|
|
|
|
|
|
1. First, create a parent directory for ``polls``, outside of your Django
|
|
|
|
|
project. Call this directory ``django-polls``.
|
|
|
|
|
|
|
|
|
|
.. admonition:: Choosing a name for your app
|
|
|
|
|
|
|
|
|
|
When choosing a name for your package, check resources like PyPI to avoid
|
|
|
|
|
naming conflicts with existing packages. It's often useful to prepend
|
|
|
|
|
``django-`` to your module name when creating a package to distribute.
|
|
|
|
|
This helps others looking for Django apps identify your app as Django
|
|
|
|
|
specific.
|
|
|
|
|
|
|
|
|
|
2. Move the ``polls`` directory into the ``django-polls`` directory.
|
|
|
|
|
|
2013-04-24 15:27:03 +08:00
|
|
|
|
3. Create a file ``django-polls/README.rst`` with the following contents::
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
=====
|
|
|
|
|
Polls
|
|
|
|
|
=====
|
|
|
|
|
|
|
|
|
|
Polls is a simple Django app to conduct Web-based polls. For each
|
|
|
|
|
question, visitors can choose between a fixed number of answers.
|
|
|
|
|
|
|
|
|
|
Detailed documentation is in the "docs" directory.
|
|
|
|
|
|
|
|
|
|
Quick start
|
|
|
|
|
-----------
|
|
|
|
|
|
|
|
|
|
1. Add "polls" to your INSTALLED_APPS setting like this::
|
|
|
|
|
|
|
|
|
|
INSTALLED_APPS = (
|
|
|
|
|
...
|
|
|
|
|
'polls',
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
2. Include the polls URLconf in your project urls.py like this::
|
|
|
|
|
|
|
|
|
|
url(r'^polls/', include('polls.urls')),
|
|
|
|
|
|
|
|
|
|
3. Run `python manage.py syncdb` to create the polls models.
|
|
|
|
|
|
|
|
|
|
4. Start the development server and visit http://127.0.0.1:8000/admin/
|
2013-04-24 15:41:06 +08:00
|
|
|
|
to create a poll (you'll need the Admin app enabled).
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
5. Visit http://127.0.0.1:8000/polls/ to participate in the poll.
|
|
|
|
|
|
|
|
|
|
4. Create a ``django-polls/LICENSE`` file. Choosing a license is beyond the
|
|
|
|
|
scope of this tutorial, but suffice it to say that code released publicly
|
|
|
|
|
without a license is *useless*. Django and many Django-compatible apps are
|
|
|
|
|
distributed under the BSD license; however, you're free to pick your own
|
|
|
|
|
license. Just be aware that your licensing choice will affect who is able
|
|
|
|
|
to use your code.
|
|
|
|
|
|
|
|
|
|
5. Next we'll create a ``setup.py`` file which provides details about how to
|
|
|
|
|
build and install the app. A full explanation of this file is beyond the
|
|
|
|
|
scope of this tutorial, but the `distribute docs
|
|
|
|
|
<http://packages.python.org/distribute/setuptools.html>`_ have a good explanation.
|
|
|
|
|
Create a file ``django-polls/setup.py`` with the following contents::
|
|
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
from setuptools import setup
|
|
|
|
|
|
2013-04-24 15:27:03 +08:00
|
|
|
|
README = open(os.path.join(os.path.dirname(__file__), 'README.rst')).read()
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
# allow setup.py to be run from any path
|
|
|
|
|
os.chdir(os.path.normpath(os.path.join(os.path.abspath(__file__), os.pardir)))
|
|
|
|
|
|
|
|
|
|
setup(
|
2013-04-24 15:27:03 +08:00
|
|
|
|
name='django-polls',
|
|
|
|
|
version='0.1',
|
|
|
|
|
packages=['polls'],
|
|
|
|
|
include_package_data=True,
|
|
|
|
|
license='BSD License', # example license
|
|
|
|
|
description='A simple Django app to conduct Web-based polls.',
|
|
|
|
|
long_description=README,
|
|
|
|
|
url='http://www.example.com/',
|
|
|
|
|
author='Your Name',
|
|
|
|
|
author_email='yourname@example.com',
|
|
|
|
|
classifiers=[
|
2012-10-31 03:53:56 +08:00
|
|
|
|
'Environment :: Web Environment',
|
|
|
|
|
'Framework :: Django',
|
|
|
|
|
'Intended Audience :: Developers',
|
|
|
|
|
'License :: OSI Approved :: BSD License', # example license
|
|
|
|
|
'Operating System :: OS Independent',
|
|
|
|
|
'Programming Language :: Python',
|
|
|
|
|
'Programming Language :: Python :: 2.6',
|
|
|
|
|
'Programming Language :: Python :: 2.7',
|
|
|
|
|
'Topic :: Internet :: WWW/HTTP',
|
|
|
|
|
'Topic :: Internet :: WWW/HTTP :: Dynamic Content',
|
|
|
|
|
],
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
.. admonition:: I thought you said we were going to use ``distribute``?
|
|
|
|
|
|
|
|
|
|
Distribute is a drop-in replacement for ``setuptools``. Even though we
|
|
|
|
|
appear to import from ``setuptools``, since we have ``distribute``
|
|
|
|
|
installed, it will override the import.
|
|
|
|
|
|
|
|
|
|
6. Only Python modules and packages are included in the package by default. To
|
|
|
|
|
include additional files, we'll need to create a ``MANIFEST.in`` file. The
|
|
|
|
|
distribute docs referred to in the previous step discuss this file in more
|
2013-04-24 15:27:03 +08:00
|
|
|
|
details. To include the templates, the ``README.rst`` and our ``LICENSE``
|
|
|
|
|
file, create a file ``django-polls/MANIFEST.in`` with the following
|
|
|
|
|
contents::
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
include LICENSE
|
2013-04-24 15:27:03 +08:00
|
|
|
|
include README.rst
|
2013-05-06 18:19:52 +08:00
|
|
|
|
recursive-include polls/static *
|
2012-10-31 03:53:56 +08:00
|
|
|
|
recursive-include polls/templates *
|
|
|
|
|
|
|
|
|
|
7. It's optional, but recommended, to include detailed documentation with your
|
|
|
|
|
app. Create an empty directory ``django-polls/docs`` for future
|
|
|
|
|
documentation. Add an additional line to ``django-polls/MANIFEST.in``::
|
|
|
|
|
|
|
|
|
|
recursive-include docs *
|
|
|
|
|
|
|
|
|
|
Note that the ``docs`` directory won't be included in your package unless
|
|
|
|
|
you add some files to it. Many Django apps also provide their documentation
|
|
|
|
|
online through sites like `readthedocs.org <http://readthedocs.org>`_.
|
|
|
|
|
|
|
|
|
|
8. Try building your package with ``python setup.py sdist`` (run from inside
|
|
|
|
|
``django-polls``). This creates a directory called ``dist`` and builds your
|
|
|
|
|
new package, ``django-polls-0.1.tar.gz``.
|
|
|
|
|
|
|
|
|
|
For more information on packaging, see `The Hitchhiker's Guide to Packaging
|
|
|
|
|
<http://guide.python-distribute.org/quickstart.html>`_.
|
|
|
|
|
|
|
|
|
|
Using your own package
|
|
|
|
|
======================
|
|
|
|
|
|
|
|
|
|
Since we moved the ``polls`` directory out of the project, it's no longer
|
|
|
|
|
working. We'll now fix this by installing our new ``django-polls`` package.
|
|
|
|
|
|
2012-11-01 07:56:53 +08:00
|
|
|
|
.. admonition:: Installing as a user library
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
2012-11-01 07:56:53 +08:00
|
|
|
|
The following steps install ``django-polls`` as a user library. Per-user
|
|
|
|
|
installs have a lot of advantages over installing the package system-wide,
|
|
|
|
|
such as being usable on systems where you don't have administrator access
|
|
|
|
|
as well as preventing the package from affecting system services and other
|
|
|
|
|
users of the machine. Python 2.6 added support for user libraries, so if
|
|
|
|
|
you are using an older version this won't work, but Django 1.5 requires
|
|
|
|
|
Python 2.6 or newer anyway.
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
2012-11-01 07:56:53 +08:00
|
|
|
|
Note that per-user installations can still affect the behavior of system
|
|
|
|
|
tools that run as that user, so ``virtualenv`` is a more robust solution
|
|
|
|
|
(see below).
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
1. Inside ``django-polls/dist``, untar the new package
|
|
|
|
|
``django-polls-0.1.tar.gz`` (e.g. ``tar xzvf django-polls-0.1.tar.gz``). If
|
|
|
|
|
you're using Windows, you can download the command-line tool bsdtar_ to do
|
|
|
|
|
this, or you can use a GUI-based tool such as 7-zip_.
|
|
|
|
|
|
|
|
|
|
2. Change into the directory created in step 1 (e.g. ``cd django-polls-0.1``).
|
|
|
|
|
|
|
|
|
|
3. If you're using GNU/Linux, Mac OS X or some other flavor of Unix, enter the
|
2012-11-01 07:56:53 +08:00
|
|
|
|
command ``python setup.py install --user`` at the shell prompt. If you're
|
|
|
|
|
using Windows, start up a command shell and run the command
|
|
|
|
|
``setup.py install --user``.
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
With luck, your Django project should now work correctly again. Run the
|
|
|
|
|
server again to confirm this.
|
|
|
|
|
|
|
|
|
|
4. To uninstall the package, use pip (you already :ref:`installed it
|
|
|
|
|
<installing-reusable-apps-prerequisites>`, right?)::
|
|
|
|
|
|
2012-11-01 07:56:53 +08:00
|
|
|
|
pip uninstall django-polls
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
.. _bsdtar: http://gnuwin32.sourceforge.net/packages/bsdtar.htm
|
|
|
|
|
.. _7-zip: http://www.7-zip.org/
|
|
|
|
|
.. _pip: http://pypi.python.org/pypi/pip
|
|
|
|
|
|
|
|
|
|
Publishing your app
|
|
|
|
|
===================
|
|
|
|
|
|
|
|
|
|
Now that we've packaged and tested ``django-polls``, it's ready to share with
|
|
|
|
|
the world! If this wasn't just an example, you could now:
|
|
|
|
|
|
|
|
|
|
* Email the package to a friend.
|
|
|
|
|
|
|
|
|
|
* Upload the package on your Web site.
|
|
|
|
|
|
|
|
|
|
* Post the package on a public repository, such as `The Python Package Index
|
|
|
|
|
(PyPI) <http://guide.python-distribute.org/contributing.html#pypi-info>`_.
|
|
|
|
|
|
|
|
|
|
For more information on PyPI, see the `Quickstart
|
|
|
|
|
<http://guide.python-distribute.org/quickstart.html#register-your-package-with-the-python-package-index-pypi>`_
|
|
|
|
|
section of The Hitchhiker's Guide to Packaging. One detail this guide mentions
|
|
|
|
|
is choosing the license under which your code is distributed.
|
|
|
|
|
|
|
|
|
|
Installing Python packages with virtualenv
|
|
|
|
|
==========================================
|
|
|
|
|
|
2012-11-01 07:56:53 +08:00
|
|
|
|
Earlier, we installed the polls app as a user library. This has some
|
2012-10-31 03:53:56 +08:00
|
|
|
|
disadvantages:
|
|
|
|
|
|
2012-11-01 07:56:53 +08:00
|
|
|
|
* Modifying the user libraries can affect other Python software on your system.
|
2012-10-31 03:53:56 +08:00
|
|
|
|
|
|
|
|
|
* You won't be able to run multiple versions of this package (or others with
|
|
|
|
|
the same name).
|
|
|
|
|
|
|
|
|
|
Typically, these situations only arise once you're maintaining several Django
|
|
|
|
|
projects. When they do, the best solution is to use `virtualenv
|
|
|
|
|
<http://www.virtualenv.org/>`_. This tool allows you to maintain multiple
|
|
|
|
|
isolated Python environments, each with its own copy of the libraries and
|
|
|
|
|
package namespace.
|