<!-- PROJECT SHIELDS -->
<!--
*** I'm using markdown "reference style" links for readability.
*** Reference links are enclosed in brackets [ ] instead of parentheses ( ).
*** See the bottom of this document for the declaration of the reference variables
*** for contributors-url, forks-url, etc. This is an optional, concise syntax you may use.
*** https://www.markdownguide.org/basic-syntax/#reference-style-links
-->
[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)
[![Downloads](https://static.pepy.tech/badge/sanic-security)](https://pepy.tech/project/sanic-security)
[![Conda Downloads](https://img.shields.io/conda/dn/conda-forge/sanic-security.svg)](https://anaconda.org/conda-forge/sanic-security)
<!-- PROJECT LOGO -->
<br />
<p align="center">
<h3 align="center">Sanic Security</h3>
<p align="center">
An async security library for the Sanic framework.
</p>
</p>
<!-- TABLE OF CONTENTS -->
## Table of Contents
* [About the Project](#about-the-project)
* [Getting Started](#getting-started)
* [Prerequisites](#prerequisites)
* [Installation](#installation)
* [Configuration](#configuration)
* [Usage](#usage)
* [OAuth](#oauth)
* [Authentication](#authentication)
* [CAPTCHA](#captcha)
* [Two-step Verification](#two-step-verification)
* [Authorization](#authorization)
* [Testing](#testing)
* [Tortoise](#tortoise)
* [Contributing](#contributing)
* [License](#license)
* [Versioning](#versioning)
* [Support](https://discord.gg/JHpZkMfKTJ)
<!-- ABOUT THE PROJECT -->
## About The Project
Sanic Security is an authentication, authorization, and verification library designed for use with the
[Sanic](https://github.com/huge-success/sanic) framework.
* OAuth2 integration
* Login, registration, and authentication with refresh mechanisms
* Role based authorization with wildcard permissions
* Image & audio CAPTCHA
* Two-factor authentication
* Two-step verification
* Logging & auditing
Visit [security.na-stewart.com](https://security.na-stewart.com) for documentation.
<!-- GETTING STARTED -->
## Getting Started
In order to get started, please install [PyPI](https://pypi.org/).
### Installation
* Install the Sanic Security pip package.
```shell
pip3 install sanic-security
````
* Install the Sanic Security pip package with the `cryptography` dependency included.
If you're planning on encoding or decoding JWTs using certain digital signature algorithms (like RSA or ECDSA which use
the public secret and private secret), you will need to install the `cryptography` library. This can be installed explicitly, or
as an extra requirement.
```shell
pip3 install sanic-security[crypto]
````
* Install the Sanic Security pip package with the `httpx-oauth` dependency included.
If you're planning on utilizing OAuth, you will need to install the `httpx-oauth` library. This can be installed explicitly, or
as an extra requirement.
```shell
pip3 install sanic-security[oauth]
````
* Update Sanic Security if already installed.
```shell
pip3 install sanic-security --upgrade
```
### Configuration
Sanic Security configuration is merely an object that can be modified either using dot-notation or like a
dictionary.
For example:
```python
from sanic_security.configuration import config as security_config
security_config.SECRET = "This is a big secret. Shhhhh"
security_config["CAPTCHA_FONT"] = "./resources/captcha-font.ttf"
```
You can also use the update() method like on regular dictionaries.
Any environment variables defined with the SANIC_SECURITY_ prefix will be applied to the config. For example, setting
SANIC_SECURITY_SECRET will be loaded by the application automatically and fed into the SECRET config variable.
You can load environment variables with a different prefix via `security_config.load_environment_variables("NEW_PREFIX_")` method.
* Default configuration values:
| Key | Value | Description |
|---------------------------------------|------------------------------|-----------------------------------------------------------------------------------------------------------------------------------|
| **SECRET** | This is a big secret. Shhhhh | The secret used for generating and signing JWTs. This should be a string unique to your application. Keep it safe. |
| **PUBLIC_SECRET** | None | The secret used for verifying and decoding JWTs and can be publicly shared. This should be a string unique to your application. |
| **OAUTH_CLIENT** | None | The client ID provided by the OAuth provider, this is used to identify the application making the OAuth request. |
| **OAUTH_SECRET** | None | The client secret provided by the OAuth provider, this is used in conjunction with the client ID to authenticate the application. |
| **SESSION_SAMESITE** | Strict | The SameSite attribute of session cookies. |
| **SESSION_SECURE** | True | The Secure attribute of session cookies. |
| **SESSION_HTTPONLY** | True | The HttpOnly attribute of session cookies. HIGHLY recommended that you do not turn this off, unless you know what you are doing. |
| **SESSION_DOMAIN** | None | The Domain attribute of session cookies. |
| **SESSION_ENCODING_ALGORITHM** | HS256 | The algorithm used to encode and decode session JWT's. |
| **SESSION_PREFIX** | tkn | Prefix attached to the beginning of session cookies. |
| **MAX_CHALLENGE_ATTEMPTS** | 3 | The maximum amount of session challenge attempts allowed. |
| **CAPTCHA_SESSION_EXPIRATION** | 180 | The amount of seconds till captcha session expiration on creation. Setting to 0 will disable expiration. |
| **CAPTCHA_FONT** | captcha-font.ttf | The file path to the font being used for captcha generation. Several fonts can be used by separating them via comma. |
| **CAPTCHA_VOICE** | captcha-voice/ | The directory of the voice library being used for audio captcha generation. |
| **TWO_STEP_SESSION_EXPIRATION** | 300 | The amount of seconds till two-step session expiration on creation. Setting to 0 will disable expiration. |
| **AUTHENTICATION_SESSION_EXPIRATION** | 86400 | The amount of seconds till authentication session expiration on creation. Setting to 0 will disable expiration. |
| **AUTHENTICATION_REFRESH_EXPIRATION** | 604800 | The amount of seconds till authentication refresh expiration. Setting to 0 will disable refresh mechanism. |
| **ALLOW_LOGIN_WITH_USERNAME** | False | Allows login via username; unique constraint is disabled when set to false. |
| **INITIAL_ADMIN_EMAIL** | admin@example.com | Email used when creating the initial admin account. |
| **INITIAL_ADMIN_PASSWORD** | admin123 | Password used when creating the initial admin account. |
## Usage
Sanic Security's authentication and verification functionality is session based. A new session will be created for the user after the user logs in or requests some form of verification (two-step, captcha). The session data is then encoded into a JWT and stored on a cookie on the user’s browser. The session cookie is then sent
along with every subsequent request. The server can then compare the session stored on the cookie against the session information stored in the database to verify user’s identity and send a response with the corresponding state.
* Initialize Sanic Security as follows:
```python
initialize_security(app)
initialize_oauth(app) # Remove if not utilizing OAuth
if __name__ == "__main__":
app.run(host="127.0.0.1", port=8000, workers=1, debug=True)
```
The tables in the below examples represent example [request form-data](https://sanicframework.org/en/guide/basics/request.html#form).
## OAuth
Provides users with a familiar experience by having them register/login using their existing credentials from other trusted services (such as Google, Discord, etc.).
This feature is designed to complement existing protocols by linking a Sanic Security account with the user's OAuth credentials. As a result, developers can leverage all of Sanic Security's capabilities including robust session handling and account management.
* Define OAuth clients
You can [utilize various OAuth clients](https://frankie567.github.io/httpx-oauth/reference/httpx_oauth.clients/) based on your needs or [customize one](https://frankie567.github.io/httpx-oauth/usage/).
ID and secret should be stored and referenced via configuration.
```python
discord_oauth = DiscordOAuth2(
"1325594509043830895",
"WNMYbkDJjGlC0ej60qM-50tC9mMy0EXa",
)
google_oauth = GoogleOAuth2(
"480512993828-e2e9tqtl2b8or62hc4l7hpoh478s3ni1.apps.googleusercontent.com",
"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",
)
```
* Redirect to authorization URL
```python
@app.route("api/security/oauth", methods=["GET", "POST"])
async def on_oauth_request(request):
return redirect(
await google_oauth.get_authorization_url(
"http://localhost:8000/api/security/oauth/callback",
scope=google_oauth.base_scopes,
)
)
```
* Handle OAuth callback
```python
@app.get("api/security/oauth/callback")
async def on_oauth_callback(request):
token_info, authentication_session = await oauth_callback(
request, google_oauth, "http://localhost:8000/api/security/oauth/callback"
)
response = json(
"Authorization successful.",
{"token_info": token_info, "auth_session": authentication_session.json},
)
oauth_encode(response, token_info)
authentication_session.encode(response)
return response
```
* Get access token
```python
@app.get("api/security/oauth/token")
async def on_oauth_token(request):
token_info = await decode_oauth(request, google_oauth)
return json(
"Access token retrieved.",
token_info,
)
```
* Requires access token (This method is not called directly and instead used as a decorator)
```python
@app.get("api/security/oauth/token")
@requires_oauth(google_oauth)
async def on_oauth_token(request):
return json(
"Access token retrieved.",
request.ctx.oauth,
)
```
## Authentication
* Registration (With two-step account verification)
Phone can be null or empty.
| Key | Value |
|--------------|---------------------|
| **username** | example |
| **email** | example@example.com |
| **phone** | 19811354186 |
| **password** | examplepass |
```python
@app.post("api/security/register")
async def on_register(request):
account = await register(request)
two_step_session = await request_two_step_verification(request, account)
await email_code(
account.email, two_step_session.code # Code = 24KF19
) # Custom method for emailing verification code.
response = json(
"Registration successful! Email verification required.", account.json
)
two_step_session.encode(response)
return response
```
* Verify Account
Verifies the client's account via two-step session code.
| Key | Value |
|----------|--------|
| **code** | 24KF19 |
```python
@app.put("api/security/verify")
async def on_verify(request):
two_step_session = await verify_account(request)
return json(
"You have verified your account and may login!", two_step_session.bearer.json
)
```
* Login (With two-factor authentication)
Credentials are retrieved via header are constructed by first combining the username and the password with a colon
(aladdin:opensesame), and then by encoding the resulting string in base64 (YWxhZGRpbjpvcGVuc2VzYW1l).
Here is an example authorization header: `Authorization: Basic YWxhZGRpbjpvcGVuc2VzYW1l`. You can use a username
as well as an email for login if `ALLOW_LOGIN_WITH_USERNAME` is true in the config.
```python
@app.post("api/security/login")
async def on_login(request):
authentication_session = await login(request, require_second_factor=True)
two_step_session = await request_two_step_verification(
request, authentication_session.bearer
)
await email_code(
authentication_session.bearer.email, two_step_session.code # Code = XGED2U
) # Custom method for emailing verification code.
response = json(
"Login successful! Two-factor authentication required.",
authentication_session.bearer.json,
)
authentication_session.encode(response)
two_step_session.encode(response)
return response
```
If this isn't desired, you can pass an account and password attempt directly into the login method instead.
* Fulfill Second Factor
Fulfills client authentication session's second factor requirement via two-step session code.
| Key | Value |
|----------|--------|
| **code** | XGED2U |
```python
@app.put("api/security/fulfill-2fa")
async def on_two_factor_authentication(request):
authentication_session = await fulfill_second_factor(request)
response = json(
"Authentication session second-factor fulfilled! You are now authenticated.",
authentication_session.bearer.json,
)
return response
```
* Anonymous Login
Simply create a new session and encode it.
```python
@app.post("api/security/login/anon")
async def on_anonymous_login(request):
authentication_session = await AuthenticationSession.new(request)
response = json(
"Anonymous client now associated with session!", authentication_session.json
)
authentication_session.encode(response)
return response
```
* Logout
```python
@app.post("api/security/logout")
async def on_logout(request):
authentication_session = await logout(request)
response = json("Logout successful!", authentication_session.json)
await oauth_revoke(request, response, google_oauth) # Remove if not utilizing OAuth
return response
```
* Authenticate
```python
@app.post("api/security/auth")
async def on_authenticate(request):
authentication_session = await authenticate(request)
response = json(
"You have been authenticated.",
authentication_session.json,
)
return response
```
* Requires Authentication (This method is not called directly and instead used as a decorator)
```python
@app.post("api/security/auth")
@requires_authentication
async def on_authenticate(request):
response = json("You have been authenticated.", request.ctx.session.json)
return response
```
## CAPTCHA
Protects against spam and malicious activities by ensuring that only real humans can complete certain actions like
submitting a form or creating an account. A font and voice library for CAPTCHA challenges is included in the repository,
or you can download/create your own and specify its path in the configuration.
* Request CAPTCHA
```python
@app.get("api/security/captcha")
async def on_captcha_img_request(request):
captcha_session = await request_captcha(request)
response = raw(
captcha_session.get_image(), content_type="image/jpeg"
) # Captcha: LJ0F3U
captcha_session.encode(response)
return response
```
* Request CAPTCHA Audio
```python
@app.get("api/security/captcha/audio")
async def on_captcha_audio_request(request):
captcha_session = await CaptchaSession.decode(request)
return raw(captcha_session.get_audio(), content_type="audio/mpeg")
```
* Attempt CAPTCHA
| Key | Value |
|-------------|--------|
| **captcha** | LJ0F3U |
```python
@app.post("api/security/captcha")
async def on_captcha(request):
captcha_session = await captcha(request)
return json("Captcha attempt successful!", captcha_session.json)
```
* Requires CAPTCHA (This method is not called directly and instead used as a decorator)
| Key | Value |
|-------------|--------|
| **captcha** | LJ0F3U |
```python
@app.post("api/security/captcha")
@requires_captcha
async def on_captcha(request):
return json("Captcha attempt successful!", request.ctx.session.json)
```
## Two-step Verification
Two-step verification should be integrated with other custom functionalities, such as forgot password recovery.
* Request Two-step Verification
| Key | Value |
|-------------|---------------------|
| **email** | example@example.com |
```python
@app.post("api/security/two-step/request")
async def on_two_step_request(request):
two_step_session = await request_two_step_verification(request) # Code = T2I58I
await email_code(
two_step_session.bearer.email, two_step_session.code
) # Custom method for emailing verification code.
response = json("Verification request successful!", two_step_session.json)
two_step_session.encode(response)
return response
```
* Resend Two-step Verification Code
```python
@app.post("api/security/two-step/resend")
async def on_two_step_resend(request):
two_step_session = await TwoStepSession.decode(request) # Code = T2I58I
await email_code(
two_step_session.bearer.email, two_step_session.code
) # Custom method for emailing verification code.
return json("Verification code resend successful!", two_step_session.json)
```
* Attempt Two-step Verification
| Key | Value |
|----------|--------|
| **code** | T2I58I |
```python
@app.post("api/security/two-step")
async def on_two_step_verification(request):
two_step_session = await two_step_verification(request)
response = json("Two-step verification attempt successful!", two_step_session.json)
return response
```
* Requires Two-step Verification (This method is not called directly and instead used as a decorator)
| Key | Value |
|----------|--------|
| **code** | T2I58I |
```python
@app.post("api/security/two-step")
@requires_two_step_verification
async def on_two_step_verification(request):
response = json(
"Two-step verification attempt successful!", request.ctx.session.json
)
return response
```
## Authorization
Sanic Security uses role based authorization with wildcard permissions.
Roles are created for various job functions. The permissions to perform certain operations are assigned to specific roles.
Users are assigned particular roles, and through those role assignments acquire the permissions needed to perform
particular system functions. Since users are not assigned permissions directly, but only acquire them through their
role (or roles), management of individual user rights becomes a matter of simply assigning appropriate roles to the
user's account; this simplifies common operations, such as adding a user, or changing a user's department.
Wildcard permissions support the concept of multiple levels or parts. For example, you could grant a user the permission
`printer:query`, `printer:query,delete`, or `printer:*`.
* Assign Role
```python
await assign_role(
"Chat Room Moderator",
account,
"Can read and delete messages in all chat rooms, suspend and mute accounts, and control voice chat.",
"channels:view,delete",
"voice:*",
"account:suspend,mute",
)
```
* Check Permissions
```python
@app.post("api/security/perms")
async def on_check_perms(request):
authentication_session = await check_permissions(
request, "channels:view", "voice:*"
)
return text("Account is authorized.")
```
* Require Permissions (This method is not called directly and instead used as a decorator.)
```python
@app.post("api/security/perms")
@requires_permission("channels:view", "voice:*")
async def on_check_perms(request):
return text("Account is authorized.")
```
* Check Roles
```python
@app.post("api/security/roles")
async def on_check_roles(request):
authentication_session = await check_roles(request, "Chat Room Moderator")
return text("Account is authorized.")
```
* Require Roles (This method is not called directly and instead used as a decorator)
```python
@app.post("api/security/roles")
@requires_role("Chat Room Moderator")
async def on_check_roles(request):
return text("Account is authorized.")
```
## Testing
* Set the `TEST_DATABASE_URL` configuration value.
* Make sure the test Sanic instance (`test/server.py`) is running on your machine.
* Run the test client (`test/tests.py`) for results.
## Tortoise
Sanic Security uses [Tortoise ORM](https://tortoise-orm.readthedocs.io/en/latest/index.html) for database operations.
Tortoise ORM is an easy-to-use asyncio ORM (Object Relational Mapper).
* Initialise your models and database like so:
```python
async def init():
await Tortoise.init(
db_url="sqlite://db.sqlite3",
modules={"models": ["sanic_security.models", "app.models"]},
)
await Tortoise.generate_schemas()
```
or
```python
register_tortoise(
app,
db_url="sqlite://db.sqlite3",
modules={"models": ["sanic_security.models", "app.models"]},
generate_schemas=True,
)
```
* Define your models like so:
```python
from tortoise.models import Model
from tortoise import fields
class Tournament(Model):
id = fields.IntField(pk=True)
name = fields.TextField()
```
* Use it like so:
```python
# Create instance by save
tournament = Tournament(name="New Tournament")
await tournament.save()
# Or by .create()
await Tournament.create(name="Another Tournament")
# Now search for a record
tour = await Tournament.filter(name__contains="Another").first()
print(tour.name)
```
<!-- CONTRIBUTING -->
## Contributing
Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are **greatly appreciated**.
1. Fork the Project
2. Create your Feature Branch (`git checkout -b feature/AmazingFeature`)
3. Commit your Changes (`git commit -m 'Add some AmazingFeature'`)
4. Push to the Branch (`git push origin feature/AmazingFeature`)
5. Open a Pull Request
<!-- LICENSE -->
## License
Distributed under the MIT License. See `LICENSE` for more information.
<!-- Versioning -->
## Versioning
**0.0.0**
* MAJOR version when you make incompatible API changes.
* MINOR version when you add functionality in a backwards compatible manner.
* PATCH version when you make backwards compatible bug fixes.
[https://semver.org/](https://semver.org/)
Raw data
{
"_id": null,
"home_page": null,
"name": "sanic-security",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.8",
"maintainer_email": null,
"keywords": "security, authentication, authorization, verification, async, sanic",
"author": null,
"author_email": "Aidan Stewart <me@na-stewart.com>",
"download_url": "https://files.pythonhosted.org/packages/70/d0/d711233089b4778d7eedb5b82fc832a66d63c8cbfdc3e0a9a23b3aa379f9/sanic_security-1.16.5.tar.gz",
"platform": null,
"description": "<!-- PROJECT SHIELDS -->\r\n<!--\r\n*** I'm using markdown \"reference style\" links for readability.\r\n*** Reference links are enclosed in brackets [ ] instead of parentheses ( ).\r\n*** See the bottom of this document for the declaration of the reference variables\r\n*** for contributors-url, forks-url, etc. This is an optional, concise syntax you may use.\r\n*** https://www.markdownguide.org/basic-syntax/#reference-style-links\r\n-->\r\n\r\n[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)\r\n[![Downloads](https://static.pepy.tech/badge/sanic-security)](https://pepy.tech/project/sanic-security)\r\n[![Conda Downloads](https://img.shields.io/conda/dn/conda-forge/sanic-security.svg)](https://anaconda.org/conda-forge/sanic-security)\r\n\r\n\r\n<!-- PROJECT LOGO -->\r\n<br />\r\n<p align=\"center\">\r\n <h3 align=\"center\">Sanic Security</h3>\r\n <p align=\"center\">\r\n An async security library for the Sanic framework.\r\n </p>\r\n</p>\r\n\r\n\r\n<!-- TABLE OF CONTENTS -->\r\n## Table of Contents\r\n\r\n* [About the Project](#about-the-project)\r\n* [Getting Started](#getting-started)\r\n * [Prerequisites](#prerequisites)\r\n * [Installation](#installation)\r\n * [Configuration](#configuration)\r\n* [Usage](#usage)\r\n * [OAuth](#oauth)\r\n * [Authentication](#authentication)\r\n * [CAPTCHA](#captcha)\r\n * [Two-step Verification](#two-step-verification)\r\n * [Authorization](#authorization)\r\n * [Testing](#testing)\r\n * [Tortoise](#tortoise)\r\n* [Contributing](#contributing)\r\n* [License](#license)\r\n* [Versioning](#versioning)\r\n* [Support](https://discord.gg/JHpZkMfKTJ)\r\n\r\n<!-- ABOUT THE PROJECT -->\r\n## About The Project\r\n\r\nSanic Security is an authentication, authorization, and verification library designed for use with the \r\n[Sanic](https://github.com/huge-success/sanic) framework.\r\n\r\n* OAuth2 integration\r\n* Login, registration, and authentication with refresh mechanisms\r\n* Role based authorization with wildcard permissions\r\n* Image & audio CAPTCHA\r\n* Two-factor authentication\r\n* Two-step verification\r\n* Logging & auditing\r\n\r\nVisit [security.na-stewart.com](https://security.na-stewart.com) for documentation.\r\n\r\n<!-- GETTING STARTED -->\r\n## Getting Started\r\n\r\nIn order to get started, please install [PyPI](https://pypi.org/).\r\n\r\n### Installation\r\n\r\n* Install the Sanic Security pip package.\r\n```shell\r\npip3 install sanic-security\r\n````\r\n\r\n* Install the Sanic Security pip package with the `cryptography` dependency included.\r\n\r\nIf you're planning on encoding or decoding JWTs using certain digital signature algorithms (like RSA or ECDSA which use \r\nthe public secret and private secret), you will need to install the `cryptography` library. This can be installed explicitly, or \r\nas an extra requirement.\r\n\r\n```shell\r\npip3 install sanic-security[crypto]\r\n````\r\n\r\n* Install the Sanic Security pip package with the `httpx-oauth` dependency included.\r\n\r\nIf you're planning on utilizing OAuth, you will need to install the `httpx-oauth` library. This can be installed explicitly, or \r\nas an extra requirement.\r\n\r\n```shell\r\npip3 install sanic-security[oauth]\r\n````\r\n\r\n* Update Sanic Security if already installed.\r\n\r\n```shell\r\npip3 install sanic-security --upgrade\r\n```\r\n\r\n### Configuration\r\n\r\nSanic Security configuration is merely an object that can be modified either using dot-notation or like a \r\ndictionary.\r\n\r\nFor example: \r\n\r\n```python\r\nfrom sanic_security.configuration import config as security_config\r\n\r\nsecurity_config.SECRET = \"This is a big secret. Shhhhh\"\r\nsecurity_config[\"CAPTCHA_FONT\"] = \"./resources/captcha-font.ttf\"\r\n```\r\n\r\nYou can also use the update() method like on regular dictionaries.\r\n\r\nAny environment variables defined with the SANIC_SECURITY_ prefix will be applied to the config. For example, setting \r\nSANIC_SECURITY_SECRET will be loaded by the application automatically and fed into the SECRET config variable.\r\n\r\nYou can load environment variables with a different prefix via `security_config.load_environment_variables(\"NEW_PREFIX_\")` method.\r\n\r\n* Default configuration values:\r\n\r\n| Key | Value | Description |\r\n|---------------------------------------|------------------------------|-----------------------------------------------------------------------------------------------------------------------------------|\r\n| **SECRET** | This is a big secret. Shhhhh | The secret used for generating and signing JWTs. This should be a string unique to your application. Keep it safe. |\r\n| **PUBLIC_SECRET** | None | The secret used for verifying and decoding JWTs and can be publicly shared. This should be a string unique to your application. |\r\n| **OAUTH_CLIENT** | None | The client ID provided by the OAuth provider, this is used to identify the application making the OAuth request. |\r\n| **OAUTH_SECRET** | None | The client secret provided by the OAuth provider, this is used in conjunction with the client ID to authenticate the application. |\r\n| **SESSION_SAMESITE** | Strict | The SameSite attribute of session cookies. |\r\n| **SESSION_SECURE** | True | The Secure attribute of session cookies. |\r\n| **SESSION_HTTPONLY** | True | The HttpOnly attribute of session cookies. HIGHLY recommended that you do not turn this off, unless you know what you are doing. |\r\n| **SESSION_DOMAIN** | None | The Domain attribute of session cookies. |\r\n| **SESSION_ENCODING_ALGORITHM** | HS256 | The algorithm used to encode and decode session JWT's. |\r\n| **SESSION_PREFIX** | tkn | Prefix attached to the beginning of session cookies. |\r\n| **MAX_CHALLENGE_ATTEMPTS** | 3 | The maximum amount of session challenge attempts allowed. |\r\n| **CAPTCHA_SESSION_EXPIRATION** | 180 | The amount of seconds till captcha session expiration on creation. Setting to 0 will disable expiration. |\r\n| **CAPTCHA_FONT** | captcha-font.ttf | The file path to the font being used for captcha generation. Several fonts can be used by separating them via comma. |\r\n| **CAPTCHA_VOICE** | captcha-voice/ | The directory of the voice library being used for audio captcha generation. |\r\n| **TWO_STEP_SESSION_EXPIRATION** | 300 | The amount of seconds till two-step session expiration on creation. Setting to 0 will disable expiration. |\r\n| **AUTHENTICATION_SESSION_EXPIRATION** | 86400 | The amount of seconds till authentication session expiration on creation. Setting to 0 will disable expiration. |\r\n| **AUTHENTICATION_REFRESH_EXPIRATION** | 604800 | The amount of seconds till authentication refresh expiration. Setting to 0 will disable refresh mechanism. |\r\n| **ALLOW_LOGIN_WITH_USERNAME** | False | Allows login via username; unique constraint is disabled when set to false. |\r\n| **INITIAL_ADMIN_EMAIL** | admin@example.com | Email used when creating the initial admin account. |\r\n| **INITIAL_ADMIN_PASSWORD** | admin123 | Password used when creating the initial admin account. |\r\n\r\n## Usage\r\n\r\nSanic Security's authentication and verification functionality is session based. A new session will be created for the user after the user logs in or requests some form of verification (two-step, captcha). The session data is then encoded into a JWT and stored on a cookie on the user\u2019s browser. The session cookie is then sent\r\nalong with every subsequent request. The server can then compare the session stored on the cookie against the session information stored in the database to verify user\u2019s identity and send a response with the corresponding state.\r\n\r\n* Initialize Sanic Security as follows:\r\n```python\r\ninitialize_security(app)\r\ninitialize_oauth(app) # Remove if not utilizing OAuth\r\nif __name__ == \"__main__\":\r\n app.run(host=\"127.0.0.1\", port=8000, workers=1, debug=True)\r\n```\r\n\r\nThe tables in the below examples represent example [request form-data](https://sanicframework.org/en/guide/basics/request.html#form).\r\n\r\n## OAuth\r\n\r\nProvides users with a familiar experience by having them register/login using their existing credentials from other trusted services (such as Google, Discord, etc.).\r\n\r\nThis feature is designed to complement existing protocols by linking a Sanic Security account with the user's OAuth credentials. As a result, developers can leverage all of Sanic Security's capabilities including robust session handling and account management.\r\n\r\n* Define OAuth clients\r\n\r\nYou can [utilize various OAuth clients](https://frankie567.github.io/httpx-oauth/reference/httpx_oauth.clients/) based on your needs or [customize one](https://frankie567.github.io/httpx-oauth/usage/).\r\nID and secret should be stored and referenced via configuration.\r\n\r\n```python\r\ndiscord_oauth = DiscordOAuth2(\r\n \"1325594509043830895\",\r\n \"WNMYbkDJjGlC0ej60qM-50tC9mMy0EXa\",\r\n)\r\ngoogle_oauth = GoogleOAuth2(\r\n \"480512993828-e2e9tqtl2b8or62hc4l7hpoh478s3ni1.apps.googleusercontent.com\",\r\n \"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\",\r\n)\r\n```\r\n\r\n* Redirect to authorization URL\r\n\r\n```python\r\n@app.route(\"api/security/oauth\", methods=[\"GET\", \"POST\"])\r\nasync def on_oauth_request(request):\r\n return redirect(\r\n await google_oauth.get_authorization_url(\r\n \"http://localhost:8000/api/security/oauth/callback\",\r\n scope=google_oauth.base_scopes,\r\n )\r\n )\r\n```\r\n\r\n* Handle OAuth callback\r\n\r\n```python\r\n@app.get(\"api/security/oauth/callback\")\r\nasync def on_oauth_callback(request):\r\n token_info, authentication_session = await oauth_callback(\r\n request, google_oauth, \"http://localhost:8000/api/security/oauth/callback\"\r\n )\r\n response = json(\r\n \"Authorization successful.\",\r\n {\"token_info\": token_info, \"auth_session\": authentication_session.json},\r\n )\r\n oauth_encode(response, token_info)\r\n authentication_session.encode(response)\r\n return response\r\n```\r\n\r\n* Get access token \r\n\r\n```python\r\n@app.get(\"api/security/oauth/token\")\r\nasync def on_oauth_token(request):\r\n token_info = await decode_oauth(request, google_oauth)\r\n return json(\r\n \"Access token retrieved.\",\r\n token_info,\r\n )\r\n```\r\n\r\n* Requires access token (This method is not called directly and instead used as a decorator)\r\n\r\n```python\r\n@app.get(\"api/security/oauth/token\")\r\n@requires_oauth(google_oauth)\r\nasync def on_oauth_token(request):\r\n return json(\r\n \"Access token retrieved.\",\r\n request.ctx.oauth,\r\n )\r\n```\r\n\r\n## Authentication\r\n \r\n* Registration (With two-step account verification)\r\n\r\nPhone can be null or empty.\r\n\r\n| Key | Value |\r\n|--------------|---------------------|\r\n| **username** | example |\r\n| **email** | example@example.com |\r\n| **phone** | 19811354186 |\r\n| **password** | examplepass |\r\n\r\n```python\r\n@app.post(\"api/security/register\")\r\nasync def on_register(request):\r\n account = await register(request)\r\n two_step_session = await request_two_step_verification(request, account)\r\n await email_code(\r\n account.email, two_step_session.code # Code = 24KF19\r\n ) # Custom method for emailing verification code.\r\n response = json(\r\n \"Registration successful! Email verification required.\", account.json\r\n )\r\n two_step_session.encode(response)\r\n return response\r\n```\r\n\r\n* Verify Account\r\n\r\nVerifies the client's account via two-step session code.\r\n\r\n| Key | Value |\r\n|----------|--------|\r\n| **code** | 24KF19 |\r\n\r\n```python\r\n@app.put(\"api/security/verify\")\r\nasync def on_verify(request):\r\n two_step_session = await verify_account(request)\r\n return json(\r\n \"You have verified your account and may login!\", two_step_session.bearer.json\r\n )\r\n```\r\n\r\n* Login (With two-factor authentication)\r\n\r\nCredentials are retrieved via header are constructed by first combining the username and the password with a colon \r\n(aladdin:opensesame), and then by encoding the resulting string in base64 (YWxhZGRpbjpvcGVuc2VzYW1l). \r\nHere is an example authorization header: `Authorization: Basic YWxhZGRpbjpvcGVuc2VzYW1l`. You can use a username \r\nas well as an email for login if `ALLOW_LOGIN_WITH_USERNAME` is true in the config.\r\n\r\n```python\r\n@app.post(\"api/security/login\")\r\nasync def on_login(request):\r\n authentication_session = await login(request, require_second_factor=True)\r\n two_step_session = await request_two_step_verification(\r\n request, authentication_session.bearer\r\n )\r\n await email_code(\r\n authentication_session.bearer.email, two_step_session.code # Code = XGED2U\r\n ) # Custom method for emailing verification code.\r\n response = json(\r\n \"Login successful! Two-factor authentication required.\",\r\n authentication_session.bearer.json,\r\n )\r\n authentication_session.encode(response)\r\n two_step_session.encode(response)\r\n return response\r\n```\r\n\r\nIf this isn't desired, you can pass an account and password attempt directly into the login method instead.\r\n\r\n* Fulfill Second Factor\r\n\r\nFulfills client authentication session's second factor requirement via two-step session code.\r\n\r\n| Key | Value |\r\n|----------|--------|\r\n| **code** | XGED2U |\r\n\r\n```python\r\n@app.put(\"api/security/fulfill-2fa\")\r\nasync def on_two_factor_authentication(request):\r\n authentication_session = await fulfill_second_factor(request)\r\n response = json(\r\n \"Authentication session second-factor fulfilled! You are now authenticated.\",\r\n authentication_session.bearer.json,\r\n )\r\n return response\r\n```\r\n\r\n* Anonymous Login\r\n\r\nSimply create a new session and encode it.\r\n\r\n```python\r\n@app.post(\"api/security/login/anon\")\r\nasync def on_anonymous_login(request):\r\n authentication_session = await AuthenticationSession.new(request)\r\n response = json(\r\n \"Anonymous client now associated with session!\", authentication_session.json\r\n )\r\n authentication_session.encode(response)\r\n return response\r\n```\r\n\r\n* Logout\r\n\r\n```python\r\n@app.post(\"api/security/logout\")\r\nasync def on_logout(request):\r\n authentication_session = await logout(request)\r\n response = json(\"Logout successful!\", authentication_session.json)\r\n await oauth_revoke(request, response, google_oauth) # Remove if not utilizing OAuth\r\n return response\r\n```\r\n\r\n* Authenticate\r\n\r\n```python\r\n@app.post(\"api/security/auth\")\r\nasync def on_authenticate(request):\r\n authentication_session = await authenticate(request)\r\n response = json(\r\n \"You have been authenticated.\",\r\n authentication_session.json,\r\n )\r\n return response\r\n```\r\n\r\n* Requires Authentication (This method is not called directly and instead used as a decorator)\r\n\r\n```python\r\n@app.post(\"api/security/auth\")\r\n@requires_authentication\r\nasync def on_authenticate(request):\r\n response = json(\"You have been authenticated.\", request.ctx.session.json)\r\n return response\r\n```\r\n\r\n## CAPTCHA\r\n\r\nProtects against spam and malicious activities by ensuring that only real humans can complete certain actions like \r\nsubmitting a form or creating an account. A font and voice library for CAPTCHA challenges is included in the repository, \r\nor you can download/create your own and specify its path in the configuration.\r\n\r\n* Request CAPTCHA\r\n\r\n```python\r\n@app.get(\"api/security/captcha\")\r\nasync def on_captcha_img_request(request):\r\n captcha_session = await request_captcha(request)\r\n response = raw(\r\n captcha_session.get_image(), content_type=\"image/jpeg\"\r\n ) # Captcha: LJ0F3U\r\n captcha_session.encode(response)\r\n return response\r\n```\r\n\r\n* Request CAPTCHA Audio\r\n\r\n```python\r\n@app.get(\"api/security/captcha/audio\")\r\nasync def on_captcha_audio_request(request):\r\n captcha_session = await CaptchaSession.decode(request)\r\n return raw(captcha_session.get_audio(), content_type=\"audio/mpeg\")\r\n```\r\n\r\n* Attempt CAPTCHA\r\n\r\n| Key | Value |\r\n|-------------|--------|\r\n| **captcha** | LJ0F3U |\r\n\r\n```python\r\n@app.post(\"api/security/captcha\")\r\nasync def on_captcha(request):\r\n captcha_session = await captcha(request)\r\n return json(\"Captcha attempt successful!\", captcha_session.json)\r\n```\r\n\r\n* Requires CAPTCHA (This method is not called directly and instead used as a decorator)\r\n\r\n| Key | Value |\r\n|-------------|--------|\r\n| **captcha** | LJ0F3U |\r\n\r\n```python\r\n@app.post(\"api/security/captcha\")\r\n@requires_captcha\r\nasync def on_captcha(request):\r\n return json(\"Captcha attempt successful!\", request.ctx.session.json)\r\n```\r\n\r\n## Two-step Verification\r\n\r\nTwo-step verification should be integrated with other custom functionalities, such as forgot password recovery.\r\n\r\n* Request Two-step Verification\r\n\r\n| Key | Value |\r\n|-------------|---------------------|\r\n| **email** | example@example.com |\r\n\r\n```python\r\n@app.post(\"api/security/two-step/request\")\r\nasync def on_two_step_request(request):\r\n two_step_session = await request_two_step_verification(request) # Code = T2I58I\r\n await email_code(\r\n two_step_session.bearer.email, two_step_session.code\r\n ) # Custom method for emailing verification code.\r\n response = json(\"Verification request successful!\", two_step_session.json)\r\n two_step_session.encode(response)\r\n return response\r\n``` \r\n\r\n* Resend Two-step Verification Code\r\n\r\n```python\r\n@app.post(\"api/security/two-step/resend\")\r\nasync def on_two_step_resend(request):\r\n two_step_session = await TwoStepSession.decode(request) # Code = T2I58I\r\n await email_code(\r\n two_step_session.bearer.email, two_step_session.code\r\n ) # Custom method for emailing verification code.\r\n return json(\"Verification code resend successful!\", two_step_session.json)\r\n```\r\n\r\n* Attempt Two-step Verification\r\n\r\n| Key | Value |\r\n|----------|--------|\r\n| **code** | T2I58I |\r\n\r\n```python\r\n@app.post(\"api/security/two-step\")\r\nasync def on_two_step_verification(request):\r\n two_step_session = await two_step_verification(request)\r\n response = json(\"Two-step verification attempt successful!\", two_step_session.json)\r\n return response\r\n```\r\n\r\n* Requires Two-step Verification (This method is not called directly and instead used as a decorator)\r\n\r\n| Key | Value |\r\n|----------|--------|\r\n| **code** | T2I58I |\r\n\r\n```python\r\n@app.post(\"api/security/two-step\")\r\n@requires_two_step_verification\r\nasync def on_two_step_verification(request):\r\n response = json(\r\n \"Two-step verification attempt successful!\", request.ctx.session.json\r\n )\r\n return response\r\n```\r\n\r\n## Authorization\r\n\r\nSanic Security uses role based authorization with wildcard permissions.\r\n\r\nRoles are created for various job functions. The permissions to perform certain operations are assigned to specific roles. \r\nUsers are assigned particular roles, and through those role assignments acquire the permissions needed to perform \r\nparticular system functions. Since users are not assigned permissions directly, but only acquire them through their \r\nrole (or roles), management of individual user rights becomes a matter of simply assigning appropriate roles to the \r\nuser's account; this simplifies common operations, such as adding a user, or changing a user's department. \r\n\r\nWildcard permissions support the concept of multiple levels or parts. For example, you could grant a user the permission\r\n`printer:query`, `printer:query,delete`, or `printer:*`.\r\n\r\n* Assign Role\r\n\r\n```python\r\nawait assign_role(\r\n \"Chat Room Moderator\",\r\n account,\r\n \"Can read and delete messages in all chat rooms, suspend and mute accounts, and control voice chat.\",\r\n \"channels:view,delete\",\r\n \"voice:*\",\r\n \"account:suspend,mute\",\r\n)\r\n```\r\n\r\n* Check Permissions\r\n\r\n```python\r\n@app.post(\"api/security/perms\")\r\nasync def on_check_perms(request):\r\n authentication_session = await check_permissions(\r\n request, \"channels:view\", \"voice:*\"\r\n )\r\n return text(\"Account is authorized.\")\r\n```\r\n\r\n* Require Permissions (This method is not called directly and instead used as a decorator.)\r\n\r\n```python\r\n@app.post(\"api/security/perms\")\r\n@requires_permission(\"channels:view\", \"voice:*\")\r\nasync def on_check_perms(request):\r\n return text(\"Account is authorized.\")\r\n```\r\n\r\n* Check Roles\r\n\r\n```python\r\n@app.post(\"api/security/roles\")\r\nasync def on_check_roles(request):\r\n authentication_session = await check_roles(request, \"Chat Room Moderator\")\r\n return text(\"Account is authorized.\")\r\n```\r\n\r\n* Require Roles (This method is not called directly and instead used as a decorator)\r\n\r\n```python\r\n@app.post(\"api/security/roles\")\r\n@requires_role(\"Chat Room Moderator\")\r\nasync def on_check_roles(request):\r\n return text(\"Account is authorized.\")\r\n```\r\n\r\n## Testing\r\n\r\n* Set the `TEST_DATABASE_URL` configuration value.\r\n\r\n* Make sure the test Sanic instance (`test/server.py`) is running on your machine.\r\n\r\n* Run the test client (`test/tests.py`) for results.\r\n\r\n## Tortoise\r\n\r\nSanic Security uses [Tortoise ORM](https://tortoise-orm.readthedocs.io/en/latest/index.html) for database operations.\r\n\r\nTortoise ORM is an easy-to-use asyncio ORM (Object Relational Mapper).\r\n\r\n* Initialise your models and database like so: \r\n\r\n```python\r\nasync def init():\r\n await Tortoise.init(\r\n db_url=\"sqlite://db.sqlite3\",\r\n modules={\"models\": [\"sanic_security.models\", \"app.models\"]},\r\n )\r\n await Tortoise.generate_schemas()\r\n```\r\n\r\nor\r\n\r\n```python\r\nregister_tortoise(\r\n app,\r\n db_url=\"sqlite://db.sqlite3\",\r\n modules={\"models\": [\"sanic_security.models\", \"app.models\"]},\r\n generate_schemas=True,\r\n)\r\n```\r\n\r\n* Define your models like so:\r\n\r\n```python\r\nfrom tortoise.models import Model\r\nfrom tortoise import fields\r\n\r\n\r\nclass Tournament(Model):\r\n id = fields.IntField(pk=True)\r\n name = fields.TextField()\r\n```\r\n\r\n* Use it like so:\r\n\r\n```python\r\n# Create instance by save\r\ntournament = Tournament(name=\"New Tournament\")\r\nawait tournament.save()\r\n\r\n# Or by .create()\r\nawait Tournament.create(name=\"Another Tournament\")\r\n\r\n# Now search for a record\r\ntour = await Tournament.filter(name__contains=\"Another\").first()\r\nprint(tour.name)\r\n```\r\n\r\n<!-- CONTRIBUTING -->\r\n## Contributing\r\n\r\nContributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are **greatly appreciated**.\r\n\r\n1. Fork the Project\r\n2. Create your Feature Branch (`git checkout -b feature/AmazingFeature`)\r\n3. Commit your Changes (`git commit -m 'Add some AmazingFeature'`)\r\n4. Push to the Branch (`git push origin feature/AmazingFeature`)\r\n5. Open a Pull Request\r\n\r\n\r\n<!-- LICENSE -->\r\n## License\r\n\r\nDistributed under the MIT License. See `LICENSE` for more information.\r\n\r\n<!-- Versioning -->\r\n## Versioning\r\n\r\n**0.0.0**\r\n\r\n* MAJOR version when you make incompatible API changes.\r\n\r\n* MINOR version when you add functionality in a backwards compatible manner.\r\n\r\n* PATCH version when you make backwards compatible bug fixes.\r\n\r\n[https://semver.org/](https://semver.org/)\r\n",
"bugtrack_url": null,
"license": null,
"summary": "An async security library for the Sanic framework.",
"version": "1.16.5",
"project_urls": {
"Documentation": "https://security.na-stewart.com/",
"Repository": "https://github.com/na-stewart/sanic-security"
},
"split_keywords": [
"security",
" authentication",
" authorization",
" verification",
" async",
" sanic"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "2fe1aa43d44fd72b8491a3704e6e908dd800d6613ea6da7021e029e78f574265",
"md5": "2dbc643d34d3c7aef06fb76ec923a7b7",
"sha256": "1fb4c96a54296acbff8d4902c8d8db0f7324712d7069a68daabc6f240c97a21f"
},
"downloads": -1,
"filename": "sanic_security-1.16.5-py3-none-any.whl",
"has_sig": false,
"md5_digest": "2dbc643d34d3c7aef06fb76ec923a7b7",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.8",
"size": 40492,
"upload_time": "2025-01-13T07:46:59",
"upload_time_iso_8601": "2025-01-13T07:46:59.368378Z",
"url": "https://files.pythonhosted.org/packages/2f/e1/aa43d44fd72b8491a3704e6e908dd800d6613ea6da7021e029e78f574265/sanic_security-1.16.5-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "70d0d711233089b4778d7eedb5b82fc832a66d63c8cbfdc3e0a9a23b3aa379f9",
"md5": "bfdde307b7f2bd576f8202fa86099cd7",
"sha256": "77c975fa397b879f57b4d149a9541199c5cf2e422938043c26c87d1c444a6bc6"
},
"downloads": -1,
"filename": "sanic_security-1.16.5.tar.gz",
"has_sig": false,
"md5_digest": "bfdde307b7f2bd576f8202fa86099cd7",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.8",
"size": 37608,
"upload_time": "2025-01-13T07:47:01",
"upload_time_iso_8601": "2025-01-13T07:47:01.986667Z",
"url": "https://files.pythonhosted.org/packages/70/d0/d711233089b4778d7eedb5b82fc832a66d63c8cbfdc3e0a9a23b3aa379f9/sanic_security-1.16.5.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2025-01-13 07:47:01",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "na-stewart",
"github_project": "sanic-security",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"lcname": "sanic-security"
}