Skip to content

A Django 1.6 base template with HTML5Boilerplate and Twitter Bootstrap

License

Notifications You must be signed in to change notification settings

atleta/django-base-template

 
 

Repository files navigation

Creating a new project using this template

This needs to be done in two steps due to the limitations of the django project template solution:

  1. Use the standard django-admin command:
$ django-admin.py startproject --template <path_or_url_of_template> <project_name> <dir>

Where path_or_url_of_template points to the template project (normally the directory where this README file is located), project_name is the name of the project as you'd like to call it and dir is the destination directory (can also be .).

  1. Run the fixref script from the root of the newly created project to fix project name references in some of the generated files (where django didn't do so):
$ fixrefs <project_name>

Where project_name should be the same as above. After running this script you can remove it.

Project structure

Directory layout

env - python virtualenv directory *not* checked into the VCS
        (version control system, e.g. git or hg). You need
        to create it on first checkout and keep it updated
        (see below)
conf - deploy time configuration files
docs - documentation (not used)
goalcast +     source tree of the project
         |
         + apps      django apps that consitute the project. Each
         |           app has its own directory. (NOTE: these are
         |           the locally developed apps, not 3rd party
         |           ones. 3rd party code lives in the virtualenv.)
         + libs      python modules, that don't belong to any app.
         |
         + settings  Settings live here. Each runtime environment
         |               (see below) has its own setting. Detailed
         |               explanation below.
         + wsgi.py   WSGI file for running the project with a wsgi
                       server.
media - Media files (images, etc.) will be uploaded here (if the
          app has an upload functionality). Contents are NOT
          checked into the VCS, used at run time only.
requirements - the requirement files for each runtime environment.
                 The requirements are generated from the *.in
                 files (these are the ones to edit manually) to
                 freeze the versions. Both the .in files and
                 the generated .txt files are checked into the
                 VCS.
static - static files (e.g. js & css files, images, etc.)
templates - django templates
tests - test code lives here. (Don't put tests under the source tree!)
fabfile.py - deployment script (see deployment instructions below)
requirements.txt - root requirements file for deployment PaaS targets
                     that may need it (e.g. Heroku). It simply includes
                     requirements/deployment.txt .

Runtime environments & settings & requirements

To understand how settings and requirements are handled, we have to introduce the concept of the 'runtime environments'. These are simply the different configurations that the app will be run in, e.g.: development, deployment, staging, etc.

  • development - used for locally developing the app.
  • deployment - used for deploying the app. This includes both production and staging. To keep differences between the two minimal, we don't use a separate production environment, but only override the settings for staging. (See below.)
  • staging - used for trying/testing the app in an environment that is as close to production as possible.

Settings

The settings are structured according to the environments above. Each environment has its own setting file, but some of them depend on the other:

  • base.py - holds the settings common to all environments. (Most settings will belong here, e.g. 3rd party and local apps used, etc.)
  • development.py - Settings used for development. This may contain development specific django apps (e.g. django-debug-toolbar, more verbose logging, test runner setup, etc.)
  • local.py - This one will be loaded by development.py (at the end) if it exists. It may contain local (developer/machine specific) settings and thus it should not be checked into the VCS as it will be different for every project member.
  • deployment.py - Settings for deployment ('server'). We don't have a separate production setting, but only override the settings that differ in the staging settings. (We could have a production setting if necessary, that should include deployment, just as staging does. In that case, the environment name should be updated in the fabfile.)
  • staging.py - contains (overrides) settings that differ from the production environment. E.g.: database credentials, logging verbosity, etc. Ideally this file is very minimal.

Requirements

Requirements are also structured according to the environments, but are a bit simpler. We only have base, development and deployment ones. Note, that there is no staging requirements as that would mean that the staging and production server differ too much to be meaningful.

Experimental: requirements (*.txt files) are generated from the *.in files. The .in files should pin the requirements where it's necessary (e.g. needing a package version of at least X) and should only include packages that are needed by the project (and not packages that are only depended on by other packages). The requirements are then generated using the pip-compile tool into the according .txt files (e.g. deployment.in -> deployment.txt).

The .txt files will contain all requirements for that environment (i.e. there will be no base.txt included by deployment.txt). The .txt files should also be checked into the VCS but not edited manually.

When deploying the project, the deploy script will use the matching .txt file from the requirements directory. To keep the local (development) requirements up to date, you should run:

(env) $ pip install -r requirements/development.txt

Initial setup

Before being able to run the project, you'll at least have to set up the database. (Remember, local settings go into the local.py file.) You can use whatever database you want, but it's best to use the same as production as different databases may (and do) behave differently and that might make tests break (or pass and then break in production). PostgreSQL is the suggested db for production.

After setting up the db, you should run

(env) $ ./manage migrate

To create the initial schema. You'll also need to run this from time to time after pulling other people's code from the VCS repository (as they may have changed the model).

Running and working with the app locally

Instead of the standard runserver manage command (./manage runserver) you can use

(env) $ ./manage runserver_plus

to run the app. This will enhance the server with the Werkzeug interactive debugger that will allow you to interact with your code (debug it) in the browser, when it throws an exception instead of returning a valid response.

Similarly, instead of ./manage shell you can use

(env) $ ./manage shell_plus

and have your models (and a few frequently needed classes and modules) be automatically imported.

Installation

To install the system first you need to provision a new server (install and setup external dependencies) then run the deployment script. To update the app, you just need to run the deployment again.

Both actions need the development requirements to be installed.

Provisioning

Provisioning is done using ansible (documentation: https://docs.ansible.com/ ).

The related files are located in the provisioning subdirectory of the project. The below commands should be run from this directory.

Starting from scratch, e.g. with a vanilla Linode VPS, you do the provisioning like this (all commands are to be run on your local machine): 0. edit hosts file (if you haven't done so far) to include your machines.

  1. install your SSH public key to the server using the add-root.sh script:
(env) $ ./add-root.sh xxx.xxx.xxx.xxx

Where xxx.xxx.xxx.xxx is the IP (or the FQDN) of the target machine

(env) $ ./provision_all

Without going into the details here's a short explanation for most of the files used in the provisioning process:

  • hosts - the inventory file containing the hosts we want to provision to. Hosts can have different roles (separate db, web server, etc.). This simple setup doesn't use this feature.

  • playbook.yml - the provisioning script that describes what and how should be installed and set up in the hosts. It will work on debian based systems (including ubuntu).

    It does roughly the following:

    • add 3rd party repositories for nginx, postgresql and rabbitmq, so that it can install up to date versions. (The ones that ship with debian are usually too old to be useable.)
    • Install packages need by the app and by its dependencies.
    • create a database user and a database for the app, set the password. NOTE: these must match the ones provided in the settings (deployment.py and/or staging.py).
    • Create and set up access rights for the directories to be used by the app and the app deploy script. (Note that the rights must be in agreement with the user the app will run under, i.e. the setting in conf/supervisord_autumn.conf .)
  • files - contains files that get installed during the provisionin step. At the moment it holds the nginx (web server) base configuration.

  • roles - this directory is empty, see roles in the [ansible documentation] (http://docs.ansible.com/ansible/playbooks_roles.html)

  • pre_requisits.sh - a simple script than installs the pre-requisits needed to run ansible on the target machines. (This is not needed for e.g. linode VPS machines, as those already have these packages installed.)

  • provision_all - a simple script that executes ansible with the playbook.yml and the hosts inventory file, thus provisioning all hosts that are listed in the hosts file

  • add-root.sh - a simple convenience script to add the ssh public key (RSA) of the current user to the root account of the specified machine. This is needed to avoid ansible constantly asking for a password. (The key can be added in any other way, of course.)

  • test - contains files and a script that allows running the whole process on Docker, creating a docker image as a result. Theoretically this could be used as a deployment format as well (though the app itself is NOT included at this step yet), but mainly intended for testing the provisioning process.```

Deployment

The deloyment script should be run against a previously provisioned machine. The deployment uses fabric (http://www.fabfile.org/).

To install the app on the staging machine run: (see explanation for targets below)

(env) $ fab deploy:target=staging

Where 'deploy' is the name of the task and the 'target' parameter is set to staging.

This command has to be run from the project root directory, where the deployment script (fabfile.py) is located. The caveat is that it will start from any other subdirectory as well, but it will not be able to run without errors.

The deployment script's deploy task does roughly the following:

  • Do a build (build task):
    • put the project files into a tar archive
    • copy (with templating as needed) the config files as described in conf/deploy.map into the tar file.
      • For now only a single file is added: the one that selects the environment (e.g. staging, deployment), i.e. which setting to use.
    • compress the archive
  • Deploy the archive created by the build:
    • upload the archive into the /tmp dir on the remote machine
    • create a new timestamped subdirectory in the deployment directory (see settings at the top of the fabfile).
    • uncompress the uploaded archive there
    • create a virtualenv in the timestamped directory, and use pip to install the requirements matching the environment (see the target parameter above and the settings at the top of the fabfile). E.g.: staging or deployment
    • run collectstatic and migrations
    • upload the nginx and the supervisor config files. (Note, that "supervisorct reload" is not being run, thus the changes in the config file won't take effect until done so. This may be added to the script.)
    • stop the running instance of the app (which was the old version up to this point)
    • link 'current' to the timestamped directory we've just deployed into
    • start the app (the new version)
    • tell nginx to reload its config

The target parameter

The project structure supports multiple targets (as can be seen in the settings and the requirements directories). Which target an installation will use can be selected (and specified on the command line) at deploy time.with the 'target' argument. (Note that the 'build' task, if called separately, also needs this parameter.)

Build/deploy configuration settings for each target can be found at the top of the fabfile. The settings are self explanatory, but in short you can select the requirements file to use, the settings file to use and of course the list of machines to deploy to. (You can also set the deployment target directory that will hold the 'current' link and the timestamped installation directories.)

About

A Django 1.6 base template with HTML5Boilerplate and Twitter Bootstrap

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Python 53.2%
  • HTML 28.4%
  • Shell 13.1%
  • Jinja 4.0%
  • JavaScript 1.3%