Amazon AWS API Gateway WSGI wrapper
Project description
# Serverless WSGI
[![npm package](https://nodei.co/npm/serverless-wsgi.png?downloads=true&downloadRank=true&stars=true)](https://nodei.co/npm/serverless-wsgi/)
[![serverless](http://public.serverless.com/badges/v3.svg)](http://www.serverless.com)
[![Build Status](https://travis-ci.org/logandk/serverless-wsgi.png?branch=master)](https://travis-ci.org/logandk/serverless-wsgi)
[![Coverage Status](https://coveralls.io/repos/github/logandk/serverless-wsgi/badge.svg?branch=master)](https://coveralls.io/github/logandk/serverless-wsgi?branch=master)
[![Dependency Status](https://david-dm.org/logandk/serverless-wsgi.png)](https://david-dm.org/logandk/serverless-wsgi)
[![Dev Dependency Status](https://david-dm.org/logandk/serverless-wsgi/dev-status.png)](https://david-dm.org/logandk/serverless-wsgi?type=dev)
A Serverless v1.x plugin to build your deploy Python WSGI applications using Serverless. Compatible
WSGI application frameworks include Flask, Django and Pyramid - for a complete list, see:
http://wsgi.readthedocs.io/en/latest/frameworks.html.
### Features
- Transparently converts API Gateway requests to and from standard WSGI requests
- Supports anything you'd expect from WSGI such as redirects, cookies, file uploads etc.
- Automatically downloads Python packages that you specify in `requirements.txt` and deploys them along with your application
- Convenient `wsgi serve` command for serving your application locally during development
## Install
```
npm install --save serverless-wsgi
```
Add the plugin to your `serverless.yml` file and set the WSGI application:
```yaml
plugins:
- serverless-wsgi
```
## Flask configuration example
This example assumes that you have intialized your application as `app` inside `api.py`.
```
project
├── api.py
├── requirements.txt
└── serverless.yml
```
### api.py
A regular Flask application.
```python
from flask import Flask
app = Flask(__name__)
@app.route("/cats")
def cats():
return "Cats"
@app.route("/dogs/<id>")
def dog(id):
return "Dog"
```
### serverless.yml
Load the plugin and set the `custom.wsgi.app` configuration in `serverless.yml` to the
module path of your Flask application.
All functions that will use WSGI need to have `wsgi.handler` set as the Lambda handler and
use the default `lambda-proxy` integration for API Gateway. This configuration example treats
API Gateway as a transparent proxy, passing all requests directly to your Flask application,
and letting the application handle errors, 404s etc.
```yaml
service: example
provider:
name: aws
runtime: python2.7
plugins:
- serverless-wsgi
functions:
api:
handler: wsgi.handler
events:
- http: ANY /
- http: ANY {proxy+}
custom:
wsgi:
app: api.app
```
### requirements.txt
Add Flask to the application bundle.
```
Flask==0.12.2
```
## Deployment
Simply run the serverless deploy command as usual:
```
$ sls deploy
Serverless: Packaging Python WSGI handler...
Serverless: Packaging required Python packages...
Serverless: Packaging service...
Serverless: Removing old service versions...
Serverless: Uploading CloudFormation file to S3...
Serverless: Uploading service .zip file to S3...
Serverless: Updating Stack...
Serverless: Checking Stack update progress...
..........
Serverless: Stack update finished...
```
## Other frameworks
Set `custom.wsgi.app` in `serverless.yml` according to your WSGI callable:
- For Pyramid, use [make_wsgi_app](http://docs.pylonsproject.org/projects/pyramid/en/latest/api/config.html#pyramid.config.Configurator.make_wsgi_app) to intialize the callable
- Django is configured for WSGI by default, set the callable to `<project_name>.wsgi.application`. See https://docs.djangoproject.com/en/1.10/howto/deployment/wsgi/ for more information.
## Usage
### Automatic requirement packaging
You'll need to include any packages that your application uses in the bundle
that's deployed to AWS Lambda. This plugin helps you out by doing this automatically,
as long as you specify your required packages in a `requirements.txt` file in the root
of your Serverless service path:
```
Flask==0.12.2
requests==2.18.3
```
For more information, see https://pip.readthedocs.io/en/1.1/requirements.html.
You can use the requirement packaging functionality of _serverless-wsgi_ without the WSGI
handler itself by including the plugin in your `serverless.yml` configuration, without specifying
the `custom.wsgi.app` setting. This will omit the WSGI handler from the package, but include
any requirements specified in `requirements.txt`.
If you don't want to use automatic requirement packaging you can set `custom.wsgi.packRequirements` to false:
```yaml
custom:
wsgi:
app: api.app
packRequirements: false
```
For a more advanced approach to packaging requirements, consider using https://github.com/UnitedIncome/serverless-python-requirements.
### Python version
Python is used for packaging requirements and serving the app when invoking `sls wsgi serve`. By
default, the current runtime setting is expected to be the name of the Python binary in `PATH`,
for instance `python3.6`. If this is not the name of your Python binary, override it using the
`pythonBin` option:
```yaml
custom:
wsgi:
app: api.app
pythonBin: python3
```
### Local server
For convenience, a `sls wsgi serve` command is provided to run your WSGI application
locally. This command requires the `werkzeug` Python package to be installed,
and acts as a simple wrapper for starting werkzeug's built-in HTTP server.
By default, the server will start on port 5000.
```
$ sls wsgi serve
* Running on http://localhost:5000/ (Press CTRL+C to quit)
* Restarting with stat
* Debugger is active!
```
Configure the port using the `-p` parameter:
```
$ sls wsgi serve -p 8000
* Running on http://localhost:8000/ (Press CTRL+C to quit)
* Restarting with stat
* Debugger is active!
```
When running locally, an environment variable named `IS_OFFLINE` will be set to `"True"`.
So, if you want to know when the application is running locally, check `os.environ["IS_OFFLINE"]`.
### Explicit routes
If you'd like to be explicit about which routes and HTTP methods should pass through to your
application, see the following example:
```yaml
service: example
provider:
name: aws
runtime: python2.7
plugins:
- serverless-wsgi
functions:
api:
handler: wsgi.handler
events:
- http:
path: cats
method: get
integration: lambda-proxy
- http:
path: dogs/{id}
method: get
integration: lambda-proxy
custom:
wsgi:
app: api.app
```
### Custom domain names
If you use custom domain names with API Gateway, you might have a base path that is
at the beginning of your path, such as the stage (`/dev`, `/stage`, `/prod`). You
can pass in an `API_GATEWAY_BASE_PATH` environment variable so your WSGI app can
handle it correctly.
The example below uses the [serverless-domain-manager](https://github.com/amplify-education/serverless-domain-manager)
plugin to handle custom domains in API Gateway:
```yaml
service: example
provider:
name: aws
runtime: python2.7
environment:
API_GATEWAY_BASE_PATH: ${self:custom.customDomain.basePath}
plugins:
- serverless-wsgi
- serverless-domain-manager
functions:
api:
handler: wsgi.handler
events:
- http: ANY /
- http: ANY {proxy+}
custom:
wsgi:
app: api.app
customDomain:
basePath: ${opt:stage}
domainName: mydomain.name.com
stage: ${opt:stage}
createRoute53Record: true
```
### File uploads
In order to accept file uploads from HTML forms, make sure to add `multipart/form-data` to
the list of content types with _Binary Support_ in your API Gateway API. The
[serverless-apigw-binary](https://github.com/maciejtreder/serverless-apigw-binary)
Serverless plugin can be used to automate this process.
Keep in mind that, when building Serverless applications, uploading
[directly to S3](http://docs.aws.amazon.com/AmazonS3/latest/dev/UsingHTTPPOST.html)
from the browser is usually the preferred approach.
### Raw context and event
The raw context and event from AWS Lambda are both accessible through the WSGI
request. The following example shows how to access them when using Flask:
```python
from flask import Flask, request
app = Flask(__name__)
@app.route("/")
def index():
print(request.environ['context'])
print(request.environ['event'])
```
### Text MIME types
By default, all MIME types starting with `text/` and the following whitelist are sent
through API Gateway in plain text. All other MIME types will have their response body
base64 encoded (and the `isBase64Encoded` API Gateway flag set) in order to be
delivered by API Gateway as binary data.
This is the default whitelist of plain text MIME types:
- `application/json`
- `application/javascript`
- `application/xml`
- `application/vnd.api+json`
In order to add additional plain text MIME types to this whitelist, use the
`textMimeTypes` configuration option:
```yaml
custom:
wsgi:
app: api.app
textMimeTypes:
- application/custom+json
- application/vnd.company+json
```
## Usage without Serverless
The AWS API Gateway to WSGI mapping module is available on PyPI in the
`serverless-wsgi` package.
Use this package if you need to deploy Python Lambda functions to handle
API Gateway events directly, without using the Serverless framework.
```
pip install serverless-wsgi
```
Initialize your WSGI application and in your Lambda event handler, call
the request mapper:
```python
import app # Replace with your actual application
import serverless_wsgi
# If you need to send additional content types as text, add then directly
# to the whitelist:
#
# serverless_wsgi.TEXT_MIME_TYPES.append("application/custom+json")
def handle(event, context):
return serverless_wsgi.handle_request(app.app, event, context)
```
# Thanks
Thanks to [Zappa](https://github.com/Miserlou/Zappa), which has been both the
inspiration and source of several implementations that went into this project.
Thanks to [chalice](https://github.com/awslabs/chalice) for the
requirement packaging implementation.
[![npm package](https://nodei.co/npm/serverless-wsgi.png?downloads=true&downloadRank=true&stars=true)](https://nodei.co/npm/serverless-wsgi/)
[![serverless](http://public.serverless.com/badges/v3.svg)](http://www.serverless.com)
[![Build Status](https://travis-ci.org/logandk/serverless-wsgi.png?branch=master)](https://travis-ci.org/logandk/serverless-wsgi)
[![Coverage Status](https://coveralls.io/repos/github/logandk/serverless-wsgi/badge.svg?branch=master)](https://coveralls.io/github/logandk/serverless-wsgi?branch=master)
[![Dependency Status](https://david-dm.org/logandk/serverless-wsgi.png)](https://david-dm.org/logandk/serverless-wsgi)
[![Dev Dependency Status](https://david-dm.org/logandk/serverless-wsgi/dev-status.png)](https://david-dm.org/logandk/serverless-wsgi?type=dev)
A Serverless v1.x plugin to build your deploy Python WSGI applications using Serverless. Compatible
WSGI application frameworks include Flask, Django and Pyramid - for a complete list, see:
http://wsgi.readthedocs.io/en/latest/frameworks.html.
### Features
- Transparently converts API Gateway requests to and from standard WSGI requests
- Supports anything you'd expect from WSGI such as redirects, cookies, file uploads etc.
- Automatically downloads Python packages that you specify in `requirements.txt` and deploys them along with your application
- Convenient `wsgi serve` command for serving your application locally during development
## Install
```
npm install --save serverless-wsgi
```
Add the plugin to your `serverless.yml` file and set the WSGI application:
```yaml
plugins:
- serverless-wsgi
```
## Flask configuration example
This example assumes that you have intialized your application as `app` inside `api.py`.
```
project
├── api.py
├── requirements.txt
└── serverless.yml
```
### api.py
A regular Flask application.
```python
from flask import Flask
app = Flask(__name__)
@app.route("/cats")
def cats():
return "Cats"
@app.route("/dogs/<id>")
def dog(id):
return "Dog"
```
### serverless.yml
Load the plugin and set the `custom.wsgi.app` configuration in `serverless.yml` to the
module path of your Flask application.
All functions that will use WSGI need to have `wsgi.handler` set as the Lambda handler and
use the default `lambda-proxy` integration for API Gateway. This configuration example treats
API Gateway as a transparent proxy, passing all requests directly to your Flask application,
and letting the application handle errors, 404s etc.
```yaml
service: example
provider:
name: aws
runtime: python2.7
plugins:
- serverless-wsgi
functions:
api:
handler: wsgi.handler
events:
- http: ANY /
- http: ANY {proxy+}
custom:
wsgi:
app: api.app
```
### requirements.txt
Add Flask to the application bundle.
```
Flask==0.12.2
```
## Deployment
Simply run the serverless deploy command as usual:
```
$ sls deploy
Serverless: Packaging Python WSGI handler...
Serverless: Packaging required Python packages...
Serverless: Packaging service...
Serverless: Removing old service versions...
Serverless: Uploading CloudFormation file to S3...
Serverless: Uploading service .zip file to S3...
Serverless: Updating Stack...
Serverless: Checking Stack update progress...
..........
Serverless: Stack update finished...
```
## Other frameworks
Set `custom.wsgi.app` in `serverless.yml` according to your WSGI callable:
- For Pyramid, use [make_wsgi_app](http://docs.pylonsproject.org/projects/pyramid/en/latest/api/config.html#pyramid.config.Configurator.make_wsgi_app) to intialize the callable
- Django is configured for WSGI by default, set the callable to `<project_name>.wsgi.application`. See https://docs.djangoproject.com/en/1.10/howto/deployment/wsgi/ for more information.
## Usage
### Automatic requirement packaging
You'll need to include any packages that your application uses in the bundle
that's deployed to AWS Lambda. This plugin helps you out by doing this automatically,
as long as you specify your required packages in a `requirements.txt` file in the root
of your Serverless service path:
```
Flask==0.12.2
requests==2.18.3
```
For more information, see https://pip.readthedocs.io/en/1.1/requirements.html.
You can use the requirement packaging functionality of _serverless-wsgi_ without the WSGI
handler itself by including the plugin in your `serverless.yml` configuration, without specifying
the `custom.wsgi.app` setting. This will omit the WSGI handler from the package, but include
any requirements specified in `requirements.txt`.
If you don't want to use automatic requirement packaging you can set `custom.wsgi.packRequirements` to false:
```yaml
custom:
wsgi:
app: api.app
packRequirements: false
```
For a more advanced approach to packaging requirements, consider using https://github.com/UnitedIncome/serverless-python-requirements.
### Python version
Python is used for packaging requirements and serving the app when invoking `sls wsgi serve`. By
default, the current runtime setting is expected to be the name of the Python binary in `PATH`,
for instance `python3.6`. If this is not the name of your Python binary, override it using the
`pythonBin` option:
```yaml
custom:
wsgi:
app: api.app
pythonBin: python3
```
### Local server
For convenience, a `sls wsgi serve` command is provided to run your WSGI application
locally. This command requires the `werkzeug` Python package to be installed,
and acts as a simple wrapper for starting werkzeug's built-in HTTP server.
By default, the server will start on port 5000.
```
$ sls wsgi serve
* Running on http://localhost:5000/ (Press CTRL+C to quit)
* Restarting with stat
* Debugger is active!
```
Configure the port using the `-p` parameter:
```
$ sls wsgi serve -p 8000
* Running on http://localhost:8000/ (Press CTRL+C to quit)
* Restarting with stat
* Debugger is active!
```
When running locally, an environment variable named `IS_OFFLINE` will be set to `"True"`.
So, if you want to know when the application is running locally, check `os.environ["IS_OFFLINE"]`.
### Explicit routes
If you'd like to be explicit about which routes and HTTP methods should pass through to your
application, see the following example:
```yaml
service: example
provider:
name: aws
runtime: python2.7
plugins:
- serverless-wsgi
functions:
api:
handler: wsgi.handler
events:
- http:
path: cats
method: get
integration: lambda-proxy
- http:
path: dogs/{id}
method: get
integration: lambda-proxy
custom:
wsgi:
app: api.app
```
### Custom domain names
If you use custom domain names with API Gateway, you might have a base path that is
at the beginning of your path, such as the stage (`/dev`, `/stage`, `/prod`). You
can pass in an `API_GATEWAY_BASE_PATH` environment variable so your WSGI app can
handle it correctly.
The example below uses the [serverless-domain-manager](https://github.com/amplify-education/serverless-domain-manager)
plugin to handle custom domains in API Gateway:
```yaml
service: example
provider:
name: aws
runtime: python2.7
environment:
API_GATEWAY_BASE_PATH: ${self:custom.customDomain.basePath}
plugins:
- serverless-wsgi
- serverless-domain-manager
functions:
api:
handler: wsgi.handler
events:
- http: ANY /
- http: ANY {proxy+}
custom:
wsgi:
app: api.app
customDomain:
basePath: ${opt:stage}
domainName: mydomain.name.com
stage: ${opt:stage}
createRoute53Record: true
```
### File uploads
In order to accept file uploads from HTML forms, make sure to add `multipart/form-data` to
the list of content types with _Binary Support_ in your API Gateway API. The
[serverless-apigw-binary](https://github.com/maciejtreder/serverless-apigw-binary)
Serverless plugin can be used to automate this process.
Keep in mind that, when building Serverless applications, uploading
[directly to S3](http://docs.aws.amazon.com/AmazonS3/latest/dev/UsingHTTPPOST.html)
from the browser is usually the preferred approach.
### Raw context and event
The raw context and event from AWS Lambda are both accessible through the WSGI
request. The following example shows how to access them when using Flask:
```python
from flask import Flask, request
app = Flask(__name__)
@app.route("/")
def index():
print(request.environ['context'])
print(request.environ['event'])
```
### Text MIME types
By default, all MIME types starting with `text/` and the following whitelist are sent
through API Gateway in plain text. All other MIME types will have their response body
base64 encoded (and the `isBase64Encoded` API Gateway flag set) in order to be
delivered by API Gateway as binary data.
This is the default whitelist of plain text MIME types:
- `application/json`
- `application/javascript`
- `application/xml`
- `application/vnd.api+json`
In order to add additional plain text MIME types to this whitelist, use the
`textMimeTypes` configuration option:
```yaml
custom:
wsgi:
app: api.app
textMimeTypes:
- application/custom+json
- application/vnd.company+json
```
## Usage without Serverless
The AWS API Gateway to WSGI mapping module is available on PyPI in the
`serverless-wsgi` package.
Use this package if you need to deploy Python Lambda functions to handle
API Gateway events directly, without using the Serverless framework.
```
pip install serverless-wsgi
```
Initialize your WSGI application and in your Lambda event handler, call
the request mapper:
```python
import app # Replace with your actual application
import serverless_wsgi
# If you need to send additional content types as text, add then directly
# to the whitelist:
#
# serverless_wsgi.TEXT_MIME_TYPES.append("application/custom+json")
def handle(event, context):
return serverless_wsgi.handle_request(app.app, event, context)
```
# Thanks
Thanks to [Zappa](https://github.com/Miserlou/Zappa), which has been both the
inspiration and source of several implementations that went into this project.
Thanks to [chalice](https://github.com/awslabs/chalice) for the
requirement packaging implementation.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
serverless-wsgi-1.5.1.tar.gz
(10.3 kB
view details)
Built Distribution
File details
Details for the file serverless-wsgi-1.5.1.tar.gz
.
File metadata
- Download URL: serverless-wsgi-1.5.1.tar.gz
- Upload date:
- Size: 10.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/1.11.0 pkginfo/1.4.2 requests/2.19.1 setuptools/40.2.0 requests-toolbelt/0.8.0 tqdm/4.25.0 CPython/3.6.5
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 731e170d631a7194feb122b6f89c9a316e7b45cc35744c966f51f670dd2698ad |
|
MD5 | 253a630be60eb8e29c51c977fc5b5d2b |
|
BLAKE2b-256 | 49eb715d05a0335707556e86aebca77f8bb8fa78c9217edb14484f5b76e1bf8c |
File details
Details for the file serverless_wsgi-1.5.1-py2.py3-none-any.whl
.
File metadata
- Download URL: serverless_wsgi-1.5.1-py2.py3-none-any.whl
- Upload date:
- Size: 12.6 kB
- Tags: Python 2, Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/1.11.0 pkginfo/1.4.2 requests/2.19.1 setuptools/40.2.0 requests-toolbelt/0.8.0 tqdm/4.25.0 CPython/3.6.5
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 1535f16a7f3f35e2e97aaf416641b395e7694f1dc92c36faee3036f734405d5a |
|
MD5 | 17c7ac479ae431bec4c78ecb9c16fddb |
|
BLAKE2b-256 | d2042ba38d08ded3d3e86cc1b0e76ce4fd6a098ec3ae98876983290ec28fbb1c |