[![CI Build Status](https://img.shields.io/github/actions/workflow/status/StevenMapes/django-aws-api-gateway-websockets/main.yml?branch=main&style=for-the-badge)](https://github.com/StevenMapes/django-aws-api-gateway-websockets/actions)
[![Coverage](https://img.shields.io/badge/Coverage-96%25-success?style=for-the-badge)](https://github.com/StevenMapes/django-aws-api-gateway-websockets/actions?workclow=CI)
[![PyPi](https://img.shields.io/pypi/v/django-aws-api-gateway-websockets.svg?style=for-the-badge)](https://pypi.org/project/django-aws-api-gateway-websockets/)
![Code Style](https://img.shields.io/badge/code%20style-black-000000.svg?style=for-the-badge)
![Pre-Commit Enabled](https://img.shields.io/badge/pre--commit-enabled-brightgreen?logo=pre-commit&logoColor=white&style=for-the-badge)
[![Read The Docs](https://img.shields.io/readthedocs/django-mysql?style=for-the-badge)](https://django-aws-api-gateway-websockets.readthedocs.io/)
[![PyPi Downloads](https://img.shields.io/pypi/dd/django-aws-api-gateway-websockets)](https://pypistats.org/packages/django-aws-api-gateway-websockets)
# Django AWS API Gateway Websockets
It is the aim of this project to create a uniform way to record websocket connections, associate the Django user who established the connection and then retrieve that user within each request.
This project is designed to work exclusively with AWS API Gateway.
It is not intended to be a replacement of [Django Channels](https://github.com/django/channels) instead this project allows you to add [WebSockets](https://en.wikipedia.org/wiki/WebSocket) support into your project by writing normal HTTP request-response views whilst allowing [AWS API Gateway](https://aws.amazon.com/api-gateway/) to worry about the WebSocket connection.
This project introduced a new [Class-Based-View](https://docs.djangoproject.com/en/dev/topics/class-based-views/) to handle connections, disconnections, routing, basic security checks and ensuring that the User object is available within every request.
The project will keep track of which users created which WebSockets, which ones are active and will allow you to send messages back down the socket to the client via Boto3.
Please refer to the installation notes and Getting Start Guides.
# Security Concerns
**IMPORTANT:**: In order to work the dispatch method requires the ```csrf_exempt``` decorator to be added. This has
already been added as a class decorator on the base view but if you overload the dispatch method you will need to add
it back to avoid receiving CSRF Token failures.
# Python and Django Support
This project supports Python 3.8 to 3.14 and Django 4.2 to 5.2. It may work with older versions of Django, but I am not
actively supporting anything below 4.2.
| **Python/Django** | **4.2** | **5.0** | **5.1** | **5.2** |
|--------------------|----------|----------|---------|----------|
| 3.8 | Y | N | N/A | N/A |
| 3.9 | Y | N | N/A | N/A |
| 3.10 | Y | Y | Y | Y |
| 3.11 | Y | Y | Y | Y |
| 3.12 | Y | Y | Y | Y |
| 3.13 | Y | Y | Y | Y |
| 3.14 | N | N | Y | Y |
* Python 3.11 only works with Django 4.1.3+
# Installation
You can install this package from pip using
```
pip install django-aws-api-gateway-websockets
```
## settings.py
Add ```django_aws_api_gateway_websockets``` into ```INSTALLED_APPS```
### IMPORTANT
If your site is **not** already running cross-origin you will need to update some settings and flush the sessions to ensure the primary domain and subdomain will work.
Because the API Gateway will run from a subdomain you need ensure the cookies are set-up to allow subdomains to read them.
Assuming your site runs from **www.example.com** and you wanted to use **ws.www.example.com** for websockets you would need to
set the below CSRF and COOKIE settings (Django 4 examples shown)
```
# CSRF
CSRF_COOKIE_SAMESITE=Lax
# CSRF_TRUSTED_ORIGINS=www.example.com,ws.www.example.com ## For Django 3*
## For Django 4+
CSRF_TRUSTED_ORIGINS=https://www.example.com,https://ws.www.example.com
CSRF_COOKIE_DOMAIN='.www.example.com'
# Sessions
SESSION_COOKIE_SAMESITE='Lax'
SESSION_COOKIE_NAME='mysessionid'
SESSION_COOKIE_DOMAIN='.www.example.com'
```
If you wanted to use **www.example.com** for the main site and **ws.example.com** for websockets you'd need to use
these settings
```
# CSRF
CSRF_COOKIE_SAMESITE=Lax
CSRF_TRUSTED_ORIGINS=https://www.example.com,https://ws.www.example.com
CSRF_COOKIE_DOMAIN='.example.com'
# Sessions
SESSION_COOKIE_SAMESITE='Lax'
SESSION_COOKIE_NAME='mysessionid'
SESSION_COOKIE_DOMAIN='.example.com'
```
**NOTE:** You need to rename the SESSION cookie. In the example I have renamed if from ```sessionid``` to ```mysessionid```. This will ensure that any old cookies are ignored.
### Flushing Sessions
Because you are changing the session cookie you will also need to flush any cached sessions using ```python manage.py clearsessions```.
## Clearing Stale Websocket connections
The websocket connections will become stale over time and some housekeeping is required. To help there is a management
command clearWebSocketSessions that can be run to delete the closed connections from the database. Simply run
```
python manage.py clearWebSocketSessions
```
I recommend setting this as a scheduled task.
# AWS Setup
In order for this package to create the API Gateway, it's routes, integration, custom domain and to publish messages
you will need to assign the correct permission to the IAM User/Role following best practices of restrictive permission.
If you are using a EC2/ECS then you should be using an IAM Role otherwise use a user.
This package **does not** include creating an AWS Certificate as you may already have one. You should create that
yourself. If you do not know how then see the [Appendix](#Appendix) section at the end of this file.
## IAM Policy
You'll need to grant the IAM permission to allow this project to create the API Gateway, create the domain mappings and
to execute the API to send messages from the server to the client(s).
I'm still reviewing the "minimum required permissions" but this project has been tested with the
following IAM policy which you can copy and paste into the JSON editor within the AWS console:
```
{
"Version": "2012-10-17",
"Statement": [
{
"Sid": "DjangoApiGatewayPolicy01",
"Effect": "Allow",
"Action": [
"apigateway:GET",
"apigateway:PATCH",
"apigateway:POST",
"apigateway:PUT",
"execute-api:*",
"iam:CreateServiceLinkedRole"
],
"Resource": [
"arn:aws:apigateway:*::/apis",
"arn:aws:apigateway:*::/apis/*",
"arn:aws:apigateway:*::/apis/*/authorizers",
"arn:aws:apigateway:*::/apis/*/authorizers/*",
"arn:aws:apigateway:*::/apis/*/cors",
"arn:aws:apigateway:*::/apis/*/deployments",
"arn:aws:apigateway:*::/apis/*/deployments/*",
"arn:aws:apigateway:*::/apis/*/exports/*",
"arn:aws:apigateway:*::/apis/*/integrations",
"arn:aws:apigateway:*::/apis/*/integrations/*",
"arn:aws:apigateway:*::/apis/*/integrations/*/integrationresponses",
"arn:aws:apigateway:*::/apis/*/integrations/*/integrationresponses/*",
"arn:aws:apigateway:*::/apis/*/models",
"arn:aws:apigateway:*::/apis/*/models/*",
"arn:aws:apigateway:*::/apis/*/models/*/template",
"arn:aws:apigateway:*::/apis/*/routes",
"arn:aws:apigateway:*::/apis/*/routes/*",
"arn:aws:apigateway:*::/apis/*/routes/*/requestparameters/*",
"arn:aws:apigateway:*::/apis/*/routes/*/routeresponses",
"arn:aws:apigateway:*::/apis/*/routes/*/routeresponses/*",
"arn:aws:apigateway:*::/apis/*/stages",
"arn:aws:apigateway:*::/apis/*/stages/*",
"arn:aws:apigateway:*::/apis/*/stages/*/accesslogsettings",
"arn:aws:apigateway:*::/apis/*/stages/*/cache/authorizers",
"arn:aws:apigateway:*::/apis/*/stages/*/routesettings/*",
"arn:aws:apigateway:{AWS-REGION-NAME}::/domainnames",
"arn:aws:apigateway:{AWS-REGION-NAME}::/domainnames/*/apimappings",
"arn:aws:apigateway:{AWS-REGION-NAME}::/domainnames/*/apimappings/*",
"arn:aws:execute-api:{AWS-REGION-NAME}:{AWS-ACCOUNT-NUMBER}:*/*/*/*",
"arn:aws:iam::{AWS-ACCOUNT-NUMBER}:role/aws-service-role/ops.apigateway.amazonaws.com/AWSServiceRoleForAPIGateway"
]
}
]
}
```
You will need to edit the permissio and replace the following:
1. ```{AWS-REGION-NANE}``` with the correct AWS region you are using, E.G ```eu-west-1```. If you wish to grant access to all regions then replace this placeholder with an ```*```
2. ```{AWS-ACCOUNT-NUMBER}``` with your account number E.G: 123456789101
This policy grants permissions to ensure the API Gateway(s) will be created, the custom domain name mapped to the
gateway and that you can send messages from the server to clients. The AWS Service role is required as it's used when
you create a custom domain name for API Gateway. If you do this via the console it will create the role for you so we
need to ensure the IAM user has the permission in order to replicate this.
Once you have created your API Gateway(s) you may wish to follow AWS best practice and restrict of revoke the
permissions to the API Gateway(s) you have created. Because I do not know what you will name your gateway, the
permissions above will allow you to add/edit and API gateway on your account.
# Getting Started
The core files within this project are:
1. ```django_aws_api_gateway_websockets.views.WebSocketView``` - The base class-based view from which you should extend
2. ```django_aws_api_gateway_websockets.models.ApiGateway``` - A model for managing the API Gateway. A Django Admin
page is included along with custom actions to create the API Gateway and configure a Custom Domain. For those with
projects not using Django Admin there are two management commands that perform the same actions.
3. ```django_aws_api_gateway_websockets.models.WebSocketSession``` - The websocket session store. Every connection
writes to this model which contains a method to send a message to the connection. The QuerySet of the objects model
manager has been extended to include a method to send messages to all records included within a queryset.
## Django
### URLS.py
Edit your urls.py file and add an entry for the URL you wish API Gateway to call. **IMPORTANT** The slug parameter
must be called "route". This will be populated by API Gateway with the route it uses E.G. $connect, $default or
$disconnect
E.G.
```
path("ws/<slug:route>", ExampleWebSocketView.as_view(), name="example_websocket")
```
### Creating the Views
Subclass the ```WebSocketView``` and implement methods where the name of the method is the name of the route the
API Gateway has been setup to use. There are already methods for $connect and $disconnect you just need to implement
a method for ```default``` along with any other custom routes you have created. The methods are selected dynamically
via the ```dispatch``` method with any leading dollar sign being remove.
The methods take the ```request``` parameter and only needs to return a response if you wish to return a negative HTTP
response such as a HttpResponseBadRequest otherwise there is no need to return anything.
```
from django_aws_api_gateway_websockets.views import WebSocketView
class ExampleWebSocketView(WebSocketView):
"""Custom Websocket view."""
def default(self, request, *args, **kwargs) -> JsonResponse:
"""Add the logic you wish to make here when you receive a message.
create your JSON response that you will handle within the Javascript
"""
logger.debug(f"body {self.body}")
```
If you want to send a response to the websocket that made the request then you need to call the ```send_message()``` on
the WebSocketSession that is being used. See the example below
```
from django_aws_api_gateway_websockets.views import WebSocketView
class ExampleWebSocketView(WebSocketView):
"""Custom Websocket view."""
def default(self, request, *args, **kwargs) -> JsonResponse:
# Do stuff
...
# Send a message back to the client - i.e unicast
self.websocket_session.send_message({"key1": "value1", "key2": "value2"})
```
If you are using the "channels" to group WebSocket connections together for multicasting, that is one-to-many
communication then you can use the following example
```
from django_aws_api_gateway_websockets.models import WebSocketSession
from django_aws_api_gateway_websockets.views import WebSocketView
class ExampleWebSocketView(WebSocketView):
"""Custom Websocket view."""
def default(self, request, *args, **kwargs) -> JsonResponse:
# Do stuff
...
# Multicast a message to ALL CONNECTED clients on the same "channel"
WebSocketSession.objects.filter(
channel_name=self.websocket_session.channel_name
).send_message({"key": "value})
# Multicast a message to ALL CONNECTED clients who are with the
# channel_nme "my-example-channel"
WebSocketSession.objects.filter(
channel_name=my-example-channel
).send_message(
{
"key": "value
}
)
```
#### Using the Route Selection Key value to call specific methods
API Gateway works by routing messages based on the "Route Selection Key". This project sets you up with a default route
so that you have a catch-all route but the Route Selection Key is preserved and is used by the dispatch method when
selecting the method to use to handle the request. This means that you can write individual methods to handle each
route individually for cleaner, more testable code.
In the example that follows the default Route Selection Key of **action** is being used.
Assume that two series of sends are made to the WebSocket. The first with the payload:
```{"action": "test", "value": "hello world"}``` and the second ```{"action": "help", "value": "Help Me"}```. You can
either handle these within the catch-all ```default``` method or you can write individual methods for each action
```
from django_aws_api_gateway_websockets.models import WebSocketSession
from django_aws_api_gateway_websockets.views import WebSocketView
class ExampleWebSocketView(WebSocketView):
"""Custom Websocket view."""
def test(self, request, *args, **kwargs) -> JsonResponse:
print(self.body.get("value")
# Prints "hello world"
def help(self, request, *args, **kwargs) -> JsonResponse:
print(self.body.get("value")
# Prints "Help Me"
```
**Remember**: The "action" key is the default ```route_selection_key```, if you chose to use a different one when
setting up the websocket make sure to update the ```route_selection_key``` class property to use the same value
### Debugging the View
Sometimes you the view may return a HTTP400 that you wish to debug further. In order to help with this you can pass
```debug=True``` into the ```as_view()``` method. The class will then call the private method ```_debug(msg)``` passing
in a string. By default this method will update a list property called ```debug_log``` with the message string but
you may wish to simply overload the method and call your logger.
E.G.
```
def _debug(self, msg: str):
if self.debug:
logger.debug(msg)
```
This can help track the issue which may be as simply as sending a message from the client that is missing the
```route_select_key```.
## Example of sending a message from the server to the client
To send a message to a specific connection simple load its ```WebSocketSession``` record and then call the
```send_message``` method passing in a JSON compatible dictionary of the payload you wish to send to the client.
### Sending a message to one connection
```python
from django_aws_api_gateway_websockets.models import WebSocketSession
obj = WebSocketSession.objects.get(pk=1)
obj.send_message({"type": "example", "msg": "This is a message"})
```
### Sending a message to ALL active connections associated with the same channel
```python
from django_aws_api_gateway_websockets.models import WebSocketSession
WebSocketSession.objects.filter(channel_name="Chatroom 1").send_message(
{"msg": "This is a a sample message"}
)
```
The ```WebSocketSessionQuerySet.send_message``` method automatically adds a filter of ```connected=True```
## Django Admin
Three Django Admin pages will be added to your project under the app _Django AWS APIGateway WebSockets_. Those pages
allow you to view and manage the three base models.
### Creating an API Gateway Endpoint
**Important** This section assumes that you are using an IAM account with the permissions listed earlier.
Using the Django Admin page create a new API Gateway record using the following for reference:
1. **API Name** - The human friendly API Name
2. **API Description** - Optional
3. **Default channel name** - Fill this in if you want all connections to this Websocket to also be associated with
the same "channel" otherwise leave it blank. "Channels" are groups of web socket connections and nothing more.
4. **Target base Endpoint** - This is the full URL path to the view you wish to use to handle the requests **excluding**
the ```route``` slug portion that will be automatically appended.
5. **Certificate ARN** - You'll need to manually create certificate within AWS. Once you have, copy the ARN into this field
6. **Hosted Zone ID** - If you use Route53 then you'll need to enter the Hosted Zone ID here if you wish to use a custom
domain name with the API Gateway Endpoint
7. **API Key Selection Expression** - In most cases leave this as the default value. See the AWS docs for more
8. **Route selection expression** - As per the above. This is the field that maps the "action" key within the payload
as being the key to determine the route to take. If you change this then you must overload the
```route_selection_key``` of the view
9. **Route key** - This is the default root key. In most cases you will not need to change this.
10. **Stage Name** - The name you wish to give to the staging. Currently this package does not support multiple stages.
If you leave it blank it will default to "production"
11. **Stage description" - Optional
12. **Tags** - Currently not implement but these will be used to create the tags with AWS
13. **API ID** - This will be populated when the API is created.
14. **API Endpoint** - This will be populated when the API is created.
15. **API Gateway Domain Name** - This will be populated when you run the Custom Domain setup. The value that appears
here is the value to which you should your DNS CNAME entry should point.
16. **API Mapping ID** - This will be populated when the API is created.
Once you have created the record within the database simply select it from the Django Admin list view, choose
**Create API Gateway** action from the actions list and click Go. The API Gateway record will be created within your
account. When it's ready the "API Created" column will show as True.
Once the API has been created you can now add a custom domain name mapping by choosing the row again and this time
selecting the **Create Custom Domain record for the API**. This will create the Custom Domain record and will associate
it with the stage name you entered earlier. Once it's completed the **Custom Domain Created** flag will be set as True.
At this point you can open the record where you'll find that the ```API Gateway Domain Name``` has been populated.
## Django Management Commands
If you are not using Django Admin then you can populate the apigateway database table manually using the same list
as shown above.
Once you've populated those fields you can then run the two actions as management commands rather than via Django Admin.
```
python manage.py createApiGateway --pk=1
```
```
python manage.py createCustomDomain --pk=1
```
The same actions will run as above.
## Adding Additional Routes
This project will route all requests to one URL by default however that is not always what you will need,
sometimes you will want to route requests to different URLs to send requests to different views potentially within
different apps. API Gateway support this by using _"integrations"_ and _"routes"_ that use unique "route keys" to
identify requests and then route those requests to a URL.
To support this approach this project uses the ```ApiGatewayAdditionalRoute``` model entries to map a route key to a chosen URL.
They are available to manage as inline forms on the main ```ApiGateway``` admin or as their own admin page.
The models are set-up to auto-deploy the new route if the ```ApiGateway``` has already been deployed. If it has not been
deployed already then these routes will be setup during the main deployment.
From the client side you choose the route you wish to take by setting the value of the **action** to the route key you
wish to match.
## Gotchas and debugging
### Failure to connect to websockets
#### Differing required headers
The most common reasons for the websocket failing to connect is due to different required headers. The base view is
set-up with two lists of expected headers. ```required_headers``` and ```additional_required_headers```. If you are
deploying to an EC2 server then you shouldn't have to change these but if you are deploying else where or are testing
locally you may find that you need to change some of these. During development of this library I was using an
[NGROK](https://ngrok.com/) network edge tunnel and found that the "X-Real-Ip" and "Connection" headers were being lost
during which is why they were moved to the additional_required_headers. If you find this is the case for you then simply
overload the class property and set it to an empty list.
# Client Side Integration (Javascript)
This section will guide you through two common ways of connecting to and using this project from a webpage.
### Basic Integration
Below is a very basic integration using the WebSockets API built into browsers. It does not handle reconnecting dropped
websockets, see the next section for that.
**WARNING**: This method will create a WebSocket that will timeout after around 10 minutes.
The below example assumes you created the API Gateway to work on the custom domain name ws.example.com
```javascript
let wss_url = 'wss://ws.example.com';
let regDeskWSocket = new WebSocket(wss_url);
regDeskWSocket.onmessage = function(event) {
// Take your action here to handle messages being received
console.log(event);
let msg = JSON.parse(event.data);
console.log(msg);
};
```
You can set the channel by using the **channel** querystring parameter during the connection
```javascript
let wss_url = 'wss://ws.example.com?channel=my+example+channel';
let exampleWS = new WebSocket(wss_url);
exampleWS.onmessage = function(event) {
// Take your action here to handle messages being received
console.log(event);
let msg = JSON.parse(event.data);
console.log(msg);
};
```
### Reconnecting WebSockets
Websockets can disconnect due top a variety of reasons to work around this here are some links to libraries of proposed
solutions
1. [Stack Overflow- WebSocket: How to automatically reconnect after it dies](https://stackoverflow.com/questions/22431751/websocket-how-to-automatically-reconnect-after-it-dies)
2. [JS library - reconnecting-websocket](https://github.com/joewalnes/reconnecting-websocket)
The below example is using the JS library. Note you just include the lib and then use the
```ReconnectingWebSocket``` class rather than ```WebSocket```:
```html
<script src="https://cdnjs.cloudflare.com/ajax/libs/reconnecting-websocket/1.0.0/reconnecting-websocket.min.js" integrity="sha512-B4skI5FiLurS86aioJx9VfozI1wjqrn6aTdJH+YQUmCZum/ZibPBTX55k5d9XM6EsKePDInkLVrN7vPmJxc1qA==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
<script>
let wss_url = 'wss://ws.example.com';
let exampleWS = new ReconnectingWebSocket(wss_url);
exampleWS.onmessage = function(event) {
// Take your action here to handle messages being received
console.log(event);
let msg = JSON.parse(event.data);
console.log(msg);
};
</script>
```
### Sending a message from the client to the server
Both the example above use the same method.
```javascript
let wss_url = 'wss://ws.example.com?channel=my+example+channel';
let exampleWS = new WebSocket(wss_url); // Or use ReconnectingWebSocket it does not matter
// Send a message
exampleWS.send(JSON.stringify({"action": "custom", "message": "What is this"}))
```
**IMPORTANT** The value of ```action``` determines the route that is used by **API Gateway**. By default, the only
routes that are set-up are ```$connect```, ```$disconnect``` and ```default```. Any messages sent to unknown routes on
the API Gateway are delivered to the ```default``` route. So if you created a custom route called ```bob``` and then
sent the following message from the client:
```
exampleWS.send(JSON.stringify({"action": "bob", "message": "What is this"}))
```
API Gateway will route this to the endpoint set for the "bob" route. This will be calling your view with the route slugs
value being assigned to **bob**. The ```dispatch``` method of the view will then look for a method on the class called
```bob```. If one is found then it will be invoked otherwise the ```default``` method will be called.
# Appendix
## Creating an SSL Certificate within AWS Certificate Manager
Please refer to the [official AWS documentation](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html)
# Found a Bug?
Issues are tracked via GitHub issues at the [project issue page](https://github.com/StevenMapes/django-aws-api-gateway-websockets/issues)
# Have A Feature Request?
Feature requests can be raised by creating an issue within the [project issue page](https://github.com/StevenMapes/django-aws-api-gateway-websockets/issues), but please create the issue with "Feature Request -" at the start of the issue
# Testing
To run the tests use
```
coverage erase && \
python -W error::DeprecationWarning -W error::PendingDeprecationWarning -m coverage run --parallel -m pytest --ds tests.settings && \
coverage combine && \
coverage report
```
# Compiling Requirements
Run ```pip install pip-tools``` then run ```python requirements/compile.py``` to generate the various requirements files.
I use two local VIRTUALENVS to build the requirements, one running Python3.8 and the other running Python 3.11.
Also require ```pytest-django``` for testing
# tox
# Contributing
- [Check for open issues](https://github.com/StevenMapes/django-aws-api-gateway-websockets/issues) at the project issue page or open a new issue to start a discussion about a feature or bug.
- Fork the [repository on GitHub](https://github.com/StevenMapes/django-aws-api-gateway-websockets) to start making changes.
- Clone the repository
- Initialise pre-commit by running ```pre-commit install```
- Install requirements from one of the requirement files depending on the versions of Python and Django you wish to use.
- Add a test case to show that the bug is fixed or the feature is implemented correctly.
- Test using ```python -W error::DeprecationWarning -W error::PendingDeprecationWarning -m coverage run --parallel -m pytest --ds tests.settings```
- Create a pull request, tagging the issue, bug me until I can merge your pull request. Also, don't forget to add yourself to AUTHORS.
Raw data
{
"_id": null,
"home_page": "https://github.com/StevenMapes/django-aws-api-gateway-websockets",
"name": "django-aws-api-gateway-websockets",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.8",
"maintainer_email": null,
"keywords": "Django, WebSockets, AWS, API Gateway, Amazon WebServices",
"author": "Steven Mapes",
"author_email": "Steven Mapes <steve@stevenmapes.com>",
"download_url": "https://files.pythonhosted.org/packages/0c/9d/293f104337cf8bb6a34bc4ff68f210677e227ccbeae66a0cde4816a17fd1/django_aws_api_gateway_websockets-1.1.1.tar.gz",
"platform": null,
"description": "[![CI Build Status](https://img.shields.io/github/actions/workflow/status/StevenMapes/django-aws-api-gateway-websockets/main.yml?branch=main&style=for-the-badge)](https://github.com/StevenMapes/django-aws-api-gateway-websockets/actions)\n[![Coverage](https://img.shields.io/badge/Coverage-96%25-success?style=for-the-badge)](https://github.com/StevenMapes/django-aws-api-gateway-websockets/actions?workclow=CI)\n[![PyPi](https://img.shields.io/pypi/v/django-aws-api-gateway-websockets.svg?style=for-the-badge)](https://pypi.org/project/django-aws-api-gateway-websockets/)\n![Code Style](https://img.shields.io/badge/code%20style-black-000000.svg?style=for-the-badge)\n![Pre-Commit Enabled](https://img.shields.io/badge/pre--commit-enabled-brightgreen?logo=pre-commit&logoColor=white&style=for-the-badge)\n[![Read The Docs](https://img.shields.io/readthedocs/django-mysql?style=for-the-badge)](https://django-aws-api-gateway-websockets.readthedocs.io/)\n\n[![PyPi Downloads](https://img.shields.io/pypi/dd/django-aws-api-gateway-websockets)](https://pypistats.org/packages/django-aws-api-gateway-websockets)\n\n# Django AWS API Gateway Websockets\nIt is the aim of this project to create a uniform way to record websocket connections, associate the Django user who established the connection and then retrieve that user within each request.\n\nThis project is designed to work exclusively with AWS API Gateway.\n\nIt is not intended to be a replacement of [Django Channels](https://github.com/django/channels) instead this project allows you to add [WebSockets](https://en.wikipedia.org/wiki/WebSocket) support into your project by writing normal HTTP request-response views whilst allowing [AWS API Gateway](https://aws.amazon.com/api-gateway/) to worry about the WebSocket connection.\n\nThis project introduced a new [Class-Based-View](https://docs.djangoproject.com/en/dev/topics/class-based-views/) to handle connections, disconnections, routing, basic security checks and ensuring that the User object is available within every request.\n\nThe project will keep track of which users created which WebSockets, which ones are active and will allow you to send messages back down the socket to the client via Boto3.\n\nPlease refer to the installation notes and Getting Start Guides.\n\n# Security Concerns\n**IMPORTANT:**: In order to work the dispatch method requires the ```csrf_exempt``` decorator to be added. This has\nalready been added as a class decorator on the base view but if you overload the dispatch method you will need to add\nit back to avoid receiving CSRF Token failures.\n\n# Python and Django Support\nThis project supports Python 3.8 to 3.14 and Django 4.2 to 5.2. It may work with older versions of Django, but I am not\nactively supporting anything below 4.2.\n\n| **Python/Django** | **4.2** | **5.0** | **5.1** | **5.2** |\n|--------------------|----------|----------|---------|----------|\n| 3.8 | Y | N | N/A | N/A | \n| 3.9 | Y | N | N/A | N/A | \n| 3.10 | Y | Y | Y | Y |\n| 3.11 | Y | Y | Y | Y |\n| 3.12 | Y | Y | Y | Y |\n| 3.13 | Y | Y | Y | Y |\n| 3.14 | N | N | Y | Y |\n\n* Python 3.11 only works with Django 4.1.3+\n\n# Installation\nYou can install this package from pip using\n```\npip install django-aws-api-gateway-websockets\n```\n\n## settings.py\nAdd ```django_aws_api_gateway_websockets``` into ```INSTALLED_APPS``` \n\n### IMPORTANT\nIf your site is **not** already running cross-origin you will need to update some settings and flush the sessions to ensure the primary domain and subdomain will work.\n\nBecause the API Gateway will run from a subdomain you need ensure the cookies are set-up to allow subdomains to read them.\nAssuming your site runs from **www.example.com** and you wanted to use **ws.www.example.com** for websockets you would need to \nset the below CSRF and COOKIE settings (Django 4 examples shown)\n```\n# CSRF\nCSRF_COOKIE_SAMESITE=Lax\n# CSRF_TRUSTED_ORIGINS=www.example.com,ws.www.example.com ## For Django 3*\n## For Django 4+\nCSRF_TRUSTED_ORIGINS=https://www.example.com,https://ws.www.example.com\nCSRF_COOKIE_DOMAIN='.www.example.com'\n\n# Sessions\nSESSION_COOKIE_SAMESITE='Lax'\nSESSION_COOKIE_NAME='mysessionid'\nSESSION_COOKIE_DOMAIN='.www.example.com'\n```\n\nIf you wanted to use **www.example.com** for the main site and **ws.example.com** for websockets you'd need to use \nthese settings\n\n```\n# CSRF\nCSRF_COOKIE_SAMESITE=Lax\nCSRF_TRUSTED_ORIGINS=https://www.example.com,https://ws.www.example.com\nCSRF_COOKIE_DOMAIN='.example.com'\n\n# Sessions\nSESSION_COOKIE_SAMESITE='Lax'\nSESSION_COOKIE_NAME='mysessionid'\nSESSION_COOKIE_DOMAIN='.example.com'\n```\n\n**NOTE:** You need to rename the SESSION cookie. In the example I have renamed if from ```sessionid``` to ```mysessionid```. This will ensure that any old cookies are ignored.\n\n### Flushing Sessions\nBecause you are changing the session cookie you will also need to flush any cached sessions using ```python manage.py clearsessions```.\n\n## Clearing Stale Websocket connections\nThe websocket connections will become stale over time and some housekeeping is required. To help there is a management\ncommand clearWebSocketSessions that can be run to delete the closed connections from the database. Simply run\n```\npython manage.py clearWebSocketSessions\n``` \nI recommend setting this as a scheduled task.\n\n# AWS Setup\nIn order for this package to create the API Gateway, it's routes, integration, custom domain and to publish messages\nyou will need to assign the correct permission to the IAM User/Role following best practices of restrictive permission.\n\nIf you are using a EC2/ECS then you should be using an IAM Role otherwise use a user.\n\nThis package **does not** include creating an AWS Certificate as you may already have one. You should create that \nyourself. If you do not know how then see the [Appendix](#Appendix) section at the end of this file.\n\n## IAM Policy\nYou'll need to grant the IAM permission to allow this project to create the API Gateway, create the domain mappings and\nto execute the API to send messages from the server to the client(s).\n\nI'm still reviewing the \"minimum required permissions\" but this project has been tested with the \nfollowing IAM policy which you can copy and paste into the JSON editor within the AWS console:\n\n```\n{\n \"Version\": \"2012-10-17\",\n \"Statement\": [\n {\n \"Sid\": \"DjangoApiGatewayPolicy01\",\n \"Effect\": \"Allow\",\n \"Action\": [\n \"apigateway:GET\",\n \"apigateway:PATCH\",\n \"apigateway:POST\",\n \"apigateway:PUT\",\n \"execute-api:*\",\n \"iam:CreateServiceLinkedRole\"\n ],\n \"Resource\": [\n \"arn:aws:apigateway:*::/apis\",\n \"arn:aws:apigateway:*::/apis/*\",\n \"arn:aws:apigateway:*::/apis/*/authorizers\",\n \"arn:aws:apigateway:*::/apis/*/authorizers/*\",\n \"arn:aws:apigateway:*::/apis/*/cors\",\n \"arn:aws:apigateway:*::/apis/*/deployments\",\n \"arn:aws:apigateway:*::/apis/*/deployments/*\",\n \"arn:aws:apigateway:*::/apis/*/exports/*\",\n \"arn:aws:apigateway:*::/apis/*/integrations\",\n \"arn:aws:apigateway:*::/apis/*/integrations/*\",\n \"arn:aws:apigateway:*::/apis/*/integrations/*/integrationresponses\",\n \"arn:aws:apigateway:*::/apis/*/integrations/*/integrationresponses/*\",\n \"arn:aws:apigateway:*::/apis/*/models\",\n \"arn:aws:apigateway:*::/apis/*/models/*\",\n \"arn:aws:apigateway:*::/apis/*/models/*/template\",\n \"arn:aws:apigateway:*::/apis/*/routes\",\n \"arn:aws:apigateway:*::/apis/*/routes/*\",\n \"arn:aws:apigateway:*::/apis/*/routes/*/requestparameters/*\",\n \"arn:aws:apigateway:*::/apis/*/routes/*/routeresponses\",\n \"arn:aws:apigateway:*::/apis/*/routes/*/routeresponses/*\",\n \"arn:aws:apigateway:*::/apis/*/stages\",\n \"arn:aws:apigateway:*::/apis/*/stages/*\",\n \"arn:aws:apigateway:*::/apis/*/stages/*/accesslogsettings\",\n \"arn:aws:apigateway:*::/apis/*/stages/*/cache/authorizers\",\n \"arn:aws:apigateway:*::/apis/*/stages/*/routesettings/*\",\n \"arn:aws:apigateway:{AWS-REGION-NAME}::/domainnames\",\n \"arn:aws:apigateway:{AWS-REGION-NAME}::/domainnames/*/apimappings\",\n \"arn:aws:apigateway:{AWS-REGION-NAME}::/domainnames/*/apimappings/*\",\n \"arn:aws:execute-api:{AWS-REGION-NAME}:{AWS-ACCOUNT-NUMBER}:*/*/*/*\",\n \"arn:aws:iam::{AWS-ACCOUNT-NUMBER}:role/aws-service-role/ops.apigateway.amazonaws.com/AWSServiceRoleForAPIGateway\"\n ]\n }\n ]\n}\n```\n\nYou will need to edit the permissio and replace the following: \n1. ```{AWS-REGION-NANE}``` with the correct AWS region you are using, E.G ```eu-west-1```. If you wish to grant access to all regions then replace this placeholder with an ```*```\n2. ```{AWS-ACCOUNT-NUMBER}``` with your account number E.G: 123456789101 \n\nThis policy grants permissions to ensure the API Gateway(s) will be created, the custom domain name mapped to the \ngateway and that you can send messages from the server to clients. The AWS Service role is required as it's used when \nyou create a custom domain name for API Gateway. If you do this via the console it will create the role for you so we\nneed to ensure the IAM user has the permission in order to replicate this.\n\nOnce you have created your API Gateway(s) you may wish to follow AWS best practice and restrict of revoke the \npermissions to the API Gateway(s) you have created. Because I do not know what you will name your gateway, the \npermissions above will allow you to add/edit and API gateway on your account.\n\n# Getting Started\n\nThe core files within this project are:\n\n1. ```django_aws_api_gateway_websockets.views.WebSocketView``` - The base class-based view from which you should extend\n2. ```django_aws_api_gateway_websockets.models.ApiGateway``` - A model for managing the API Gateway. A Django Admin \npage is included along with custom actions to create the API Gateway and configure a Custom Domain. For those with\nprojects not using Django Admin there are two management commands that perform the same actions. \n3. ```django_aws_api_gateway_websockets.models.WebSocketSession``` - The websocket session store. Every connection \nwrites to this model which contains a method to send a message to the connection. The QuerySet of the objects model \nmanager has been extended to include a method to send messages to all records included within a queryset. \n\n## Django\n### URLS.py\nEdit your urls.py file and add an entry for the URL you wish API Gateway to call. **IMPORTANT** The slug parameter \nmust be called \"route\". This will be populated by API Gateway with the route it uses E.G. $connect, $default or \n$disconnect\n\nE.G. \n```\npath(\"ws/<slug:route>\", ExampleWebSocketView.as_view(), name=\"example_websocket\")\n```\n\n### Creating the Views\nSubclass the ```WebSocketView``` and implement methods where the name of the method is the name of the route the \nAPI Gateway has been setup to use. There are already methods for $connect and $disconnect you just need to implement\na method for ```default``` along with any other custom routes you have created. The methods are selected dynamically\nvia the ```dispatch``` method with any leading dollar sign being remove.\n\nThe methods take the ```request``` parameter and only needs to return a response if you wish to return a negative HTTP\nresponse such as a HttpResponseBadRequest otherwise there is no need to return anything.\n\n```\nfrom django_aws_api_gateway_websockets.views import WebSocketView\n\nclass ExampleWebSocketView(WebSocketView):\n \"\"\"Custom Websocket view.\"\"\"\n\n def default(self, request, *args, **kwargs) -> JsonResponse:\n \"\"\"Add the logic you wish to make here when you receive a message.\n create your JSON response that you will handle within the Javascript\n \"\"\"\n logger.debug(f\"body {self.body}\")\n\n```\n\nIf you want to send a response to the websocket that made the request then you need to call the ```send_message()``` on\nthe WebSocketSession that is being used. See the example below\n\n```\nfrom django_aws_api_gateway_websockets.views import WebSocketView\n\nclass ExampleWebSocketView(WebSocketView):\n \"\"\"Custom Websocket view.\"\"\"\n\n def default(self, request, *args, **kwargs) -> JsonResponse:\n # Do stuff \n ... \n \n # Send a message back to the client - i.e unicast\n self.websocket_session.send_message({\"key1\": \"value1\", \"key2\": \"value2\"})\n\n```\n\nIf you are using the \"channels\" to group WebSocket connections together for multicasting, that is one-to-many \ncommunication then you can use the following example\n\n```\nfrom django_aws_api_gateway_websockets.models import WebSocketSession\nfrom django_aws_api_gateway_websockets.views import WebSocketView\n\n\nclass ExampleWebSocketView(WebSocketView):\n \"\"\"Custom Websocket view.\"\"\"\n\n def default(self, request, *args, **kwargs) -> JsonResponse:\n # Do stuff \n ... \n \n # Multicast a message to ALL CONNECTED clients on the same \"channel\"\n WebSocketSession.objects.filter(\n channel_name=self.websocket_session.channel_name\n ).send_message({\"key\": \"value})\n\n # Multicast a message to ALL CONNECTED clients who are with the \n # channel_nme \"my-example-channel\"\n WebSocketSession.objects.filter(\n channel_name=my-example-channel\n ).send_message(\n {\n \"key\": \"value\n }\n )\n\n```\n\n#### Using the Route Selection Key value to call specific methods\nAPI Gateway works by routing messages based on the \"Route Selection Key\". This project sets you up with a default route\nso that you have a catch-all route but the Route Selection Key is preserved and is used by the dispatch method when \nselecting the method to use to handle the request. This means that you can write individual methods to handle each \nroute individually for cleaner, more testable code.\n\nIn the example that follows the default Route Selection Key of **action** is being used. \nAssume that two series of sends are made to the WebSocket. The first with the payload: \n```{\"action\": \"test\", \"value\": \"hello world\"}``` and the second ```{\"action\": \"help\", \"value\": \"Help Me\"}```. You can\neither handle these within the catch-all ```default``` method or you can write individual methods for each action\n\n```\nfrom django_aws_api_gateway_websockets.models import WebSocketSession\nfrom django_aws_api_gateway_websockets.views import WebSocketView\n\n\nclass ExampleWebSocketView(WebSocketView):\n \"\"\"Custom Websocket view.\"\"\"\n\n def test(self, request, *args, **kwargs) -> JsonResponse:\n print(self.body.get(\"value\")\n # Prints \"hello world\" \n\n def help(self, request, *args, **kwargs) -> JsonResponse:\n print(self.body.get(\"value\")\n # Prints \"Help Me\" \n```\n\n**Remember**: The \"action\" key is the default ```route_selection_key```, if you chose to use a different one when \nsetting up the websocket make sure to update the ```route_selection_key``` class property to use the same value\n\n\n### Debugging the View\nSometimes you the view may return a HTTP400 that you wish to debug further. In order to help with this you can pass\n```debug=True``` into the ```as_view()``` method. The class will then call the private method ```_debug(msg)``` passing\nin a string. By default this method will update a list property called ```debug_log``` with the message string but \nyou may wish to simply overload the method and call your logger.\n\nE.G.\n```\ndef _debug(self, msg: str):\n if self.debug:\n logger.debug(msg)\n```\n\nThis can help track the issue which may be as simply as sending a message from the client that is missing the \n```route_select_key```.\n\n## Example of sending a message from the server to the client\nTo send a message to a specific connection simple load its ```WebSocketSession``` record and then call the \n```send_message``` method passing in a JSON compatible dictionary of the payload you wish to send to the client.\n\n### Sending a message to one connection\n```python\nfrom django_aws_api_gateway_websockets.models import WebSocketSession\n\nobj = WebSocketSession.objects.get(pk=1)\nobj.send_message({\"type\": \"example\", \"msg\": \"This is a message\"})\n```\n\n### Sending a message to ALL active connections associated with the same channel \n```python\nfrom django_aws_api_gateway_websockets.models import WebSocketSession\n\nWebSocketSession.objects.filter(channel_name=\"Chatroom 1\").send_message(\n {\"msg\": \"This is a a sample message\"}\n)\n```\n\nThe ```WebSocketSessionQuerySet.send_message``` method automatically adds a filter of ```connected=True``` \n\n## Django Admin\nThree Django Admin pages will be added to your project under the app _Django AWS APIGateway WebSockets_. Those pages \nallow you to view and manage the three base models.\n\n### Creating an API Gateway Endpoint\n**Important** This section assumes that you are using an IAM account with the permissions listed earlier.\n\nUsing the Django Admin page create a new API Gateway record using the following for reference:\n\n1. **API Name** - The human friendly API Name\n2. **API Description** - Optional\n3. **Default channel name** - Fill this in if you want all connections to this Websocket to also be associated with \nthe same \"channel\" otherwise leave it blank. \"Channels\" are groups of web socket connections and nothing more.\n4. **Target base Endpoint** - This is the full URL path to the view you wish to use to handle the requests **excluding**\nthe ```route``` slug portion that will be automatically appended.\n5. **Certificate ARN** - You'll need to manually create certificate within AWS. Once you have, copy the ARN into this field\n6. **Hosted Zone ID** - If you use Route53 then you'll need to enter the Hosted Zone ID here if you wish to use a custom\ndomain name with the API Gateway Endpoint\n7. **API Key Selection Expression** - In most cases leave this as the default value. See the AWS docs for more\n8. **Route selection expression** - As per the above. This is the field that maps the \"action\" key within the payload \nas being the key to determine the route to take. If you change this then you must overload the \n```route_selection_key``` of the view\n9. **Route key** - This is the default root key. In most cases you will not need to change this.\n10. **Stage Name** - The name you wish to give to the staging. Currently this package does not support multiple stages.\nIf you leave it blank it will default to \"production\"\n11. **Stage description\" - Optional\n12. **Tags** - Currently not implement but these will be used to create the tags with AWS\n13. **API ID** - This will be populated when the API is created.\n14. **API Endpoint** - This will be populated when the API is created.\n15. **API Gateway Domain Name** - This will be populated when you run the Custom Domain setup. The value that appears \nhere is the value to which you should your DNS CNAME entry should point. \n16. **API Mapping ID** - This will be populated when the API is created.\n\nOnce you have created the record within the database simply select it from the Django Admin list view, choose \n**Create API Gateway** action from the actions list and click Go. The API Gateway record will be created within your\naccount. When it's ready the \"API Created\" column will show as True.\n\nOnce the API has been created you can now add a custom domain name mapping by choosing the row again and this time \nselecting the **Create Custom Domain record for the API**. This will create the Custom Domain record and will associate\nit with the stage name you entered earlier. Once it's completed the **Custom Domain Created** flag will be set as True.\n\nAt this point you can open the record where you'll find that the ```API Gateway Domain Name``` has been populated.\n\n## Django Management Commands\nIf you are not using Django Admin then you can populate the apigateway database table manually using the same list \nas shown above.\n\nOnce you've populated those fields you can then run the two actions as management commands rather than via Django Admin.\n\n```\npython manage.py createApiGateway --pk=1\n```\n\n```\npython manage.py createCustomDomain --pk=1\n```\n\nThe same actions will run as above.\n\n## Adding Additional Routes\nThis project will route all requests to one URL by default however that is not always what you will need, \nsometimes you will want to route requests to different URLs to send requests to different views potentially within \ndifferent apps. API Gateway support this by using _\"integrations\"_ and _\"routes\"_ that use unique \"route keys\" to \nidentify requests and then route those requests to a URL.\n\nTo support this approach this project uses the ```ApiGatewayAdditionalRoute``` model entries to map a route key to a chosen URL.\nThey are available to manage as inline forms on the main ```ApiGateway``` admin or as their own admin page.\n\nThe models are set-up to auto-deploy the new route if the ```ApiGateway``` has already been deployed. If it has not been\ndeployed already then these routes will be setup during the main deployment.\n\nFrom the client side you choose the route you wish to take by setting the value of the **action** to the route key you\nwish to match.\n\n\n## Gotchas and debugging\n### Failure to connect to websockets\n#### Differing required headers\nThe most common reasons for the websocket failing to connect is due to different required headers. The base view is \nset-up with two lists of expected headers. ```required_headers``` and ```additional_required_headers```. If you are \ndeploying to an EC2 server then you shouldn't have to change these but if you are deploying else where or are testing\nlocally you may find that you need to change some of these. During development of this library I was using an\n[NGROK](https://ngrok.com/) network edge tunnel and found that the \"X-Real-Ip\" and \"Connection\" headers were being lost\nduring which is why they were moved to the additional_required_headers. If you find this is the case for you then simply\noverload the class property and set it to an empty list.\n\n\n# Client Side Integration (Javascript)\nThis section will guide you through two common ways of connecting to and using this project from a webpage.\n\n### Basic Integration\nBelow is a very basic integration using the WebSockets API built into browsers. It does not handle reconnecting dropped\nwebsockets, see the next section for that.\n\n**WARNING**: This method will create a WebSocket that will timeout after around 10 minutes. \n\nThe below example assumes you created the API Gateway to work on the custom domain name ws.example.com\n```javascript\nlet wss_url = 'wss://ws.example.com';\nlet regDeskWSocket = new WebSocket(wss_url);\nregDeskWSocket.onmessage = function(event) {\n // Take your action here to handle messages being received\n console.log(event);\n let msg = JSON.parse(event.data);\n console.log(msg);\n};\n```\n\nYou can set the channel by using the **channel** querystring parameter during the connection \n\n```javascript\nlet wss_url = 'wss://ws.example.com?channel=my+example+channel';\nlet exampleWS = new WebSocket(wss_url);\nexampleWS.onmessage = function(event) {\n // Take your action here to handle messages being received\n console.log(event);\n let msg = JSON.parse(event.data);\n console.log(msg);\n};\n```\n\n### Reconnecting WebSockets\nWebsockets can disconnect due top a variety of reasons to work around this here are some links to libraries of proposed\nsolutions\n\n1. [Stack Overflow- WebSocket: How to automatically reconnect after it dies](https://stackoverflow.com/questions/22431751/websocket-how-to-automatically-reconnect-after-it-dies)\n2. [JS library - reconnecting-websocket](https://github.com/joewalnes/reconnecting-websocket)\n\nThe below example is using the JS library. Note you just include the lib and then use the \n```ReconnectingWebSocket``` class rather than ```WebSocket```:\n\n```html\n<script src=\"https://cdnjs.cloudflare.com/ajax/libs/reconnecting-websocket/1.0.0/reconnecting-websocket.min.js\" integrity=\"sha512-B4skI5FiLurS86aioJx9VfozI1wjqrn6aTdJH+YQUmCZum/ZibPBTX55k5d9XM6EsKePDInkLVrN7vPmJxc1qA==\" crossorigin=\"anonymous\" referrerpolicy=\"no-referrer\"></script>\n<script>\nlet wss_url = 'wss://ws.example.com';\nlet exampleWS = new ReconnectingWebSocket(wss_url);\nexampleWS.onmessage = function(event) {\n // Take your action here to handle messages being received\n console.log(event);\n let msg = JSON.parse(event.data);\n console.log(msg);\n};\n</script>\n```\n\n### Sending a message from the client to the server\nBoth the example above use the same method.\n\n```javascript\nlet wss_url = 'wss://ws.example.com?channel=my+example+channel';\nlet exampleWS = new WebSocket(wss_url); // Or use ReconnectingWebSocket it does not matter\n\n// Send a message\nexampleWS.send(JSON.stringify({\"action\": \"custom\", \"message\": \"What is this\"}))\n```\n\n**IMPORTANT** The value of ```action``` determines the route that is used by **API Gateway**. By default, the only \nroutes that are set-up are ```$connect```, ```$disconnect``` and ```default```. Any messages sent to unknown routes on \nthe API Gateway are delivered to the ```default``` route. So if you created a custom route called ```bob``` and then \nsent the following message from the client:\n\n```\nexampleWS.send(JSON.stringify({\"action\": \"bob\", \"message\": \"What is this\"}))\n```\n\nAPI Gateway will route this to the endpoint set for the \"bob\" route. This will be calling your view with the route slugs\nvalue being assigned to **bob**. The ```dispatch``` method of the view will then look for a method on the class called\n```bob```. If one is found then it will be invoked otherwise the ```default``` method will be called.\n\n# Appendix\n\n## Creating an SSL Certificate within AWS Certificate Manager\nPlease refer to the [official AWS documentation](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html)\n\n# Found a Bug?\nIssues are tracked via GitHub issues at the [project issue page](https://github.com/StevenMapes/django-aws-api-gateway-websockets/issues)\n\n# Have A Feature Request?\nFeature requests can be raised by creating an issue within the [project issue page](https://github.com/StevenMapes/django-aws-api-gateway-websockets/issues), but please create the issue with \"Feature Request -\" at the start of the issue\n\n# Testing\nTo run the tests use\n\n```\ncoverage erase && \\\npython -W error::DeprecationWarning -W error::PendingDeprecationWarning -m coverage run --parallel -m pytest --ds tests.settings && \\\ncoverage combine && \\\ncoverage report\n```\n\n# Compiling Requirements\nRun ```pip install pip-tools``` then run ```python requirements/compile.py``` to generate the various requirements files.\nI use two local VIRTUALENVS to build the requirements, one running Python3.8 and the other running Python 3.11.\n\nAlso require ```pytest-django``` for testing\n\n# tox\n\n# Contributing\n- [Check for open issues](https://github.com/StevenMapes/django-aws-api-gateway-websockets/issues) at the project issue page or open a new issue to start a discussion about a feature or bug.\n- Fork the [repository on GitHub](https://github.com/StevenMapes/django-aws-api-gateway-websockets) to start making changes.\n- Clone the repository\n- Initialise pre-commit by running ```pre-commit install```\n- Install requirements from one of the requirement files depending on the versions of Python and Django you wish to use.\n- Add a test case to show that the bug is fixed or the feature is implemented correctly.\n- Test using ```python -W error::DeprecationWarning -W error::PendingDeprecationWarning -m coverage run --parallel -m pytest --ds tests.settings```\n- Create a pull request, tagging the issue, bug me until I can merge your pull request. Also, don't forget to add yourself to AUTHORS.\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "Created to allow Django projects to be used as a HTTP backend for AWS API Gateway websockets",
"version": "1.1.1",
"project_urls": {
"Change log": "https://github.com/StevenMapes/django-aws-api-gateway-websockets/blob/main/CHANGELOG.md",
"Homepage": "https://github.com/StevenMapes/django-aws-api-gateway-websockets",
"Twitter": "https://twitter.com/stevenamapes"
},
"split_keywords": [
"django",
" websockets",
" aws",
" api gateway",
" amazon webservices"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "0c9d293f104337cf8bb6a34bc4ff68f210677e227ccbeae66a0cde4816a17fd1",
"md5": "a4048baed22acfafd19aaa7c6fbc9435",
"sha256": "4b048b9bdcf3fb9bc22464548dd6b60e72320e868dc7575391b40444a127cf92"
},
"downloads": -1,
"filename": "django_aws_api_gateway_websockets-1.1.1.tar.gz",
"has_sig": false,
"md5_digest": "a4048baed22acfafd19aaa7c6fbc9435",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.8",
"size": 42731,
"upload_time": "2025-01-16T15:03:20",
"upload_time_iso_8601": "2025-01-16T15:03:20.443596Z",
"url": "https://files.pythonhosted.org/packages/0c/9d/293f104337cf8bb6a34bc4ff68f210677e227ccbeae66a0cde4816a17fd1/django_aws_api_gateway_websockets-1.1.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2025-01-16 15:03:20",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "StevenMapes",
"github_project": "django-aws-api-gateway-websockets",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"requirements": [],
"tox": true,
"lcname": "django-aws-api-gateway-websockets"
}