Please note: this is not an official Google product.
This contains a boilerplate AppEngine application meant to provide a secure base on which to build additional functionality. Structure:
- / - top level directory for common files, e.g. app.yaml
- /js - directory for uncompiled Javascript resources.
- /src - directory for all source code
- /static - directory for static content
- /templates - directory for Django/Jinja2 templates your app renders.
- /templates/soy - directory for Closure Templates your application uses.
Javascript resources for your application can be written using Closure, and compiled by Google's Closure Compiler (detailed below in the dependencies section).
The scaffold provides the following basic security guarantees by default through
a set of base classes found in src/base/handlers.py
. These handlers:
- Set assorted security headers (Strict-Transport-Security, X-Frame-Options,
X-XSS-Protection, X-Content-Type-Options, Content-Security-Policy) with
strong default values to help avoid attacks like Cross-Site Scripting (XSS)
and Cross-Site Script Inclusion. See
_SetCommonResponseHeaders()
andSetAjaxResponseHeaders()
. - Prevent the XSS-prone construction of HTML via string concatenation by
forcing the use of a template system (Django/Jinja2 supported). The
template systems have non-contextual autoescaping enabled by default.
See the
render()
,render_json()
methods inBaseHandler
andBaseAjaxHandler
. For contextual autoescaping, you should use Closure Templates in strict mode (https://developers.google.com/closure/templates/docs/security). - Test for the presence of headers that guarantee requests to Cron or
Task endpoints are made by the AppEngine serving environment or an
application administrator. See the
dispatch()
method inBaseCronHandler
andBaseTaskHandler
. - Verify XSRF tokens by default on authenticated requests using any verb other
that GET, HEAD, or OPTIONS. See the
_RequestContainsValidXsrfToken()
method for more information.
In addition to the protections above, the scaffold monkey patches assorted APIs
that use insecure or dangerous defaults (see src/base/api_fixer.py
).
Obviously no framework is perfect, and the flexibility of Python offers many ways for a motivated developer to circumvent the protections offered. Under the assumption that developers are not malicious, using the scaffold should centralize many security mechanisms, provide safe defaults, and structure the code in a way that facilitates security review.
Sample implementations can be found in src/handlers.py
. These demonstrate
basic functionality, and should be removed / replaced by code specific to
your application.
These instructions have been tested with the following software:
- node.js >= 0.8.0
- 0.8.0 is the minimum required to build with Grunt.
- git
- curl
An alternative to the Grunt build is provided via the util.sh
shell script.
pushd .
mkdir $HOME/bin; cd $HOME/bin
npm install grunt-cli
- Alternatively,
sudo npm install -g grunt-cli
will install system-wide and you may skip the next step.
- Alternatively,
export PATH=$HOME/bin/node_modules/grunt-cli/bin:$PATH
- It is advisable to add this to login profile scripts (.bashrc, etc.).
- Visit https://developers.google.com/appengine/downloads, copy URL of "Linux/Other Platforms" zip file for current AppEngine SDK. Do this regardless of whether you are on Linux or OS X.
curl -O <url on clipboard>
unzip google_appengine_*.zip
mkdir google_closure; cd google_closure
curl -O https://dl.google.com/closure-compiler/compiler-latest.zip
unzip compiler-latest.zip; cd ..
mkdir google_closure_templates; cd google_closure_templates
curl -O https://dl.google.com/closure-templates/closure-templates-for-javascript-latest.zip
unzip closure-templates-for-javascript-latest.zip
popd
To install dependencies for unit testing:
sudo easy_install pip
sudo pip install unittest2
These instructions assume a working directory of the repository root.
All users should run:
git submodule init
git submodule update
Grunt users should also run:
npm install
To run unit tests:
python run_tests.py ~/bin/google_appengine src
To run the development appserver locally:
grunt clean
grunt
grunt appengine:run:app
Note that the development appserver will be running on a snapshot of code at the time you run it. If you make changes, you can run the various Grunt tasks in order to propagate them to the local appserver. For instance:
grunt copy
will refresh the source code (local and third party), static files,
and templates. You can run grunt closureSoys
and/or grunt closureBuilder
before grunt copy
if you need to rebuild your Closure Templates or Closure
Javascript.
If you are not using Grunt, simply run:
util.sh -d
To deploy to AppEngine:
grunt clean
grunt --appid=<appid>
grunt appengine:update:app --appid=<appid>
Specifying --appid=
will override any value set in config.json
. You may
modify the config.json
file to avoid having to pass this parameter on
every invocation.
If you are not using Grunt, simply run:
util.sh -p <appid>
Files in js/
are compiled by the Closure Compiler (if available) and placed in
out/static/app.js
.
Closure templates are compiled by the Closure Template Compiler (if available)
and placed in out/static/app.soy.js
.
The /static
and /template
directories are replicated in out/
, and the
files in src/
are rebased into out/
(so src/base/foo.py
becomes
out/base/foo.py
).
-
The AppEngine SDK should be present in the directory:
$HOME/bin/google_appengine/
You can find / download this at: https://developers.google.com/appengine/downloads
-
(Optional, if using Google Closure): The Google Closure Compiler (and a suitable Java runtime), located at:
$HOME/bin/google_closure/
You can find / download this at: https://github.com/google/closure-compiler
You will need all the files from this archive in the above directory: compiler-latest.zip
The compiler is invoked with the default namespace of 'app.' The compiled
Javascript is written to out/static/app.js
.
You will also need the Closure Library (in the closure-library submodule of this repository).
You can find more on the Closure Library here: https://github.com/google/closure-library
To use it, you will need to check out the code as a submodule by running the following commands from the base directory of this repository:
git submodule add <https://github.com/google/closure-library/> closure-library
git commit -m "Initial import of Closure Library"
-
(Optional, if using Closure Templates): The Closure Template compiler (in addition to the Closure Compiler), located at:
$HOME/bin/google_closure_templates
You can find / download Closure Templates at: https://github.com/google/closure-templates
You will need all the files from this archive in the above directory: closure-templates-for-javascript-latest.zip
You can build this using the ant target "zips-for-release", or download a prebuilt version (the URL is in the Dependency Setup section).
The deployment script checks for the presence of .soy files in templates/soy.
If found, they are compiled to a single Javascript file using the
SoyToJsSrcCompiler.jar in the previously mentioned directory. The resulting
Javascript file is stored in static/app.soy.js, alongside the soyutils.js
library provided with the Closure Templates bundle that is necessary to include
on any page you plan to use Closure Templates.