From 7266008a851ec32d8dcc042da0ec2dcdec19d37c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Nazar=C3=A9=20da=20Piedade?= Date: Sun, 15 May 2022 02:04:28 +0100 Subject: [PATCH] docs(pt): prepare the ground to start translate documentation to portuguese --- src/.vuepress/config.js | 3 + .../locales/abstract/locale_abstract_pt.js | 7 + .../locales/details/locale_detail_pt.js | 12 + src/.vuepress/locales/details/nav/nav_pt.js | 101 +++ .../locales/details/sidebar/sidebar_pt.js | 195 +++++ src/.vuepress/locales/index.js | 4 + src/pt/README.md | 23 + src/pt/guide/README.md | 87 +++ src/pt/guide/advanced/README.md | 1 + src/pt/guide/advanced/class-based-views.md | 191 +++++ src/pt/guide/advanced/proxy-headers.md | 436 +++++++++++ src/pt/guide/advanced/signals.md | 277 +++++++ src/pt/guide/advanced/streaming.md | 163 +++++ src/pt/guide/advanced/versioning.md | 157 ++++ src/pt/guide/advanced/websockets.md | 58 ++ src/pt/guide/basics/README.md | 1 + src/pt/guide/basics/app.md | 238 ++++++ src/pt/guide/basics/cookies.md | 76 ++ src/pt/guide/basics/handlers.md | 120 ++++ src/pt/guide/basics/headers.md | 234 ++++++ src/pt/guide/basics/listeners.md | 234 ++++++ src/pt/guide/basics/middleware.md | 201 ++++++ src/pt/guide/basics/request.md | 196 +++++ src/pt/guide/basics/response.md | 185 +++++ src/pt/guide/basics/routing.md | 676 ++++++++++++++++++ src/pt/guide/basics/tasks.md | 125 ++++ src/pt/guide/best-practices/README.md | 1 + src/pt/guide/best-practices/blueprints.md | 388 ++++++++++ src/pt/guide/best-practices/decorators.md | 183 +++++ src/pt/guide/best-practices/exceptions.md | 496 +++++++++++++ src/pt/guide/best-practices/logging.md | 93 +++ src/pt/guide/best-practices/testing.md | 3 + src/pt/guide/deployment/README.md | 1 + src/pt/guide/deployment/configuration.md | 259 +++++++ src/pt/guide/deployment/development.md | 72 ++ src/pt/guide/deployment/docker.md | 1 + src/pt/guide/deployment/kubernetes.md | 1 + src/pt/guide/deployment/nginx.md | 218 ++++++ src/pt/guide/deployment/running.md | 281 ++++++++ src/pt/guide/deployment/server-choice.md | 1 + src/pt/guide/getting-started.md | 90 +++ src/pt/guide/how-to/README.md | 1 + src/pt/guide/how-to/assets/images/lake.jpg | Bin 0 -> 141444 bytes src/pt/guide/how-to/authentication.md | 114 +++ src/pt/guide/how-to/autodiscovery.md | 161 +++++ src/pt/guide/how-to/cors.md | 135 ++++ src/pt/guide/how-to/csrf.md | 1 + src/pt/guide/how-to/db.md | 1 + src/pt/guide/how-to/decorators.md | 1 + src/pt/guide/how-to/ipv6.md | 0 src/pt/guide/how-to/mounting.md | 52 ++ src/pt/guide/how-to/orm.md | 292 ++++++++ src/pt/guide/how-to/request-id-logging.md | 0 src/pt/guide/how-to/serialization.md | 1 + src/pt/guide/how-to/server-sent-events.md | 1 + src/pt/guide/how-to/static-redirects.md | 112 +++ src/pt/guide/how-to/task-queue.md | 1 + src/pt/guide/how-to/tls.md | 168 +++++ src/pt/guide/how-to/toc.md | 21 + src/pt/guide/how-to/validation.md | 1 + src/pt/guide/how-to/websocket-feed.md | 1 + src/pt/guide/release-notes/v21.12.md | 507 +++++++++++++ src/pt/guide/release-notes/v21.3.md | 264 +++++++ src/pt/guide/release-notes/v21.6.md | 342 +++++++++ src/pt/guide/release-notes/v21.9.md | 234 ++++++ src/pt/guide/release-notes/v22.3.md | 221 ++++++ src/pt/help.md | 29 + src/pt/org/README.md | 1 + src/pt/org/policies.md | 61 ++ src/pt/org/scope.md | 264 +++++++ src/pt/plugins/README.md | 0 src/pt/plugins/sanic-ext/README.md | 0 src/pt/plugins/sanic-ext/configuration.md | 239 +++++++ src/pt/plugins/sanic-ext/convenience.md | 86 +++ src/pt/plugins/sanic-ext/getting-started.md | 78 ++ src/pt/plugins/sanic-ext/http/README.md | 0 src/pt/plugins/sanic-ext/http/cors.md | 86 +++ src/pt/plugins/sanic-ext/http/methods.md | 137 ++++ src/pt/plugins/sanic-ext/injection.md | 274 +++++++ src/pt/plugins/sanic-ext/openapi.md | 7 + src/pt/plugins/sanic-ext/openapi/README.md | 0 src/pt/plugins/sanic-ext/openapi/advanced.md | 10 + src/pt/plugins/sanic-ext/openapi/autodoc.md | 131 ++++ src/pt/plugins/sanic-ext/openapi/basic.md | 70 ++ .../plugins/sanic-ext/openapi/decorators.md | 456 ++++++++++++ src/pt/plugins/sanic-ext/openapi/security.md | 91 +++ src/pt/plugins/sanic-ext/openapi/ui.md | 26 + src/pt/plugins/sanic-ext/validation.md | 127 ++++ src/pt/plugins/sanic-testing/README.md | 0 src/pt/plugins/sanic-testing/clients.md | 112 +++ .../plugins/sanic-testing/getting-started.md | 83 +++ 91 files changed, 11089 insertions(+) create mode 100644 src/.vuepress/locales/abstract/locale_abstract_pt.js create mode 100644 src/.vuepress/locales/details/locale_detail_pt.js create mode 100644 src/.vuepress/locales/details/nav/nav_pt.js create mode 100644 src/.vuepress/locales/details/sidebar/sidebar_pt.js create mode 100644 src/pt/README.md create mode 100644 src/pt/guide/README.md create mode 100644 src/pt/guide/advanced/README.md create mode 100644 src/pt/guide/advanced/class-based-views.md create mode 100644 src/pt/guide/advanced/proxy-headers.md create mode 100644 src/pt/guide/advanced/signals.md create mode 100644 src/pt/guide/advanced/streaming.md create mode 100644 src/pt/guide/advanced/versioning.md create mode 100644 src/pt/guide/advanced/websockets.md create mode 100644 src/pt/guide/basics/README.md create mode 100644 src/pt/guide/basics/app.md create mode 100644 src/pt/guide/basics/cookies.md create mode 100644 src/pt/guide/basics/handlers.md create mode 100644 src/pt/guide/basics/headers.md create mode 100644 src/pt/guide/basics/listeners.md create mode 100644 src/pt/guide/basics/middleware.md create mode 100644 src/pt/guide/basics/request.md create mode 100644 src/pt/guide/basics/response.md create mode 100644 src/pt/guide/basics/routing.md create mode 100644 src/pt/guide/basics/tasks.md create mode 100644 src/pt/guide/best-practices/README.md create mode 100644 src/pt/guide/best-practices/blueprints.md create mode 100644 src/pt/guide/best-practices/decorators.md create mode 100644 src/pt/guide/best-practices/exceptions.md create mode 100644 src/pt/guide/best-practices/logging.md create mode 100644 src/pt/guide/best-practices/testing.md create mode 100644 src/pt/guide/deployment/README.md create mode 100644 src/pt/guide/deployment/configuration.md create mode 100644 src/pt/guide/deployment/development.md create mode 100644 src/pt/guide/deployment/docker.md create mode 100644 src/pt/guide/deployment/kubernetes.md create mode 100644 src/pt/guide/deployment/nginx.md create mode 100644 src/pt/guide/deployment/running.md create mode 100644 src/pt/guide/deployment/server-choice.md create mode 100644 src/pt/guide/getting-started.md create mode 100644 src/pt/guide/how-to/README.md create mode 100644 src/pt/guide/how-to/assets/images/lake.jpg create mode 100644 src/pt/guide/how-to/authentication.md create mode 100644 src/pt/guide/how-to/autodiscovery.md create mode 100644 src/pt/guide/how-to/cors.md create mode 100644 src/pt/guide/how-to/csrf.md create mode 100644 src/pt/guide/how-to/db.md create mode 100644 src/pt/guide/how-to/decorators.md create mode 100644 src/pt/guide/how-to/ipv6.md create mode 100644 src/pt/guide/how-to/mounting.md create mode 100644 src/pt/guide/how-to/orm.md create mode 100644 src/pt/guide/how-to/request-id-logging.md create mode 100644 src/pt/guide/how-to/serialization.md create mode 100644 src/pt/guide/how-to/server-sent-events.md create mode 100644 src/pt/guide/how-to/static-redirects.md create mode 100644 src/pt/guide/how-to/task-queue.md create mode 100644 src/pt/guide/how-to/tls.md create mode 100644 src/pt/guide/how-to/toc.md create mode 100644 src/pt/guide/how-to/validation.md create mode 100644 src/pt/guide/how-to/websocket-feed.md create mode 100644 src/pt/guide/release-notes/v21.12.md create mode 100644 src/pt/guide/release-notes/v21.3.md create mode 100644 src/pt/guide/release-notes/v21.6.md create mode 100644 src/pt/guide/release-notes/v21.9.md create mode 100644 src/pt/guide/release-notes/v22.3.md create mode 100644 src/pt/help.md create mode 100644 src/pt/org/README.md create mode 100644 src/pt/org/policies.md create mode 100644 src/pt/org/scope.md create mode 100644 src/pt/plugins/README.md create mode 100644 src/pt/plugins/sanic-ext/README.md create mode 100644 src/pt/plugins/sanic-ext/configuration.md create mode 100644 src/pt/plugins/sanic-ext/convenience.md create mode 100644 src/pt/plugins/sanic-ext/getting-started.md create mode 100644 src/pt/plugins/sanic-ext/http/README.md create mode 100644 src/pt/plugins/sanic-ext/http/cors.md create mode 100644 src/pt/plugins/sanic-ext/http/methods.md create mode 100644 src/pt/plugins/sanic-ext/injection.md create mode 100644 src/pt/plugins/sanic-ext/openapi.md create mode 100644 src/pt/plugins/sanic-ext/openapi/README.md create mode 100644 src/pt/plugins/sanic-ext/openapi/advanced.md create mode 100644 src/pt/plugins/sanic-ext/openapi/autodoc.md create mode 100644 src/pt/plugins/sanic-ext/openapi/basic.md create mode 100644 src/pt/plugins/sanic-ext/openapi/decorators.md create mode 100644 src/pt/plugins/sanic-ext/openapi/security.md create mode 100644 src/pt/plugins/sanic-ext/openapi/ui.md create mode 100644 src/pt/plugins/sanic-ext/validation.md create mode 100644 src/pt/plugins/sanic-testing/README.md create mode 100644 src/pt/plugins/sanic-testing/clients.md create mode 100644 src/pt/plugins/sanic-testing/getting-started.md diff --git a/src/.vuepress/config.js b/src/.vuepress/config.js index 69d977e6eb..64f0dd59d7 100644 --- a/src/.vuepress/config.js +++ b/src/.vuepress/config.js @@ -43,6 +43,7 @@ let site_config = { "/zh/": locales.locale_abstract_zh, "/ja/": locales.locale_abstract_ja, "/ko/": locales.locale_abstract_ko, + "/pt/": locales.locale_abstract_pt, }, markdown: { @@ -94,6 +95,7 @@ let site_config = { "/zh/": "新的", "/ja/": "新しい", "/ko/": "새로운", + "/pt/": "NOVO", }, }, ], @@ -135,6 +137,7 @@ let site_config = { "/zh/": locales.locale_detail_zh, "/ja/": locales.locale_detail_ja, "/ko/": locales.locale_detail_ko, + "/pt/": locales.locale_detail_pt, }, author: { name: "Sanic Community Organization", diff --git a/src/.vuepress/locales/abstract/locale_abstract_pt.js b/src/.vuepress/locales/abstract/locale_abstract_pt.js new file mode 100644 index 0000000000..98ae39e295 --- /dev/null +++ b/src/.vuepress/locales/abstract/locale_abstract_pt.js @@ -0,0 +1,7 @@ +module.exports = { + lang: "pt-PT", + title: "A Abstração Sanic", + current: "Recente com a versão 22.3", + description: + "A Sanic é um servidor web e uma abstração web da Python 3.7+ que é escrita ir rápido. Ela permite o uso da sintaxe de async/await adicionada à Python 3.5, a qual torna o seu código não-bloqueante e rápido.", +}; diff --git a/src/.vuepress/locales/details/locale_detail_pt.js b/src/.vuepress/locales/details/locale_detail_pt.js new file mode 100644 index 0000000000..573e59335b --- /dev/null +++ b/src/.vuepress/locales/details/locale_detail_pt.js @@ -0,0 +1,12 @@ +const sidebar_en = require("./sidebar/sidebar_pt"); +const nav_en = require("./nav/nav_pt"); +module.exports = { + selectText: "Línguas", + label: "Português", + editLinkText: "Edite esta página no GitHub", + helpText: "Ajudar", + algolia: {}, + current: "Recente com a versão 21.12", + nav: nav_pt, + sidebar: sidebar_pt, +}; diff --git a/src/.vuepress/locales/details/nav/nav_pt.js b/src/.vuepress/locales/details/nav/nav_pt.js new file mode 100644 index 0000000000..1ea01fa058 --- /dev/null +++ b/src/.vuepress/locales/details/nav/nav_pt.js @@ -0,0 +1,101 @@ +module.exports = [ + { + text: "Página Inicial", + link: "/pt/", + }, + // { text: "Anúncios", link: "/announcements" }, + { + text: "Documentação", + items: [ + { + text: "Guia de Usuário", + items: [ + { + text: "Geral", + link: "/pt/guide/getting-started.md", + }, + { + text: "Básicos", + link: "/pt/guide/basics/app.md", + }, + { + text: "Avançados", + link: "/pt/guide/advanced/class-based-views.md", + }, + { + text: "Boas Práticas", + link: "/pt/guide/best-practices/blueprints.md", + }, + { + text: "Executando e Desdobrando", + link: "/pt/guide/deployment/configuration.md", + }, + { + text: "Como...", + link: "/pt/guide/how-to/toc.md", + }, + { + text: "Notas do Último Lançamento", + link: "/pt/guide/release-notes/v21.12.md", + }, + ], + }, + { + text: "Plugins Oficiais", + items: [ + { + text: "Extensões de Sanic", + link: "/pt/plugins/sanic-ext/getting-started.md", + }, + { + text: "Testando a Sanic", + link: "/pt/plugins/sanic-testing/getting-started.md", + }, + ], + }, + { + text: "Documentação da API", + items: [ + { + text: "Ver a Documentação da API", + link: "https://sanic.readthedocs.io", + }, + ], + }, + { + text: "Documentação da Organização", + items: [ + { + text: "Políticas", + link: "/pt/org/policies.md", + }, + { + text: "D.O.M.Í.N.I.O", + link: "/pt/org/scope.md", + }, + ], + }, + ], + }, + { + text: "Comunidade", + items: [ + { + text: "Fóruns", + link: "https://community.sanicframework.org/", + }, + { + text: "Discord", + link: "https://discord.gg/FARQzAEMAA", + }, + { + text: "Twitter", + link: "https://twitter.com/sanicframework", + }, + { + text: "Patrocínio", + link: "https://opencollective.com/sanic-org/", + }, + ], + }, +]; diff --git a/src/.vuepress/locales/details/sidebar/sidebar_pt.js b/src/.vuepress/locales/details/sidebar/sidebar_pt.js new file mode 100644 index 0000000000..9f1868e513 --- /dev/null +++ b/src/.vuepress/locales/details/sidebar/sidebar_pt.js @@ -0,0 +1,195 @@ +module.exports = { + "/pt/guide/": [ + { + title: "Geral", + sidebarDepth: 1, + children: ["/pt/guide/", "/pt/guide/getting-started.md"], + }, + { + title: "Básicos", + sidebarDepth: 1, + children: [ + "/pt/guide/basics/app.md", + "/pt/guide/basics/handlers.md", + "/pt/guide/basics/request.md", + "/pt/guide/basics/response.md", + "/pt/guide/basics/routing.md", + "/pt/guide/basics/listeners.md", + "/pt/guide/basics/middleware.md", + "/pt/guide/basics/headers.md", + "/pt/guide/basics/cookies.md", + "/pt/guide/basics/tasks.md", + ], + }, + { + title: "Avançado", + sidebarDepth: 1, + children: [ + "/pt/guide/advanced/class-based-views.md", + "/pt/guide/advanced/proxy-headers.md", + "/pt/guide/advanced/streaming.md", + "/pt/guide/advanced/websockets.md", + "/pt/guide/advanced/versioning.md", + "/pt/guide/advanced/signals.md", + ], + }, + { + title: "Boas Práticas", + sidebarDepth: 1, + children: [ + "/pt/guide/best-practices/blueprints.md", + "/pt/guide/best-practices/exceptions.md", + "/pt/guide/best-practices/decorators.md", + "/pt/guide/best-practices/logging.md", + "/pt/guide/best-practices/testing.md", + ], + }, + { + title: "Executando e Desdobrando", + sidebarDepth: 2, + children: [ + "/pt/guide/deployment/configuration.md", + "/pt/guide/deployment/development.md", + "/pt/guide/deployment/running.md", + // "/guide/deployment/server-choice.md", + "/pt/guide/deployment/nginx.md", + // "/guide/deployment/docker.md", + // "/guide/deployment/kubernetes.md", + ], + }, + { + title: "Como...", + sidebarDepth: 1, + children: [ + "/pt/guide/how-to/toc.md", + "/pt/guide/how-to/mounting.md", + "/pt/guide/how-to/authentication.md", + "/pt/guide/how-to/autodiscovery.md", + "/pt/guide/how-to/cors.md", + // "/guide/how-to/db.md", + // "/guide/how-to/decorators.md", + // "/guide/how-to/validation.md", + // "/guide/how-to/csrf.md", + // "/guide/how-to/serialization.md", + // "/pt/guide/how-to/sqlalchemy.md", + "/pt/guide/how-to/orm.md", + "/pt/guide/how-to/static-redirects.md", + // "/guide/how-to/task-queue.md", + "/pt/guide/how-to/tls.md", + // "/guide/how-to/websocket-feed.md", + // "/guide/how-to/server-sent-events.md", + ], + }, + { + title: "Notas de Lançamanto", + sidebarDepth: 1, + children: [ + "/pt/guide/release-notes/v22.3.md", + "/pt/guide/release-notes/v21.12.md", + "/pt/guide/release-notes/v21.9.md", + "/pt/guide/release-notes/v21.6.md", + "/pt/guide/release-notes/v21.3.md", + ], + }, + { + title: "Plugins", + sidebarDepth: 1, + children: [ + ["/pt/plugins/sanic-ext/getting-started.md", "Extensões de Sanic"], + ["/pt/plugins/sanic-testing/getting-started.md", "Testes da Sanic"], + ], + }, + { + title: "Organização", + sidebarDepth: 1, + children: ["/pt/org/policies.md", "/pt/org/scope.md"], + }, + ], + + "/pt/plugins/": [ + { + title: "Guia de Usuário", + sidebarDepth: 1, + children: [ + ["/pt/guide/", "Geral"], + ["/pt/guide/basics/app.md", "Básicos"], + ["/pt/guide/advanced/class-based-views.md", "Avançado"], + ["/pt/guide/best-practices/blueprints.md", "Boas Práticas"], + ["/pt/guide/deployment/configuration.md", "Executando e Desdobrando"], + ["/pt/guide/how-to/toc.md", "Como..."], + ], + }, + { + title: "Extensões de Sanic", + sidebarDepth: 1, + children: [ + "/pt/plugins/sanic-ext/getting-started.md", + { + title: "Recursos da HTTP", + children: [ + "/pt/plugins/sanic-ext/http/methods.md", + "/pt/plugins/sanic-ext/http/cors.md", + ], + }, + "/pt/plugins/sanic-ext/convenience.md", + "/pt/plugins/sanic-ext/injection.md", + { + title: "OpenAPI", + sidebarDepth: 2, + children: [ + "/pt/plugins/sanic-ext/openapi/basic.md", + "/pt/plugins/sanic-ext/openapi/ui.md", + "/pt/plugins/sanic-ext/openapi/decorators.md", + "/pt/plugins/sanic-ext/openapi/advanced.md", + "/pt/plugins/sanic-ext/openapi/autodoc.md", + "/pt/plugins/sanic-ext/openapi/security.md", + ], + }, + "/pt/plugins/sanic-ext/validation.md", + "/pt/plugins/sanic-ext/configuration.md", + ], + }, + { + title: "Testes da Sanic", + sidebarDepth: 1, + children: [ + "/pt/plugins/sanic-testing/getting-started.md", + "/pt/plugins/sanic-testing/clients.md", + ], + }, + { + title: "Organização", + sidebarDepth: 1, + children: ["/pt/org/policies.md", "/pt/org/scope.md"], + }, + ], + + "/pt/org/": [ + { + title: "Guia de Usuário", + sidebarDepth: 1, + children: [ + ["/pt/guide/", "Geral"], + ["/pt/guide/basics/app.md", "Básicos"], + ["/pt/guide/advanced/class-based-views.md", "Avançado"], + ["/pt/guide/best-practices/blueprints.md", "Boas Práticas"], + ["/pt/guide/deployment/configuration.md", "Executando e Desdobrando"], + ["/pt/guide/how-to/toc.md", "Como..."], + ], + }, + { + title: "Plugins", + sidebarDepth: 1, + children: [ + ["/pt/plugins/sanic-ext/getting-started.md", "Extensões de Sanic"], + ["/pt/plugins/sanic-testing/getting-started.md", "Testes da Sanic"], + ], + }, + { + title: "Organização", + sidebarDepth: 1, + collapsable: false, + children: ["/pt/org/policies.md", "/pt/org/scope.md"], + }, + ], +}; diff --git a/src/.vuepress/locales/index.js b/src/.vuepress/locales/index.js index c7d19e9bc5..0a9cfa7063 100644 --- a/src/.vuepress/locales/index.js +++ b/src/.vuepress/locales/index.js @@ -2,18 +2,22 @@ const locale_abstract_en = require("./abstract/locale_abstract_en"); const locale_abstract_ko = require("./abstract/locale_abstract_ko"); const locale_abstract_zh = require("./abstract/locale_abstract_zh"); const locale_abstract_ja = require("./abstract/locale_abstract_ja"); +const locale_abstract_pt = require("./abstract/locale_abstract_pt"); const locale_detail_en = require("./details/locale_detail_en"); const locale_detail_ko = require("./details/locale_detail_ko"); const locale_detail_zh = require("./details/locale_detail_zh"); const locale_detail_ja = require("./details/locale_detail_ja"); +const locale_detail_pt = require("./details/locale_detail_pt"); module.exports = { locale_abstract_en, locale_abstract_ko, locale_abstract_zh, locale_abstract_ja, + locale_abstract_pt, locale_detail_en, locale_detail_ko, locale_detail_zh, locale_detail_ja, + locale_detail_pt, }; diff --git a/src/pt/README.md b/src/pt/README.md new file mode 100644 index 0000000000..94a55b293a --- /dev/null +++ b/src/pt/README.md @@ -0,0 +1,23 @@ +--- +home: true +heroImage: https://raw.githubusercontent.com/huge-success/sanic-assets/master/png/sanic-framework-logo-400x97.png +heroText: Build fast. Run fast. +tagline: Next generation Python web server/framework +actionText: Get Started → +actionLink: /en/guide/ +features: +- title: Simple and lightweight + details: Intuitive API with smart defaults and no bloat allows you to get straight to work building your app. +- title: Unopinionated and flexible + details: Build the way you want to build without letting your tooling constrain you. +- title: Performant and scalable + details: Built from the ground up with speed and scalability as a main concern. It is ready to power web applications big and small. +- title: Production ready + details: Out of the box, it comes bundled with a web server ready to power your web applications. +- title: Trusted by millions + details: Sanic is one of the overall most popular frameworks on PyPI, and the top async enabled framework +- title: Community driven + details: The project is maintained and run by the community for the community. +pageClass: landing-page +logo: false +--- diff --git a/src/pt/guide/README.md b/src/pt/guide/README.md new file mode 100644 index 0000000000..d9cec6d26d --- /dev/null +++ b/src/pt/guide/README.md @@ -0,0 +1,87 @@ +--- +pageClass: intro +--- + +# Introduction + +Sanic is a Python 3.7+ web server and web framework that’s written to go fast. It allows the usage of the async/await syntax added in Python 3.5, which makes your code non-blocking and speedy. + +| | | +|---------|-------------------------------------------------------------------------------------------------------------------------| +| Build | [![Build Status][]][1] [![AppVeyor Build Status][]][2] [![Codecov]][3] | +| Docs | [![Documentation]][4] | +| Package | [![PyPI][]][5] [![PyPI version][]][5] [![PyPI Wheel][]][6] [![Supported implementations][]][6] [![Code style black]][7] | +| Support | [![Forums][]][8] [![Discord][]][9] [![Awesome Sanic List]][10] | +| Stats | [![Downloads][]][11] [![Downloads][12]][11] | + +## What is it? + +First things first, before you jump in the water, you should know that Sanic is different than other frameworks. + +Right there in that first sentence there is a huge mistake because Sanic is _both_ a **framework** and a **web server**. In the deployment section we will talk a little bit more about this. + +But, remember, out of the box Sanic comes with everything you need to write, deploy, and scale a production grade web application. :rocket: + +## Goal + +> To provide a simple way to get up and running a highly performant HTTP server that is easy to build, to expand, and ultimately to scale. +## Features + +---:1 + +- Built in _fast_ web server +- Production ready +- Highly scalable +- ASGI compliant +- Simple and intuitive API design +- By the community, for the community +:--:1 + +:--- + + + +## Sponsor + +Check out [open collective](https://opencollective.com/sanic-org) to learn more about helping to fund Sanic. + + +## Join the Community + +The main channel for discussion is at the [community forums](https://community.sanicframework.org/). There also is a [Discord Server](https://discord.gg/FARQzAEMAA) for live discussion and chat. + +The Stackoverflow `[sanic]` tag is [actively monitored](https://stackoverflow.com/questions/tagged/sanic) by project maintainers. + +## Contribution + +We are always happy to have new contributions. We have [marked issues good for anyone looking to get started](https://github.com/sanic-org/sanic/issues?q=is%3Aopen+is%3Aissue+label%3Abeginner), and welcome [questions/answers/discussion on the forums](https://community.sanicframework.org/). Please take a look at our [Contribution guidelines](https://github.com/sanic-org/sanic/blob/master/CONTRIBUTING.rst). + +## Who we are + + + +[Build Status]: https://travis-ci.com/sanic-org/sanic.svg?branch=master +[1]: https://travis-ci.com/sanic-org/sanic +[AppVeyor Build Status]: https://ci.appveyor.com/api/projects/status/d8pt3ids0ynexi8c/branch/master?svg=true +[2]: https://ci.appveyor.com/project/sanic-org/sanic +[Codecov]: https://codecov.io/gh/sanic-org/sanic/branch/master/graph/badge.svg +[3]: https://codecov.io/gh/sanic-org/sanic +[Documentation]: https://readthedocs.org/projects/sanic/badge/?version=latest +[4]: http://sanic.readthedocs.io/en/latest/?badge=latest +[PyPI]: https://img.shields.io/pypi/v/sanic.svg +[5]: https://pypi.python.org/pypi/sanic/ +[PyPI version]: https://img.shields.io/pypi/pyversions/sanic.svg +[PyPI Wheel]: https://img.shields.io/pypi/wheel/sanic.svg +[6]: https://pypi.python.org/pypi/sanic +[Supported implementations]: https://img.shields.io/pypi/implementation/sanic.svg +[Code style black]: https://img.shields.io/badge/code%20style-black-000000.svg +[7]: https://github.com/ambv/black +[Forums]: https://img.shields.io/badge/forums-community-ff0068.svg +[8]: https://community.sanicframework.org/ +[Discord]: https://img.shields.io/discord/812221182594121728?logo=discord +[9]: https://discord.gg/FARQzAEMAA +[Awesome Sanic List]: https://cdn.rawgit.com/sindresorhus/awesome/d7305f38d29fed78fa85652e3a63e154dd8e8829/media/badge.svg +[10]: https://github.com/mekicha/awesome-sanic +[Downloads]: https://pepy.tech/badge/sanic/month +[11]: https://pepy.tech/project/sanic +[12]: https://pepy.tech/badge/sanic/week diff --git a/src/pt/guide/advanced/README.md b/src/pt/guide/advanced/README.md new file mode 100644 index 0000000000..931fae06c0 --- /dev/null +++ b/src/pt/guide/advanced/README.md @@ -0,0 +1 @@ +# Advanced diff --git a/src/pt/guide/advanced/class-based-views.md b/src/pt/guide/advanced/class-based-views.md new file mode 100644 index 0000000000..c1e042a25e --- /dev/null +++ b/src/pt/guide/advanced/class-based-views.md @@ -0,0 +1,191 @@ +# Class Based Views + +## Why use them? + +---:1 + +### The problem + +A common pattern when designing an API is to have multiple functionality on the same endpoint that depends upon the HTTP method. + +While both of these options work, they are not good design practices and may be hard to maintain over time as your project grows. +:--:1 +```python +@app.get("/foo") +async def foo_get(request): + ... + +@app.post("/foo") +async def foo_post(request): + ... + +@app.put("/foo") +async def foo_put(request): + ... + +@app.route("/bar", methods=["GET", "POST", "PATCH"]) +async def bar(request): + if request.method == "GET": + ... + elif request.method == "POST": + ... + elif request.method == "PATCH": + ... +``` +:--- + +---:1 + +### The solution + +Class-based views are simply classes that implement response behavior to requests. They provide a way to compartmentalize handling of different HTTP request types at the same endpoint. +:--:1 +```python +from sanic.views import HTTPMethodView + +class FooBar(HTTPMethodView): + async def get(self, request): + ... + + async def post(self, request): + ... + + async def put(self, request): + ... + +app.add_route(FooBar.as_view(), "/foobar") +``` +:--- + +## Defining a view + +A class-based view should subclass `HTTPMethodView`. You can then implement class methods with the name of the corresponding HTTP method. If a request is received that has no defined method, a `405: Method not allowed` response will be generated. + +---:1 + +To register a class-based view on an endpoint, the `app.add_route` method is used. The first argument should be the defined class with the method `as_view` invoked, and the second should be the URL endpoint. + +The available methods are: + +- get +- post +- put +- patch +- delete +- head +- options +:--:1 +```python +from sanic.views import HTTPMethodView +from sanic.response import text + +class SimpleView(HTTPMethodView): + + def get(self, request): + return text("I am get method") + + # You can also use async syntax + async def post(self, request): + return text("I am post method") + + def put(self, request): + return text("I am put method") + + def patch(self, request): + return text("I am patch method") + + def delete(self, request): + return text("I am delete method") + +app.add_route(SimpleView.as_view(), "/") +``` +:--- + +## Path parameters + +---:1 + +You can use path parameters exactly as discussed in [the routing section](/guide/basics/routing.md). +:--:1 +```python +class NameView(HTTPMethodView): + + def get(self, request, name): + return text("Hello {}".format(name)) + +app.add_route(NameView.as_view(), "/") +``` +:--- + +## Decorators + +As discussed in [the decorators section](/guide/best-practices/decorators.md), often you will need to add functionality to endpoints with the use of decorators. You have two options with CBV: + +1. Apply to _all_ HTTP methods in the view +2. Apply individually to HTTP methods in the view + +Let's see what the options look like: + +---:1 + +### Apply to all methods + +If you want to add any decorators to the class, you can set the `decorators` class variable. These will be applied to the class when `as_view` is called. +:--:1 +```python +class ViewWithDecorator(HTTPMethodView): + decorators = [some_decorator_here] + + def get(self, request, name): + return text("Hello I have a decorator") + + def post(self, request, name): + return text("Hello I also have a decorator") + +app.add_route(ViewWithDecorator.as_view(), "/url") +``` +:--- + +---:1 + +### Apply to individual methods + +But if you just want to decorate some methods and not all methods, you can as shown here. +:--:1 +```python +class ViewWithSomeDecorator(HTTPMethodView): + + @staticmethod + @some_decorator_here + def get(request, name): + return text("Hello I have a decorator") + + def post(self, request, name): + return text("Hello I don"t have any decorators") + + @some_decorator_here + def patch(self, request, name): + return text("Hello I have a decorator") +``` +:--- + +## Generating a URL +---:1 + +This works just like [generating any other URL](/guide/basics/routing.md#generating-a-url), except that the class name is a part of the endpoint. +:--:1 +```python +@app.route("/") +def index(request): + url = app.url_for("SpecialClassView") + return redirect(url) + + +class SpecialClassView(HTTPMethodView): + def get(self, request): + return text("Hello from the Special Class View!") + + +app.add_route(SpecialClassView.as_view(), "/special_class_view") +``` +:--- diff --git a/src/pt/guide/advanced/proxy-headers.md b/src/pt/guide/advanced/proxy-headers.md new file mode 100644 index 0000000000..0e06deb66b --- /dev/null +++ b/src/pt/guide/advanced/proxy-headers.md @@ -0,0 +1,436 @@ +# Proxy configuration + +When you use a reverse proxy server (e.g. nginx), the value of `request.ip` will contain the IP of a proxy, typically `127.0.0.1`. Almost always, this is **not** what you will want. + +Sanic may be configured to use proxy headers for determining the true client IP, available as `request.remote_addr`. The full external URL is also constructed from header fields _if available_. + +::: tip Heads up +Without proper precautions, a malicious client may use proxy headers to spoof its own IP. To avoid such issues, Sanic does not use any proxy headers unless explicitly enabled. +::: + +---:1 + +Services behind reverse proxies must configure one or more of the following [configuration values](/guide/deployment/configuration.md): + +- `FORWARDED_SECRET` +- `REAL_IP_HEADER` +- `PROXIES_COUNT` +:--:1 +```python +app.config.FORWARDED_SECRET = "super-duper-secret" +app.config.REAL_IP_HEADER = "CF-Connecting-IP" +app.config.PROXIES_COUNT = 2 +``` +:--- + +## Forwarded header + +In order to use the `Forwarded` header, you should set `app.config.FORWARDED_SECRET` to a value known to the trusted proxy server. The secret is used to securely identify a specific proxy server. + +Sanic ignores any elements without the secret key, and will not even parse the header if no secret is set. + +All other proxy headers are ignored once a trusted forwarded element is found, as it already carries complete information about the client. + +To learn more about the `Forwarded` header, read the related [MDN](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Forwarded) and [Nginx](https://www.nginx.com/resources/wiki/start/topics/examples/forwarded/) articles. + +## Traditional proxy headers + +### IP Headers + +When your proxy forwards you the IP address in a known header, you can tell Sanic what that is with the `REAL_IP_HEADER` config value. + +### X-Forwarded-For + +This header typically contains a chain of IP addresses through each layer of a proxy. Setting `PROXIES_COUNT` tells Sanic how deep to look to get an actual IP address for the client. This value should equal the _expected_ number of IP addresses in the chain. + +### Other X-headers + +If a client IP is found by one of these methods, Sanic uses the following headers for URL parts: + +- x-forwarded-proto +- x-forwarded-host +- x-forwarded-port +- x-forwarded-path +- x-scheme + +## Examples + +In the following examples, all requests will assume that the endpoint looks like this: +```python +@app.route("/fwd") +async def forwarded(request): + return json( + { + "remote_addr": request.remote_addr, + "scheme": request.scheme, + "server_name": request.server_name, + "server_port": request.server_port, + "forwarded": request.forwarded, + } + ) +``` +---:1 +--- + +##### Example 1 +Without configured FORWARDED_SECRET, x-headers should be respected +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: for=1.1.1.1, for=injected;host=", for="[::2]";proto=https;host=me.tld;path="/app/";secret=mySecret,for=broken;;secret=b0rked, for=127.0.0.3;scheme=http;port=1234' \ + -H "X-Real-IP: 127.0.0.2" \ + -H "X-Forwarded-For: 127.0.1.1" \ + -H "X-Scheme: ws" \ + -H "Host: local.site" | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "127.0.0.2", + "scheme": "ws", + "server_name": "local.site", + "server_port": 80, + "forwarded": { + "for": "127.0.0.2", + "proto": "ws" + } +} +``` +:--- +--- +---:1 + +##### Example 2 +FORWARDED_SECRET now configured +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: for=1.1.1.1, for=injected;host=", for="[::2]";proto=https;host=me.tld;path="/app/";secret=mySecret,for=broken;;secret=b0rked, for=127.0.0.3;scheme=http;port=1234' \ + -H "X-Real-IP: 127.0.0.2" \ + -H "X-Forwarded-For: 127.0.1.1" \ + -H "X-Scheme: ws" \ + -H "Host: local.site" | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "[::2]", + "scheme": "https", + "server_name": "me.tld", + "server_port": 443, + "forwarded": { + "for": "[::2]", + "proto": "https", + "host": "me.tld", + "path": "/app/", + "secret": "mySecret" + } +} +``` +:--- +--- +---:1 + +##### Example 3 +Empty Forwarded header -> use X-headers +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H "X-Real-IP: 127.0.0.2" \ + -H "X-Forwarded-For: 127.0.1.1" \ + -H "X-Scheme: ws" \ + -H "Host: local.site" | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "127.0.0.2", + "scheme": "ws", + "server_name": "local.site", + "server_port": 80, + "forwarded": { + "for": "127.0.0.2", + "proto": "ws" + } +} +``` +:--- +--- +---:1 + +##### Example 4 +Header present but not matching anything +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H "Forwarded: nomatch" | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "", + "scheme": "http", + "server_name": "localhost", + "server_port": 8000, + "forwarded": {} +} + +``` +:--- +--- +---:1 + +##### Example 5 +Forwarded header present but no matching secret -> use X-headers +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H "Forwarded: for=1.1.1.1;secret=x, for=127.0.0.1" \ + -H "X-Real-IP: 127.0.0.2" | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "127.0.0.2", + "scheme": "http", + "server_name": "localhost", + "server_port": 8000, + "forwarded": { + "for": "127.0.0.2" + } +} +``` +:--- +--- +---:1 + +##### Example 6 +Different formatting and hitting both ends of the header +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: Secret="mySecret";For=127.0.0.4;Port=1234' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "127.0.0.4", + "scheme": "http", + "server_name": "localhost", + "server_port": 1234, + "forwarded": { + "secret": "mySecret", + "for": "127.0.0.4", + "port": 1234 + } +} +``` +:--- +--- +---:1 + +##### Example 7 +Test escapes (modify this if you see anyone implementing quoted-pairs) +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: for=test;quoted="\,x=x;y=\";secret=mySecret' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "test", + "scheme": "http", + "server_name": "localhost", + "server_port": 8000, + "forwarded": { + "for": "test", + "quoted": "\\,x=x;y=\\", + "secret": "mySecret" + } +} +``` +:--- +--- +---:1 + +##### Example 8 +Secret insulated by malformed field #1 +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: for=test;secret=mySecret;b0rked;proto=wss;' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "test", + "scheme": "http", + "server_name": "localhost", + "server_port": 8000, + "forwarded": { + "for": "test", + "secret": "mySecret" + } +} +``` +:--- +--- +---:1 + +##### Example 9 +Secret insulated by malformed field #2 +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: for=test;b0rked;secret=mySecret;proto=wss' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "", + "scheme": "wss", + "server_name": "localhost", + "server_port": 8000, + "forwarded": { + "secret": "mySecret", + "proto": "wss" + } +} +``` +:--- +--- +---:1 + +##### Example 10 +Unexpected termination should not lose existing acceptable values +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: b0rked;secret=mySecret;proto=wss' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "", + "scheme": "wss", + "server_name": "localhost", + "server_port": 8000, + "forwarded": { + "secret": "mySecret", + "proto": "wss" + } +} +``` +:--- +--- +---:1 + +##### Example 11 +Field normalization +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "mySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: PROTO=WSS;BY="CAFE::8000";FOR=unknown;PORT=X;HOST="A:2";PATH="/With%20Spaces%22Quoted%22/sanicApp?key=val";SECRET=mySecret' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "", + "scheme": "wss", + "server_name": "a", + "server_port": 2, + "forwarded": { + "proto": "wss", + "by": "[cafe::8000]", + "host": "a:2", + "path": "/With Spaces\"Quoted\"/sanicApp?key=val", + "secret": "mySecret" + } +} +``` +:--- +--- +---:1 + +##### Example 12 +Using "by" field as secret +```python +app.config.PROXIES_COUNT = 1 +app.config.REAL_IP_HEADER = "x-real-ip" +app.config.FORWARDED_SECRET = "_proxySecret" +``` +```bash +$ curl localhost:8000/fwd \ + -H 'Forwarded: for=1.2.3.4; by=_proxySecret' | jq +``` +:--:1 +```bash +# curl response +{ + "remote_addr": "1.2.3.4", + "scheme": "http", + "server_name": "localhost", + "server_port": 8000, + "forwarded": { + "for": "1.2.3.4", + "by": "_proxySecret" + } +} + +``` +:--- diff --git a/src/pt/guide/advanced/signals.md b/src/pt/guide/advanced/signals.md new file mode 100644 index 0000000000..9d2c522f36 --- /dev/null +++ b/src/pt/guide/advanced/signals.md @@ -0,0 +1,277 @@ +# Signals + +Signals provide a way for one part of your application to tell another part that something happened. + +```python +@app.signal("user.registration.created") +async def send_registration_email(**context): + await send_email(context["email"], template="registration") + +@app.post("/register") +async def handle_registration(request): + await do_registration(request) + await request.app.dispatch( + "user.registration.created", + context={"email": request.json.email} + }) +``` + +## Adding a signal + +---:1 +The API for adding a signal is very similar to adding a route. +:--:1 +```python +async def my_signal_handler(): + print("something happened") + +app.add_signal(my_signal_handler, "something.happened.ohmy") +``` +:--- + +---:1 +But, perhaps a slightly more convenient method is to use the built-in decorators. +:--:1 +```python +@app.signal("something.happened.ohmy") +async def my_signal_handler(): + print("something happened") +``` +:--- + +---:1 +Signals can also be declared on blueprints +:--:1 +```python +bp = Blueprint("foo") + +@bp.signal("something.happened.ohmy") +async def my_signal_handler(): + print("something happened") +``` +:--- + +## Built-in signals + +In addition to creating a new signal, there are a number of built-in signals that are dispatched from Sanic itself. These signals exist to provide developers with more opportunities to add functionality into the request and server lifecycles. + +---:1 + + +You can attach them just like any other signal to an application or blueprint instance. + +:--:1 + +```python +@app.signal("http.lifecycle.complete") +async def my_signal_handler(conn_info): + print("Connection has been closed") +``` +:--- + +These signals are the signals that are available, along with the arguments that the handlers take, and the conditions that attach (if any). + + +| Event name | Arguments | Conditions | +| -------------------------- | ------------------------------- | --------------------------------------------------------- | +| `http.routing.before` | request | | +| `http.routing.after` | request, route, kwargs, handler | | +| `http.lifecycle.begin` | conn_info | | +| `http.lifecycle.read_head` | head | | +| `http.lifecycle.request` | request | | +| `http.lifecycle.handle` | request | | +| `http.lifecycle.read_body` | body | | +| `http.lifecycle.exception` | request, exception | | +| `http.lifecycle.response` | request, response | | +| `http.lifecycle.send` | data | | +| `http.lifecycle.complete` | conn_info | | +| `http.middleware.before` | request, response | `{"attach_to": "request"}` or `{"attach_to": "response"}` | +| `http.middleware.after` | request, response | `{"attach_to": "request"}` or `{"attach_to": "response"}` | +| `server.init.before` | app, loop | | +| `server.init.after` | app, loop | | +| `server.shutdown.before` | app, loop | | +| `server.shutdown.after` | app, loop | | + +---:1 +To make using the built-in signals easier, there is an `Enum` object that contains all of the allowed built-ins. With a modern IDE this will help so that you do not need to remember the full list of event names as strings. +:--:1 +```python +from sanic.signals import Event + +@app.signal(Event.HTTP_LIFECYCLE_COMPLETE) +async def my_signal_handler(conn_info): + print("Connection has been closed") +``` +:--- + +## Events + +---:1 +Signals are based off of an _event_. An event, is simply a string in the following pattern: +:--:1 +``` +namespace.reference.action +``` +:--- + +::: tip +Events must have three parts. If you do not know what to use, try these patterns: + +- `my_app.something.happened` +- `sanic.notice.hello` +::: + +### Event parameters + +---:1 +An event can be "dynamic" and declared using the same syntax as [path parameters](../basics/routing.md#path-parameters). This allows matching based upon arbitrary values. +:--:1 +```python +@app.signal("foo.bar.") +async def signal_handler(thing): + print(f"[signal_handler] {thing=}") + +@app.get("/") +async def trigger(request): + await app.dispatch("foo.bar.baz") + return response.text("Done.") +``` +:--- + +Checkout [path parameters](../basics/routing.md#path-parameters) for more information on allowed type definitions. + +::: warning +Only the third part of an event (the "action") may be dynamic: + +- `foo.bar.` :ok: +- `foo..baz` :x: +::: + +### Waiting + +---:1 +In addition to executing a signal handler, your application can wait for an event to be triggered. +:--:1 +```python +await app.event("foo.bar.baz") +``` +:--- + +---:1 +**IMPORTANT**: waiting is a blocking function. Therefore, you likely will want this to run in a [background task](../basics/tasks.md). +:--:1 +```python +async def wait_for_event(app): + while True: + print("> waiting") + await app.event("foo.bar.baz") + print("> event found\n") + +@app.after_server_start +async def after_server_start(app, loop): + app.add_task(wait_for_event(app)) +``` +:--- + +---:1 +If your event was defined with a dynamic path, you can use `*` to catch any action. +:--:1 +```python +@app.signal("foo.bar.") + +... + +await app.event("foo.bar.*") +``` +:--- + +## Dispatching + +*In the future, Sanic will dispatch some events automatically to assist developers to hook into life cycle events.* + +---:1 +Dispatching an event will do two things: + +1. execute any signal handlers defined on the event, and +2. resolve anything that is "waiting" for the event to complete. +:--:1 +```python +@app.signal("foo.bar.") +async def foo_bar(thing): + print(f"{thing=}") + +await app.dispatch("foo.bar.baz") +``` +``` +thing=baz +``` +:--- + +### Context + +---:1 +Sometimes you may find the need to pass extra information into the signal handler. In our first example above, we wanted our email registration process to have the email address for the user. +:--:1 +```python +@app.signal("user.registration.created") +async def send_registration_email(**context): + print(context) + +await app.dispatch( + "user.registration.created", + context={"hello": "world"} +) +``` +``` +{'hello': 'world'} +``` +:--- + +::: tip FYI +Signals are dispatched in a background task. +::: + +### Blueprints + +Dispatching blueprint signals works similar in concept to [middleware](../basics/middleware.md). Anything that is done from the app level, will trickle down to the blueprints. However, dispatching on a blueprint, will only execute the signals that are defined on that blueprint. + +---:1 +Perhaps an example is easier to explain: +:--:1 +```python +bp = Blueprint("bp") + +app_counter = 0 +bp_counter = 0 + +@app.signal("foo.bar.baz") +def app_signal(): + nonlocal app_counter + app_counter += 1 + +@bp.signal("foo.bar.baz") +def bp_signal(): + nonlocal bp_counter + bp_counter += 1 +``` +:--- + +---:1 +Running `app.dispatch("foo.bar.baz")` will execute both signals. +:--:1 +```python +await app.dispatch("foo.bar.baz") +assert app_counter == 1 +assert bp_counter == 1 +``` +:--- + +---:1 +Running `bp.dispatch("foo.bar.baz")` will execute only the blueprint signal. +:--:1 +```python +await bp.dispatch("foo.bar.baz") +assert app_counter == 1 +assert bp_counter == 2 +``` +:--- diff --git a/src/pt/guide/advanced/streaming.md b/src/pt/guide/advanced/streaming.md new file mode 100644 index 0000000000..c02a6e7b56 --- /dev/null +++ b/src/pt/guide/advanced/streaming.md @@ -0,0 +1,163 @@ +# Streaming + +## Request streaming + +Sanic allows you to stream data sent by the client to begin processing data as the bytes arrive. + +---:1 + +When enabled on an endpoint, you can stream the request body using `await request.stream.read()`. + +That method will return `None` when the body is completed. +:--:1 +```python +from sanic.views import stream + +class SimpleView(HTTPMethodView): + @stream + async def post(self, request): + result = "" + while True: + body = await request.stream.read() + if body is None: + break + result += body.decode("utf-8") + return text(result) +``` +:--- + +---:1 + +It also can be enabled with a keyword argument in the decorator... +:--:1 +```python +@app.post("/stream", stream=True) +async def handler(request): + ... + body = await request.stream.read() + ... +``` +:--- + +---:1 + +... or the `add_route()` method. +:--:1 +```python +bp.add_route( + bp_handler, + "/bp_stream", + methods=["POST"], + stream=True, +) +``` +:--- + +::: tip FYI +Only post, put and patch decorators have stream argument. +::: + +## Response streaming + +---:1 + +Sanic allows you to stream content to the client with an instance of `StreamingHTTPResponse`. There is also a `sanic.response.stream` convenience method. + +This method accepts a coroutine callback which is passed an object that can control writing to the client. +:--:1 +```python +from sanic.response import stream + +@app.route("/") +async def test(request): + async def sample_streaming_fn(response): + await response.write("foo,") + await response.write("bar") + + return stream(sample_streaming_fn, content_type="text/csv") +``` +:--- + +This is useful in situations where you want to stream content to the client that originates in an external service, like a database. For example, you can stream database records to the client with the asynchronous cursor that `asyncpg` provides. + +```python +@app.route("/") +async def index(request): + async def stream_from_db(response): + conn = await asyncpg.connect(database='test') + async with conn.transaction(): + async for record in conn.cursor('SELECT generate_series(0, 10)'): + await response.write(record[0]) + + return stream(stream_from_db) +``` + +::: tip FYI +If a client supports HTTP/1.1, Sanic will use [chunked transfer encoding](https://en.wikipedia.org/wiki/Chunked_transfer_encoding); you can explicitly enable or disable it using chunked option of the stream function. +::: + +---:1 + +The coroutine callback pattern using `stream` is not needed. It is the *old style* of streaming, and should be replaced with the newer inline streaming. You now are able to stream the response directly in the handler. + +:--:1 +```python +@app.route("/") +async def test(request): + response = await request.respond(content_type="text/csv") + await response.send("foo,") + await response.send("bar") + + # Optionally, you can explicitly end the stream by calling: + await response.eof() +``` +:--- + +In the example above `await response.eof()` is called as a convenience method to replace `await response.send("", True)`. It should be called **one time** *after* your handler has determined that it has nothing left to send back to the client. + + +## File streaming + +---:1 + +Sanic provides `sanic.response.file_stream` function that is useful when you want to send a large file. It returns a `StreamingHTTPResponse` object and will use chunked transfer encoding by default; for this reason Sanic doesn’t add `Content-Length` HTTP header in the response. + +A typical use case might be streaming an video file. +:--:1 +```python +@app.route("/mp4") +async def handler_file_stream(request): + return await response.file_stream( + "/path/to/sample.mp4", + chunk_size=1024, + mime_type="application/metalink4+xml", + headers={ + "Content-Disposition": 'Attachment; filename="nicer_name.meta4"', + "Content-Type": "application/metalink4+xml", + }, + ) +``` +:--- + +---:1 + +If you want to use the `Content-Length` header, you can disable chunked transfer encoding and add it manually simply by adding the `Content-Length` header. + +:--:1 +```python +from aiofiles import os as async_os +from sanic.response import file_stream + +@app.route("/") +async def index(request): + file_path = "/srv/www/whatever.png" + + file_stat = await async_os.stat(file_path) + headers = {"Content-Length": str(file_stat.st_size)} + + return await file_stream( + file_path, + headers=headers, + ) +``` +:--- diff --git a/src/pt/guide/advanced/versioning.md b/src/pt/guide/advanced/versioning.md new file mode 100644 index 0000000000..f5339de513 --- /dev/null +++ b/src/pt/guide/advanced/versioning.md @@ -0,0 +1,157 @@ +# Versioning + +It is standard practice in API building to add versions to your endpoints. This allows you to easily differentiate incompatible endpoints when you try and change your API down the road in a breaking manner. + +Adding a version will add a `/v{version}` url prefix to your endpoints. + +The version can be a `int`, `float`, or `str`. Acceptable values: + +- `1`, `2`, `3` +- `1.1`, `2.25`, `3.0` +- `"1"`, `"v1"`, `"v1.1"` + +## Per route + +---:1 + +You can pass a version number to the routes directly. +:--:1 +```python +# /v1/text +@app.route("/text", version=1) +def handle_request(request): + return response.text("Hello world! Version 1") + +# /v2/text +@app.route("/text", version=2) +def handle_request(request): + return response.text("Hello world! Version 2") +``` +:--- + +## Per Blueprint + +---:1 + +You can also pass a version number to the blueprint, which will apply to all routes in that blueprint. +:--:1 +```python +bp = Blueprint("test", url_prefix="/foo", version=1) + +# /v1/foo/html +@bp.route("/html") +def handle_request(request): + return response.html("

Hello world!

") +``` +:--- + +## Per Blueprint Group + +---:1 +In order to simplify the management of the versioned blueprints, you can provide a version number in the blueprint +group. The same will be inherited to all the blueprint grouped under it if the blueprints don't already override the +same information with a value specified while creating a blueprint instance. + +When using blueprint groups for managing the versions, the following order is followed to apply the Version prefix to +the routes being registered. + +1. Route Level configuration +2. Blueprint level configuration +3. Blueprint Group level configuration + +If we find a more pointed versioning specification, we will pick that over the more generic versioning specification +provided under the Blueprint or Blueprint Group +:--:1 +```python +from sanic.blueprints import Blueprint +from sanic.response import json + +bp1 = Blueprint( + name="blueprint-1", + url_prefix="/bp1", + version=1.25, +) +bp2 = Blueprint( + name="blueprint-2", + url_prefix="/bp2", +) + +group = Blueprint.group( + [bp1, bp2], + url_prefix="/bp-group", + version="v2", +) + +# GET /v1.25/bp-group/bp1/endpoint-1 +@bp1.get("/endpoint-1") +async def handle_endpoint_1_bp1(request): + return json({"Source": "blueprint-1/endpoint-1"}) + +# GET /v2/bp-group/bp2/endpoint-2 +@bp2.get("/endpoint-1") +async def handle_endpoint_1_bp2(request): + return json({"Source": "blueprint-2/endpoint-1"}) + +# GET /v1/bp-group/bp2/endpoint-2 +@bp2.get("/endpoint-2", version=1) +async def handle_endpoint_2_bp2(request): + return json({"Source": "blueprint-2/endpoint-2"}) +``` +:--- + +## Version prefix + +As seen above, the `version` that is applied to a route is **always** the first segment in the generated URI path. Therefore, to make it possible to add path segments before the version, every place that a `version` argument is passed, you can also pass `version_prefix`. + +The `version_prefix` argument can be defined in: + +- `app.route` and `bp.route` decorators (and all the convenience decorators also) +- `Blueprint` instantiation +- `Blueprint.group` constructor +- `BlueprintGroup` instantiation +- `app.blueprint` registration + +If there are definitions in multiple places, a more specific definition overrides a more general. This list provides that hierarchy. + +The default value of `version_prefix` is `/v`. + +---:1 +An often requested feature is to be able to mount versioned routes on `/api`. This can easily be accomplished with `version_prefix`. +:--:1 +```python +# /v1/my/path +app.route("/my/path", version=1, version_prefix="/api/v") +``` +:--- + +---:1 +Perhaps a more compelling usage is to load all `/api` routes into a single `BlueprintGroup`. +:--:1 +```python +# /v1/my/path +app = Sanic(__name__) +v2ip = Blueprint("v2ip", url_prefix="/ip", version=2) +api = Blueprint.group(v2ip, version_prefix="/api/version") + +# /api/version2/ip +@v2ip.get("/") +async def handler(request): + return text(request.ip) + +app.blueprint(api) +``` +:--- + +We can therefore learn that a route's URI is: + +``` +version_prefix + version + url_prefix + URI definition +``` + +::: tip +Just like with `url_prefix`, it is possible to define path parameters inside a `version_prefix`. It is perfectly legitimate to do this. Just remember that every route will have that parameter injected into the handler. + +```python +version_prefix="//v" +``` +::: diff --git a/src/pt/guide/advanced/websockets.md b/src/pt/guide/advanced/websockets.md new file mode 100644 index 0000000000..b3310aef34 --- /dev/null +++ b/src/pt/guide/advanced/websockets.md @@ -0,0 +1,58 @@ +# Websockets + +Sanic provides an easy to use abstraction on top of [websockets](https://websockets.readthedocs.io/en/stable/). + + +## Routing + +---:1 + +Websocket handlers can be hooked up to the router similar to regular handlers. +:--:1 +```python +async def feed(request, ws): + pass + +app.add_websocket_route(feed, "/feed") +``` +```python +@app.websocket("/feed") +async def feed(request, ws): + pass +``` +:--- + +## Handler + + +---:1 + +Typically, a websocket handler will want to hold open a loop. + +It can then use the `send()` and `recv()` methods on the second object injected into the handler. + +This example is a simple endpoint that echos back to the client messages that it receives. +:--:1 +```python + +@app.websocket("/feed") +async def feed(request, ws): + while True: + data = "hello!" + print("Sending: " + data) + await ws.send(data) + data = await ws.recv() + print("Received: " + data) +``` +:--- +## Configuration + +See [configuration section](/guide/deployment/configuration.md) for more details. +```python +app.config.WEBSOCKET_MAX_SIZE = 2 ** 20 +app.config.WEBSOCKET_MAX_QUEUE = 32 +app.config.WEBSOCKET_READ_LIMIT = 2 ** 16 +app.config.WEBSOCKET_WRITE_LIMIT = 2 ** 16 +app.config.WEBSOCKET_PING_INTERVAL = 20 +app.config.WEBSOCKET_PING_TIMEOUT = 20 +``` diff --git a/src/pt/guide/basics/README.md b/src/pt/guide/basics/README.md new file mode 100644 index 0000000000..25dcc52044 --- /dev/null +++ b/src/pt/guide/basics/README.md @@ -0,0 +1 @@ +# Basics diff --git a/src/pt/guide/basics/app.md b/src/pt/guide/basics/app.md new file mode 100644 index 0000000000..6eb1409f6d --- /dev/null +++ b/src/pt/guide/basics/app.md @@ -0,0 +1,238 @@ +# Sanic Application + +## Instance + +---:1 +The most basic building block is the `Sanic()` instance. It is not required, but the custom is to instantiate this in a file called `server.py`. +:--:1 +```python +# /path/to/server.py + +from sanic import Sanic + +app = Sanic("MyHelloWorldApp") +``` +:--- + +## Application context + +Most applications will have the need to share/reuse data or objects across different parts of the code base. The most common example is DB connections. + +---:1 +In versions of Sanic prior to v21.3, this was commonly done by attaching an attribute to the application instance +:--:1 +```python +# Raises a warning as deprecated feature in 21.3 +app = Sanic("MyApp") +app.db = Database() +``` +:--- + +---:1 +Because this can create potential problems with name conflicts, and to be consistent with [request context](./request.md#context) objects, v21.3 introduces application level context object. +:--:1 +```python +# Correct way to attach objects to the application +app = Sanic("MyApp") +app.ctx.db = Database() +``` +:--- + +## App Registry + +---:1 + +When you instantiate a Sanic instance, that can be retrieved at a later time from the Sanic app registry. This can be useful, for example, if you need to access your Sanic instance from a location where it is not otherwise accessible. +:--:1 +```python +# ./path/to/server.py +from sanic import Sanic + +app = Sanic("my_awesome_server") + +# ./path/to/somewhere_else.py +from sanic import Sanic + +app = Sanic.get_app("my_awesome_server") +``` +:--- + +---:1 + +If you call `Sanic.get_app("non-existing")` on an app that does not exist, it will raise `SanicException` by default. You can, instead, force the method to return a new instance of Sanic with that name. +:--:1 +```python +app = Sanic.get_app( + "non-existing", + force_create=True, +) +``` +:--- + +---:1 +If there is **only one** Sanic instance registered, then calling `Sanic.get_app()` with no arguments will return that instance +:--:1 +```python +Sanic("My only app") + +app = Sanic.get_app() +``` +:--- + +## Configuration + +---:1 +Sanic holds the configuration in the `config` attribute of the `Sanic` instance. Configuration can be modified **either** using dot-notation **OR** like a dictionary. +:--:1 +```python +app = Sanic('myapp') + +app.config.DB_NAME = 'appdb' +app.config['DB_USER'] = 'appuser' + +db_settings = { + 'DB_HOST': 'localhost', + 'DB_NAME': 'appdb', + 'DB_USER': 'appuser' +} +app.config.update(db_settings) +``` +:--- + +::: tip Heads up +Config keys _should_ be uppercase. But, this is mainly by convention, and lowercase will work most of the time. +``` +app.config.GOOD = "yay!" +app.config.bad = "boo" +``` +::: + +There is much [more detail about configuration](/guide/deployment/configuration.md) later on. + + +## Customization + +The Sanic application instance can be customized for your application needs in a variety of ways at instantiation. + +### Custom configuration +---:1 + +This simplest form of custom configuration would be to pass your own object directly into that Sanic application instance + +If you create a custom configuration object, it is *highly* recommended that you subclass the Sanic `Config` option to inherit its behavior. You could use this option for adding properties, or your own set of custom logic. + +:--:1 +```python +from sanic.config import Config + +class MyConfig(Config): + FOO = "bar" + +app = Sanic(..., config=MyConfig()) +``` +:--- + +---:1 +A useful example of this feature would be if you wanted to use a config file in a form that differs from what is [supported](../deployment/configuration.md#using-sanic-update-config). +:--:1 +```python +from sanic import Sanic, text +from sanic.config import Config + +class TomlConfig(Config): + def __init__(self, *args, path: str, **kwargs): + super().__init__(*args, **kwargs) + + with open(path, "r") as f: + self.apply(toml.load(f)) + + def apply(self, config): + self.update(self._to_uppercase(config)) + + def _to_uppercase(self, obj: Dict[str, Any]) -> Dict[str, Any]: + retval: Dict[str, Any] = {} + for key, value in obj.items(): + upper_key = key.upper() + if isinstance(value, list): + retval[upper_key] = [ + self._to_uppercase(item) for item in value + ] + elif isinstance(value, dict): + retval[upper_key] = self._to_uppercase(value) + else: + retval[upper_key] = value + return retval + +toml_config = TomlConfig(path="/path/to/config.toml") +app = Sanic(toml_config.APP_NAME, config=toml_config) +``` +:--- +### Custom context +---:1 + +By default, the application context is a [`SimpleNamespace()`](https://docs.python.org/3/library/types.html#types.SimpleNamespace) that allows you to set any properties you want on it. However, you also have the option of passing any object whatsoever instead. + +:--:1 +```python +app = Sanic(..., ctx=1) +``` + +```python +app = Sanic(..., ctx={}) +``` + +```python +class MyContext: + ... + +app = Sanic(..., ctx=MyContext()) +``` +:--- +### Custom requests +---:1 +It is sometimes helpful to have your own `Request` class, and tell Sanic to use that instead of the default. One example is if you wanted to modify the default `request.id` generator. + +::: tip Important + +It is important to remember that you are passing the *class* not an instance of the class. + +::: +:--:1 +```python +import time + +from sanic import Request, Sanic, text + + +class NanoSecondRequest(Request): + @classmethod + def generate_id(*_): + return time.time_ns() + + +app = Sanic(..., request_class=NanoSecondRequest) + + +@app.get("/") +async def handler(request): + return text(str(request.id)) +``` +:--- + +### Custom error handler + +---:1 +See [exception handling](../best-practices/exceptions.md#custom-error-handling) for more +:--:1 +```python +from sanic.handlers import ErrorHandler + +class CustomErrorHandler(ErrorHandler): + def default(self, request, exception): + ''' handles errors that have no error handlers assigned ''' + # You custom error handling logic... + return super().default(request, exception) + +app = Sanic(..., error_handler=CustomErrorHandler()) +``` +:--- diff --git a/src/pt/guide/basics/cookies.md b/src/pt/guide/basics/cookies.md new file mode 100644 index 0000000000..4db4308b36 --- /dev/null +++ b/src/pt/guide/basics/cookies.md @@ -0,0 +1,76 @@ +# Cookies + +## Reading + +---:1 + +Cookies can be accessed via the `Request` object’s `cookies` dictionary. +:--:1 +```python +@app.route("/cookie") +async def test(request): + test_cookie = request.cookies.get("test") + return text("Test cookie: {}".format(test_cookie)) +``` +:--- + + +## Writing + +---:1 + +When returning a response, cookies can be set on the `Response` object: `response.cookies`. This object is an instance of `CookieJar` which is a special sort of dictionary that automatically will write the response headers for you. +:--:1 +```python +@app.route("/cookie") +async def test(request): + response = text("There's a cookie up in this response") + response.cookies["test"] = "It worked!" + response.cookies["test"]["domain"] = ".yummy-yummy-cookie.com" + response.cookies["test"]["httponly"] = True + return response +``` +:--- + +Response cookies can be set like dictionary values and have the following parameters available: + +- `expires: datetime` - The time for the cookie to expire on the client’s browser. +- `path: str` - The subset of URLs to which this cookie applies. Defaults to `/`. +- `comment: str` - A comment (metadata). +- `domain: str` - Specifies the domain for which the cookie is valid. An explicitly specified domain must always start with a dot. +- `max-age: int` - Number of seconds the cookie should live for. +- `secure: bool` - Specifies whether the cookie will only be sent via HTTPS. +- `httponly: bool` - Specifies whether the cookie cannot be read by JavaScript. +- `samesite: str` - Default is browser dependent, specification states (Lax, Strict, and None) are valid values. + +## Deleting + +---:1 + +Cookies can be removed semantically or explicitly. +:--:1 +```python +@app.route("/cookie") +async def test(request): + response = text("Time to eat some cookies muahaha") + + # This cookie will be set to expire in 0 seconds + del response.cookies["kill_me"] + + # This cookie will self destruct in 5 seconds + response.cookies["short_life"] = "Glad to be here" + response.cookies["short_life"]["max-age"] = 5 + del response.cookies["favorite_color"] + + # This cookie will remain unchanged + response.cookies["favorite_color"] = "blue" + response.cookies["favorite_color"] = "pink" + del response.cookies["favorite_color"] + + return response +``` +:--- + +## Eating + +I like cookies :cookie: diff --git a/src/pt/guide/basics/handlers.md b/src/pt/guide/basics/handlers.md new file mode 100644 index 0000000000..c90f8aca6c --- /dev/null +++ b/src/pt/guide/basics/handlers.md @@ -0,0 +1,120 @@ +# Handlers + +The next important building block are your _handlers_. These are also sometimes called "views". + +In Sanic, a handler is any callable that takes at least a `Request` instance as an argument, and returns either an `HTTPResponse` instance, or a coroutine that does the same. + + + +---:1 + +Huh? :confused: + +It is a **function**; either synchronous or asynchronous. + +The job of the handler is to respond to an endpoint and do something. This is where the majority of your business logic will go. +:--:1 +```python +def i_am_a_handler(request): + return HTTPResponse() + +async def i_am_ALSO_a_handler(request): + return HTTPResponse() +``` +:--- + +::: tip Heads up +If you want to learn more about encapsulating your logic, checkout [class based views](/guide/advanced/class-based-views.md). +::: +---:1 +Then, all you need to do is wire it up to an endpoint. We'll learn more about [routing soon](./routing.md). + +Let's look at a practical example. + +- We use a convenience decorator on our app instance: `@app.get()` +- And a handy convenience method for generating out response object: `text()` + +Mission accomplished :muscle: +:--:1 +```python +from sanic.response import text + +@app.get("/foo") +async def foo_handler(request): + return text("I said foo!") +``` +:--- + +--- + +## A word about _async_... + +---:1 + +It is entirely possible to write handlers that are synchronous. + +In this example, we are using the _blocking_ `time.sleep()` to simulate 100ms of processing time. Perhaps this represents fetching data from a DB, or a 3rd-party website. + +Using four (4) worker processes and a common benchmarking tool: + +- **956** requests in 30.10s +- Or, about **31.76** requests/second +:--:1 +```python +@app.get("/sync") +def sync_handler(request): + time.sleep(0.1) + return text("Done.") +``` +:--- + +---:1 + +Just by changing to the asynchronous alternative `asyncio.sleep()`, we see an incredible change in performance. :rocket: + +Using the same four (4) worker processes: + +- **115,590** requests in 30.08s +- Or, about **3,843.17** requests/second + +:flushed: + +Okay... this is a ridiculously overdramatic result. And any benchmark you see is inherently very biased. This example is meant to over-the-top show the benefit of `async/await` in the web world. Results will certainly vary. Tools like Sanic and other async Python libraries are not magic bullets that make things faster. They make them _more efficient_. + +In our example, the asynchronous version is so much better because while one request is sleeping, it is able to start another one, and another one, and another one, and another one... + +But, this is the point! Sanic is fast because it takes the available resources and squeezes performance out of them. It can handle many requests concurrently, which means more requests per second. + +:--:1 +```python +@app.get("/async") +async def async_handler(request): + await asyncio.sleep(0.1) + return text("Done.") +``` +:--- + +::: warning A common mistake! + +Don't do this! You need to ping a website. What do you use? `pip install your-fav-request-library` :see_no_evil: + +Instead, try using a client that is `async/await` capable. Your server will thank you. Avoid using blocking tools, and favor those that play well in the asynchronous ecosystem. If you need recommendations, check out [Awesome Sanic](https://github.com/mekicha/awesome-sanic). + +Sanic uses [httpx](https://www.python-httpx.org/) inside of its testing package (sanic-testing) :wink:. + +::: + +--- + +## A fully annotated handler + +For those that are using type annotations... + +```python +from sanic.response import HTTPResponse, text +from sanic.request import Request + +@app.get("/typed") +async def typed_handler(request: Request) -> HTTPResponse: + return text("Done.") +``` diff --git a/src/pt/guide/basics/headers.md b/src/pt/guide/basics/headers.md new file mode 100644 index 0000000000..8640124d74 --- /dev/null +++ b/src/pt/guide/basics/headers.md @@ -0,0 +1,234 @@ +# Headers + +Request and response headers are available in the `Request` and `HTTPResponse` objects, respectively. They make use of the [`multidict` package](https://multidict.readthedocs.io/en/stable/multidict.html#cimultidict) that allows a single key to have multiple values. + +::: tip FYI + +Header keys are converted to *lowercase* when parsed. Capitalization is not considered for headers. + +::: + +## Request + +Sanic does attempt to do some normalization on request headers before presenting them to the developer, and also make some potentially meaningful extractions for common use cases. + +---:1 + +#### Tokens + +Authorization tokens in the form `Token ` or `Bearer ` are extracted to the request object: `request.token`. + +:--:1 + +```python +@app.route("/") +async def handler(request): + return text(request.token) +``` + +```bash +$ curl localhost:8000 \ + -H "Authorization: Token ABCDEF12345679" +ABCDEF12345679 +``` + +```bash +$ curl localhost:8000 \ + -H "Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c" +eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c +``` + +:--- + +---:1 + +#### Proxy headers + +Sanic has special handling for proxy headers. See the [proxy headers](/guide/advanced/proxy-headers.md) section for more details. + +#### Host header and dynamic URL construction + +The *effective host* is available via `request.host`. This is not necessarily the same as the host header, as it prefers proxy-forwarded host and can be forced by the server name setting. + +Webapps should generally use this accessor so that they can function the same no matter how they are deployed. The actual host header, if needed, can be found via `request.headers` + +The effective host is also used in dynamic URL construction via `request.url_for`, which uses the request to determine the external address of a handler. + +::: tip Be wary of malicious clients +These URLs can be manipulated by sending misleading host headers. `app.url_for` should be used instead if this is a concern. +::: + +:--:1 + +```python +app.config.SERVER_NAME = "https://example.com" + +@app.route("/hosts", name="foo") +async def handler(request): + return json( + { + "effective host": request.host, + "host header": request.headers.get("host"), + "forwarded host": request.forwarded.get("host"), + "you are here": request.url_for("foo"), + } + ) +``` + +```bash +$ curl localhost:8000/hosts +{ + "effective host": "example.com", + "host header": "localhost:8000", + "forwarded host": null, + "you are here": "https://example.com/hosts" +} +``` + +:--- + +---:1 +#### Other headers + +All request headers are available on `request.headers`, and can be accessed in dictionary form. Capitalization is not considered for headers, and can be accessed using either uppercase or lowercase keys. + +:--:1 + +```python +@app.route("/") +async def handler(request): + return json( + { + "foo_weakref": request.headers["foo"], + "foo_get": request.headers.get("Foo"), + "foo_getone": request.headers.getone("FOO"), + "foo_getall": request.headers.getall("fOo"), + "all": list(request.headers.items()), + } + ) +``` + +```bash +$ curl localhost:9999/headers -H "Foo: one" -H "FOO: two"|jq +{ + "foo_weakref": "one", + "foo_get": "one", + "foo_getone": "one", + "foo_getall": [ + "one", + "two" + ], + "all": [ + [ + "host", + "localhost:9999" + ], + [ + "user-agent", + "curl/7.76.1" + ], + [ + "accept", + "*/*" + ], + [ + "foo", + "one" + ], + [ + "foo", + "two" + ] + ] +} +``` + +:--- + +::: tip FYI +💡 The request.headers object is one of a few types that is a dictionary with each value being a list. This is because HTTP allows a single key to be reused to send multiple values. + +Most of the time you will want to use the .get() or .getone() methods to access the first element and not a list. If you do want a list of all items, you can use .getall(). +::: + +#### Request ID + +---:1 + +Often it is convenient or necessary to track a request by its `X-Request-ID` header. You can easily access that as: `request.id`. + +:--:1 + +```python +@app.route("/") +async def handler(request): + return text(request.id) +``` + +```bash +$ curl localhost:8000 \ + -H "X-Request-ID: ABCDEF12345679" +ABCDEF12345679 +``` + +:--- + +## Response + +Sanic will automatically set the following response headers (when appropriate) for you: + +- `content-length` +- `content-type` +- `connection` +- `transfer-encoding` + +In most circumstances, you should never need to worry about setting these headers. + +---:1 + +Any other header that you would like to set can be done either in the route handler, or a response middleware. + +:--:1 + +```python +@app.route("/") +async def handler(request): + return text("Done.", headers={"content-language": "en-US"}) + +@app.middleware("response") +async def add_csp(request, response): + response.headers["content-security-policy"] = "default-src 'none'; script-src 'self'; connect-src 'self'; img-src 'self'; style-src 'self';base-uri 'self';form-action 'self'" +``` + +:--- + +---:1 + +A common [middleware](middleware.md) you might want is to add a `X-Request-ID` header to every response. As stated above: `request.id` will provide the ID from the incoming request. But, even if no ID was supplied in the request headers, one will be automatically supplied for you. + +[See API docs for more details](https://sanic.readthedocs.io/en/latest/sanic/api_reference.html#sanic.request.Request.id) + +:--:1 + +```python +@app.route("/") +async def handler(request): + return text(str(request.id)) + +@app.on_response +async def add_request_id_header(request, response): + response.headers["X-Request-ID"] = request.id +``` + +```bash +$ curl localhost:8000 -i +HTTP/1.1 200 OK +X-Request-ID: 805a958e-9906-4e7a-8fe0-cbe83590431b +content-length: 36 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +805a958e-9906-4e7a-8fe0-cbe83590431b +``` + +:--- diff --git a/src/pt/guide/basics/listeners.md b/src/pt/guide/basics/listeners.md new file mode 100644 index 0000000000..e150f94bf9 --- /dev/null +++ b/src/pt/guide/basics/listeners.md @@ -0,0 +1,234 @@ +# Listeners + +Sanic provides you with eight (8) opportunities to inject an operation into the life cycle of your application server. This does not include the [signals](../advanced/signals.md), which allow further injection customization. + +There are two (2) that run **only** on your main Sanic process (ie, once per call to `sanic server.app`.) + +- `main_process_start` +- `main_process_stop` + +::: new NEW in v22.3 +There are also two (2) that run **only** in a reloader process if auto-reload has been turned on. + +- `reload_process_start` +- `reload_process_stop` +::: + +There are four (4) that enable you to execute startup/teardown code as your server starts or closes. + +- `before_server_start` +- `after_server_start` +- `before_server_stop` +- `after_server_stop` + +The life cycle of a worker process looks like this: + +```mermaid +sequenceDiagram +autonumber +participant Process +participant Worker +participant Listener +participant Handler +Note over Process: sanic server.app +loop + Process->>Listener: @app.main_process_start + Listener->>Handler: Invoke event handler +end +Process->>Worker: Run workers +loop Start each worker + loop + Worker->>Listener: @app.before_server_start + Listener->>Handler: Invoke event handler + end + Note over Worker: Server status: started + loop + Worker->>Listener: @app.after_server_start + Listener->>Handler: Invoke event handler + end + Note over Worker: Server status: ready +end +Process->>Worker: Graceful shutdown +loop Stop each worker + loop + Worker->>Listener: @app.before_server_stop + Listener->>Handler: Invoke event handler + end + Note over Worker: Server status: stopped + loop + Worker->>Listener: @app.after_server_stop + Listener->>Handler: Invoke event handler + end + Note over Worker: Server status: closed +end +loop + Process->>Listener: @app.main_process_stop + Listener->>Handler: Invoke event handler +end +Note over Process: exit +``` + +The reloader process live outside of this worker process inside of a process that is responsible for starting and stopping the Sanic processes. Consider the following example: + +```python +@app.reload_process_start +async def reload_start(*_): + print(">>>>>> reload_start <<<<<<") + + +@app.main_process_start +async def main_start(*_): + print(">>>>>> main_start <<<<<<") +``` + +If this application were run with auto-reload turned on, the `reload_start` function would be called once. This is contrasted with `main_start`, which would be run every time a file is save and the reloader restarts the applicaition process. + +## Attaching a listener + +---:1 + +The process to setup a function as a listener is similar to declaring a route. + +The currently running `Sanic()` instance is injected into the listener. +:--:1 +```python +async def setup_db(app): + app.ctx.db = await db_setup() + +app.register_listener(setup_db, "before_server_start") +``` +:--- + +---:1 + +The `Sanic` app instance also has a convenience decorator. +:--:1 +```python +@app.listener("before_server_start") +async def setup_db(app): + app.ctx.db = await db_setup() +``` +:--- + +::: new NEW in v22.3 +---:1 +Prior to v22.3, both the application instance and the current event loop were injected into the function. However, only the application instance is injected by default. If your function signature will accept both, then both the application and the loop will be injected as shown here. +:--:1 +```python +@app.listener("before_server_start") +async def setup_db(app, loop): + app.ctx.db = await db_setup() +``` +::: + +---:1 + +You can shorten the decorator even further. This is helpful if you have an IDE with autocomplete. + +:--:1 +```python +@app.before_server_start +async def setup_db(app): + app.ctx.db = await db_setup() +``` +:--- + +## Order of execution + +Listeners are executed in the order they are declared during startup, and reverse order of declaration during teardown + +| | Phase | Order | +|-----------------------|-----------------|---------| +| `main_process_start` | main startup | regular :smiley: | +| `before_server_start` | worker startup | regular :smiley: | +| `after_server_start` | worker startup | regular :smiley: | +| `before_server_stop` | worker shutdown | reverse :upside_down_face: | +| `after_server_stop` | worker shutdown | reverse :upside_down_face: | +| `main_process_stop` | main shutdown | reverse :upside_down_face: | + +Given the following setup, we should expect to see this in the console if we run two workers. + +---:1 + +```python +@app.listener("before_server_start") +async def listener_1(app, loop): + print("listener_1") + +@app.before_server_start +async def listener_2(app, loop): + print("listener_2") + +@app.listener("after_server_start") +async def listener_3(app, loop): + print("listener_3") + +@app.after_server_start +async def listener_4(app, loop): + print("listener_4") + +@app.listener("before_server_stop") +async def listener_5(app, loop): + print("listener_5") + +@app.before_server_stop +async def listener_6(app, loop): + print("listener_6") + +@app.listener("after_server_stop") +async def listener_7(app, loop): + print("listener_7") + +@app.after_server_stop +async def listener_8(app, loop): + print("listener_8") +``` +:--:1 +```bash{3-7,13,19-22} +[pid: 1000000] [INFO] Goin' Fast @ http://127.0.0.1:9999 +[pid: 1000000] [INFO] listener_0 +[pid: 1111111] [INFO] listener_1 +[pid: 1111111] [INFO] listener_2 +[pid: 1111111] [INFO] listener_3 +[pid: 1111111] [INFO] listener_4 +[pid: 1111111] [INFO] Starting worker [1111111] +[pid: 1222222] [INFO] listener_1 +[pid: 1222222] [INFO] listener_2 +[pid: 1222222] [INFO] listener_3 +[pid: 1222222] [INFO] listener_4 +[pid: 1222222] [INFO] Starting worker [1222222] +[pid: 1111111] [INFO] Stopping worker [1111111] +[pid: 1222222] [INFO] Stopping worker [1222222] +[pid: 1222222] [INFO] listener_6 +[pid: 1222222] [INFO] listener_5 +[pid: 1222222] [INFO] listener_8 +[pid: 1222222] [INFO] listener_7 +[pid: 1111111] [INFO] listener_6 +[pid: 1111111] [INFO] listener_5 +[pid: 1111111] [INFO] listener_8 +[pid: 1111111] [INFO] listener_7 +[pid: 1000000] [INFO] listener_9 +[pid: 1000000] [INFO] Server Stopped +``` +In the above example, notice how there are three processes running: + +- `pid: 1000000` - The *main* process +- `pid: 1111111` - Worker 1 +- `pid: 1222222` - Worker 2 + +*Just because our example groups all of one worker and then all of another, in reality since these are running on separate processes, the ordering between processes is not guaranteed. But, you can be sure that a single worker will **always** maintain its order.* +:--- + + +::: tip FYI +The practical result of this is that if the first listener in `before_server_start` handler setups a database connection, listeners that are registered after it can rely upon that connection being alive both when they are started and stopped. +::: + +## ASGI Mode + +If you are running your application with an ASGI server, then make note of the following changes: + +- `reload_process_start` and `reload_process_stop` will be **ignored** +- `main_process_start` and `main_process_stop` will be **ignored** +- `before_server_start` will run as early as it can, and will be before `after_server_start`, but technically, the server is already running at that point +- `after_server_stop` will run as late as it can, and will be after `before_server_stop`, but technically, the server is still running at that point diff --git a/src/pt/guide/basics/middleware.md b/src/pt/guide/basics/middleware.md new file mode 100644 index 0000000000..ed43bb4424 --- /dev/null +++ b/src/pt/guide/basics/middleware.md @@ -0,0 +1,201 @@ +# Middleware + +Whereas listeners allow you to attach functionality to the lifecycle of a worker process, middleware allows you to attach functionality to the lifecycle of an HTTP stream. + +You can execute middleware either _before_ the handler is executed, or _after_. + +```mermaid +sequenceDiagram +autonumber +participant Worker +participant Middleware +participant MiddlewareHandler +participant RouteHandler +Note over Worker: Incoming HTTP request +loop + Worker->>Middleware: @app.on_request + Middleware->>MiddlewareHandler: Invoke middleware handler + MiddlewareHandler-->>Worker: Return response (optional) +end +rect rgba(255, 13, 104, .1) +Worker->>RouteHandler: Invoke route handler +RouteHandler->>Worker: Return response +end +loop + Worker->>Middleware: @app.on_response + Middleware->>MiddlewareHandler: Invoke middleware handler + MiddlewareHandler-->>Worker: Return response (optional) +end +Note over Worker: Deliver response +``` +## Attaching middleware + +---:1 + +This should probably look familiar by now. All you need to do is declare when you would like the middleware to execute: on the `request` or on the `response`. +:--:1 +```python +async def extract_user(request): + request.ctx.user = await extract_user_from_request(request) + +app.register_middleware(extract_user, "request") +``` +:--- + +---:1 + +Again, the `Sanic` app instance also has a convenience decorator. +:--:1 +```python +@app.middleware("request") +async def extract_user(request): + request.ctx.user = await extract_user_from_request(request) +``` +:--- + +---:1 + +Response middleware receives both the `request` and `response` arguments. +:--:1 +```python +@app.middleware('response') +async def prevent_xss(request, response): + response.headers["x-xss-protection"] = "1; mode=block" +``` +:--- + +---:1 + +You can shorten the decorator even further. This is helpful if you have an IDE with autocomplete. +:--:1 +```python +@app.on_request +async def extract_user(request): + ... + +@app.on_response +async def prevent_xss(request, response): + ... +``` +:--- + +## Modification + +---:1 + +Middleware can modify the request or response parameter it is given, _as long as it does not return it_. + +#### Order of execution + +1. Request middleware: `add_key` +2. Route handler: `index` +3. Response middleware: `prevent_xss` +4. Response middleware: `custom_banner` +:--:1 +```python +@app.middleware("request") +async def add_key(request): + # Arbitrary data may be stored in request context: + request.ctx.foo = "bar" + + +@app.middleware("response") +async def custom_banner(request, response): + response.headers["Server"] = "Fake-Server" + + +@app.middleware("response") +async def prevent_xss(request, response): + response.headers["x-xss-protection"] = "1; mode=block" + + +@app.get("/") +async def index(request): + return text(request.ctx.foo) + +``` +:--- + + +---:1 +You can modify the `request.match_info`. A useful feature that could be used, for example, in middleware to convert `a-slug` to `a_slug`. +:--:1 +```python +@app.on_request +def convert_slug_to_underscore(request: Request): + request._match_info["slug"] = request._match_info["slug"].replace("-", "_") + + +@app.get("/") +async def handler(request, slug): + return text(slug) +``` +``` +$ curl localhost:9999/foo-bar-baz +foo_bar_baz +``` +:--- +## Responding early + +---:1 + +If middleware returns a `HTTPResponse` object, the request will stop processing and the response will be returned. If this occurs to a request before the route handler is reached, the handler will **not** be called. Returning a response will also prevent any further middleware from running. + +::: tip +You can return a `None` value to stop the execution of the middleware handler to allow the request to process as normal. This can be useful when using early return to avoid processing requests inside of that middleware handler. +::: +:--:1 +```python +@app.middleware("request") +async def halt_request(request): + return text("I halted the request") + +@app.middleware("response") +async def halt_response(request, response): + return text("I halted the response") +``` +:--- + +#### Order of execution + +Request middleware is executed in the order declared. Response middleware is executed in **reverse order**. + +Given the following setup, we should expect to see this in the console. + +---:1 + +```python +@app.middleware("request") +async def middleware_1(request): + print("middleware_1") + + +@app.middleware("request") +async def middleware_2(request): + print("middleware_2") + + +@app.middleware("response") +async def middleware_3(request, response): + print("middleware_3") + + +@app.middleware("response") +async def middleware_4(request, response): + print("middleware_4") + +@app.get("/handler") +async def handler(request): + print("~ handler ~") + return text("Done.") +``` +:--:1 +```bash +middleware_1 +middleware_2 +~ handler ~ +middleware_4 +middleware_3 +[INFO][127.0.0.1:44788]: GET http://localhost:8000/handler 200 5 +``` +:--- diff --git a/src/pt/guide/basics/request.md b/src/pt/guide/basics/request.md new file mode 100644 index 0000000000..a8580098a7 --- /dev/null +++ b/src/pt/guide/basics/request.md @@ -0,0 +1,196 @@ +# Request + +The `Request` instance contains **a lot** of helpful information available on its parameters. Refer to the [API documentation](https://sanic.readthedocs.io/) for full details. + +## Body + +:::: tabs +::: tab JSON + +**Parameter**: `request.json` +**Description**: The parsed JSON object + +```bash +$ curl localhost:8000 -d '{"foo": "bar"}' +``` + +```python +>>> print(request.json) +{'foo': 'bar'} +``` +::: + +::: tab Raw + +**Parameter**: `request.body` +**Description**: The raw bytes from the request body + +```bash +$ curl localhost:8000 -d '{"foo": "bar"}' +``` + +```python +>>> print(request.body) +b'{"foo": "bar"}' +``` +::: + +::: tab Form + +**Parameter**: `request.form` +**Description**: The form data + +```bash +$ curl localhost:8000 -d 'foo=bar' +``` + +```python +>>> print(request.body) +b'foo=bar' + +>>> print(request.form) +{'foo': ['bar']} + +>>> print(request.form.get("foo")) +bar + +>>> print(request.form.getlist("foo")) +['bar'] +``` + +::: tip FYI +:bulb: The `request.form` object is one of a few types that is a dictionary with each value being a list. This is because HTTP allows a single key to be reused to send multiple values. + +Most of the time you will want to use the `.get()` method to access the first element and not a list. If you do want a list of all items, you can use `.getlist()`. +::: + +::: tab Uploaded + +**Parameter**: `request.files` +**Description**: The files uploaded to the server + +```bash +$ curl -F 'my_file=@/path/to/TEST' http://localhost:8000 +``` + +```python +>>> print(request.body) +b'--------------------------cb566ad845ad02d3\r\nContent-Disposition: form-data; name="my_file"; filename="TEST"\r\nContent-Type: application/octet-stream\r\n\r\nhello\n\r\n--------------------------cb566ad845ad02d3--\r\n' + +>>> print(request.files) +{'my_file': [File(type='application/octet-stream', body=b'hello\n', name='TEST')]} + +>>> print(request.files.get("my_file")) +File(type='application/octet-stream', body=b'hello\n', name='TEST') + +>>> print(request.files.getlist("my_file")) +[File(type='application/octet-stream', body=b'hello\n', name='TEST')] +``` +::: tip FYI +:bulb: The `request.files` object is one of a few types that is a dictionary with each value being a list. This is because HTTP allows a single key to be reused to send multiple values. + +Most of the time you will want to use the `.get()` method to access the first element and not a list. If you do want a list of all items, you can use `.getlist()`. +::: + +:::: + +## Context + +### Request context + +The `request.ctx` object is your playground to store whatever information you need to about the request. + +This is often used to store items like authenticated user details. We will get more into [middleware](./middleware.md) later, but here is a simple example. + +```python +@app.on_request +async def run_before_handler(request): + request.ctx.user = await fetch_user_by_token(request.token) + +@app.route('/hi') +async def hi_my_name_is(request): + return text("Hi, my name is {}".format(request.ctx.user.name)) +``` + +A typical use case would be to store the user object acquired from database in an authentication middleware. Keys added are accessible to all later middleware as well as the handler over the duration of the request. + +Custom context is reserved for applications and extensions. Sanic itself makes no use of it. + +### Connection context + +---:1 + +Often times your API will need to serve multiple concurrent (or consecutive) requests to the same client. This happens, for example, very often with progressive web apps that need to query multiple endpoints to get data. + +The HTTP protocol calls for an easing of overhead time caused by the connection with the use of [keep alive headers](../deployment/configuration.md#keep-alive-timeout). + +When multiple requests share a single connection, Sanic provides a context object to allow those requests to share state. + +:--:1 +```python +@app.on_request +async def increment_foo(request): + if not hasattr(request.conn_info.ctx, "foo"): + request.conn_info.ctx.foo = 0 + request.conn_info.ctx.foo += 1 + +@app.get("/") +async def count_foo(request): + return text(f"request.conn_info.ctx.foo={request.conn_info.ctx.foo}") +``` + +```bash +$ curl localhost:8000 localhost:8000 localhost:8000 +request.conn_info.ctx.foo=1 +request.conn_info.ctx.foo=2 +request.conn_info.ctx.foo=3 +``` +:--- + +## Parameters + +---:1 +Values that are extracted from the path are injected into the handler as parameters, or more specifically as keyword arguments. There is much more detail about this in the [Routing section](./routing.md). +:--:1 +```python +@app.route('/tag/') +async def tag_handler(request, tag): + return text("Tag - {}".format(tag)) +``` +:--- + + +## Arguments + +There are two attributes on the `request` instance to get query parameters: + +- `request.args` +- `request.query_args` + +```bash +$ curl http://localhost:8000\?key1\=val1\&key2\=val2\&key1\=val3 +``` + +```python +>>> print(request.args) +{'key1': ['val1', 'val3'], 'key2': ['val2']} + +>>> print(request.args.get("key1")) +val1 + +>>> print(request.args.getlist("key1")) +['val1', 'val3'] + +>>> print(request.query_args) +[('key1', 'val1'), ('key2', 'val2'), ('key1', 'val3')] + +>>> print(request.query_string) +key1=val1&key2=val2&key1=val3 + +``` + +::: tip FYI +:bulb: The `request.args` object is one of a few types that is a dictionary with each value being a list. This is because HTTP allows a single key to be reused to send multiple values. + +Most of the time you will want to use the `.get()` method to access the first element and not a list. If you do want a list of all items, you can use `.getlist()`. +::: diff --git a/src/pt/guide/basics/response.md b/src/pt/guide/basics/response.md new file mode 100644 index 0000000000..e12d2af06c --- /dev/null +++ b/src/pt/guide/basics/response.md @@ -0,0 +1,185 @@ +# Response + +All [handlers](./handlers.md) **must** return a response object, and [middleware](./middleware.md) may optionally return a response object. + +## Methods + +The easiest way to generate a response object is to use one of the nine (9) convenience methods. + +:::: tabs + +::: tab Text + +**Default Content-Type**: `text/plain; charset=utf-8` +**Description**: Returns plain text + +```python +from sanic.response import text + +@app.route("/") +async def handler(request): + return text("Hi 😎") +``` +::: +::: tab HTML + +**Default Content-Type**: `text/html; charset=utf-8` +**Description**: Returns an HTML document + +```python +from sanic.response import html + +@app.route("/") +async def handler(request): + return html('
Hi 😎
') +``` +::: +::: tab JSON + +**Default Content-Type**: `application/json` +**Description**: Returns a JSON document + +```python +from sanic.response import json + +@app.route("/") +async def handler(request): + return json({"foo": "bar"}) +``` + +By default, Sanic ships with [`ujson`](https://github.com/ultrajson/ultrajson) as its JSON encoder of choice. It is super simple to change this if you want. + +```python +from orjson import dumps + +json({"foo": "bar"}, dumps=dumps) +``` + +If `ujson` is not installed, it will fall back to the standard library `json` module. + +You may additionally declare which implementation to use globally across your application at initialization: + +```python +from orjson import dumps + +app = Sanic(..., dumps=dumps) +``` +::: +::: tab File + +**Default Content-Type**: N/A +**Description**: Returns a file + + +```python +from sanic.response import file + +@app.route("/") +async def handler(request): + return await file("/path/to/whatever.png") +``` + +Sanic will examine the file, and try and guess its mime type and use an appropriate value for the content type. You could be explicit, if you would like: + +```python +file("/path/to/whatever.png", mime_type="image/png") +``` + +You can also choose to override the file name: + +```python +file("/path/to/whatever.png", filename="super-awesome-incredible.png") +``` +::: +::: tab Streaming + +**Default Content-Type**: `text/plain; charset=utf-8` +**Description**: Streams data to a client + +```python +from sanic.response import stream + +@app.route("/") +async def handler(request): + return stream(streaming_fn) + +async def streaming_fn(response): + await response.write('foo') + await response.write('bar') +``` +By default, Sanic will stream back to the client using chunked encoding if the client supports it. You can disable this: + +```python +stream(streaming_fn, chunked=False) +``` +::: +::: tab "File Streaming" + +**Default Content-Type**: N/A +**Description**: Streams a file to a client, useful when streaming large files, like a video + +```python +from sanic.response import file_stream + +@app.route("/") +async def handler(request): + return await file_stream("/path/to/whatever.mp4") +``` + +Like the `file()` method, `file_stream()` will attempt to determine the mime type of the file. +::: +::: tab Raw + +**Default Content-Type**: `application/octet-stream` +**Description**: Send raw bytes without encoding the body + +```python +from sanic.response import raw + +@app.route("/") +async def handler(request): + return raw(b"raw bytes") +``` +::: +::: tab Redirect + +**Default Content-Type**: `text/html; charset=utf-8` +**Description**: Send a `302` response to redirect the client to a different path + +```python +from sanic.response import redirect + +@app.route("/") +async def handler(request): + return redirect("/login") +``` + +::: +::: tab Empty + +**Default Content-Type**: N/A +**Description**: For responding with an empty message as defined by [RFC 2616](https://tools.ietf.org/search/rfc2616#section-7.2.1) + +```python +from sanic.response import empty + +@app.route("/") +async def handler(request): + return empty() +``` + +Defaults to a `204` status. +::: +:::: + +## Default status + +The default HTTP status code for the response is `200`. If you need to change it, it can be done by the response method. + + +```python +@app.post("/") +async def create_new(request): + new_thing = await do_create(request) + return json({"created": True, "id": new_thing.thing_id}, status=201) +``` diff --git a/src/pt/guide/basics/routing.md b/src/pt/guide/basics/routing.md new file mode 100644 index 0000000000..26907d875f --- /dev/null +++ b/src/pt/guide/basics/routing.md @@ -0,0 +1,676 @@ +# Routing + +---:1 + +So far we have seen a lot of this decorator in different forms. + +But what is it? And how do we use it? +:--:1 +```python +@app.route("/stairway") +... + +@app.get("/to") +... + +@app.post("/heaven") +... +``` +:--- + +## Adding a route + +---:1 + +The most basic way to wire up a handler to an endpoint is with `app.add_route()`. + +See [API docs](https://sanic.readthedocs.io/en/stable/sanic/api_reference.html#sanic.app.Sanic.url_for) for more details. +:--:1 +```python +async def handler(request): + return text("OK") + +app.add_route(handler, "/test") +``` +:--- + +---:1 + +By default, routes are available as an HTTP `GET` call. You can change a handler to respond to one or more HTTP methods. +:--:1 +```python +app.add_route( + handler, + '/test', + methods=["POST", "PUT"], +) +``` +:--- + +---:1 + +Using the decorator syntax, the previous example is identical to this. +:--:1 +```python +@app.route('/test', methods=["POST", "PUT"]) +async def handler(request): + return text('OK') +``` +:--- + +## HTTP methods + +Each of the standard HTTP methods has a convenience decorator. + +:::: tabs +::: tab GET + +```python +@app.get('/test') +async def handler(request): + return text('OK') +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/GET) +::: +::: tab POST + +```python +@app.post('/test') +async def handler(request): + return text('OK') +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/POST) +::: +::: tab PUT + +```python +@app.put('/test') +async def handler(request): + return text('OK') +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/PUT) +::: +::: tab PATCH + +```python +@app.patch('/test') +async def handler(request): + return text('OK') +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/PATCH) +::: +::: tab DELETE + +```python +@app.delete('/test') +async def handler(request): + return text('OK') +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/DELETE) +::: +::: tab HEAD + +```python +@app.head('/test') +async def handler(request): + return empty() +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/HEAD) +::: +::: tab OPTIONS + +```python +@app.options('/test') +async def handler(request): + return empty() +``` + +[MDN Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/OPTIONS) +::: +:::: + +::: warning +By default, Sanic will **only** consume the incoming request body on non-safe HTTP methods (`POST`, `PUT`, `PATCH`). If you want to receive data in the HTTP request on any other method, you will need to do one of the following two options: + +**Option #1 - Tell Sanic to consume the body using `ignore_body`** +```python +@app.delete("/path", ignore_body=False) +async def handler(_): + ... +``` + +**Option #2 - Manually consume the body in the handler using `receive_body`** +```python +@app.delete("/path") +async def handler(request: Request): + await request.receive_body() +``` +::: + +## Path parameters + +---:1 + +Sanic allows for pattern matching, and for extracting values from URL paths. These parameters are then injected as keyword arguments in the route handler. +:--:1 +```python +@app.get("/tag/") +async def tag_handler(request, tag): + return text("Tag - {}".format(tag)) +``` +:--- + +---:1 + +You can declare a type for the parameter. This will be enforced when matching, and also will type cast the variable. +:--:1 +```python +@app.get("/foo/") +async def uuid_handler(request, foo_id: UUID): + return text("UUID - {}".format(foo_id)) +``` +:--- + +### Supported types + +:::: tabs + +::: tab str + +```python +@app.route("/path/to/") +async def handler(request, foo: str): + ... +``` +**Regular expression applied**: `r"[^/]+")` +**Cast type**: `str` +**Example matches**: +- `/path/to/Bob` +- `/path/to/Python%203` + +::: new NEW in v22.3 +`str` will *not* match on empty strings. See `strorempty` for this behavior. + +::: +::: tab "strorempty 🌟" + +::: new NEW in v22.3 + +```python +@app.route("/path/to/") +async def handler(request, foo: str): + ... +``` +**Regular expression applied**: `r"[^/]*")` +**Cast type**: `str` +**Example matches**: +- `/path/to/Bob` +- `/path/to/Python%203` +- `/path/to/` + +Unlike the `str` path parameter type, `strorempty` can also match on an empty string path segment. + +::: +::: tab int + +```python +@app.route("/path/to/") +async def handler(request, foo: int): + ... +``` +**Regular expression applied**: `r"-?\d+")` +**Cast type**: `int` +**Example matches**: +- `/path/to/10` +- `/path/to/-10` + +_Does not match float, hex, octal, etc_ +::: +::: tab float + +```python +@app.route("/path/to/") +async def handler(request, foo: float): + ... +``` +**Regular expression applied**: `r"-?(?:\d+(?:\.\d*)?|\.\d+)")` +**Cast type**: `float` +**Example matches**: +- `/path/to/10` +- `/path/to/-10` +- `/path/to/1.5` + +::: +::: tab alpha + +```python +@app.route("/path/to/") +async def handler(request, foo: str): + ... +``` +**Regular expression applied**: `r"[A-Za-z]+")` +**Cast type**: `str` +**Example matches**: +- `/path/to/Bob` +- `/path/to/Python` + +_Does not match a digit, or a space or other special character_ +::: +::: tab slug + +```python +@app.route("/path/to/") +async def handler(request, article: str): + ... +``` +**Regular expression applied**: `r"[a-z0-9]+(?:-[a-z0-9]+)*")` +**Cast type**: `str` +**Example matches**: +- `/path/to/some-news-story` +- `/path/to/or-has-digits-123` + +::: +::: tab path + +```python +@app.route("/path/to/") +async def handler(request, foo: str): + ... +``` +**Regular expression applied**: `r"[^/].*?")` +**Cast type**: `str` +**Example matches**: +- `/path/to/hello` +- `/path/to/hello.txt` +- `/path/to/hello/world.txt` + +::: warning +Because this will match on `/`, you should be careful and thoroughly test your patterns that use `path` so they do not capture traffic intended for another endpoint. +::: +::: tab ymd + +```python +@app.route("/path/to/") +async def handler(request, foo: datetime.date): + ... +``` +**Regular expression applied**: `r"^([12]\d{3}-(0[1-9]|1[0-2])-(0[1-9]|[12]\d|3[01]))"` +**Cast type**: `datetime.date` +**Example matches**: +- `/path/to/2021-03-28` +::: + +::: tab uuid + +```python +@app.route("/path/to/") +async def handler(request, foo: UUID): + ... +``` +**Regular expression applied**: `r"[A-Fa-f0-9]{8}-[A-Fa-f0-9]{4}-[A-Fa-f0-9]{4}-[A-Fa-f0-9]{4}-[A-Fa-f0-9]{12}"` +**Cast type**: `UUID` +**Example matches**: +- `/path/to/123a123a-a12a-1a1a-a1a1-1a12a1a12345` + +::: + +::: tab "ext 🌟" + +::: new NEW in v22.3 + +```python +@app.route("/path/to/") +async def handler(request, foo: UUID): + ... +``` +**Regular expression applied**: n/a +**Cast type**: *varies* +**Example matches**: + +| definition | example | filename | extension | +| --------------------------------- | ----------- | ----------- | ---------- | +| \ | page.txt | `"page"` | `"txt"` | +| \ | cat.jpg | `"cat"` | `"jpg"` | +| \ | cat.jpg | `"cat"` | `"jpg"` | +| | 123.txt | `123` | `"txt"` | +| | 123.svg | `123` | `"svg"` | +| | 3.14.tar.gz | `3.14` | `"tar.gz"` | + +File extensions can be matched using the special `ext` parameter type. It uses a special format that allows you to specify other types of parameter types as the file name, and one or more specific extensions as shown in the example table above. + +It does *not* support the `path` parameter type. + +::: + +::: tab regex + +```python +@app.route(r"/path/to/") +async def handler(request, foo: str): + ... +``` +**Regular expression applied**: _whatever you insert_ +**Cast type**: `str` +**Example matches**: +- `/path/to/2021-01-01` + +This gives you the freedom to define specific matching patterns for your use case. + +In the example shown, we are looking for a date that is in `YYYY-MM-DD` format. + +:::: + +### Regex Matching + + + +More often than not, compared with complex routing, the above example is too simple, and we use a completely different routing matching pattern, so here we will explain the advanced usage of regex matching in detail. + +Sometimes, you want to match a part of a route: + +```text +/image/123456789.jpg +``` + +If you wanted to match the file pattern, but only capture the numeric portion, you need to do some regex fun 😄: + +```python +app.route(r"/image/\d+)\.jpg>") +``` + +Further, these should all be acceptable: + +```python +@app.get(r"/") # matching on the full pattern +@app.get(r"/") # defining a single matching group +@app.get(r"/[a-z]{3}).txt>") # defining a single named matching group +@app.get(r"/[a-z]{3}).(?:txt)>") # defining a single named matching group, with one or more non-matching groups +``` + +Also, if using a named matching group, it must be the same as the segment label. + +```python +@app.get(r"/\d+).jpg>") # OK +@app.get(r"/\d+).jpg>") # NOT OK +``` + +For more regular usage methods, please refer to [Regular expression operations](https://docs.python.org/3/library/re.html) + +## Generating a URL + +---:1 + +Sanic provides a method to generate URLs based on the handler method name: `app.url_for()`. This is useful if you want to avoid hardcoding url paths into your app; instead, you can just reference the handler name. +:--:1 +```python +@app.route('/') +async def index(request): + # generate a URL for the endpoint `post_handler` + url = app.url_for('post_handler', post_id=5) + + # Redirect to `/posts/5` + return redirect(url) + +@app.route('/posts/') +async def post_handler(request, post_id): + ... +``` +:--- + +---:1 + +You can pass any arbitrary number of keyword arguments. Anything that is _not_ a request parameter will be implemented as a part of the query string. +:--:1 +```python +>>> app.url_for( + "post_handler", + post_id=5, + arg_one="one", + arg_two="two", +) +'/posts/5?arg_one=one&arg_two=two' +``` +:--- + +---:1 + +Also supported is passing multiple values for a single query key. +:--:1 +```python +>>> app.url_for( + "post_handler", + post_id=5, + arg_one=["one", "two"], +) +'/posts/5?arg_one=one&arg_one=two' +``` +:--- + +### Special keyword arguments + +See [API Docs]() for more details. + +```python +>>> app.url_for("post_handler", post_id=5, arg_one="one", _anchor="anchor") +'/posts/5?arg_one=one#anchor' + +# _external requires you to pass an argument _server or set SERVER_NAME in app.config if not url will be same as no _external +>>> app.url_for("post_handler", post_id=5, arg_one="one", _external=True) +'//server/posts/5?arg_one=one' + +# when specifying _scheme, _external must be True +>>> app.url_for("post_handler", post_id=5, arg_one="one", _scheme="http", _external=True) +'http://server/posts/5?arg_one=one' + +# you can pass all special arguments at once +>>> app.url_for("post_handler", post_id=5, arg_one=["one", "two"], arg_two=2, _anchor="anchor", _scheme="http", _external=True, _server="another_server:8888") +'http://another_server:8888/posts/5?arg_one=one&arg_one=two&arg_two=2#anchor' +``` + +### Customizing a route name + +---:1 + +A custom route name can be used by passing a `name` argument while registering the route. +:--:1 +```python +@app.get("/get", name="get_handler") +def handler(request): + return text("OK") +``` +:--- + +---:1 + +Now, use this custom name to retrieve the URL +:--:1 +```python +>>> app.url_for("get_handler", foo="bar") +'/get?foo=bar' +``` +:--- + +## Websockets routes + +---:1 + +Websocket routing works similar to HTTP methods. +:--:1 +```python +async def handler(request, ws): + message = "Start" + while True: + await ws.send(message) + message = await ws.recv() + +app.add_websocket_route(handler, "/test") +``` +:--- + +---:1 + +It also has a convenience decorator. +:--:1 +```python +@app.websocket("/test") +async def handler(request, ws): + message = "Start" + while True: + await ws.send(message) + message = await ws.recv() +``` +:--- + +Read the [websockets section](/guide/advanced/websockets.md) to learn more about how they work. + +## Strict slashes + + +---:1 + +Sanic routes can be configured to strictly match on whether or not there is a trailing slash: `/`. This can be configured at a few levels and follows this order of precedence: + +1. Route +2. Blueprint +3. BlueprintGroup +4. Application + +:--:1 +```python +# provide default strict_slashes value for all routes +app = Sanic(__file__, strict_slashes=True) +``` + +```python +# overwrite strict_slashes value for specific route +@app.get("/get", strict_slashes=False) +def handler(request): + return text("OK") +``` + +```python +# it also works for blueprints +bp = Blueprint(__file__, strict_slashes=True) + +@bp.get("/bp/get", strict_slashes=False) +def handler(request): + return text("OK") +``` + +```python +bp1 = Blueprint(name="bp1", url_prefix="/bp1") +bp2 = Blueprint( + name="bp2", + url_prefix="/bp2", + strict_slashes=False, +) + +# This will enforce strict slashes check on the routes +# under bp1 but ignore bp2 as that has an explicitly +# set the strict slashes check to false +group = Blueprint.group([bp1, bp2], strict_slashes=True) +``` +:--- + +## Static files + +---:1 + +In order to serve static files from Sanic, use `app.static()`. + +The order of arguments is important: + +1. Route the files will be served from +2. Path to the files on the server + +See [API docs]() for more details. +:--:1 +```python +app.static("/static", "/path/to/directory") +``` +:--- + +---:1 + +You can also serve individual files. +:--:1 +```python +app.static("/", "/path/to/index.html") +``` +:--- + +---:1 + +It is also sometimes helpful to name your endpoint +:--:1 +```python +app.static( + "/user/uploads", + "/path/to/uploads", + name="uploads", +) +``` +:--- + +---:1 + +Retrieving the URLs works similar to handlers. But, we can also add the `filename` argument when we need a specific file inside a directory. +:--:1 +```python +>>> app.url_for( + "static", + name="static", + filename="file.txt", +) +'/static/file.txt' + +```python +>>> app.url_for( + "static", + name="uploads", + filename="image.png", +) +'/user/uploads/image.png' + +``` +:--- + +::: tip +If you are going to have multiple `static()` routes, then it is *highly* suggested that you manually name them. This will almost certainly alleviate potential hard to discover bugs. + +```python +app.static("/user/uploads", "/path/to/uploads", name="uploads") +app.static("/user/profile", "/path/to/profile", name="profile_pics") +``` +::: + +## Route context + +---:1 +When a route is defined, you can add any number of keyword arguments with a `ctx_` prefix. These values will be injected into the route `ctx` object. +:--:1 +```python +@app.get("/1", ctx_label="something") +async def handler1(request): + ... + +@app.get("/2", ctx_label="something") +async def handler2(request): + ... + +@app.get("/99") +async def handler99(request): + ... + +@app.on_request +async def do_something(request): + if request.route.ctx.label == "something": + ... +``` +:--- diff --git a/src/pt/guide/basics/tasks.md b/src/pt/guide/basics/tasks.md new file mode 100644 index 0000000000..a8dcce4026 --- /dev/null +++ b/src/pt/guide/basics/tasks.md @@ -0,0 +1,125 @@ +# Background tasks + +## Creating Tasks +It is often desirable and very convenient to make usage of [tasks](https://docs.python.org/3/library/asyncio-task.html#asyncio.create_task) in async Python. Sanic provides a convenient method to add tasks to the currently **running** loop. It is somewhat similar to `asyncio.create_task`. For adding tasks before the 'App' loop is running, see next section. + +```python +async def notify_server_started_after_five_seconds(): + await asyncio.sleep(5) + print('Server successfully started!') + +app.add_task(notify_server_started_after_five_seconds()) +``` + +---:1 + +Sanic will attempt to automatically inject the app, passing it as an argument to the task. +:--:1 +```python +async def auto_inject(app): + await asyncio.sleep(5) + print(app.name) + +app.add_task(auto_inject) +``` +:--- + +---:1 + +Or you can pass the `app` argument explicitly. +:--:1 +```python +async def explicit_inject(app): + await asyncio.sleep(5) + print(app.name) + +app.add_task(explicit_inject(app)) +``` +:--- + +## Adding tasks before `app.run` + +It is possible to add background tasks before the App is run ie. before `app.run`. To add a task before the App is run, it is recommended to not pass the coroutine object (ie. one created by calling the `async` callable), but instead just pass the callable and Sanic will create the coroutine object on **each worker**. Note: the tasks that are added such are run as `before_server_start` jobs and thus run on every worker (and not in the main process). This has certain consequences, please read [this comment](https://github.com/sanic-org/sanic/issues/2139#issuecomment-868993668) on [this issue](https://github.com/sanic-org/sanic/issues/2139) for further details. + +To add work on the main process, consider adding work to [`@app.main_process_start`](./listeners.md). Note: the workers won't start until this work is completed. + +---:1 + +Example to add a task before `app.run` +:---:1 +```python +async def slow_work(): + ... + +async def even_slower(num): + ... + +app = Sanic(...) +app.add_task(slow_work) # Note: we are passing the callable and not coroutine object ... +app.add_task(even_slower(10)) # ... or we can call the function and pass the coroutine. +app.run(...) +``` + +## Named tasks + +_This is only supported in Python 3.8+_ + +---:1 +When creating a task, you can ask Sanic to keep track of it for you by providing a `name`. + +:--:1 +```python +app.add_task(slow_work, name="slow_task") +``` +:--- + +---:1 +You can now retrieve that task instance from anywhere in your application using `get_task`. + +:--:1 +```python +task = app.get_task("slow_task") +``` +:--- + +---:1 +If that task needs to be cancelled, you can do that with `cancel_task`. Make sure that you `await` it. + +:--:1 +```python +await app.cancel_task("slow_task") +``` +:--- + +---:1 +All registered tasks can be found in the `app.tasks` property. To prevent cancelled tasks from filling up, you may want to run `app.purge_tasks` that will clear out any completed or cancelled tasks. + +:--:1 +```python +app.purge_tasks() +``` +:--- + +This pattern can be particularly useful with `websockets`: + +```python +async def receiver(ws): + while True: + message = await ws.recv() + if not message: + break + print(f"Received: {message}") + +@app.websocket("/feed") +async def feed(request, ws): + task_name = f"receiver:{request.id}" + request.app.add_task(receiver(ws), name=task_name) + try: + while True: + await request.app.event("my.custom.event") + await ws.send("A message") + finally: + # When the websocket closes, let's cleanup the task + await request.app.cancel_task(task_name) + request.app.purge_tasks() +::: diff --git a/src/pt/guide/best-practices/README.md b/src/pt/guide/best-practices/README.md new file mode 100644 index 0000000000..874ca7f3c6 --- /dev/null +++ b/src/pt/guide/best-practices/README.md @@ -0,0 +1 @@ +# Best Practices diff --git a/src/pt/guide/best-practices/blueprints.md b/src/pt/guide/best-practices/blueprints.md new file mode 100644 index 0000000000..648e1ba9b1 --- /dev/null +++ b/src/pt/guide/best-practices/blueprints.md @@ -0,0 +1,388 @@ +# Blueprints + +## Overview + +Blueprints are objects that can be used for sub-routing within an application. Instead of adding routes to the application instance, blueprints define similar methods for adding routes, which are then registered with the application in a flexible and pluggable manner. + +Blueprints are especially useful for larger applications, where your application logic can be broken down into several groups or areas of responsibility. + +## Creating and registering + +---:1 + +First, you must create a blueprint. It has a very similar API as the `Sanic()` app instance with many of the same decorators. +:--:1 +```python +# ./my_blueprint.py +from sanic.response import json +from sanic import Blueprint + +bp = Blueprint("my_blueprint") + +@bp.route("/") +async def bp_root(request): + return json({"my": "blueprint"}) +``` +:--- + + +---:1 + +Next, you register it with the app instance. +:--:1 +```python +from sanic import Sanic +from my_blueprint import bp + +app = Sanic(__name__) +app.blueprint(bp) +``` +:--- + +Blueprints also have the same `websocket()` decorator and `add_websocket_route` method for implementing websockets. + +---:1 + +Beginning in v21.12, a Blueprint may be registered before or after adding objects to it. Previously, only objects attached to the Blueprint at the time of registration would be loaded into application instance. +:--:1 +```python +app.blueprint(bp) + +@bp.route("/") +async def bp_root(request): + ... +``` +:--- +## Copying + +---:1 + +Blueprints along with everything that is attached to them can be copied to new instances using the `copy()` method. The only required argument is to pass it a new `name`. However, you could also use this to override any of the values from the old blueprint. + +:--:1 + +```python +v1 = Blueprint("Version1", version=1) + +@v1.route("/something") +def something(request): + pass + +v2 = v1.copy("Version2", version=2) + +app.blueprint(v1) +app.blueprint(v2) +``` + +``` +Available routes: +/v1/something +/v2/something + +``` + +:--- + +## Blueprint groups + +Blueprints may also be registered as part of a list or tuple, where the registrar will recursively cycle through any sub-sequences of blueprints and register them accordingly. The Blueprint.group method is provided to simplify this process, allowing a ‘mock’ backend directory structure mimicking what’s seen from the front end. Consider this (quite contrived) example: + +```text +api/ +├──content/ +│ ├──authors.py +│ ├──static.py +│ └──__init__.py +├──info.py +└──__init__.py +app.py +``` + +---:1 + +#### First blueprint + +:--:1 +```python +# api/content/authors.py +from sanic import Blueprint + +authors = Blueprint("content_authors", url_prefix="/authors") +``` +:--- + +---:1 + +#### Second blueprint + +:--:1 +```python +# api/content/static.py +from sanic import Blueprint + +static = Blueprint("content_static", url_prefix="/static") +``` +:--- + +---:1 + +#### Blueprint group + +:--:1 +```python +# api/content/__init__.py +from sanic import Blueprint +from .static import static +from .authors import authors + +content = Blueprint.group(static, authors, url_prefix="/content") +``` +:--- + +---:1 + +#### Third blueprint + +:--:1 +```python +# api/info.py +from sanic import Blueprint + +info = Blueprint("info", url_prefix="/info") +``` +:--- + +---:1 + +#### Another blueprint group + +:--:1 +```python +# api/__init__.py +from sanic import Blueprint +from .content import content +from .info import info + +api = Blueprint.group(content, info, url_prefix="/api") +``` +:--- + +---:1 + +#### Main server + +All blueprints are now registered + +:--:1 +```python +# app.py +from sanic import Sanic +from .api import api + +app = Sanic(__name__) +app.blueprint(api) +``` +:--- + +## Middleware + +---:1 + +Blueprints can also have middleware that is specifically registered for its endpoints only. +:--:1 +```python +@bp.middleware +async def print_on_request(request): + print("I am a spy") + +@bp.middleware("request") +async def halt_request(request): + return text("I halted the request") + +@bp.middleware("response") +async def halt_response(request, response): + return text("I halted the response") +``` +:--- + +---:1 + +Similarly, using blueprint groups, it is possible to apply middleware to an entire group of nested blueprints. +:--:1 +```python +bp1 = Blueprint("bp1", url_prefix="/bp1") +bp2 = Blueprint("bp2", url_prefix="/bp2") + +@bp1.middleware("request") +async def bp1_only_middleware(request): + print("applied on Blueprint : bp1 Only") + +@bp1.route("/") +async def bp1_route(request): + return text("bp1") + +@bp2.route("/") +async def bp2_route(request, param): + return text(param) + +group = Blueprint.group(bp1, bp2) + +@group.middleware("request") +async def group_middleware(request): + print("common middleware applied for both bp1 and bp2") + +# Register Blueprint group under the app +app.blueprint(group) +``` +:--- + +## Exceptions + +---:1 + +Just like other [exception handling](./exceptions.md), you can define blueprint specific handlers. +:--:1 +```python +@bp.exception(NotFound) +def ignore_404s(request, exception): + return text("Yep, I totally found the page: {}".format(request.url)) +``` +:--- + +## Static files + +---:1 + +Blueprints can also have their own static handlers +:--:1 +```python +bp = Blueprint("bp", url_prefix="/bp") +bp.static("/web/path", "/folder/to/serve") +bp.static("/web/path", "/folder/to/server", name="uploads") +``` +:--- + +---:1 + +Which can then be retrieved using `url_for()`. See [routing](/guide/basics/routing.md) for more information. +:--:1 +```python +>>> print(app.url_for("static", name="bp.uploads", filename="file.txt")) +'/bp/web/path/file.txt' +``` +:--- + +## Listeners + +---:1 + +Blueprints can also implement [listeners](/guide/basics/listeners.md). +:--:1 +```python +@bp.listener("before_server_start") +async def before_server_start(app, loop): + ... + +@bp.listener("after_server_stop") +async def after_server_stop(app, loop): + ... +``` +:--- + +## Versioning + +As discussed in the [versioning section](/guide/advanced/versioning.md), blueprints can be used to implement different versions of a web API. + +---:1 + +The `version` will be prepended to the routes as `/v1` or `/v2`, etc. +:--:1 +```python +auth1 = Blueprint("auth", url_prefix="/auth", version=1) +auth2 = Blueprint("auth", url_prefix="/auth", version=2) +``` +:--- + +---:1 + +When we register our blueprints on the app, the routes `/v1/auth` and `/v2/auth` will now point to the individual blueprints, which allows the creation of sub-sites for each API version. +:--:1 +```python +from auth_blueprints import auth1, auth2 + +app = Sanic(__name__) +app.blueprint(auth1) +app.blueprint(auth2) +``` +:--- + +---:1 + +It is also possible to group the blueprints under a `BlueprintGroup` entity and version multiple of them together at the +same time. +:--:1 +```python +auth = Blueprint("auth", url_prefix="/auth") +metrics = Blueprint("metrics", url_prefix="/metrics") + +group = Blueprint.group(auth, metrics, version="v1") + +# This will provide APIs prefixed with the following URL path +# /v1/auth/ and /v1/metrics +``` +:--- + +## Composable + +A `Blueprint` may be registered to multiple groups, and each of `BlueprintGroup` itself could be registered and nested further. This creates a limitless possibility `Blueprint` composition. + +---:1 +Take a look at this example and see how the two handlers are actually mounted as five (5) distinct routes. +:--:1 +```python +app = Sanic(__name__) +blueprint_1 = Blueprint("blueprint_1", url_prefix="/bp1") +blueprint_2 = Blueprint("blueprint_2", url_prefix="/bp2") +group = Blueprint.group( + blueprint_1, + blueprint_2, + version=1, + version_prefix="/api/v", + url_prefix="/grouped", + strict_slashes=True, +) +primary = Blueprint.group(group, url_prefix="/primary") + + +@blueprint_1.route("/") +def blueprint_1_default_route(request): + return text("BP1_OK") + + +@blueprint_2.route("/") +def blueprint_2_default_route(request): + return text("BP2_OK") + + +app.blueprint(group) +app.blueprint(primary) +app.blueprint(blueprint_1) + +# The mounted paths: +# /api/v1/grouped/bp1/ +# /api/v1/grouped/bp2/ +# /api/v1/primary/grouped/bp1 +# /api/v1/primary/grouped/bp2 +# /bp1 + +``` +:--- + + +## Generating a URL + +When generating a url with `url_for()`, the endpoint name will be in the form: + +```text +{blueprint_name}.{handler_name} +``` diff --git a/src/pt/guide/best-practices/decorators.md b/src/pt/guide/best-practices/decorators.md new file mode 100644 index 0000000000..d7b13b0149 --- /dev/null +++ b/src/pt/guide/best-practices/decorators.md @@ -0,0 +1,183 @@ +# Decorators + +One of the best ways to create a consistent and DRY web API is to make use of decorators to remove functionality from the handlers, and make it repeatable across your views. + +---:1 + +Therefore, it is very common to see a Sanic view handler with several decorators on it. +:--:1 +```python +@app.get("/orders") +@authorized("view_order") +@validate_list_params() +@inject_user() +async def get_order_details(request, params, user): + ... +``` +:--- + + +## Example + +Here is a starter template to help you create decorators. + +In this example, let’s say you want to check that a user is authorized to access a particular endpoint. You can create a decorator that wraps a handler function, checks a request if the client is authorized to access a resource, and sends the appropriate response. +```python +from functools import wraps +from sanic.response import json + +def authorized(): + def decorator(f): + @wraps(f) + async def decorated_function(request, *args, **kwargs): + # run some method that checks the request + # for the client's authorization status + is_authorized = await check_request_for_authorization_status(request) + + if is_authorized: + # the user is authorized. + # run the handler method and return the response + response = await f(request, *args, **kwargs) + return response + else: + # the user is not authorized. + return json({"status": "not_authorized"}, 403) + return decorated_function + return decorator + + +@app.route("/") +@authorized() +async def test(request): + return json({"status": "authorized"}) +``` + +## Templates + +Decorators are **fundamental** to building applications with Sanic. They increase the portability and maintainablity of your code. + +In paraphrasing the Zen of Python: "[decorators] are one honking great idea -- let's do more of those!" + +To make it easier to implement them, here are three examples of copy/pastable code to get you started. + +---:1 + +Don't forget to add these import statements. Although it is *not* necessary, using `@wraps` helps keep some of the metadata of your function in tact. [See docs](https://docs.python.org/3/library/functools.html#functools.wraps). Also, we use the `isawaitable` pattern here to allow the route handlers to by regular or asynchronous functions. + +:--:1 + +```python +from inspect import isawaitable +from functools import wraps +``` + +:--- + +### With args + +---:1 + +Often, you will want a decorator that will *always* need arguments. Therefore, when it is implemented you will always be calling it. + +```python +@app.get("/") +@foobar(1, 2) +async def handler(request: Request): + return text("hi") +``` + +:--:1 + +```python +def foobar(arg1, arg2): + def decorator(f): + @wraps(f) + async def decorated_function(request, *args, **kwargs): + + response = f(request, *args, **kwargs) + if isawaitable(response): + response = await response + + return response + + return decorated_function + + return decorator +``` + +:--- + +### Without args + +---:1 + +Sometimes you want a decorator that will not take arguments. When this is the case, it is a nice convenience not to have to call it + +```python +@app.get("/") +@foobar +async def handler(request: Request): + return text("hi") +``` + +:--:1 + +```python +def foobar(func): + def decorator(f): + @wraps(f) + async def decorated_function(request, *args, **kwargs): + + response = f(request, *args, **kwargs) + if isawaitable(response): + response = await response + + return response + + return decorated_function + + return decorator(func) +``` + +:--- + +### With or Without args + +---:1 + +If you want a decorator with the ability to be called or not, you can follow this pattern. Using keyword only arguments is not necessary, but might make implementation simpler. + +```python +@app.get("/") +@foobar(arg1=1, arg2=2) +async def handler(request: Request): + return text("hi") +``` + +```python +@app.get("/") +@foobar +async def handler(request: Request): + return text("hi") +``` + +:--:1 + +```python +def foobar(maybe_func=None, *, arg1=None, arg2=None): + def decorator(f): + @wraps(f) + async def decorated_function(request, *args, **kwargs): + + response = f(request, *args, **kwargs) + if isawaitable(response): + response = await response + + return response + + return decorated_function + + return decorator(maybe_func) if maybe_func else decorator +``` + +:--- diff --git a/src/pt/guide/best-practices/exceptions.md b/src/pt/guide/best-practices/exceptions.md new file mode 100644 index 0000000000..fa318ee75e --- /dev/null +++ b/src/pt/guide/best-practices/exceptions.md @@ -0,0 +1,496 @@ +# Exceptions + +## Using Sanic exceptions + +Sometimes you just need to tell Sanic to halt execution of a handler and send back a status code response. You can raise a `SanicException` for this and Sanic will do the rest for you. + +You can pass an optional `status_code` argument. By default, a SanicException will return an internal server error 500 response. + +```python +from sanic.exceptions import SanicException + +@app.route("/youshallnotpass") +async def no_no(request): + raise SanicException("Something went wrong.", status_code=501) +``` + +Sanic provides a number of standard exceptions. They each automatically will raise the appropriate HTTP status code in your response. [Check the API reference](https://sanic.readthedocs.io/en/latest/sanic/api_reference.html#module-sanic.exceptions) for more details. + +---:1 + +The more common exceptions you _should_ implement yourself include: + +- `InvalidUsage` (400) +- `Unauthorized` (401) +- `Forbidden` (403) +- `NotFound` (404) +- `ServerError` (500) + +:--:1 + +```python +from sanic import exceptions + +@app.route("/login") +async def login(request): + user = await some_login_func(request) + if not user: + raise exceptions.NotFound( + f"Could not find user with username={request.json.username}" + ) + ... +``` + +:--- + +## Exception properties + +All exceptions in Sanic derive from `SanicException`. That class has a few properties on it that assist the developer in consistently reporting their exceptions across an application. + +- `message` +- `status_code` +- `quiet` +- `context` +- `extra` + +All of these properties can be passed to the exception when it is created, but the first three can also be used as class variables as we will see. + +---:1 +### `message` + +The `message` property obviously controls the message that will be displayed as with any other exception in Python. What is particularly useful is that you can set the `message` property on the class definition allowing for easy standardization of language across an application +:--:1 +```python +class CustomError(SanicException): + message = "Something bad happened" + +raise CustomError +# or +raise CustomError("Override the default message with something else") +``` +:--- + +---:1 +### `status_code` + +This property is used to set the response code when the exception is raised. This can particularly be useful when creating custom 400 series exceptions that are usually in response to bad information coming from the client. +:--:1 +```python +class TeapotError(SanicException): + status_code = 418 + message = "Sorry, I cannot brew coffee" + +raise TeapotError +# or +raise TeapotError(status_code=400) +``` +:--- + +---:1 +### `quiet` + +By default, exceptions will be output by Sanic to the `error_logger`. Sometimes this may not be desirable, especially if you are using exceptions to trigger events in exception handlers (see [the following section](./exceptions.md#handling)). You can suppress the log output using `quiet=True`. +:--:1 +```python +class SilentError(SanicException): + message = "Something happened, but not shown in logs" + quiet = True + +raise SilentError +# or +raise InvalidUsage("blah blah", quiet=True) +``` +:--- + +---:1 + +Sometimes while debugging you may want to globally ignore the `quiet=True` property. You can force Sanic to log out all exceptions regardless of this property using `NOISY_EXCEPTIONS` + +:--:1 +```python +app.config.NOISY_EXCEPTIONS = True +``` +:--- + +---:1 + +### `extra` + +See [contextual exceptions](./exceptions.md#contextual-exceptions) + +:--:1 +```python +raise SanicException(..., extra={"name": "Adam"}) +``` +:--- + +---:1 + +### `context` + +See [contextual exceptions](./exceptions.md#contextual-exceptions) + +:--:1 +```python +raise SanicException(..., context={"foo": "bar"}) +``` +:--- + + +## Handling + +Sanic handles exceptions automatically by rendering an error page, so in many cases you don't need to handle them yourself. However, if you would like more control on what to do when an exception is raised, you can implement a handler yourself. + +Sanic provides a decorator for this, which applies to not only the Sanic standard exceptions, but **any** exception that your application might throw. + +---:1 + +The easiest method to add a handler is to use `@app.exception()` and pass it one or more exceptions. + +:--:1 + +```python +from sanic.exceptions import NotFound + +@app.exception(NotFound, SomeCustomException) +async def ignore_404s(request, exception): + return text("Yep, I totally found the page: {}".format(request.url)) +``` + +:--- + +---:1 + +You can also create a catchall handler by catching `Exception`. + +:--:1 + +```python +@app.exception(Exception) +async def catch_anything(request, exception): + ... +``` + +:--- + +---:1 + +You can also use `app.error_handler.add()` to add error handlers. + +:--:1 + +```python +async def server_error_handler(request, exception): + return text("Oops, server error", status=500) + +app.error_handler.add(Exception, server_error_handler) +``` + +:--- + +## Built-in error handling + +Sanic ships with three formats for exceptions: HTML, JSON, and text. You can see examples of them below in the [Fallback handler](#fallback-handler) section. + +---:1 + +You can control _per route_ which format to use with the `error_format` keyword argument. + +:--:1 + +```python +@app.request("/", error_format="text") +async def handler(request): + ... +``` + +:--- + + +## Custom error handling + +In some cases, you might want to add some more error handling functionality to what is provided by default. In that case, you can subclass Sanic's default error handler as such: + +```python +from sanic.handlers import ErrorHandler + +class CustomErrorHandler(ErrorHandler): + def default(self, request, exception): + ''' handles errors that have no error handlers assigned ''' + # You custom error handling logic... + return super().default(request, exception) + +app.error_handler = CustomErrorHandler() +``` + +## Fallback handler + +Sanic comes with three fallback exception handlers: + +1. HTML (*default*) +2. Text +3. JSON + +These handlers present differing levels of detail depending upon whether your application is in [debug mode](/guide/deployment/development.md) or not. + +### HTML + +```python +app.config.FALLBACK_ERROR_FORMAT = "html" +``` + +---:1 + +```python +app.config.DEBUG = True +``` + +![Error](~@assets/images/error-html-debug.png) + +:--:1 + +```python +app.config.DEBUG = False +``` + +![Error](~@assets/images/error-html-no-debug.png) + +:--- + +### Text + +```python +app.config.FALLBACK_ERROR_FORMAT = "text" +``` + +---:1 + +```python +app.config.DEBUG = True +``` + +```bash +$ curl localhost:8000/exc -i +HTTP/1.1 500 Internal Server Error +content-length: 590 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +⚠️ 500 — Internal Server Error +============================== +That time when that thing broke that other thing? That happened. + +ServerError: That time when that thing broke that other thing? That happened. while handling path /exc +Traceback of __BASE__ (most recent call last): + + ServerError: That time when that thing broke that other thing? That happened. + File /path/to/sanic/app.py, line 986, in handle_request + response = await response + + File /path/to/server.py, line 222, in exc + raise ServerError( +``` + +:--:1 + +```python +app.config.DEBUG = False +``` + +```bash +$ curl localhost:8000/exc -i +HTTP/1.1 500 Internal Server Error +content-length: 134 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +⚠️ 500 — Internal Server Error +============================== +That time when that thing broke that other thing? That happened. +``` + +:--- + +### JSON + +```python +app.config.FALLBACK_ERROR_FORMAT = "json" +``` + +---:1 + +```python +app.config.DEBUG = True +``` + +```bash +$ curl localhost:8000/exc -i +HTTP/1.1 500 Internal Server Error +content-length: 129 +connection: keep-alive +content-type: application/json + +{ + "description": "Internal Server Error", + "status": 500, + "message": "That time when that thing broke that other thing? That happened.", + "path": "/exc", + "args": {}, + "exceptions": [ + { + "type": "ServerError", + "exception": "That time when that thing broke that other thing? That happened.", + "frames": [ + { + "file": "/path/to/sanic/app.py", + "line": 986, + "name": "handle_request", + "src": "response = await response" + }, + { + "file": "/path/to/server.py", + "line": 222, + "name": "exc", + "src": "raise ServerError(" + } + ] + } + ] +} + + +``` + +:--:1 + +```python +app.config.DEBUG = False +``` + +```bash +$ curl localhost:8000/exc -i +HTTP/1.1 500 Internal Server Error +content-length: 530 +connection: keep-alive +content-type: application/json + +{ + "description": "Internal Server Error", + "status": 500, + "message": "That time when that thing broke that other thing? That happened." +} + +``` + +:--- + +### Auto + +Sanic also provides an option for guessing which fallback option to use. This is still an **experimental feature**. + +```python +app.config.FALLBACK_ERROR_FORMAT = "auto" +``` +## Contextual Exceptions + +Default exception messages that simplify the ability to consistently raise exceptions throughout your application. + +```python +class TeapotError(SanicException): + status_code = 418 + message = "Sorry, I cannot brew coffee" + +raise TeapotError +``` + +But this lacks two things: + +1. A dynamic and predictable message format +2. The ability to add additional context to an error message (more on this in a moment) + +### Dynamic and predictable message using `extra` + +Sanic exceptions can be raised using `extra` keyword arguments to provide additional information to a raised exception instance. + +```python +class TeapotError(SanicException): + status_code = 418 + + @property + def message(self): + return f"Sorry {self.extra['name']}, I cannot make you coffee" + +raise TeapotError(extra={"name": "Adam"}) +``` + +The new feature allows the passing of `extra` meta to the exception instance, which can be particularly useful as in the above example to pass dynamic data into the message text. This `extra` info object **will be suppressed** when in `PRODUCTION` mode, but displayed in `DEVELOPMENT` mode. + +---:1 +**PRODUCTION** + +![image](https://user-images.githubusercontent.com/166269/139014161-cda67cd1-843f-4ad2-9fa1-acb94a59fc4d.png) +:--:1 +**DEVELOPMENT** + +![image](https://user-images.githubusercontent.com/166269/139014121-0596b084-b3c5-4adb-994e-31ba6eba6dad.png) +:--- + +### Additional `context` to an error message + +Sanic exceptions can also be raised with a `context` argument to pass intended information along to the user about what happened. This is particularly useful when creating microservices or an API intended to pass error messages in JSON format. In this use case, we want to have some context around the exception beyond just a parseable error message to return details to the client. + +```python +raise TeapotError(context={"foo": "bar"}) +``` + +This is information **that we want** to always be passed in the error (when it is available). Here is what it should look like: + +---:1 +**PRODUCTION** + +```json +{ + "description": "I'm a teapot", + "status": 418, + "message": "Sorry Adam, I cannot make you coffee", + "context": { + "foo": "bar" + } +} +``` +:--:1 +**DEVELOPMENT** + +```json +{ + "description": "I'm a teapot", + "status": 418, + "message": "Sorry Adam, I cannot make you coffee", + "context": { + "foo": "bar" + }, + "path": "/", + "args": {}, + "exceptions": [ + { + "type": "TeapotError", + "exception": "Sorry Adam, I cannot make you coffee", + "frames": [ + { + "file": "handle_request", + "line": 83, + "name": "handle_request", + "src": "" + }, + { + "file": "/tmp/p.py", + "line": 17, + "name": "handler", + "src": "raise TeapotError(" + } + ] + } + ] +} +``` +:--- diff --git a/src/pt/guide/best-practices/logging.md b/src/pt/guide/best-practices/logging.md new file mode 100644 index 0000000000..01f797d9c7 --- /dev/null +++ b/src/pt/guide/best-practices/logging.md @@ -0,0 +1,93 @@ +# Logging + +Sanic allows you to do different types of logging (access log, error log) on the requests based on the [Python logging API](https://docs.python.org/3/howto/logging.html). You should have some basic knowledge on Python logging if you want to create a new configuration. + +## Quick Start + +---:1 + +A simple example using default settings would be like this: +:--:1 +```python +from sanic import Sanic +from sanic.log import logger +from sanic.response import text + +app = Sanic('logging_example') + +@app.route('/') +async def test(request): + logger.info('Here is your log') + return text('Hello World!') + +if __name__ == "__main__": + app.run(debug=True, access_log=True) +``` +:--- + +After the server is running, you should see logs like this. +```text +[2021-01-04 15:26:26 +0200] [1929659] [INFO] Goin' Fast @ http://127.0.0.1:8000 +[2021-01-04 15:26:26 +0200] [1929659] [INFO] Starting worker [1929659] +``` + +You can send a request to server and it will print the log messages. +```text +[2021-01-04 15:26:28 +0200] [1929659] [INFO] Here is your log +[2021-01-04 15:26:28 +0200] - (sanic.access)[INFO][127.0.0.1:44228]: GET http://localhost:8000/ 200 -1 +``` + +## Changing Sanic loggers + +To use your own logging config, simply use `logging.config.dictConfig`, or pass `log_config` when you initialize Sanic app. + +```python +app = Sanic('logging_example', log_config=LOGGING_CONFIG) + +if __name__ == "__main__": + app.run(access_log=False) +``` + +::: tip FYI +Logging in Python is a relatively cheap operation. However, if you are serving a high number of requests and performance is a concern, all of that time logging out access logs adds up and becomes quite expensive. + +This is a good opportunity to place Sanic behind a proxy (like nginx) and to do your access logging there. You will see a *significant* increase in overall performance by disabling the `access_log`. + +For optimal production performance, it is advised to run Sanic with `debug` and `access_log` disabled: `app.run(debug=False, access_log=False)` +::: + +## Configuration + +Sanic's default logging configuration is: `sanic.log.LOGGING_CONFIG_DEFAULTS`. + +---:1 +There are three loggers used in sanic, and must be defined if you want to create your own logging configuration: + +| **Logger Name** | **Use Case** | +|-----------------|-------------------------------| +| `sanic.root` | Used to log internal messages. | +| `sanic.error` | Used to log error logs. | +| `sanic.access` | Used to log access logs. | +:--:1 + +:--- + +### Log format + +In addition to default parameters provided by Python (`asctime`, `levelname`, `message`), Sanic provides additional parameters for access logger with. + +| Log Context Parameter | Parameter Value | Datatype | +|-----------------------|---------------------------------------|----------| +| `host` | `request.ip` | `str` | +| `request` | `request.method + " " + request.url` | `str` | +| `status` | `response` | `int` | +| `byte` | `len(response.body)` | `int` | + + + + +The default access log format is: + +```text +%(asctime)s - (%(name)s)[%(levelname)s][%(host)s]: %(request)s %(message)s %(status)d %(byte)d +``` diff --git a/src/pt/guide/best-practices/testing.md b/src/pt/guide/best-practices/testing.md new file mode 100644 index 0000000000..a7ac6fdaf6 --- /dev/null +++ b/src/pt/guide/best-practices/testing.md @@ -0,0 +1,3 @@ +# Testing + +See [sanic-testing](../../plugins/sanic-testing/getting-started.md) diff --git a/src/pt/guide/deployment/README.md b/src/pt/guide/deployment/README.md new file mode 100644 index 0000000000..d36c65e59e --- /dev/null +++ b/src/pt/guide/deployment/README.md @@ -0,0 +1 @@ +# Deployment diff --git a/src/pt/guide/deployment/configuration.md b/src/pt/guide/deployment/configuration.md new file mode 100644 index 0000000000..d4eb7c9cf9 --- /dev/null +++ b/src/pt/guide/deployment/configuration.md @@ -0,0 +1,259 @@ +# Configuration + +## Basics + + +---:1 + +Sanic holds the configuration in the config attribute of the application object. The configuration object is merely an object that can be modified either using dot-notation or like a dictionary. +:--:1 +```python +app = Sanic("myapp") +app.config.DB_NAME = "appdb" +app.config["DB_USER"] = "appuser" +``` +:--- + +---:1 + +You can also use the `update()` method like on regular dictionaries. +:--:1 +```python +db_settings = { + 'DB_HOST': 'localhost', + 'DB_NAME': 'appdb', + 'DB_USER': 'appuser' +} +app.config.update(db_settings) +``` +:--- + +::: tip +It is standard practice in Sanic to name your config values in **uppercase letters**. Indeed, you may experience weird behaviors if you start mixing uppercase and lowercase names. +::: + +## Loading + +### Environment variables + +---:1 + +Any environment variables defined with the `SANIC_` prefix will be applied to the Sanic config. For example, setting `SANIC_REQUEST_TIMEOUT` will be loaded by the application automatically and fed into the `REQUEST_TIMEOUT` config variable. +:--:1 +```bash +$ export SANIC_REQUEST_TIMEOUT=10 +``` +```python +>>> print(app.config.REQUEST_TIMEOUT) +10 +``` +:--- + +---:1 + +You can change the prefix that Sanic is expecting at startup. +:--:1 +```bash +$ export MYAPP_REQUEST_TIMEOUT=10 +``` +```python +>>> app = Sanic(__name__, env_prefix='MYAPP_') +>>> print(app.config.REQUEST_TIMEOUT) +10 +``` +:--- + +---:1 + +You can also disable environment variable loading completely. +:--:1 +```python +app = Sanic(__name__, load_env=False) +``` +:--- + +### Using Sanic.update_config + +The `Sanic` instance has a _very_ versatile method for loading config: `app.update_config`. You can feed it a path to a file, a dictionary, a class, or just about any other sort of object. + +#### From a file + +---:1 + +Let's say you have `my_config.py` file that looks like this. +:--:1 +```python +# my_config.py +A = 1 +B = 2 +``` +:--- + +---:1 + +You can load this as config values by passing its path to `app.update_config`. +:--:1 +```python +>>> app.update_config("/path/to/my_config.py") +>>> print(app.config.A) +1 +``` +:--- + +---:1 + +This path also accepts bash style environment variables. +:--:1 +```bash +$ export my_path="/path/to" +``` +```python +app.update_config("${my_path}/my_config.py") +``` +:--- + +::: tip +Just remember that you have to provide environment variables in the format `${environment_variable}` and that `$environment_variable` is not expanded (is treated as "plain" text). +::: +#### From a dict + +---:1 + +The `app.update_config` method also works on plain dictionaries. +:--:1 +```python +app.update_config({"A": 1, "B": 2}) +``` +:--- + +#### From a class or object + +---:1 + +You can define your own config class, and pass it to `app.update_config` +:--:1 +```python +class MyConfig: + A = 1 + B = 2 + +app.update_config(MyConfig) +``` +:--- + +---:1 + +It even could be instantiated. +:--:1 +```python +app.update_config(MyConfig()) +``` +:--- + +### Type casting + +When loading from environment variables, Sanic will attempt to cast the values to expected Python types. This particularly applies to: + +- `int` +- `float` +- `bool` + +In regards to `bool`, the following _case insensitive_ values are allowed: + +- **`True`**: `y`, `yes`, `yep`, `yup`, `t`, `true`, `on`, `enable`, `enabled`, `1` +- **`False`**: `n`, `no`, `f`, `false`, `off`, `disable`, `disabled`, `0` + +---:1 +Additionally, Sanic can be configured to cast additional types using additional type converters. This should be any callable that returns the value or raises a `ValueError`. +:--:1 +```python +app = Sanic(..., config=Config(converters=[UUID])) +``` +:--- + +## Builtin values + + +| **Variable** | **Default** | **Description** | +|---------------------------|------------------|---------------------------------------------------------------------------------------------------------------------------------------| +| ACCESS_LOG | True | Disable or enable access log | +| AUTO_EXTEND ^ | True | Control whether [Sanic Extensions](../../plugins/sanic-ext/getting-started.md) will load if it is in the existing virtual environment | +| AUTO_RELOAD | True | Control whether the application will automatically reload when a file changes | +| EVENT_AUTOREGISTER | True | When `True` using the `app.event()` method on a non-existing signal will automatically create it and not raise an exception | +| FALLBACK_ERROR_FORMAT | html | Format of error response if an exception is not caught and handled | +| FORWARDED_FOR_HEADER | X-Forwarded-For | The name of "X-Forwarded-For" HTTP header that contains client and proxy ip | +| FORWARDED_SECRET | None | Used to securely identify a specific proxy server (see below) | +| GRACEFUL_SHUTDOWN_TIMEOUT | 15.0 | How long to wait to force close non-idle connection (sec) | +| KEEP_ALIVE | True | Disables keep-alive when False | +| KEEP_ALIVE_TIMEOUT | 5 | How long to hold a TCP connection open (sec) | +| MOTD ^ | True | Whether to display the MOTD (message of the day) at startup | +| MOTD_DISPLAY ^ | {} | Key/value pairs to display additional, arbitrary data in the MOTD | +| NOISY_EXCEPTIONS ^ | False | Force all `quiet` exceptions to be logged | +| PROXIES_COUNT | None | The number of proxy servers in front of the app (e.g. nginx; see below) | +| REAL_IP_HEADER | None | The name of "X-Real-IP" HTTP header that contains real client ip | +| REGISTER | True | Whether the app registry should be enabled | +| REQUEST_BUFFER_SIZE | 65536 | Request buffer size before request is paused, default is 64 Kib | +| REQUEST_ID_HEADER | X-Request-ID | The name of "X-Request-ID" HTTP header that contains request/correlation ID | +| REQUEST_MAX_SIZE | 100000000 | How big a request may be (bytes), default is 100 megabytes | +| REQUEST_TIMEOUT | 60 | How long a request can take to arrive (sec) | +| RESPONSE_TIMEOUT | 60 | How long a response can take to process (sec) | +| USE_UVLOOP | True | Whether to override the loop policy to use `uvloop`. Supported only with `app.run`. | +| WEBSOCKET_MAX_SIZE | 2^20 | Maximum size for incoming messages (bytes) | +| WEBSOCKET_PING_INTERVAL | 20 | A Ping frame is sent every ping_interval seconds. | +| WEBSOCKET_PING_TIMEOUT | 20 | Connection is closed when Pong is not received after ping_timeout seconds | + +::: tip FYI +- The `USE_UVLOOP` value will be ignored if running with Gunicorn. Defaults to `False` on non-supported platforms (Windows). +- The `WEBSOCKET_` values will be ignored if in ASGI mode. +::: + +## Timeouts + +### REQUEST_TIMEOUT + +A request timeout measures the duration of time between the instant when a new open TCP connection is passed to the +Sanic backend server, and the instant when the whole HTTP request is received. If the time taken exceeds the +`REQUEST_TIMEOUT` value (in seconds), this is considered a Client Error so Sanic generates an `HTTP 408` response +and sends that to the client. Set this parameter's value higher if your clients routinely pass very large request payloads +or upload requests very slowly. + +### RESPONSE_TIMEOUT + +A response timeout measures the duration of time between the instant the Sanic server passes the HTTP request to the +Sanic App, and the instant a HTTP response is sent to the client. If the time taken exceeds the `RESPONSE_TIMEOUT` +value (in seconds), this is considered a Server Error so Sanic generates an `HTTP 503` response and sends that to the +client. Set this parameter's value higher if your application is likely to have long-running process that delay the +generation of a response. + +### KEEP_ALIVE_TIMEOUT + +#### What is Keep Alive? And what does the Keep Alive Timeout value do? + +`Keep-Alive` is a HTTP feature introduced in `HTTP 1.1`. When sending a HTTP request, the client (usually a web browser application) +can set a `Keep-Alive` header to indicate the http server (Sanic) to not close the TCP connection after it has send the response. +This allows the client to reuse the existing TCP connection to send subsequent HTTP requests, and ensures more efficient +network traffic for both the client and the server. + +The `KEEP_ALIVE` config variable is set to `True` in Sanic by default. If you don't need this feature in your application, +set it to `False` to cause all client connections to close immediately after a response is sent, regardless of +the `Keep-Alive` header on the request. + +The amount of time the server holds the TCP connection open is decided by the server itself. +In Sanic, that value is configured using the `KEEP_ALIVE_TIMEOUT` value. By default, it is set to 5 seconds. +This is the same default setting as the Apache HTTP server and is a good balance between allowing enough time for +the client to send a new request, and not holding open too many connections at once. Do not exceed 75 seconds unless +you know your clients are using a browser which supports TCP connections held open for that long. + +For reference: + +* Apache httpd server default keepalive timeout = 5 seconds +* Nginx server default keepalive timeout = 75 seconds +* Nginx performance tuning guidelines uses keepalive = 15 seconds +* IE (5-9) client hard keepalive limit = 60 seconds +* Firefox client hard keepalive limit = 115 seconds +* Opera 11 client hard keepalive limit = 120 seconds +* Chrome 13+ client keepalive limit > 300+ seconds + +## Proxy configuration + +See [proxy configuration section](/guide/advanced/proxy-headers.md) diff --git a/src/pt/guide/deployment/development.md b/src/pt/guide/deployment/development.md new file mode 100644 index 0000000000..866d50274b --- /dev/null +++ b/src/pt/guide/deployment/development.md @@ -0,0 +1,72 @@ +# Development + +The first thing that should be mentioned is that the webserver that is integrated into Sanic is **not** just a development server. + +It is production ready, provided you are *not* in debug mode. + +## Debug mode + +By setting the debug mode, Sanic will be more verbose in its output and will disable several run-time optimizations. + +```python +from sanic import Sanic +from sanic.response import json + +app = Sanic(__name__) + +@app.route("/") +async def hello_world(request): + return json({"hello": "world"}) + +if __name__ == "__main__": + app.run(host="0.0.0.0", port=1234, debug=True) +``` + +::: warning +Sanic's debug mode will slow down the server's performance and is therefore advised to enable it only in development environments. +::: +## Automatic Reloader + +---:1 + +Sanic offers a way to enable or disable the Automatic Reloader. The `auto_reload` argument will activate or deactivate the Automatic Reloader. Every time a Python file is changed, the reloader will restart your application automatically. This is very convenient while developing. +:--:1 +```python +app.run(auto_reload=True) +``` +:--- + +---:1 +If you have additional directories that you would like to automatically reload on file save (for example, a directory of HTML templates), you can add that at run time. +:--:1 +```python +app.run(auto_reload=True, reload_dir="/path/to/templates") +# or multiple directories +app.run(auto_reload=True, reload_dir=["/path/to/one", "/path/to/two"]) +``` +:--- + +## Best of both worlds +::: new NEW in v22.3 +---:1 +If you would like to be in debug mode **and** have the Automatic Reloader running, you can pass `dev=True`. This is equivalent to **debug + auto reload**. +:--:1 +```python +app.run(dev=True) +``` +:--- +::: + +## CLI + +It should be noted that all of these have an equivalent in the Sanic CLI: + +``` + Development: + --debug Run the server in DEBUG mode. It includes DEBUG logging, + additional context on exceptions, and other settings + not-safe for PRODUCTION, but helpful for debugging problems. + -r, --reload, --auto-reload Watch source directory for file changes and reload on changes + -R PATH, --reload-dir PATH Extra directories to watch and reload on changes + -d, --dev debug + auto reload. +``` diff --git a/src/pt/guide/deployment/docker.md b/src/pt/guide/deployment/docker.md new file mode 100644 index 0000000000..c597eaa5eb --- /dev/null +++ b/src/pt/guide/deployment/docker.md @@ -0,0 +1 @@ +# Docker diff --git a/src/pt/guide/deployment/kubernetes.md b/src/pt/guide/deployment/kubernetes.md new file mode 100644 index 0000000000..8d6340de1c --- /dev/null +++ b/src/pt/guide/deployment/kubernetes.md @@ -0,0 +1 @@ +# Kubernetes diff --git a/src/pt/guide/deployment/nginx.md b/src/pt/guide/deployment/nginx.md new file mode 100644 index 0000000000..ee5ed946c5 --- /dev/null +++ b/src/pt/guide/deployment/nginx.md @@ -0,0 +1,218 @@ +# Nginx Deployment + +## Introduction + + +Although Sanic can be run directly on Internet, it may be useful to use a proxy +server such as Nginx in front of it. This is particularly useful for running +multiple virtual hosts on the same IP, serving NodeJS or other services beside +a single Sanic app, and it also allows for efficient serving of static files. +SSL and HTTP/2 are also easily implemented on such proxy. + +We are setting the Sanic app to serve only locally at `127.0.0.1:8000`, while the +Nginx installation is responsible for providing the service to public Internet +on domain `example.com`. Static files will be served from `/var/www/`. + + +## Proxied Sanic app + +The app needs to be setup with a secret key used to identify a trusted proxy, +so that real client IP and other information can be identified. This protects +against anyone on the Internet sending fake headers to spoof their IP addresses +and other details. Choose any random string and configure it both on the app +and in Nginx config. + +```python +from sanic import Sanic +from sanic.response import text + +app = Sanic("proxied_example") +app.config.FORWARDED_SECRET = "YOUR SECRET" + +@app.get("/") +def index(request): + # This should display external (public) addresses: + return text( + f"{request.remote_addr} connected to {request.url_for('index')}\n" + f"Forwarded: {request.forwarded}\n" + ) + +if __name__ == "__main__": + app.run(host="127.0.0.1", port=8000, workers=8, access_log=False) +``` + +Since this is going to be a system service, save your code to +`/srv/sanicexample/sanicexample.py`. + +For testing, run your app in a terminal. + +## Nginx configuration + +Quite much configuration is required to allow fast transparent proxying, but +for the most part these don't need to be modified, so bear with me. + +Upstream servers need to be configured in a separate `upstream` block to enable +HTTP keep-alive, which can drastically improve performance, so we use this +instead of directly providing an upstream address in `proxy_pass` directive. In +this example, the upstream section is named by `server_name`, i.e. the public +domain name, which then also gets passed to Sanic in the `Host` header. You may +change the naming as you see fit. Multiple servers may also be provided for +load balancing and failover. + +Change the two occurrences of `example.com` to your true domain name, and +instead of `YOUR SECRET` use the secret you chose for your app. + +```nginx +upstream example.com { + keepalive 100; + server 127.0.0.1:8000; + #server unix:/tmp/sanic.sock; +} + +server { + server_name example.com; + listen 443 ssl http2 default_server; + listen [::]:443 ssl http2 default_server; + # Serve static files if found, otherwise proxy to Sanic + location / { + root /var/www; + try_files $uri @sanic; + } + location @sanic { + proxy_pass http://$server_name; + # Allow fast streaming HTTP/1.1 pipes (keep-alive, unbuffered) + proxy_http_version 1.1; + proxy_request_buffering off; + proxy_buffering off; + # Proxy forwarding (password configured in app.config.FORWARDED_SECRET) + proxy_set_header forwarded "$proxy_forwarded;secret=\"YOUR SECRET\""; + # Allow websockets and keep-alive (avoid connection: close) + proxy_set_header connection "upgrade"; + proxy_set_header upgrade $http_upgrade; + } +} +``` + +To avoid cookie visibility issues and inconsistent addresses on search engines, +it is a good idea to redirect all visitors to one true domain, always using +HTTPS: + +```nginx +# Redirect all HTTP to HTTPS with no-WWW +server { + listen 80 default_server; + listen [::]:80 default_server; + server_name ~^(?:www\.)?(.*)$; + return 301 https://$1$request_uri; +} + +# Redirect WWW to no-WWW +server { + listen 443 ssl http2; + listen [::]:443 ssl http2; + server_name ~^www\.(.*)$; + return 301 $scheme://$1$request_uri; +} +``` + +The above config sections may be placed in `/etc/nginx/sites-available/default` +or in other site configs (be sure to symlink them to `sites-enabled` if you +create new ones). + +Make sure that your SSL certificates are configured in the main config, or +add the `ssl_certificate` and `ssl_certificate_key` directives to each +`server` section that listens on SSL. + +Additionally, copy&paste all of this into `nginx/conf.d/forwarded.conf`: + +```nginx +# RFC 7239 Forwarded header for Nginx proxy_pass + +# Add within your server or location block: +# proxy_set_header forwarded "$proxy_forwarded;secret=\"YOUR SECRET\""; + +# Configure your upstream web server to identify this proxy by that password +# because otherwise anyone on the Internet could spoof these headers and fake +# their real IP address and other information to your service. + + +# Provide the full proxy chain in $proxy_forwarded +map $proxy_add_forwarded $proxy_forwarded { + default "$proxy_add_forwarded;by=\"_$hostname\";proto=$scheme;host=\"$http_host\";path=\"$request_uri\""; +} + +# The following mappings are based on +# https://www.nginx.com/resources/wiki/start/topics/examples/forwarded/ + +map $remote_addr $proxy_forwarded_elem { + # IPv4 addresses can be sent as-is + ~^[0-9.]+$ "for=$remote_addr"; + + # IPv6 addresses need to be bracketed and quoted + ~^[0-9A-Fa-f:.]+$ "for=\"[$remote_addr]\""; + + # Unix domain socket names cannot be represented in RFC 7239 syntax + default "for=unknown"; +} + +map $http_forwarded $proxy_add_forwarded { + # If the incoming Forwarded header is syntactically valid, append to it + "~^(,[ \\t]*)*([!#$%&'*+.^_`|~0-9A-Za-z-]+=([!#$%&'*+.^_`|~0-9A-Za-z-]+|\"([\\t \\x21\\x23-\\x5B\\x5D-\\x7E\\x80-\\xFF]|\\\\[\\t \\x21-\\x7E\\x80-\\xFF])*\"))?(;([!#$%&'*+.^_`|~0-9A-Za-z-]+=([!#$%&'*+.^_`|~0-9A-Za-z-]+|\"([\\t \\x21\\x23-\\x5B\\x5D-\\x7E\\x80-\\xFF]|\\\\[\\t \\x21-\\x7E\\x80-\\xFF])*\"))?)*([ \\t]*,([ \\t]*([!#$%&'*+.^_`|~0-9A-Za-z-]+=([!#$%&'*+.^_`|~0-9A-Za-z-]+|\"([\\t \\x21\\x23-\\x5B\\x5D-\\x7E\\x80-\\xFF]|\\\\[\\t \\x21-\\x7E\\x80-\\xFF])*\"))?(;([!#$%&'*+.^_`|~0-9A-Za-z-]+=([!#$%&'*+.^_`|~0-9A-Za-z-]+|\"([\\t \\x21\\x23-\\x5B\\x5D-\\x7E\\x80-\\xFF]|\\\\[\\t \\x21-\\x7E\\x80-\\xFF])*\"))?)*)?)*$" "$http_forwarded, $proxy_forwarded_elem"; + + # Otherwise, replace it + default "$proxy_forwarded_elem"; +} +``` + +::: tip Note +For installs that don't use `conf.d` and `sites-available`, all of the above +configs may also be placed inside the `http` section of the main `nginx.conf`. +::: + +Reload Nginx config after changes: + +```bash +sudo nginx -s reload +``` + +Now you should be able to connect your app on `https://example.com/`. Any 404 +errors and such will be handled by Sanic's error pages, and whenever a static +file is present at a given path, it will be served by Nginx. + +## SSL certificates + +If you haven't already configured valid certificates on your server, now is a +good time to do so. Install `certbot` and `python3-certbot-nginx`, then run + +```bash +certbot --nginx -d example.com -d www.example.com +``` + +``_ + +## Running as a service + +This part is for Linux distributions based on `systemd`. Create a unit file +`/etc/systemd/system/sanicexample.service` + +```text +[Unit] +Description=Sanic Example + +[Service] +User=nobody +WorkingDirectory=/srv/sanicexample +ExecStart=/usr/bin/env python3 sanicexample.py +Restart=always + +[Install] +WantedBy=multi-user.target +``` + +Then reload service files, start your service and enable it on boot: + +```bash +sudo systemctl daemon-reload +sudo systemctl start sanicexample +sudo systemctl enable sanicexample +``` diff --git a/src/pt/guide/deployment/running.md b/src/pt/guide/deployment/running.md new file mode 100644 index 0000000000..a29bbde2d3 --- /dev/null +++ b/src/pt/guide/deployment/running.md @@ -0,0 +1,281 @@ +# Running Sanic + +Sanic ships with its own internal web server. Under most circumstances, this is the preferred method for deployment. In addition, you can also deploy Sanic as an ASGI app bundled with an ASGI-able web server, or using gunicorn. + +## Sanic Server + +After defining an instance of `sanic.Sanic`, we can call the run method with the following keyword arguments: + +| Parameter | Default | Description | +| :-------------: | :------------: | :---------------------------------------------------------------------------------------- | +| **host** | `"127.0.0.1"` | Address to host the server on. | +| **port** | `8000` | Port to host the server on. | +| **unix** | `None` | Unix socket name to host the server on (instead of TCP). | +| **debug** | `False` | Enables debug output (slows server). | +| **ssl** | `None` | SSLContext for SSL encryption of worker(s). | +| **sock** | `None` | Socket for the server to accept connections from. | +| **workers** | `1` | Number of worker processes to spawn. | +| **loop** | `None` | An asyncio-compatible event loop. If none is specified, Sanic creates its own event loop. | +| **protocol** | `HttpProtocol` | Subclass of asyncio.protocol. | +| **access_log** | `True` | Enables log on handling requests (significantly slows server). | + +---:1 + +In the above example, we decided to turn off the access log in order to increase performance. + +:--:1 + +```python +# server.py +app = Sanic("My App") +app.run(host='0.0.0.0', port=1337, access_log=False) +``` + +:--- + +---:1 + +Now, just execute the python script that has `app.run(...)` + +:--:1 + +```bash +python server.py +``` + +:--- + +### Workers + +---:1 +By default, Sanic listens in the main process using only one CPU core. To crank up the juice, just specify the number of workers in the run arguments. +:--:1 +```python +app.run(host='0.0.0.0', port=1337, workers=4) +``` +:--- + +Sanic will automatically spin up multiple processes and route traffic between them. We recommend as many workers as you have available processors. + +---:1 +The easiest way to get the maximum CPU performance is to use the `fast` option. This will automatically run the maximum number of workers given the system constraints. +:--:1 +```python +app.run(host='0.0.0.0', port=1337, fast=True) +``` +```python +$ sanic server:app --host=0.0.0.0 --port=1337 --fast +``` +:--- + +In older versions of Sanic without the `fast` option, a common way to check this on Linux based operating systems: + +``` +$ nproc +``` + +Or, let Python do it: + +```python +import multiprocessing +workers = multiprocessing.cpu_count() +app.run(..., workers=workers) +``` + +### Running via command + +#### Sanic CLI + +---:1 +Sanic also has a simple CLI to launch via command line. + +For example, if you initialized Sanic as app in a file named `server.py`, you could run the server like so: +:--:1 +```bash +sanic server.app --host=0.0.0.0 --port=1337 --workers=4 +``` +:--- + + +Use `sanic --help` to see all the options. +```text +$ sanic --help +usage: sanic [-h] [--version] [--factory] [-s] [-H HOST] [-p PORT] [-u UNIX] [--cert CERT] [--key KEY] [--tls DIR] [--tls-strict-host] + [-w WORKERS | --fast] [--access-logs | --no-access-logs] [--debug] [-d] [-r] [-R PATH] [--motd | --no-motd] [-v] + [--noisy-exceptions | --no-noisy-exceptions] + module + + ▄███ █████ ██ ▄█▄ ██ █ █ ▄██████████ + ██ █ █ █ ██ █ █ ██ + ▀███████ ███▄ ▀ █ █ ██ ▄ █ ██ + ██ █████████ █ ██ █ █ ▄▄ + ████ ████████▀ █ █ █ ██ █ ▀██ ███████ + + To start running a Sanic application, provide a path to the module, where + app is a Sanic() instance: + + $ sanic path.to.server:app + + Or, a path to a callable that returns a Sanic() instance: + + $ sanic path.to.factory:create_app --factory + + Or, a path to a directory to run as a simple HTTP server: + + $ sanic ./path/to/static --simple + +Required +======== + Positional: + module Path to your Sanic app. Example: path.to.server:app + If running a Simple Server, path to directory to serve. Example: ./ + +Optional +======== + General: + -h, --help show this help message and exit + --version show program's version number and exit + + Application: + --factory Treat app as an application factory, i.e. a () -> callable + -s, --simple Run Sanic as a Simple Server, and serve the contents of a directory + (module arg should be a path) + + Socket binding: + -H HOST, --host HOST Host address [default 127.0.0.1] + -p PORT, --port PORT Port to serve on [default 8000] + -u UNIX, --unix UNIX location of unix socket + + TLS certificate: + --cert CERT Location of fullchain.pem, bundle.crt or equivalent + --key KEY Location of privkey.pem or equivalent .key file + --tls DIR TLS certificate folder with fullchain.pem and privkey.pem + May be specified multiple times to choose multiple certificates + --tls-strict-host Only allow clients that send an SNI matching server certs + + Worker: + -w WORKERS, --workers WORKERS Number of worker processes [default 1] + --fast Set the number of workers to max allowed + --access-logs Display access logs + --no-access-logs No display access logs + + Development: + --debug Run the server in debug mode + -d, --dev Currently is an alias for --debug. But starting in v22.3, + --debug will no longer automatically trigger auto_restart. + However, --dev will continue, effectively making it the + same as debug + auto_reload. + -r, --reload, --auto-reload Watch source directory for file changes and reload on changes + -R PATH, --reload-dir PATH Extra directories to watch and reload on changes + + Output: + --motd Show the startup display + --no-motd No show the startup display + -v, --verbosity Control logging noise, eg. -vv or --verbosity=2 [default 0] + --noisy-exceptions Output stack traces for all exceptions + --no-noisy-exceptions No output stack traces for all exceptions +``` + +#### As a module + +---:1 +It can also be called directly as a module. +:--:1 +```bash +python -m sanic server.app --host=0.0.0.0 --port=1337 --workers=4 +``` +:--- + +::: tip FYI +With either method (CLI or module), you shoud *not* invoke `app.run()` in your Python file. If you do, make sure you wrap it so that it only executes when directly run by the interpreter. +```python +if __name__ == '__main__': + app.run(host='0.0.0.0', port=1337, workers=4) +``` +::: + + +#### Sanic Simple Server + +---:1 +Sometimes you just have a directory of static files that need to be served. This especially can be handy for quickly standing up a localhost server. Sanic ships with a Simple Server, where you only need to point it at a directory. +:--:1 +```bash +sanic ./path/to/dir --simple +``` +:--- + +---:1 +This could also be paired with auto-reloading. +:--:1 +```bash +sanic ./path/to/dir --simple --reload --reload-dir=./path/to/dir +``` +:--- + +## ASGI + +Sanic is also ASGI-compliant. This means you can use your preferred ASGI webserver to run Sanic. The three main implementations of ASGI are [Daphne](http://github.com/django/daphne), [Uvicorn](https://www.uvicorn.org/), and [Hypercorn](https://pgjones.gitlab.io/hypercorn/index.html). + +::: warning +Daphne does not support the ASGI `lifespan` protocol, and therefore cannot be used to run Sanic. See [Issue #264](https://github.com/django/daphne/issues/264) for more details. +::: + +Follow their documentation for the proper way to run them, but it should look something like: + +``` +uvicorn myapp:app +hypercorn myapp:app +``` + +A couple things to note when using ASGI: + +1. When using the Sanic webserver, websockets will run using the `websockets` package. In ASGI mode, there is no need for this package since websockets are managed in the ASGI server. +2. The ASGI lifespan protocol , supports only two server events: startup and shutdown. Sanic has four: before startup, after startup, before shutdown, and after shutdown. Therefore, in ASGI mode, the startup and shutdown events will run consecutively and not actually around the server process beginning and ending (since that is now controlled by the ASGI server). Therefore, it is best to use `after_server_start` and `before_server_stop`. + +### Trio + +Sanic has experimental support for running on Trio with: + +``` +hypercorn -k trio myapp:app +``` + + +## Gunicorn + +[Gunicorn](http://gunicorn.org/) ("Green Unicorn") is a WSGI HTTP Server for UNIX based operating systems. It is a pre-fork worker model ported from Ruby’s Unicorn project. + +In order to run Sanic application with Gunicorn, you need to use the special `sanic.worker.GunicornWorker` for Gunicorn worker-class argument: + +```bash +gunicorn myapp:app --bind 0.0.0.0:1337 --worker-class sanic.worker.GunicornWorker +``` + +If your application suffers from memory leaks, you can configure Gunicorn to gracefully restart a worker after it has processed a given number of requests. This can be a convenient way to help limit the effects of the memory leak. + +See the [Gunicorn Docs](http://docs.gunicorn.org/en/latest/settings.html#max-requests) for more information. + +::: warning +When running Sanic via `gunicorn`, you are losing out on a lot of the performance benefits of `async`/`await`. Weigh your considerations carefully before making this choice. Gunicorn does provide a lot of configuration options, but it is not the best choice for getting Sanic to run at its fastest. +::: + +## Performance considerations + +---:1 +When running in production, make sure you turn off `debug`. +:--:1 +```python +app.run(..., debug=False) +``` +:--- + +---:1 +Sanic will also perform fastest if you turn off `access_log`. + +If you still require access logs, but want to enjoy this performance boost, consider using [Nginx as a proxy](./nginx.md), and letting that handle your access logging. It will be much faster than anything Python can handle. +:--:1 +```python +app.run(..., access_log=False) +``` +:--- diff --git a/src/pt/guide/deployment/server-choice.md b/src/pt/guide/deployment/server-choice.md new file mode 100644 index 0000000000..9af6345235 --- /dev/null +++ b/src/pt/guide/deployment/server-choice.md @@ -0,0 +1 @@ +# Choosing a server diff --git a/src/pt/guide/getting-started.md b/src/pt/guide/getting-started.md new file mode 100644 index 0000000000..8b115e2739 --- /dev/null +++ b/src/pt/guide/getting-started.md @@ -0,0 +1,90 @@ +# Getting Started + +Before we begin, make sure you are running Python 3.7 or higher. Currently, is known to work with Python versions 3.7, 3.8 and 3.9. + +## Install + +```bash +pip install sanic +``` + +## Hello, world application + +---:1 + +If you have ever used one of the many decorator based frameworks, this probably looks somewhat familiar to you. + +::: tip +If you are coming from Flask or another framework, there are a few important things to point out. Remember, Sanic aims for performance, flexibility, and ease of use. These guiding principles have tangible impact on the API and how it works. +::: + + + +:--:1 + +```python +from sanic import Sanic +from sanic.response import text + +app = Sanic("MyHelloWorldApp") + +@app.get("/") +async def hello_world(request): + return text("Hello, world.") +``` + +:--- + +### Important to note + +- Every request handler can either be sync (`def hello_world`) or async (`async def hello_world`). Unless you have a clear reason for it, always go with `async`. +- The `request` object is always the first argument of your handler. Other frameworks pass this around in a context variable to be imported. In the `async` world, this would not work so well and it is far easier (not to mention cleaner and more performant) to be explicit about it. +- You **must** use a response type. MANY other frameworks allow you to have a return value like this: `return "Hello, world."` or this: `return {"foo": "bar"}`. But, in order to do this implicit calling, somewhere in the chain needs to spend valuable time trying to determine what you meant. So, at the expense of this ease, Sanic has decided to require an explicit call. + +### Running + +---:1 +Let's save the above file as `server.py`. And launch it. +:--:1 +```bash +sanic server.app +``` +:--- + +::: tip +This **another** important distinction. Other frameworks come with a built in development server and explicitly say that it is _only_ intended for development use. The opposite is true with Sanic. + +**The packaged server is production ready.** +::: + +## Sanic Extensions + +Sanic intentionally aims for a clean and unopinionated feature list. The project does not want to require you to build your application in a certain way, and tries to avoid prescribing specific development patterns. There are a number of third-party plugins that are built and maintained by the community to add additional features that do not otherwise meet the requirements of the core repository. + +However, in order **to help API developers**, the Sanic organization maintains an official plugin called [Sanic Extensions](../plugins/sanic-ext/getting-started.md) to provide all sorts of goodies, including: + +- **OpenAPI** documentation with Redoc and/or Swagger +- **CORS** protection +- **Dependency injection** into route handlers +- Request query arguments and body input **validation** +- Auto create `HEAD`, `OPTIONS`, and `TRACE` endpoints +- Predefined, endpoint-specific response serializers + +The preferred method to set it up is to install it along with Sanic, but you can also install the packages on their own. + +---:1 +``` +$ pip install sanic[ext] +``` +:--:1 +``` +$ pip install sanic sanic-ext +``` +:--- + +Starting in v21.12, Sanic will automatically setup Sanic Extensions if it is in the same environment. You will also have access to two additional application properties: + +- `app.extend()` - used to configure Sanic Extensions +- `app.ext` - the `Extend` instance attached to the application + +See [the plugin documenataion](../plugins/sanic-ext/getting-started.md) for more information about how to use and work with the plugin diff --git a/src/pt/guide/how-to/README.md b/src/pt/guide/how-to/README.md new file mode 100644 index 0000000000..8e66fc0483 --- /dev/null +++ b/src/pt/guide/how-to/README.md @@ -0,0 +1 @@ +# How to ... diff --git a/src/pt/guide/how-to/assets/images/lake.jpg b/src/pt/guide/how-to/assets/images/lake.jpg new file mode 100644 index 0000000000000000000000000000000000000000..9ff601b1e799e08594a10076f0816a5433366d50 GIT binary patch literal 141444 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 *-`704uv4wbp!TW>ZY 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 literal 0 HcmV?d00001 diff --git a/src/pt/guide/how-to/authentication.md b/src/pt/guide/how-to/authentication.md new file mode 100644 index 0000000000..d652177b36 --- /dev/null +++ b/src/pt/guide/how-to/authentication.md @@ -0,0 +1,114 @@ +# Authentication + +> How do I control authentication and authorization? + +This is an _extremely_ complicated subject to cram into a few snippets. But, this should provide you with an idea on ways to tackle this problem. This example uses [JWTs](https://jwt.io/), but the concepts should be equally applicable to sessions or some other scheme. + +:::: tabs +::: tab server.py +```python +from sanic import Sanic, text + +from auth import protected +from login import login + +app = Sanic("AuthApp") +app.config.SECRET = "KEEP_IT_SECRET_KEEP_IT_SAFE" +app.blueprint(login) + + +@app.get("/secret") +@protected +async def secret(request): + return text("To go fast, you must be fast.") +``` +::: +::: tab login.py +```python +import jwt +from sanic import Blueprint, text + +login = Blueprint("login", url_prefix="/login") + + +@login.post("/") +async def do_login(request): + token = jwt.encode({}, request.app.config.SECRET) + return text(token) +``` +::: +::: tab auth.py +```python +from functools import wraps + +import jwt +from sanic import text + + +def check_token(request): + if not request.token: + return False + + try: + jwt.decode( + request.token, request.app.config.SECRET, algorithms=["HS256"] + ) + except jwt.exceptions.InvalidTokenError: + return False + else: + return True + + +def protected(wrapped): + def decorator(f): + @wraps(f) + async def decorated_function(request, *args, **kwargs): + is_authenticated = check_token(request) + + if is_authenticated: + response = await f(request, *args, **kwargs) + return response + else: + return text("You are unauthorized.", 401) + + return decorated_function + + return decorator(wrapped) +``` +This decorator pattern is taken from the [decorators page](/en/guide/best-practices/decorators.md). +::: +:::: + +```bash +$ curl localhost:9999/secret -i +HTTP/1.1 401 Unauthorized +content-length: 21 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +You are unauthorized. + +$ curl localhost:9999/login -X POST 7 ↵ +eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.e30.rjxS7ztIGt5tpiRWS8BGLUqjQFca4QOetHcZTi061DE + +$ curl localhost:9999/secret -i -H "Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.e30.rjxS7ztIGt5tpiRWS8BGLUqjQFca4QOetHcZTi061DE" +HTTP/1.1 200 OK +content-length: 29 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +To go fast, you must be fast. + +$ curl localhost:9999/secret -i -H "Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.e30.BAD" +HTTP/1.1 401 Unauthorized +content-length: 21 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +You are unauthorized. +``` + +Also, checkout some resources from the community: + +- Awesome Sanic - [Authorization](https://github.com/mekicha/awesome-sanic/blob/master/README.md#authentication) & [Session](https://github.com/mekicha/awesome-sanic/blob/master/README.md#session) +- [EuroPython 2020 - Overcoming access control in web APIs](https://www.youtube.com/watch?v=Uqgoj43ky6A) diff --git a/src/pt/guide/how-to/autodiscovery.md b/src/pt/guide/how-to/autodiscovery.md new file mode 100644 index 0000000000..4cf319357b --- /dev/null +++ b/src/pt/guide/how-to/autodiscovery.md @@ -0,0 +1,161 @@ +--- +title: Autodiscovery +--- + + +# Autodiscovery of Blueprints, Middleware, and Listeners + +> How do I autodiscover the components I am using to build my application? + +One of the first problems someone faces when building an application, is *how* to structure the project. Sanic makes heavy use of decorators to register route handlers, middleware, and listeners. And, after creating blueprints, they need to be mounted to the application. + +A possible solution is a single file in which **everything** is imported and applied to the Sanic instance. Another is passing around the Sanic instance as a global variable. Both of these solutions have their drawbacks. + +An alternative is autodiscovery. You point your application at modules (already imported, or strings), and let it wire everything up. + +:::: tabs +::: tab server.py +```python +from sanic import Sanic +from sanic.response import empty + +import blueprints +from utility import autodiscover + +app = Sanic("auto", register=True) +autodiscover( + app, + blueprints, + "parent.child", + "listeners.something", + recursive=True, +) + +app.route("/")(lambda _: empty()) +``` +```bash +[2021-03-02 21:37:02 +0200] [880451] [INFO] Goin' Fast @ http://127.0.0.1:9999 +[2021-03-02 21:37:02 +0200] [880451] [DEBUG] something +[2021-03-02 21:37:02 +0200] [880451] [DEBUG] something @ nested +[2021-03-02 21:37:02 +0200] [880451] [DEBUG] something @ level1 +[2021-03-02 21:37:02 +0200] [880451] [DEBUG] something @ level3 +[2021-03-02 21:37:02 +0200] [880451] [DEBUG] something inside __init__.py +[2021-03-02 21:37:02 +0200] [880451] [INFO] Starting worker [880451] +``` +::: +::: tab utility.py +```python + +from glob import glob +from importlib import import_module, util +from inspect import getmembers +from pathlib import Path +from types import ModuleType +from typing import Union + +from sanic.blueprints import Blueprint + + +def autodiscover( + app, *module_names: Union[str, ModuleType], recursive: bool = False +): + mod = app.__module__ + blueprints = set() + _imported = set() + + def _find_bps(module): + nonlocal blueprints + + for _, member in getmembers(module): + if isinstance(member, Blueprint): + blueprints.add(member) + + for module in module_names: + if isinstance(module, str): + module = import_module(module, mod) + _imported.add(module.__file__) + _find_bps(module) + + if recursive: + base = Path(module.__file__).parent + for path in glob(f"{base}/**/*.py", recursive=True): + if path not in _imported: + name = "module" + if "__init__" in path: + *_, name, __ = path.split("/") + spec = util.spec_from_file_location(name, path) + specmod = util.module_from_spec(spec) + _imported.add(path) + spec.loader.exec_module(specmod) + _find_bps(specmod) + + for bp in blueprints: + app.blueprint(bp) +``` +::: +::: tab blueprints/level1.py +```python +from sanic import Blueprint +from sanic.log import logger + +level1 = Blueprint("level1") + + +@level1.after_server_start +def print_something(app, loop): + logger.debug("something @ level1") +``` +::: +::: tab blueprints/one/two/level3.py +```python +from sanic import Blueprint +from sanic.log import logger + +level3 = Blueprint("level3") + + +@level3.after_server_start +def print_something(app, loop): + logger.debug("something @ level3") +``` +::: +::: tab listeners/something.py +```python +from sanic import Sanic +from sanic.log import logger + +app = Sanic.get_app("auto") + + +@app.after_server_start +def print_something(app, loop): + logger.debug("something") +``` +::: +::: tab parent/child/__init__.py +```python +from sanic import Blueprint +from sanic.log import logger + +bp = Blueprint("__init__") + + +@bp.after_server_start +def print_something(app, loop): + logger.debug("something inside __init__.py") +``` +::: +::: tab parent/child/nested.py +```python +from sanic import Blueprint +from sanic.log import logger + +nested = Blueprint("nested") + + +@nested.after_server_start +def print_something(app, loop): + logger.debug("something @ nested") +``` +::: +:::: diff --git a/src/pt/guide/how-to/cors.md b/src/pt/guide/how-to/cors.md new file mode 100644 index 0000000000..f230137703 --- /dev/null +++ b/src/pt/guide/how-to/cors.md @@ -0,0 +1,135 @@ +--- +title: CORS +--- + + +# Cross-origin resource sharing (CORS) + +> How do I configure my application for CORS? + +:::: tabs +::: tab server.py +```python +from sanic import Sanic, text + +from cors import add_cors_headers +from options import setup_options + +app = Sanic("app") + + +@app.route("/", methods=["GET", "POST"]) +async def do_stuff(request): + return text("...") + + +# Add OPTIONS handlers to any route that is missing it +app.register_listener(setup_options, "before_server_start") + +# Fill in CORS headers +app.register_middleware(add_cors_headers, "response") +``` +::: +::: tab cors.py +```python +from typing import Iterable + + +def _add_cors_headers(response, methods: Iterable[str]) -> None: + allow_methods = list(set(methods)) + if "OPTIONS" not in allow_methods: + allow_methods.append("OPTIONS") + headers = { + "Access-Control-Allow-Methods": ",".join(allow_methods), + "Access-Control-Allow-Origin": "mydomain.com", + "Access-Control-Allow-Credentials": "true", + "Access-Control-Allow-Headers": ( + "origin, content-type, accept, " + "authorization, x-xsrf-token, x-request-id" + ), + } + response.headers.extend(headers) + + +def add_cors_headers(request, response): + if request.method != "OPTIONS": + methods = [method for method in request.route.methods] + _add_cors_headers(response, methods) +``` +::: +::: tab options.py +```python +from collections import defaultdict +from typing import Dict, FrozenSet + +from sanic import Sanic, response +from sanic.router import Route + +from cors import _add_cors_headers + + +def _compile_routes_needing_options( + routes: Dict[str, Route] +) -> Dict[str, FrozenSet]: + needs_options = defaultdict(list) + # This is 21.12 and later. You will need to change this for older versions. + for route in routes.values(): + if "OPTIONS" not in route.methods: + needs_options[route.uri].extend(route.methods) + + return { + uri: frozenset(methods) for uri, methods in dict(needs_options).items() + } + + +def _options_wrapper(handler, methods): + def wrapped_handler(request, *args, **kwargs): + nonlocal methods + return handler(request, methods) + + return wrapped_handler + + +async def options_handler(request, methods) -> response.HTTPResponse: + resp = response.empty() + _add_cors_headers(resp, methods) + return resp + + +def setup_options(app: Sanic, _): + app.router.reset() + needs_options = _compile_routes_needing_options(app.router.routes_all) + for uri, methods in needs_options.items(): + app.add_route( + _options_wrapper(options_handler, methods), + uri, + methods=["OPTIONS"], + ) + app.router.finalize() +``` +::: +:::: +``` +$ curl localhost:9999/ -i +HTTP/1.1 200 OK +Access-Control-Allow-Methods: OPTIONS,POST,GET +Access-Control-Allow-Origin: mydomain.com +Access-Control-Allow-Credentials: true +Access-Control-Allow-Headers: origin, content-type, accept, authorization, x-xsrf-token, x-request-id +content-length: 3 +connection: keep-alive +content-type: text/plain; charset=utf-8 + +... + +$ curl localhost:9999/ -i -X OPTIONS +HTTP/1.1 204 No Content +Access-Control-Allow-Methods: GET,POST,OPTIONS +Access-Control-Allow-Origin: mydomain.com +Access-Control-Allow-Credentials: true +Access-Control-Allow-Headers: origin, content-type, accept, authorization, x-xsrf-token, x-request-id +connection: keep-alive +``` +Also, checkout some resources from the community: + +- [Awesome Sanic](https://github.com/mekicha/awesome-sanic/blob/master/README.md#frontend) diff --git a/src/pt/guide/how-to/csrf.md b/src/pt/guide/how-to/csrf.md new file mode 100644 index 0000000000..7f982ff12f --- /dev/null +++ b/src/pt/guide/how-to/csrf.md @@ -0,0 +1 @@ +csrf diff --git a/src/pt/guide/how-to/db.md b/src/pt/guide/how-to/db.md new file mode 100644 index 0000000000..60ca822eac --- /dev/null +++ b/src/pt/guide/how-to/db.md @@ -0,0 +1 @@ +connecting to data sources diff --git a/src/pt/guide/how-to/decorators.md b/src/pt/guide/how-to/decorators.md new file mode 100644 index 0000000000..9ef318134e --- /dev/null +++ b/src/pt/guide/how-to/decorators.md @@ -0,0 +1 @@ +decorators diff --git a/src/pt/guide/how-to/ipv6.md b/src/pt/guide/how-to/ipv6.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/guide/how-to/mounting.md b/src/pt/guide/how-to/mounting.md new file mode 100644 index 0000000000..918c422cdb --- /dev/null +++ b/src/pt/guide/how-to/mounting.md @@ -0,0 +1,52 @@ +# Application Mounting + +> How do I mount my application at some path above the root? + +```python +# server.py +from sanic import Sanic, text + +app = Sanic("app") +app.config.SERVER_NAME = "example.com/api" + + +@app.route("/foo") +def handler(request): + url = app.url_for("handler", _external=True) + return text(f"URL: {url}") +``` + +```yaml +# docker-compose.yml +version: "3.7" +services: + app: + image: nginx:alpine + ports: + - 80:80 + volumes: + - type: bind + source: ./conf + target: /etc/nginx/conf.d/default.conf +``` + +```nginx +# conf +server { + listen 80; + + # Computed data service + location /api/ { + proxy_pass http://:9999/; + proxy_set_header Host example.com; + } +} +``` +```bash +$ docker-compose up -d +$ sanic server.app --port=9999 --host=0.0.0.0 +``` +```bash +$ curl localhost/api/foo +URL: http://example.com/api/foo +``` diff --git a/src/pt/guide/how-to/orm.md b/src/pt/guide/how-to/orm.md new file mode 100644 index 0000000000..af06180bde --- /dev/null +++ b/src/pt/guide/how-to/orm.md @@ -0,0 +1,292 @@ +# ORM + +> How do I use SQLAlchemy with Sanic ? + +All ORM tools can work with Sanic, but non-async ORM tool have a impact on Sanic performance. +There are some orm packages who support + +At present, there are many ORMs that support asynchronicity. Two of the more common libraries are: + +- [SQLAlchemy 1.4](https://docs.sqlalchemy.org/en/14/changelog/changelog_14.html) +- [tortoise-orm](https://github.com/tortoise/tortoise-orm) + +Integration in to your Sanic application is fairly simple: + +## SQLAlchemy + +Because [SQLAlchemy 1.4](https://docs.sqlalchemy.org/en/14/changelog/changelog_14.html) has added native support for `asyncio`, Sanic can finally work well with SQLAlchemy. Be aware that this functionality is still considered *beta* by the SQLAlchemy project. + + +---:1 + +### Dependencies + +First, we need to install the required dependencies. In the past, the dependencies installed were `sqlalchemy` and `pymysql`, but now `sqlalchemy` and `aiomysql` are needed. + +:--:1 + +```shell +pip install -U sqlalchemy +pip install -U aiomysql +``` + +:--- + +---:1 + +### Define ORM Model + +ORM model creation remains the same. + +:--:1 + +```python +# ./models.py +from sqlalchemy import INTEGER, Column, ForeignKey, String +from sqlalchemy.orm import declarative_base, relationship + +Base = declarative_base() + + +class BaseModel(Base): + __abstract__ = True + id = Column(INTEGER(), primary_key=True) + + +class Person(BaseModel): + __tablename__ = "person" + name = Column(String()) + cars = relationship("Car") + + def to_dict(self): + return {"name": self.name, "cars": [{"brand": car.brand} for car in self.cars]} + + +class Car(BaseModel): + __tablename__ = "car" + + brand = Column(String()) + user_id = Column(ForeignKey("person.id")) + user = relationship("Person", back_populates="cars") +``` + +:--- + +---:1 + +### Create Sanic App and Async Engine + +Here we use mysql as the database, and you can also choose PostgreSQL/SQLite. Pay attention to changing the driver from `aiomysql` to `asyncpg`/`aiosqlite`. +:--:1 + + +```python +# ./server.py +from sanic import Sanic +from sqlalchemy.ext.asyncio import create_async_engine + +app = Sanic("my_app") + +bind = create_async_engine("mysql+aiomysql://root:root@localhost/test", echo=True) +``` + +:--- + +---:1 + +### Register Middlewares + +The request middleware creates an usable `AsyncSession` object and set it to `request.ctx` and `_base_model_session_ctx`. + +Thread-safe variable `_base_model_session_ctx` helps you to use the session object instead of fetching it from `request.ctx`. + + +:--:1 + +```python +# ./server.py +from contextvars import ContextVar + +from sqlalchemy.ext.asyncio import AsyncSession +from sqlalchemy.orm import sessionmaker + +_base_model_session_ctx = ContextVar("session") + +@app.middleware("request") +async def inject_session(request): + request.ctx.session = sessionmaker(bind, AsyncSession, expire_on_commit=False)() + request.ctx.session_ctx_token = _base_model_session_ctx.set(request.ctx.session) + + +@app.middleware("response") +async def close_session(request, response): + if hasattr(request.ctx, "session_ctx_token"): + _base_model_session_ctx.reset(request.ctx.session_ctx_token) + await request.ctx.session.close() +``` + +:--- + +---:1 + +### Register Routes + +According to sqlalchemy official docs, `session.query` will be legacy in 2.0, and the 2.0 way to query an ORM object is using `select`. + +:--:1 + +```python +# ./server.py +from sqlalchemy import select +from sqlalchemy.orm import selectinload +from sanic.response import json + +from models import Car, Person + + +@app.post("/user") +async def create_user(request): + session = request.ctx.session + async with session.begin(): + car = Car(brand="Tesla") + person = Person(name="foo", cars=[car]) + session.add_all([person]) + return json(person.to_dict()) + + +@app.get("/user/") +async def get_user(request, pk): + session = request.ctx.session + async with session.begin(): + stmt = select(Person).where(Person.id == pk).options(selectinload(Person.cars)) + result = await session.execute(stmt) + person = result.scalar() + + if not person: + return json({}) + + return json(person.to_dict()) +``` + +:--- + +### Send Requests + +```sh +curl --location --request POST 'http://127.0.0.1:8000/user' +{"name":"foo","cars":[{"brand":"Tesla"}]} +``` + +```sh +curl --location --request GET 'http://127.0.0.1:8000/user/1' +{"name":"foo","cars":[{"brand":"Tesla"}]} +``` + + +## Tortoise-ORM + +---:1 + +### Dependencies + +tortoise-orm's dependency is very simple, you just need install tortoise-orm. + +:--:1 + +```shell +pip install -U tortoise-orm +``` + +:--- + +---:1 + +### Define ORM Model + +If you are familiar with Django, you should find this part very familiar. + +:--:1 + +```python +# ./models.py +from tortoise import Model, fields + + +class Users(Model): + id = fields.IntField(pk=True) + name = fields.CharField(50) + + def __str__(self): + return f"I am {self.name}" +``` + +:--- + + +---:1 + +### Create Sanic App and Async Engine + +Tortoise-orm provides a set of registration interface, which is convenient for users, and you can use it to create database connection easily. + +:--:1 + +```python +# ./main.py + +from models import Users +from tortoise.contrib.sanic import register_tortoise + +app = Sanic(__name__) + + +register_tortoise( + app, db_url="mysql://root:root@localhost/test", modules={"models": ["models"]}, generate_schemas=True +) + +``` + +:--- + +---:1 + +### Register Routes + +:--:1 + +```python + +# ./main.py + +from models import Users +from sanic import Sanic, response + + +@app.route("/user") +async def list_all(request): + users = await Users.all() + return response.json({"users": [str(user) for user in users]}) + + +@app.route("/user/") +async def get_user(request, pk): + user = await Users.query(pk=pk) + return response.json({"user": str(user)}) + +if __name__ == "__main__": + app.run(port=5000) +``` + +:--- + +### Send Requests + +```sh +curl --location --request POST 'http://127.0.0.1:8000/user' +{"users":["I am foo", "I am bar"]} +``` + +```sh +curl --location --request GET 'http://127.0.0.1:8000/user/1' +{"user": "I am foo"} +``` diff --git a/src/pt/guide/how-to/request-id-logging.md b/src/pt/guide/how-to/request-id-logging.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/guide/how-to/serialization.md b/src/pt/guide/how-to/serialization.md new file mode 100644 index 0000000000..0dfc62d35b --- /dev/null +++ b/src/pt/guide/how-to/serialization.md @@ -0,0 +1 @@ +# Serialization diff --git a/src/pt/guide/how-to/server-sent-events.md b/src/pt/guide/how-to/server-sent-events.md new file mode 100644 index 0000000000..7daf86745e --- /dev/null +++ b/src/pt/guide/how-to/server-sent-events.md @@ -0,0 +1 @@ +sse diff --git a/src/pt/guide/how-to/static-redirects.md b/src/pt/guide/how-to/static-redirects.md new file mode 100644 index 0000000000..d1fc0ba741 --- /dev/null +++ b/src/pt/guide/how-to/static-redirects.md @@ -0,0 +1,112 @@ +# "Static" Redirects + +> How do I configure static redirects? + +:::: tabs +::: tab app.py +```python +### SETUP ### +import typing +import sanic, sanic.response + +# Create the Sanic app +app = sanic.Sanic(__name__) + +# This dictionary represents your "static" +# redirects. For example, these values +# could be pulled from a configuration file. +REDIRECTS = { + '/':'/hello_world', # Redirect '/' to '/hello_world' + '/hello_world':'/hello_world.html' # Redirect '/hello_world' to 'hello_world.html' +} + +# This function will return another function +# that will return the configured value +# regardless of the arguments passed to it. +def get_static_function(value:typing.Any) -> typing.Callable[..., typing.Any]: + return lambda *_, **__: value + +### ROUTING ### +# Iterate through the redirects +for src, dest in REDIRECTS.items(): + # Create the redirect response object + response:sanic.HTTPResponse = sanic.response.redirect(dest) + + # Create the handler function. Typically, + # only a sanic.Request object is passed + # to the function. This object will be + # ignored. + handler = get_static_function(response) + + # Route the src path to the handler + app.route(src)(handler) + +# Route some file and client resources +app.static('/files/', 'files') +app.static('/', 'client') + +### RUN ### +if __name__ == '__main__': + app.run( + '127.0.0.1', + 10000 + ) +``` +::: + +::: tab client/hello_world.html +```html + + + + + + + Hello World + + + +

+ Hello world! +
+ + +``` +::: + +::: tab client/hello_world.css +```css +#hello_world { + width: 1000px; + margin-left: auto; + margin-right: auto; + margin-top: 100px; + + padding: 100px; + color: aqua; + text-align: center; + font-size: 100px; + font-family: monospace; + + background-color: rgba(0, 0, 0, 0.75); + + border-radius: 10px; + box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.75); +} + +body { + background-image: url("/files/lake.jpg"); + background-repeat: no-repeat; + background-size: cover; +} +``` +::: + +::: tab files/lake.jpg +![](./assets/images/lake.jpg) +::: +:::: + +Also, checkout some resources from the community: + +- [Static Routing Example](https://github.com/Perzan/sanic-static-routing-example) diff --git a/src/pt/guide/how-to/task-queue.md b/src/pt/guide/how-to/task-queue.md new file mode 100644 index 0000000000..c5ee7c54a3 --- /dev/null +++ b/src/pt/guide/how-to/task-queue.md @@ -0,0 +1 @@ +task queue diff --git a/src/pt/guide/how-to/tls.md b/src/pt/guide/how-to/tls.md new file mode 100644 index 0000000000..e2f506bd8a --- /dev/null +++ b/src/pt/guide/how-to/tls.md @@ -0,0 +1,168 @@ +# TLS/SSL/HTTPS + +> How do I run Sanic via HTTPS? + +If you do not have TLS certificates yet, [see the end of this page](./tls.md#get-certificates-for-your-domain-names). + +## Single domain and single certificate + +---:1 +Let Sanic automatically load your certificate files, which need to be named `fullchain.pem` and `privkey.pem` in the given folder: + +:--:1 +```sh +sudo sanic myserver:app -H :: -p 443 \ + --tls /etc/letsencrypt/live/example.com/ +``` +```python +app.run("::", 443, ssl="/etc/letsencrypt/live/example.com/") +``` +:--- + +---:1 +Or, you can pass cert and key filenames separately as a dictionary: + +Additionally, `password` may be added if the key is encrypted, all fields except for the password are passed to `request.conn_info.cert`. +:--:1 +```python +ssl = { + "cert": "/path/to/fullchain.pem", + "key": "/path/to/privkey.pem", + "password": "for encrypted privkey file", # Optional +} +app.run(host="0.0.0.0", port=8443, ssl=ssl) +``` +:--- + +---:1 +Alternatively, [`ssl.SSLContext`](https://docs.python.org/3/library/ssl.html) may be passed, if you need full control over details such as which crypto algorithms are permitted. By default Sanic only allows secure algorithms, which may restrict access from very old devices. +:--:1 +```python +import ssl + +context = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) +context.load_cert_chain("certs/fullchain.pem", "certs/privkey.pem") + +app.run(host="0.0.0.0", port=8443, ssl=context) +``` +:--- + + +## Multiple domains with separate certificates + +---:1 +A list of multiple certificates may be provided, in which case Sanic chooses the one matching the hostname the user is connecting to. This occurs so early in the TLS handshake that Sanic has not sent any packets to the client yet. + +If the client sends no SNI (Server Name Indication), the first certificate on the list will be used even though on the client browser it will likely fail with a TLS error due to name mismatch. To prevent this fallback and to cause immediate disconnection of clients without a known hostname, add `None` as the first entry on the list. `--tls-strict-host` is the equivalent CLI option. +:--:1 +```python +ssl = ["certs/example.com/", "certs/bigcorp.test/"] +app.run(host="0.0.0.0", port=8443, ssl=ssl) +``` +```sh +sanic myserver:app + --tls certs/example.com/ + --tls certs/bigcorp.test/ + --tls-strict-host +``` +:--- + +::: tip +You may also use `None` in front of a single certificate if you do not wish to reveal your certificate, true hostname or site content to anyone connecting to the IP address instead of the proper DNS name. +::: + +---:1 +Dictionaries can be used on the list. This allows also specifying which domains a certificate matches to, although the names present on the certificate itself cannot be controlled from here. If names are not specified, the names from the certificate itself are used. + +To only allow connections to the main domain **example.com** and only to subdomains of **bigcorp.test**: + +:--:1 +```python +ssl = [ + None, # No fallback if names do not match! + { + "cert": "certs/example.com/fullchain.pem", + "key": "certs/example.com/privkey.pem", + "names": ["example.com", "*.bigcorp.test"], + } +] +app.run(host="0.0.0.0", port=8443, ssl=ssl) +``` +:--- + +## Accessing TLS information in handlers via `request.conn_info` fields + +* `.ssl` - is the connection secure (bool) +* `.cert` - certificate info and dict fields of the currently active cert (dict) +* `.server_name` - the SNI sent by the client (str, may be empty) + +Do note that all `conn_info` fields are per connection, where there may be many requests over time. If a proxy is used in front of your server, these requests on the same pipe may even come from different users. + +## Redirect HTTP to HTTPS, with certificate requests still over HTTP + +In addition to your normal server(s) running HTTPS, run another server for redirection, `http_redir.py`: + +```python +from sanic import Sanic, exceptions, response + +app = Sanic("http_redir") + +# Serve ACME/certbot files without HTTPS, for certificate renewals +app.static("/.well-known", "/var/www/.well-known", resource_type="dir") + +@app.exception(exceptions.NotFound, exceptions.MethodNotSupported) +def redirect_everything_else(request, exception): + server, path = request.server_name, request.path + if server and path.startswith("/"): + return response.redirect(f"https://{server}{path}", status=308) + return response.text("Bad Request. Please use HTTPS!", status=400) +``` + +It is best to setup this as a systemd unit separate of your HTTPS servers. You may need to run HTTP while initially requesting your certificates, while you cannot run the HTTPS server yet. Start for IPv4 and IPv6: + +``` +sanic http_redir:app -H 0.0.0.0 -p 80 +sanic http_redir:app -H :: -p 80 +``` + +Alternatively, it is possible to run the HTTP redirect application from the main application: + +```python +# app == Your main application +# redirect == Your http_redir application +@app.before_server_start +async def start(app, _): + app.ctx.redirect = await redirect.create_server( + port=80, return_asyncio_server=True + ) + app.add_task(runner(redirect, app.ctx.redirect)) + + +@app.before_server_stop +async def stop(app, _): + await app.ctx.redirect.close() + + +async def runner(app, app_server): + app.is_running = True + try: + app.signalize() + app.finalize() + app.state.is_started = True + await app_server.serve_forever() + finally: + app.is_running = False + app.is_stopping = True +``` + +## Get certificates for your domain names + +You can get free certificates from [Let's Encrypt](https://letsencrypt.org/). Install [certbot](https://certbot.eff.org/) via your package manager, and request a certificate: + +```sh +sudo certbot certonly --key-type ecdsa --preferred-chain "ISRG Root X1" -d example.com -d www.example.com +``` + +Multiple domain names may be added by further `-d` arguments, all stored into a single certificate which gets saved to `/etc/letsencrypt/live/example.com/` as per **the first domain** that you list here. + +The key type and preferred chain options are necessary for getting a minimal size certificate file, essential for making your server run as *fast* as possible. The chain will still contain one RSA certificate until when Let's Encrypt gets their new EC chain trusted in all major browsers, possibly around 2023. diff --git a/src/pt/guide/how-to/toc.md b/src/pt/guide/how-to/toc.md new file mode 100644 index 0000000000..b7be0f9872 --- /dev/null +++ b/src/pt/guide/how-to/toc.md @@ -0,0 +1,21 @@ +# Table of Contents + +We have compiled fully working examples to answer common questions and user cases. For the most part, the examples are as minimal as possible, but should be complete and runnable solutions. + +| Page | How do I ... | +|:-----|:------------| +| [Application mounting](./mounting.md) | ... mount my application at some path above the root? | +| [Authentication](./authentication.md) | ... control authentication and authorization? | +| [Autodiscovery](./autodiscovery.md) | ... autodiscover the components I am using to build my application? | +| [CORS](./cors.md) | ... configure my application for CORS? | +| CSRF | *Coming soon* | +| Databases | *Coming soon* | +| IPv6 | *Coming soon* | +| Request ID Logging | *Coming soon* | +| Request validation | *Coming soon* | +| Serialization | *Coming soon* | +| Server Sent Events | *Coming soon* | +| [ORM](./orm) | ... use an ORM with Sanic? | +| Task queues | *Coming soon* | +| [TLS/SSL/HTTPS](./tls.md) | ... run Sanic via HTTPS? ... redirect HTTP to HTTPS? | +| Websocket feeds | *Coming soon* | diff --git a/src/pt/guide/how-to/validation.md b/src/pt/guide/how-to/validation.md new file mode 100644 index 0000000000..d45b2dea4c --- /dev/null +++ b/src/pt/guide/how-to/validation.md @@ -0,0 +1 @@ +validation diff --git a/src/pt/guide/how-to/websocket-feed.md b/src/pt/guide/how-to/websocket-feed.md new file mode 100644 index 0000000000..ae2abc2def --- /dev/null +++ b/src/pt/guide/how-to/websocket-feed.md @@ -0,0 +1 @@ +websocket feed diff --git a/src/pt/guide/release-notes/v21.12.md b/src/pt/guide/release-notes/v21.12.md new file mode 100644 index 0000000000..75dd9288a9 --- /dev/null +++ b/src/pt/guide/release-notes/v21.12.md @@ -0,0 +1,507 @@ +# Version 21.12 + +[[toc]] + +## Introduction + +This is the final release of the version 21 [release cycle](../../org/policies.md#release-schedule). Version 21 will now enter long-term support and will be supported for two years until December 2023. + +## What to know + +More details in the [Changelog](https://sanic.readthedocs.io/en/stable/sanic/changelog.html). Notable new or breaking features, and what to upgrade... + +### Strict application and blueprint names + +In [v21.6](./v21.6.md#stricter-application-and-blueprint-names-and-deprecation) application and blueprint names were required to conform to a new set of restrictions. That change is now being enforced at startup time. + +Names **must**: + +1. Only use alphanumeric characters (`a-zA-Z0-9`) +2. May contain a hyphen (`-`) or an underscore (`_`) +3. Must begin with a letter or underscore (`a-zA-Z_`) + +### Strict application and blueprint properties + +The old leniency to allow directly setting properties of a `Sanic` or `Blueprint` object was deprecated and no longer allowed. You must use the `ctx` object. + +```python +app = Sanic("MyApp") +app.ctx.db = Database() +``` + +### Removals + +The following deprecated features no longer exist: + +- `sanic.exceptions.abort` +- `sanic.views.CompositionView` +- `sanic.response.StreamingHTTPResponse` + +### Upgrade your streaming responses (if not already) + +The `sanic.response.stream` response method has been **deprecated** and will be removed in v22.6. If you are sill using an old school streaming response, please upgrade it. + +**OLD - Deprecated** + +```python +async def sample_streaming_fn(response): + await response.write("foo,") + await response.write("bar") + +@app.route("/") +async def test(request: Request): + return stream(sample_streaming_fn, content_type="text/csv") +``` + +**Current** + +```python +async def sample_streaming_fn(response): + await response.write("foo,") + await response.write("bar") + +@app.route("/") +async def test(request: Request): + response = await request.respond(content_type="text/csv") + await response.send("foo,") + await response.send("bar") +``` + +### CLI overhaul and MOTD (Message of the Day) + +The Sanic CLI has received a fairly extensive upgrade. It adds a bunch of new features to make it on par with `app.run()`. It also includes a new MOTD display to provide quick, at-a-glance highlights about your running environment. The MOTD is TTY-aware, and therefore will be less verbose in server logs. It is mainly intended as a convenience during application development. + +``` +$ sanic --help +usage: sanic [-h] [--version] [--factory] [-s] [-H HOST] [-p PORT] [-u UNIX] [--cert CERT] [--key KEY] [--tls DIR] [--tls-strict-host] + [-w WORKERS | --fast] [--access-logs | --no-access-logs] [--debug] [-d] [-r] [-R PATH] [--motd | --no-motd] [-v] + [--noisy-exceptions | --no-noisy-exceptions] + module + + ▄███ █████ ██ ▄█▄ ██ █ █ ▄██████████ + ██ █ █ █ ██ █ █ ██ + ▀███████ ███▄ ▀ █ █ ██ ▄ █ ██ + ██ █████████ █ ██ █ █ ▄▄ + ████ ████████▀ █ █ █ ██ █ ▀██ ███████ + + To start running a Sanic application, provide a path to the module, where + app is a Sanic() instance: + + $ sanic path.to.server:app + + Or, a path to a callable that returns a Sanic() instance: + + $ sanic path.to.factory:create_app --factory + + Or, a path to a directory to run as a simple HTTP server: + + $ sanic ./path/to/static --simple + +Required +======== + Positional: + module Path to your Sanic app. Example: path.to.server:app + If running a Simple Server, path to directory to serve. Example: ./ + +Optional +======== + General: + -h, --help show this help message and exit + --version show program's version number and exit + + Application: + --factory Treat app as an application factory, i.e. a () -> callable + -s, --simple Run Sanic as a Simple Server, and serve the contents of a directory + (module arg should be a path) + + Socket binding: + -H HOST, --host HOST Host address [default 127.0.0.1] + -p PORT, --port PORT Port to serve on [default 8000] + -u UNIX, --unix UNIX location of unix socket + + TLS certificate: + --cert CERT Location of fullchain.pem, bundle.crt or equivalent + --key KEY Location of privkey.pem or equivalent .key file + --tls DIR TLS certificate folder with fullchain.pem and privkey.pem + May be specified multiple times to choose multiple certificates + --tls-strict-host Only allow clients that send an SNI matching server certs + + Worker: + -w WORKERS, --workers WORKERS Number of worker processes [default 1] + --fast Set the number of workers to max allowed + --access-logs Display access logs + --no-access-logs No display access logs + + Development: + --debug Run the server in debug mode + -d, --dev Currently is an alias for --debug. But starting in v22.3, + --debug will no longer automatically trigger auto_restart. + However, --dev will continue, effectively making it the + same as debug + auto_reload. + -r, --reload, --auto-reload Watch source directory for file changes and reload on changes + -R PATH, --reload-dir PATH Extra directories to watch and reload on changes + + Output: + --motd Show the startup display + --no-motd No show the startup display + -v, --verbosity Control logging noise, eg. -vv or --verbosity=2 [default 0] + --noisy-exceptions Output stack traces for all exceptions + --no-noisy-exceptions No output stack traces for all exceptions +``` + +### Server running modes and changes coming to `debug` + +There are now two running modes: `DEV` and `PRODUCTION`. By default, Sanic server will run under `PRODUCTION` mode. This is intended for deployments. + +Currently, `DEV` mode will operate very similarly to how `debug=True` does in older Sanic versions. However, in v22.3. `debug=True` will **no longer** enable auto-reload. If you would like to have debugging and auto-reload, you should enable `DEV` mode. + +**DEVELOPMENT** + +``` +$ sanic server:app --dev +``` + +```python +app.run(debug=True, auto_reload=True) +``` + +**PRODUCTION** + +``` +$ sanic server:app +``` + +```python +app.run() +``` + +Beginning in v22.3, `PRODUCTION` mode will no longer enable access logs by default. + +A summary of the changes are as follows: + +| Flag | Mode | Tracebacks | Logging | Access logs | Reload | Max workers | +|---------|-------|------------|---------|-------------|--------|-------------| +| --debug | DEBUG | yes | DEBUG | yes | ^1 | | +| | PROD | no | INFO ^2 | ^3 | | | +| --dev | DEBUG | yes | DEBUG | yes | yes | | +| --fast | | | | | | yes | + + +- ^1 `--debug` to deprecate auto-reloading and remove in 22.3 +- ^2 After 22.3 this moves to WARNING +- ^3 After 22.3: no + +### Max allowed workers + +You can easily spin up the maximum number of allowed workers using `--fast`. + +``` +$ sanic server:app --fast +``` + +```python +app.run(fast=True) +``` + +### First-class Sanic Extensions support + +[Sanic Extensions](../../plugins/sanic-ext/getting-started.md) provides a number of additional features specifically intended for API developers. You can now easily implement all of the functionality it has to offer without additional setup as long as the package is in the environment. These features include: + +- Auto create `HEAD`, `OPTIONS`, and `TRACE` endpoints +- CORS protection +- Predefined, endpoint-specific response serializers +- Dependency injection into route handlers +- OpenAPI documentation with Redoc and/or Swagger +- Request query arguments and body input validation + +The preferred method is to install it along with Sanic, but you can also install the packages on their own. + +---:1 +``` +$ pip install sanic[ext] +``` + +:--:1 + +``` +$ pip install sanic sanic-ext +``` + +:--- + +After that, no additional configuration is required. Sanic Extensions will be attached to your application and provide all of the additional functionality with **no further configuration**. + +If you want to change how it works, or provide additional configuration, you can change Sanic extensions using `app.extend`. The following examples are equivalent. The `Config` object is to provide helpful type annotations for IDE development. + +---:1 +```python +# This is optional, not required +app = Sanic("MyApp") +app.extend(config={"oas_url_prefix": "/apidocs"}) +``` +:--: +```python +# This is optional, not required +app = Sanic("MyApp") +app.config.OAS_URL_PREFIX = "/apidocs" +``` +:--- + +---:1 +```python +# This is optional, not required +from sanic_ext import Config + +app = Sanic("MyApp") +app.extend(config=Config(oas_url_prefix="/apidocs")) +``` +:--: + +:--- + +### Contextual exceptions + +In [v21.9](./v21.9.md#default-exception-messages) we added default messages to exceptions that simplify the ability to consistently raise exceptions throughout your application. + +```python +class TeapotError(SanicException): + status_code = 418 + message = "Sorry, I cannot brew coffee" + +raise TeapotError +``` + +But this lacked two things: + +1. A dynamic and predictable message format +2. The ability to add additional context to an error message (more on this in a moment) + +The current release allows any Sanic exception to have additional information to when raised to provide context when writing an error message: + +```python +class TeapotError(SanicException): + status_code = 418 + + @property + def message(self): + return f"Sorry {self.extra['name']}, I cannot make you coffee" + +raise TeapotError(extra={"name": "Adam"}) +``` + +The new feature allows the passing of `extra` meta to the exception instance. This `extra` info object **will be suppressed** when in `PRODUCTION` mode, but displayed in `DEVELOPMENT` mode. + +---:1 +**PRODUCTION** + +![image](https://user-images.githubusercontent.com/166269/139014161-cda67cd1-843f-4ad2-9fa1-acb94a59fc4d.png) +:--:1 +**DEVELOPMENT** + +![image](https://user-images.githubusercontent.com/166269/139014121-0596b084-b3c5-4adb-994e-31ba6eba6dad.png) +:--- + +Getting back to item 2 from above: _The ability to add additional context to an error message_ + +This is particularly useful when creating microservices or an API that you intend to pass error messages back in JSON format. In this use case, we want to have some context around the exception beyond just a parseable error message to return details to the client. + + +```python +raise TeapotError(context={"foo": "bar"}) +``` + +This is information **that we want** to always be passed in the error (when it is available). Here is what it should look like: + +---:1 +**PRODUCTION** + +```json +{ + "description": "I'm a teapot", + "status": 418, + "message": "Sorry Adam, I cannot make you coffee", + "context": { + "foo": "bar" + } +} +``` +:--:1 +**DEVELOPMENT** + +```json +{ + "description": "I'm a teapot", + "status": 418, + "message": "Sorry Adam, I cannot make you coffee", + "context": { + "foo": "bar" + }, + "extra": { + "name": "Adam", + "more": "lines", + "complex": { + "one": "two" + } + }, + "path": "/", + "args": {}, + "exceptions": [ + { + "type": "TeapotError", + "exception": "Sorry Adam, I cannot make you coffee", + "frames": [ + { + "file": "handle_request", + "line": 83, + "name": "handle_request", + "src": "" + }, + { + "file": "/tmp/p.py", + "line": 17, + "name": "handler", + "src": "raise TeapotError(" + } + ] + } + ] +} +``` +:--- + +### Background task management + +When using the `app.add_task` method to create a background task, there now is the option to pass an optional `name` keyword argument that allows it to be fetched, or cancelled. + +```python +app.add_task(dummy, name="dummy_task") +task = app.get_task("dummy_task") + +app.cancel_task("dummy_task") +``` + +### Route context kwargs in definitions + +When a route is defined, you can add any number of keyword arguments with a `ctx_` prefix. These values will be injected into the route `ctx` object. + +```python +@app.get("/1", ctx_label="something") +async def handler1(request): + ... + +@app.get("/2", ctx_label="something") +async def handler2(request): + ... + +@app.get("/99") +async def handler99(request): + ... + +@app.on_request +async def do_something(request): + if request.route.ctx.label == "something": + ... +``` + +### Blueprints can be registered at any time + +In previous versions of Sanic, there was a strict ordering of when a Blueprint could be attached to an application. If you ran `app.blueprint(bp)` *before* attaching all objects to the Blueprint instance, they would be missed. + +Now, you can attach a Blueprint at anytime and everything attached to it will be included at startup. + +### Noisy exceptions (force all exceptions to logs) + +There is a new `NOISY_EXCEPTIONS` config value. When it is `False` (which is the default), Sanic will respect the `quiet` property of any `SanicException`. This means that an exception with `quiet=True` will not be displayed to the log output. + +However, when setting `NOISY_EXCEPTIONS=True`, all exceptions will be logged regardless of the `quiet` value. + +This can be helpful when debugging. + +```python +app.config.NOISY_EXCEPTIONS = True +``` + +### Signal events as `Enum` + +There is an `Enum` with all of the built-in signal values for convenience. + +```python +from sanic.signals import Event + +@app.signal(Event.HTTP_LIFECYCLE_BEGIN) +async def connection_opened(conn_info): + ... +``` + +### Custom type casting of environment variables + +By default, Sanic will convert an `int`, `float`, or a `bool` value when applying environment variables to the `config` instance. You can extend this with your own converter: + +```python +app = Sanic(..., config=Config(converters=[UUID])) +``` + +### Disable `uvloop` by configuration value + +The usage of `uvloop` can be controlled by configuration value: + + +```python +app.config.USE_UVLOOP = False +``` + +### Run Sanic server with multiple TLS certificates + +Sanic can be run with multiple TLS certificates: + +```python +app.run( + ssl=[ + "/etc/letsencrypt/live/example.com/", + "/etc/letsencrypt/live/mysite.example/", + ] +) +``` + +## News + +### Coming Soon: Python Web Development with Sanic + +A book about Sanic is coming soon by one of the core developers, [@ahopkins](https://github.com/ahopkins). + +Learn more at [sanicbook.com](https://sanicbook.com). + +> Get equipped with the practical knowledge of working with Sanic to increase the performance and scalability of your web applications. While doing that, we will level-up your development skills as you learn to customize your application to meet the changing business needs without having to significantly over-engineer the app. + +A portion of book proceeds goes into the Sanic Community Organization to help fund the development and operation of Sanic. So, buying the book is another way you can support Sanic. + +### Dark mode for the docs + +If you have not already noticed, this Sanic website is now available in a native dark mode. You can toggle the theme at the top right of the page. + +## Thank you + +Thank you to everyone that participated in this release: :clap: + +[@adarsharegmi](https://github.com/adarsharegmi) +[@ahopkins](https://github.com/ahopkins) +[@ashleysommer](https://github.com/ashleysommer) +[@ChihweiLHBird](https://github.com/ChihweiLHBird) +[@cnicodeme](https://github.com/cnicodeme) +[@kianmeng](https://github.com/kianmeng) +[@meysam81](https://github.com/meysam81) +[@nuxion](https://github.com/nuxion) +[@prryplatypus](https://github.com/prryplatypus) +[@realDragonium](https://github.com/realDragonium) +[@SaidBySolo](https://github.com/SaidBySolo) +[@sjsadowski](https://github.com/sjsadowski) +[@Tronic](https://github.com/tronic) +[@Varriount](https://github.com/Varriount) +[@vltr](https://github.com/vltr) +[@whos4n3](https://github.com/whos4n3) + +And, a special thank you to [@miss85246](https://github.com/miss85246) and [@ZinkLu](https://github.com/ZinkLu) for their tremendous work keeping the documentation synced and translated into Chinese. + +--- + +If you enjoy the project, please consider contributing. Of course we love code contributions, but we also love contributions in any form. Consider writing some documentation, showing off use cases, joining conversations and making your voice known, and if you are able: [financial contributions](https://opencollective.com/sanic-org/). diff --git a/src/pt/guide/release-notes/v21.3.md b/src/pt/guide/release-notes/v21.3.md new file mode 100644 index 0000000000..b4447e34e2 --- /dev/null +++ b/src/pt/guide/release-notes/v21.3.md @@ -0,0 +1,264 @@ +# Version 21.3 + +[[toc]] + +## Introduction + +Sanic is now faster. + +Well, it already was fast. But with the first iteration of the v21 release, we incorporated a few major milestones that have made some tangible improvements. These encompass some ideas that have been in the works for years, and have finally made it into the released version. + +::: warning Breaking changes +Version 21.3 introduces a lot of new features. But, it also includes some breaking changes. This is why these changes were introduced after the last LTS. If you rely upon something that has been removed, you should continue to use v20.12LTS until you are able to upgrade. + +```bash +pip install "sanic>=20.12,<20.13" +pip freeze > requirements.txt +``` + +For most typical installations, you should be able to upgrade without a problem. +::: + +## What to know + +Notable new or breaking features, and what to upgrade... + +### Python 3.7+ Only + +This version drops Python 3.6 support. Version 20.12LTS will continue to support Python 3.6 until its EOL in December, 2022, and version 19.12LTS will support it until its EOL in December, 2021. + +Read more about our [LTS policy](../project/policies.md#long-term-support-v-interim-releases). + +### Streaming as first class citizen + +The biggest speed improvement came from unifying the request/response cycle into a single flow. Previously, there was a difference between regular cycles, and streaming cycles. This has been simplified under the hood, even though the API is staying the same right now for compatibility. The net benefit is that **all** requests now should see a new benefit. + +Read more about [streaming changes](../advanced/streaming.md#response-streaming). + +### Router overhaul + +The old Sanic router was based upon regular expressions. In addition it suffered from a number of quirks that made it hard to modify at run time, and resulted in some performance issues. This change has been years in the making and now [converts the router to a compiled tree at startup](https://community.sanicframework.org/t/a-fast-new-router/649/41). Look for additional improvements throughout the year. + +The outward facing API has kept backwards compatibility. However, if you were accessing anything inside the router specifically, you many notice some changes. For example: + +1. `Router.get()` has a new return value +2. `Route` is now a proper class object and not a `namedtuple` +3. If building the router manually, you will need to call `Router.finalize()` before it is usable +4. There is a new `` pattern that can be matched in your routes +5. You cannot startup an application without at least one route defined + +The router is now located in its own repository: [sanic-org/sanic-router](https://github.com/sanic-org/sanic-router) and is also its own [standalone package on PyPI](https://pypi.org/project/sanic-routing/). + +### Signals API ⭐️ + +_BETA Feature: API to be finalized in v21.6_ + +A side benefit of the new router is that it can do double duty also powering the [new signals API](https://github.com/sanic-org/sanic/issues/1630). This feature is being released for public usage now, and likely the public API will not change in its final form. + +The core ideas of this feature are: + +1. to allow the developer greater control and access to plugging into the server and request lifecycles, +2. to provide new tools to synchronize and send messages through your application, and +3. to ultimately further increase performance. + +The API introduces three new methods: + +- `@app.signal(...)` - For defining a signal handler. It looks and operates very much like a route. Whenever that signal is dispatched, this handler will be executed. +- `app.event(...)` - An awaitable that can be used anywhere in your application to pause execution until the event is triggered. +- `app.dispatch(...)` - Trigger an event and cause the signal handlers to execute. + +```python +@app.signal("foo.bar.") +async def signal_handler(thing, **kwargs): + print(f"[signal_handler] {thing=}", kwargs) + +async def wait_for_event(app): + while True: + print("> waiting") + await app.event("foo.bar.*") + print("> event found\n") + +@app.after_server_start +async def after_server_start(app, loop): + app.add_task(wait_for_event(app)) + +@app.get("/") +async def trigger(request): + await app.dispatch("foo.bar.baz") + return response.text("Done.") +``` + +### Route naming + +Routes used to be referenced by both `route.name` and `route.endpoint`. While similar, they were slightly different. Now, all routes will be **consistently** namespaced and referenced. + +``` +.[optional:.] +``` + +This new "name" is assigned to the property `route.name`. We are deprecating `route.endpoint`, and will remove that property in v21.9. Until then, it will be an alias for `route.name`. + +In addition, naming prefixes that had been in use for things like static, websocket, and blueprint routes have been removed. + +### New decorators + +Several new convenience decorators to help IDEs with autocomplete. + +```python +# Alias to @app.listener("...") +@app.before_server_start +@app.after_server_stop +@app.before_server_start +@app.after_server_stop + +# Alias to @app.middleware("...") +@app.on_request +@app.on_response +``` + +### Unquote in route + +If you have a route that uses non-ascii characters, Sanic will no longer `unquote` the text for you. You will need to specifically tell the route definition that it should do so. + +```python +@app.route("/overload/", methods=["GET"], unquote=True) +async def handler2(request, param): + return text("OK2 " + param) + +request, response = app.test_client.get("/overload/您好") +assert response.text == "OK2 您好" +``` + +If you forget to do so, your text will remain encoded. + +### Alter `Request.match_info` + +The `match_info` has always provided the data for the matched path parameters. You now have access to modify that, for example in middleware. + +```python +@app.on_request +def convert_to_snake_case(request): + request.match_info = to_snake(request.match_info) +``` + +### Version types in routes + +The `version` argument in routes can now be: + +- `str` +- `int` +- `float` + +```python +@app.route("/foo", version="2.1.1") +@app.route("/foo", version=2) +@app.route("/foo", version=2.1) +``` +### Safe method handling with body + +Route handlers for `GET`, `HEAD`, `OPTIONS` and `DELETE` will not decode any HTTP body passed to it. You can override this: + +```python +@app.delete(..., ignore_body=False) +``` + +### Application, Blueprint and Blueprint Group parity + +The `Sanic` and `Blueprint` classes share a common base. Previously they duplicated a lot of functionality, that lead to slightly different implementations between them. Now that they both inherit the same base class, developers and plugins should have a more consistent API to work with. + +Also, Blueprint Groups now also support common URL extensions like the `version` and `strict_slashes` keyword arguments. + +### Dropped `httpx` from dependencies + +There is no longer a dependency on `httpx`. + +### Removed `testing` library + +Sanic internal testing client has been removed. It is now located in its own repository: [sanic-org/sanic-testing](https://github.com/sanic-org/sanic-testing) and is also its own [standalone package on PyPI](https://pypi.org/project/sanic-testing/). + +If you have `sanic-testing` installed, it will be available and usable on your `Sanic()` application instances as before. So, the **only** change you will need to make is to add `sanic-testing` to your test suite requirements. + +### Application and connection level context (`ctx`) objects + +Version 19.9 [added ](https://github.com/sanic-org/sanic/pull/1666/files) the `request.ctx` API. This helpful construct easily allows for attaching properties and data to a request object (for example, in middleware), and reusing the information elsewhere int he application. + +Similarly, this concept is being extended in two places: + +1. the application instance, and +2. a transport connection. + +#### Application context + +A common use case is to attach properties to the app instance. For the sake of consistency, and to avoid the issue of name collision with Sanic properties, the `ctx` object now exists on `Sanic` instances. + +```python +@app.before_server_startup +async def startup_db(app, _): + # WRONG + app.db = await connect_to_db() + + # CORRECT + app.ctx.db = await connect_to_db() +``` + +#### Connection context + +When a client sends a keep alive header, Sanic will attempt to keep the transport socket [open for a period of time](../deployment/configuration.md#keep-alive-timeout). That transport object now has a `ctx` object available on it. This effectively means that multiple requests from a single client (where the transport layer is being reused) may share state. + +```python +@app.on_request +async def increment_foo(request): + if not hasattr(request.conn_info.ctx, "foo"): + request.conn_info.ctx.foo = 0 + request.conn_info.ctx.foo += 1 + +@app.get("/") +async def count_foo(request): + return text(f"request.conn_info.ctx.foo={request.conn_info.ctx.foo}") +``` + +```bash +$ curl localhost:8000 localhost:8000 localhost:8000 +request.conn_info.ctx.foo=1 +request.conn_info.ctx.foo=2 +request.conn_info.ctx.foo=3 +``` + +::: warning +Connection level context is an experimental feature, and should be finalized in v21.6. +::: + +## News + + +### A NEW frontpage 🎉 + +We have split the documentation into two. The docstrings inside the codebase will still continue to build sphinx docs to ReadTheDocs. However, it will be limited to API documentation. The new frontpage will house the "Sanic User Guide". + +The new site runs on Vuepress. Contributions are welcome. We also invite help in translating the documents. + +As a part of this, we also freshened up the RTD documentation and changed it to API docs only. + +### Chat has moved to Discord + +The Gitter chatroom has taken one step closer to being phased out. In its place we opened a [Discord server](https://discord.gg/FARQzAEMAA). + +### Open Collective + +The Sanic Community Organization has [opened a page on Open Collective](https://opencollective.com/sanic-org) to enable anyone that would like to financially support the development of Sanic. + +### 2021 Release Managers + +Thank you to @sjsadowski and @yunstanford for acting as release managers for both 2019 and 2020. This year's release managers are @ahopkins and @vltr. + +## Thank you + +Thank you to everyone that participated in this release: :clap: + +[@ahopkins](https://github.com/ahopkins) [@akshgpt7](https://github.com/akshgpt7) [@artcg](https://github.com/artcg) [@ashleysommer](https://github.com/ashleysommer) [@elis-k](https://github.com/elis-k) [@harshanarayana](https://github.com/harshanarayana) [@sjsadowski](https://github.com/sjsadowski) [@tronic](https://github.com/tronic) [@vltr](https://github.com/vltr), + +To [@ConnorZhang](https://github.com/miss85246) and [@ZinkLu](https://github.com/ZinkLu) for translating our documents into Chinese, + +--- + +Make sure to checkout the changelog to get links to all the PRs, etc. diff --git a/src/pt/guide/release-notes/v21.6.md b/src/pt/guide/release-notes/v21.6.md new file mode 100644 index 0000000000..1203dd7f7c --- /dev/null +++ b/src/pt/guide/release-notes/v21.6.md @@ -0,0 +1,342 @@ +# Version 21.6 + +[[toc]] + +## Introduction + +This is the second release of the version 21 [release cycle](../project/policies.md#release-schedule). There will be one more release in September before version 21 is "finalized" in the December long-term support version. One thing users may have noticed starting in 21.3, the router was moved to its own package: [`sanic-routing`](https://pypi.org/project/sanic-routing). This change is likely to stay for now. Starting with this release, the minimum required version is 0.7.0. + +## What to know + +More details in the [Changelog](https://sanic.readthedocs.io/en/stable/sanic/changelog.html). Notable new or breaking features, and what to upgrade... + +### Deprecation of `StreamingHTTPResponse` + +The use of `StreamingHTTPResponse` has been deprecated and will be removed in the 21.12 release. This impacts both `sanic.response.stream` and `sanic.response.file_stream`, which both under the hood instantiate `StreamingHTTPResponse`. + +Although the exact migration path has yet to be determined, `sanic.response.stream` and `sanic.response.file_stream` will continue to exist in v21.12 in some form as convenience operators. Look for more details throughout this Summer as we hope to have this finalized by the September release. + +### Deprecation of `CompositionView` + +Usage of `CompositionView` has been deprecated and will be removed in 21.12. + +### Deprecation of path parameter types: `string` and `number` + +Going forward, you should use `str` and `float` for path param types instead of `string` and `number`. + +```python +@app.get("//") +async def handler(request, foo: str, bar: float): + ... +``` + +Existing `string` and `number` types are aliased and will continue to work, but will be removed in v21.12. + +### Version 0.7 router upgrades + +This includes a number of bug fixes and more gracefully handles a wider array of edge cases than v0.6. If you experience any patterns that are not supported, [please report them](https://github.com/sanic-org/sanic-routing/issues). You can see some of the issues resolved on the `sanic-routing` [release notes](https://github.com/sanic-org/sanic-routing/releases). + +### Inline streaming with `eof()` + +Version 21.3 included [big changes in how streaming is handled](https://sanic.dev/en/guide/release-notes/v21.3.html#what-to-know). The pattern introduced will become the default (see below). As a convenience, a new `response.eof()` method has been included. It should be called once the final data has been pushed to the client: + +```python +@app.route("/") +async def test(request): + response = await request.respond(content_type="text/csv") + await response.send("foo,") + await response.send("bar") + await response.eof() + return response +``` + +### New path parameter type: `slug` + +You can now specify a dynamic path segment as a `slug` with appropriate matching: + +```python +@app.get("/articles/") +async def article(request, article_slug: str): + ... +``` + +Slugs must consist of lowercase letters or digits. They may contain a hyphen (`-`), but it cannot be the first character. + +``` +this-is-a-slug +with-123-is-also-a-slug +111-at-start-is-a-slug +NOT-a-slug +-NOT-a-slug +``` + +### Stricter application and blueprint names, and deprecation + +Your application and `Blueprint` instances must conform to a stricter set of requirements: + +1. Only consisting of alphanumeric characters +2. May contain a hyphen (`-`) or an underscore (`_`) +3. Must begin with a letter (uppercase or lowercase) + +The naming convention is similar to Python variable naming conventions, with the addition of allowing hyphens (`-`). + +The looser standard has been deprecatated. Beginning in 21.12, non-conformance will be a startup time error. + +### A new access on `Route` object: `route.uri` + +The `Route` object in v21.3 no longer had a `uri` attribute. Instead, the closes you could get was `route.path`. However, because of how `sanic-routing` works, the `path` property does *not* have a leading `/`. This has been corrected so that now there is a `route.uri` with a leading slash: + +```python +route.uri == f"/{route.path}" +``` + +### A new accessor on `Request` object impacting IPs + +To access the IP address of the incoming request, Sanic has had a convenience accessor on the request object: `request.ip`. That is not new, and comes from an underlying object that provides details about the open HTTP connection: `request.conn_info`. + +The current version adds a new `client_ip` accessor to that `conn_info` object. For IPv4, you will not notice a difference. However, for IPv6 applications, the new accessor will provide an "unwrapped" version of the address. Consider the following example: + +```python +@app.get("/") +async def handler(request): + return json( + { + "request.ip": request.ip, + "request.conn_info.client": request.conn_info.client, + "request.conn_info.client_ip": request.conn_info.client_ip, + } + ) + + +app.run(sock=my_ipv6_sock) +``` + +```bash +$ curl http://\[::1\]:8000 +{ + "request.ip": "::1", + "request.conn_info.client": "[::1]", + "request.conn_info.client_ip": "::1" +} + +``` + +### Alternate `Config` and `Sanic.ctx` objects + +You can now pass your own config and context objects to your Sanic applications. A custom configuration *should* be a subclass of `sanic.config.Config`. The context object can be anything you want, with no restrictions whatsoever. + +```python +class CustomConfig(Config): + ... + +config = CustomConfig() +app = Sanic("custom", config=config) +assert isinstance(app.config, CustomConfig) +``` + +And... + +```python +class CustomContext: + ... + +ctx = CustomContext() +app = Sanic("custom", ctx=ctx) +assert isinstance(app.ctx, CustomContext) +``` + +### Sanic CLI improvements + +1. New flag for existing feature: `--auto-reload` +2. Some new shorthand flags for existing arguments +3. New feature: `--factory` +4. New feature: `--simple` +5. New feature: `--reload-dir` + +#### Factory applications + +For applications that follow the factory pattern (a function that returns a `sanic.Sanic` instance), you can now launch your application from the Sanic CLI using the `--factory` flag. + +```python +from sanic import Blueprint, Sanic, text + +bp = Blueprint(__file__) + +@bp.get("/") +async def handler(request): + return text("😎") + +def create_app() -> Sanic: + app = Sanic(__file__) + app.blueprint(bp) + return app +``` + +You can now run it: + +```bash +$ sanic path.to:create_app --factory +``` + +#### Sanic Simple Server + +Sanic CLI now includes a simple pattern to serve a directory as a web server. It will look for an `index.html` at the directory root. + +```bash +$ sanic ./path/to/dir --simple +``` + +::: warning +This feature is still in early *beta* mode. It is likely to change in scope. +::: + +#### Additional reload directories + +When using either `debug` or `auto-reload`, you can include additional directories for Sanic to watch for new files. + +```bash +sanic ... --reload-dir=/path/to/foo --reload-dir=/path/to/bar +``` + +::: tip +You do *NOT* need to include this on your application directory. Sanic will automatically reload when any Python file in your application changes. You should use the `reload-dir` argument when you want to listen and update your application when static files are updated. +::: + +### Version prefix + +When adding `version`, your route is prefixed with `/v`. This will always be at the beginning of the path. This is not new. + +```python +# /v1/my/path +app.route("/my/path", version=1) +``` + +Now, you can alter the prefix (and therefore add path segments *before* the version). + +```python +# /api/v1/my/path +app.route("/my/path", version=1, version_prefix="/api/v") +``` + +The `version_prefix` argument is can be defined in: + +- `app.route` and `bp.route` decorators (and all the convenience decorators also) +- `Blueprint` instantiation +- `Blueprint.group` constructor +- `BlueprintGroup` instantiation +- `app.blueprint` registration + +### Signal event auto-registration + +Setting `config.EVENT_AUTOREGISTER` to `True` will allow you to await any signal event even if it has not previously been defined with a signal handler. + +```python +@app.signal("do.something.start") +async def signal_handler(): + await do_something() + await app.dispatch("do.something.complete") + +# somethere else in your app: +await app.event("do.something.complete") +``` + +### Infinitely reusable and nestable `Blueprint` and `BlueprintGroup` + +A single `Blueprint` may not be assigned and reused to multiple groups. The groups themselves can also by infinitely nested into one or more other groups. This allows for an unlimited range of composition. + +### HTTP methods as `Enum` + +Sanic now has `sanic.HTTPMethod`, which is an `Enum`. It can be used interchangeably with strings: + +```python +from sanic import Sanic, HTTPMethod + +@app.route("/", methods=["post", "PUT", HTTPMethod.PATCH]) +async def handler(...): + ... +``` + +### Expansion of `HTTPMethodView` + +Class based views may be attached now in one of three ways: + +**Option 1 - Existing** +```python +class DummyView(HTTPMethodView): + ... + +app.add_route(DummyView.as_view(), "/dummy") +``` + +**Option 2 - From `attach` method** +```python +class DummyView(HTTPMethodView): + ... + +DummyView.attach(app, "/") +``` + +**Option 3 - From class definition at `__init_subclass__`** +```python +class DummyView(HTTPMethodView, attach=app, uri="/"): + ... +``` + +Options 2 and 3 are useful if your CBV is located in another file: + +```python +from sanic import Sanic, HTTPMethodView + +class DummyView(HTTPMethodView, attach=Sanic.get_app(), uri="/"): + ... +``` + +## News + +### Discord and support forums + +If you have not already joined our community, you can become a part by joining the [Discord server](https://discord.gg/FARQzAEMAA) and the [Community Forums](https://community.sanicframework.org/). Also, follow [@sanicframework](https://twitter.com/sanicframework) on Twitter. + +### SCO 2022 elections + +The Summer 🏝/Winter ❄️ (choose your Hemisphere) is upon us. That means we will be holding elections for the SCO. This year, we will have the following positions to fill: + +- Steering Council Member (2 year term) +- Steering Council Member (2 year term) +- Steering Council Member (1 year term) +- Release Manager v22 +- Release Manager v22 + +[@vltr](https://github.com/vltr) will be staying on to complete his second year on the Steering Council. + +If you are interested in learning more, you can read about the SCO [roles and responsibilities](../project/scope.md#roles-and-responsibilities), or Adam Hopkins on Discord. + +Nominations will begin September 1. More details will be available on the Forums as we get closer. + +### New project underway + +We have added a new project to the SCO umbrella: [`sanic-ext`](https://github.com/sanic-org/sanic-ext). It is not yet released, and in heavy active development. The goal for the project will ultimately be to replace [`sanic-openapi`](https://github.com/sanic-org/sanic-openapi) with something that provides more features for web application developers, including input validation, CORS handling, and HTTP auto-method handlers. If you are interested in helping out, let us know on Discord. Look for an initial release of this project sometime (hopefully) before the September release. + +## Thank you + +Thank you to everyone that participated in this release: :clap: + +[@aaugustin](https://github.com/aaugustin) +[@ahopkins](https://github.com/ahopkins) +[@ajaygupta2790](https://github.com/ajaygupta2790) +[@ashleysommer](https://github.com/ashleysommer) +[@ENT8R](https://github.com/ent8r) +[@fredlllll](https://github.com/fredlllll) +[@graingert](https://github.com/graingert) +[@harshanarayana](https://github.com/harshanarayana) +[@jdraymon](https://github.com/jdraymon) +[@Kyle-Verhoog](https://github.com/kyle-verhoog) +[@sanjeevanahilan](https://github.com/sanjeevanahilan) +[@sjsadowski](https://github.com/sjsadowski) +[@Tronic](https://github.com/tronic) +[@vltr](https://github.com/vltr) +[@ZinkLu](https://github.com/zinklu) + +--- + +If you enjoy the project, please consider contributing. Of course we love code contributions, but we also love contributions in any form. Consider writing some documentation, showing off use cases, joining conversations and making your voice known, and if you are able, [financial contributions](https://opencollective.com/sanic-org/). diff --git a/src/pt/guide/release-notes/v21.9.md b/src/pt/guide/release-notes/v21.9.md new file mode 100644 index 0000000000..3a673be722 --- /dev/null +++ b/src/pt/guide/release-notes/v21.9.md @@ -0,0 +1,234 @@ +# Version 21.9 + +[[toc]] + +## Introduction + +This is the third release of the version 21 [release cycle](../../org/policies.md#release-schedule). Version 21 will be "finalized" in the December long-term support version release. + +## What to know + +More details in the [Changelog](https://sanic.readthedocs.io/en/stable/sanic/changelog.html). Notable new or breaking features, and what to upgrade... + +### Removal of config values: `WEBSOCKET_READ_LIMIT`, `WEBSOCKET_WRITE_LIMIT` and `WEBSOCKET_MAX_QUEUE` + +With the complete overhaul of the websocket implementation, these configuration values were removed. There currently is not a plan to replace them. + +### Deprecation of default value of `FALLBACK_ERROR_FORMAT` + +When no error handler is attached, Sanic has used `html` as the fallback format-type. This has been deprecated and will change to `text` starting in v22.3. While the value of this has changed to `auto`, it will still continue to use HTML as the last resort thru v21.12LTS before changing. + +### `ErrorHandler.lookup` signature deprecation + +The `ErrorHandler.lookup` now **requires** two positional arguments: + +```python +def lookup(self, exception, route_name: Optional[str]): +``` + +A non-conforming method will cause Blueprint-specific exception handlers to not properly attach. + +### Reminder of upcoming removals + +As a reminder, the following items have already been deprecated, and will be removed in version 21.12LTS + +- `CompositionView` +- `load_env` (use `env_prefix` instead) +- Sanic objects (application instances, blueprints, and routes) must by alphanumeric conforming to: `^[a-zA-Z][a-zA-Z0-9_\-]*$` +- Arbitrary assignment of objects to application and blueprint instances (use `ctx` instead; removal of this has been bumped from 21.9 to 21.12) + +### Overhaul of websockets + +There has been a huge overhaul to the handling of websocket connections. Thanks to [@aaugustin](https://github.com/aaugustin) the [`websockets`](https://websockets.readthedocs.io/en/stable/index.html) now has a new implementation that allows Sanic to handle the I/O of websocket connections on its own. Therefore, Sanic has bumped the minimum version to `websockets>=10.0`. + +The change should mostly be unnoticeable to developers, except that some of the oddities around websocket handlers in Sanic have been corrected. For example, you now should be able to catch the `CancellError` yourself when someone disconnects: + +```python +@app.websocket("/") +async def handler(request, ws): + try: + while True: + await asyncio.sleep(0.25) + except asyncio.CancelledError: + print("User closed connection") +``` + +### Built-in signals + +Version [21.3](./v21.3.md) introduced [signals](../advanced/signals.md). Now, Sanic dispatches signal events **from within the codebase** itself. This means that developers now have the ability to hook into the request/response cycle at a much closer level than before. + +Previously, if you wanted to inject some logic you were limited to middleware. Think of integrated signals as _super_-middleware. The events that are dispatched now include: + +- `http.lifecycle.begin` +- `http.lifecycle.complete` +- `http.lifecycle.exception` +- `http.lifecycle.handle` +- `http.lifecycle.read_body` +- `http.lifecycle.read_head` +- `http.lifecycle.request` +- `http.lifecycle.response` +- `http.lifecycle.send` +- `http.middleware.after` +- `http.middleware.before` +- `http.routing.after` +- `http.routing.before` +- `server.init.after` +- `server.init.before` +- `server.shutdown.after` +- `server.shutdown.before` + +::: tip Note +The `server` signals are the same as the four (4) main server listener events. In fact, those listeners themselves are now just convenience wrappers to signal implementations. +::: + +### Smarter `auto` exception formatting + +Sanic will now try to respond with an appropriate exception format based upon the endpoint and the client. For example, if your endpoint always returns a `sanic.response.json` object, then any exceptions will automatically be formatted in JSON. The same is true for `text` and `html` responses. + +Furthermore, you now can _explicitly_ control which formatter to use on a route-by-route basis using the route definition: + +```python +@app.route("/", error_format="json") +async def handler(request): + pass +``` + +### Blueprint copying + +Blueprints can be copied to new instances. This will carry forward everything attached to it, like routes, middleware, etc. + +```python +v1 = Blueprint("Version1", version=1) + +@v1.route("/something") +def something(request): + pass + +v2 = v1.copy("Version2", version=2) + +app.blueprint(v1) +app.blueprint(v2) +``` + +``` +/v1/something +/v2/something +``` +### Blueprint group convenience methods + +Blueprint groups should now have all of the same methods available to them as regular Blueprints. With this, along with Blueprint copying, Blueprints should now be very composable and flexible. + +### Accept header parsing + +Sanic `Request` objects can parse an `Accept` header to provide an ordered list of the client's content-type preference. You can simply access it as an accessor: + +```python +print(request.accept) +# ["*/*"] +``` + +It also is capable of handling wildcard matching. For example, assuming the incoming request included: + +``` +Accept: */* +``` + +Then, the following is `True`: + +```python +"text/plain" in request.accept +``` + +### Default exception messages + +Any exception that derives from `SanicException` can now define a default exception message. This makes it more convenient and maintainable to reuse the same exception in multiple places without running into DRY issues with the message that the exception provides. + +```python +class TeaError(SanicException): + message = "Tempest in a teapot" + + +raise TeaError +``` + +### Type annotation conveniences + +It is now possible to control the path parameter types using Python's type annotations. Instead of doing this: + +```python +@app.route("///") +def handler(request: Request, one: int, two: float, three: UUID): + ... +``` + +You can now simply do this: + +```python +@app.route("///") +def handler(request: Request, one: int, two: float, three: UUID): + ... +``` + +Both of these examples will result in the same routing principles to be applied. + +### Explicit static resource type + +You can now explicitly tell a `static` endpoint whether it is supposed to treat the resource as a file or a directory: + +```python +static("/", "/path/to/some/file", resource_type="file")) +``` + +## News + +### Release of `sanic-ext` and deprecation of `sanic-openapi` + +One of the core principles of Sanic is that it is meant to be a tool, not a dictator. As the frontpage of this website states: + +> Build the way you want to build without letting your tooling constrain you. + +This means that a lot of common features used (specifically by Web API developers) do not exist in the `sanic` repository. This is for good reason. Being unopinionated provides the developer freedom and flexibility. + +But, sometimes you do not want to have to build and rebuild the same things. Sanic has until now really relied upon the awesome support of the community to fill in the gaps with plugins. + +From the early days, there has been an official `sanic-openapi` package that offered the ability to create OpenAPI documentation based upon your application. But, that project has been plagued over the years and has not been given as much priority as the main project. + +Starting with the release of v21.9, the SCO is deprecating the `sanic-openapi` package and moving it to maintenance mode. This means that it will continue to get updates as needed to maintain it for the current future, but it will not receive any new feature enhancements. + +A new project called `sanic-ext` is taking its place. This package provides not only the ability to build OAS3 documentation, but fills in many of the gaps that API developers may want in their applications. For example, out of the box it will setup CORS, and auto enable `HEAD` and `OPTIONS` responses where needed. It also has the ability validate incoming data using either standard library Dataclasses or Pydantic models. + +The list of goodies includes: +- CORS protection +- incoming request validation +- auto OAS3 documentation using Redoc and/or Swagger UI +- auto `HEAD`, `OPTIONS`, and `TRACE` responses +- dependency injection +- response serialization + +This project is still in `alpha` mode for now and is subject to change. While it is considered to be production capable, there may be some need to change the API as we continue to add features. + +Checkout the [documentation](../../plugins/sanic-ext/getting-started.md) for more details. + + +## Thank you + +Thank you to everyone that participated in this release: :clap: + +[@aaugustin](https://github.com/aaugustin) +[@ahopkins](https://github.com/ahopkins) +[@ashleysommer](https://github.com/ashleysommer) +[@cansarigol3megawatt](https://github.com/cansarigol3megawatt) +[@ChihweiLHBird](https://github.com/ChihweiLHBird) +[@gluhar2006](https://github.com/gluhar2006) +[@komar007](https://github.com/komar007) +[@ombe1229](https://github.com/ombe1229) +[@prryplatypus](https://github.com/prryplatypus) +[@SaidBySolo](https://github.com/SaidBySolo) +[@Tronic](https://github.com/tronic) +[@vltr](https://github.com/vltr) + +And, a special thank you to [@miss85246](https://github.com/miss85246) and [@ZinkLu](https://github.com/ZinkLu) for their tremendous work keeping the documentation synced and translated into Chinese. + +--- + +If you enjoy the project, please consider contributing. Of course we love code contributions, but we also love contributions in any form. Consider writing some documentation, showing off use cases, joining conversations and making your voice known, and if you are able, [financial contributions](https://opencollective.com/sanic-org/). diff --git a/src/pt/guide/release-notes/v22.3.md b/src/pt/guide/release-notes/v22.3.md new file mode 100644 index 0000000000..4bc98ad66c --- /dev/null +++ b/src/pt/guide/release-notes/v22.3.md @@ -0,0 +1,221 @@ +# Version 22.3 + +[[toc]] + +## Introduction + +This is the first release of the version 22 [release cycle](../../org/policies.md#release-schedule). All of the standard SCO libraries are now entering the same release cycle and will follow the same versioning pattern. Those packages are: + +- [`sanic-routing`](https://github.com/sanic-org/sanic-routing) +- [`sanic-testing`](https://github.com/sanic-org/sanic-testing) +- [`sanic-ext`](https://github.com/sanic-org/sanic-ext) + +## What to know + +More details in the [Changelog](https://sanic.readthedocs.io/en/stable/sanic/changelog.html). Notable new or breaking features, and what to upgrade... + +### Application multi-serve + +The Sanic server now has an API to allow you to run multiple applications side-by-side in the same process. This is done by calling `app.prepare(...)` on one or more application instances, one or many times. Each time it should be bound to a unique host/port combination. Then, you begin serving the applications by calling `Sanic.serve()`. + +```python +app = Sanic("One") +app2 = Sanic("Two") + +app.prepare(port=9999) +app.prepare(port=9998) +app.prepare(port=9997) +app2.prepare(port=8888) +app2.prepare(port=8887) + +Sanic.serve() +``` + +In the above snippet, there are two applications that will be run concurrently and bound to multiple ports. This feature is *not* supported in the CLI. + +This pattern is meant to be an alternative to running `app.run(...)`. It should be noted that `app.run` is now just a shorthand for the above pattern and is still fully supported. + +### 👶 *BETA FEATURE* - New path parameter type: file extensions + +A very common pattern is to create a route that dynamically generates a file. The endpoint is meant to match on a file with an extension. There is a new path parameter to match files: ``. + +```python +@app.get("/path/to/") +async def handler(request, filename, ext): + ... +``` + +This will catch any pattern that ends with a file extension. You may, however want to expand this by specifying which extensions, and also by using other path parameter types for the file name. + +For example, if you want to catch a `.jpg` file that is only numbers: + +```python +@app.get("/path/to/") +async def handler(request, filename, ext): + ... +``` + +Some potential examples: + +| definition | example | filename | extension | +| --------------------------------- | ----------- | ----------- | ---------- | +| \ | page.txt | `"page"` | `"txt"` | +| \ | cat.jpg | `"cat"` | `"jpg"` | +| \ | cat.jpg | `"cat"` | `"jpg"` | +| | 123.txt | `123` | `"txt"` | +| | 123.svg | `123` | `"svg"` | +| | 3.14.tar.gz | `3.14` | `"tar.gz"` | + +### 🚨 *BREAKING CHANGE* - Path parameter matching of non-empty strings + +A dynamic path parameter will only match on a non-empty string. + +Previously a route with a dynamic string parameter (`/` or `/`) would match on any string, including empty strings. It will now only match a non-empty string. To retain the old behavior, you should use the new parameter type: `/`. + +```python +@app.get("/path/to/") +async def handler(request, foo) + ... +``` + +### 🚨 *BREAKING CHANGE* - `sanic.worker.GunicornWorker` has been removed + +Departing from our normal deprecation policy, the `GunicornWorker` was removed as a part of the process of upgrading the Sanic server to include multi-serve. This decision was made largely in part because even while it existed it was not an optimal strategy for deploying Sanic. + +If you want to deploy Sanic using `gunicorn`, then you are advised to do it using [the strategy implemented by `uvicorn`](https://www.uvicorn.org/#running-with-gunicorn). This will effectively run Sanic as an ASGI application through `uvicorn`. You can upgrade to this pattern by installing `uvicorn`: + +``` +pip install uvicorn +``` + +Then, you should be able to run it with a pattern like this: + +``` +gunicorn path.to.sanic:app -k uvicorn.workers.UvicornWorker +``` + +### Authorization header parsing + +The `Authorization` header has been partially parseable for some time now. You have been able to use `request.token` to gain access to a header that was in one of the following two forms: + +``` +Authorization: Token +Authorization: Bearer +``` + +Sanic can now parse more credential types like `BASIC`: + +``` +Authorization: Basic Z2lsLWJhdGVzOnBhc3N3b3JkMTIz +``` + +This can be accessed now as `request.credentials`: + +```python +print(request.credentials) +# Credentials(auth_type='Basic', token='Z2lsLWJhdGVzOnBhc3N3b3JkMTIz', _username='gil-bates', _password='password123') +``` + +### CLI arguments optionally injected into application factory + +Sanic will now attempt to inject the parsed CLI arguments into your factory if you are using one. + +```python +def create_app(args): + app = Sanic("MyApp") + print(args) + return app +``` +``` +$sanic p:create_app --factory +Namespace(module='p:create_app', factory=True, simple=False, host='127.0.0.1', port=8000, unix='', cert=None, key=None, tls=None, tlshost=False, workers=1, fast=False, access_log=False, debug=False, auto_reload=False, path=None, dev=False, motd=True, verbosity=None, noisy_exceptions=False) +``` + +If you are running the CLI with `--factory`, you also have the option of passing arbitrary arguments to the command, which will be injected into the argument `Namespace`. + +``` +sanic p:create_app --factory --foo=bar +Namespace(module='p:create_app', factory=True, simple=False, host='127.0.0.1', port=8000, unix='', cert=None, key=None, tls=None, tlshost=False, workers=1, fast=False, access_log=False, debug=False, auto_reload=False, path=None, dev=False, motd=True, verbosity=None, noisy_exceptions=False, foo='bar') +``` + +### New reloader process listener events + +When running Sanic server with auto-reload, there are two new events that trigger a listener *only* on the reloader process: + +- `reload_process_start` +- `reload_process_stop` + +These are only triggered if the reloader is running. + +```python +@app.reload_process_start +async def reload_start(*_): + print(">>>>>> reload_start <<<<<<") + + +@app.reload_process_stop +async def reload_stop(*_): + print(">>>>>> reload_stop <<<<<<") +``` + +### The event loop is no longer a required argument of a listener + +You can leave out the `loop` argument of a listener. Both of these examples work as expected: + +```python +@app.before_server_start +async def without(app): + ... + +@app.before_server_start +async def with(app, loop): + ... +``` + +### Removal - Debug mode does not automatically start the reloader + +When running with `--debug` or `debug=True`, the Sanic server will not automatically start the auto-reloader. This feature of doing both on debug was deprecated in v21 and removed in this release. If you would like to have *both* debug mode and auto-reload, you can use `--dev` or `dev=True`. + +**dev = debug mode + auto reloader** + +### Deprecation - Loading of lower case environment variables + +Sanic loads prefixed environment variables as configuration values. It has not distinguished between uppercase and lowercase as long as the prefix matches. However, it has always been the convention that the keys should be uppercase. This is deprecated and you will receive a warning if the value is not uppercase. In v22.9 only uppercase and prefixed keys will be loaded. + +## News + +### Packt publishes new book on Sanic web development + +---:1 +There is a new book on **Python Web Development with Sanic** by [@ahopkins](https://github.com/ahopkins). The book is endorsed by the SCO and part of the proceeds of all sales go directly to the SCO for further development of Sanic. + +You can learn more at [sanicbook.com](https://sanicbook.com/) +:--:1 +![Python Web Development with Sanic](https://sanicbook.com/images/SanicCoverFinal.png) +:--- + +## Thank you + +Thank you to everyone that participated in this release: :clap: + +[@aericson](https://github.com/aericson) +[@ahankinson](https://github.com/ahankinson) +[@ahopkins](https://github.com/ahopkins) +[@ariebovenberg](https://github.com/ariebovenberg) +[@ashleysommer](https://github.com/ashleysommer) +[@Bluenix2](https://github.com/Bluenix2) +[@ChihweiLHBird](https://github.com/ChihweiLHBird) +[@dotlambda](https://github.com/dotlambda) +[@eric-spitler](https://github.com/eric-spitler) +[@howzitcdf](https://github.com/howzitcdf) +[@jonra1993](https://github.com/jonra1993) +[@prryplatypus](https://github.com/prryplatypus) +[@raphaelauv](https://github.com/raphaelauv) +[@SaidBySolo](https://github.com/SaidBySolo) +[@SerGeRybakov](https://github.com/SerGeRybakov) +[@Tronic](https://github.com/Tronic) + + +--- + +If you enjoy the project, please consider contributing. Of course we love code contributions, but we also love contributions in any form. Consider writing some documentation, showing off use cases, joining conversations and making your voice known, and if you are able: [financial contributions](https://opencollective.com/sanic-org/). diff --git a/src/pt/help.md b/src/pt/help.md new file mode 100644 index 0000000000..3c4f2cf078 --- /dev/null +++ b/src/pt/help.md @@ -0,0 +1,29 @@ +--- +layout: BlankLayout +--- + +# Need some help? + +As an active community of developers, we try to support each other. If you need some help, try one of the following: + +---:1 + +### Discord :speech_balloon: + +Best place to turn for quick answers and live chat + +`#sanic-support` channel on the [Discord server](https://discord.gg/FARQzAEMAA) + +:--:1 + +### Community Forums :busts_in_silhouette: + +Good for sharing snippets of code and longer support queries + +`Questions and Help` category on the [Forums](https://community.sanicframework.org/c/questions-and-help/6) + +:--- + +--- + +We also actively monitor the `[sanic]` tag on [Stack Overflow](https://stackoverflow.com/questions/tagged/sanic). diff --git a/src/pt/org/README.md b/src/pt/org/README.md new file mode 100644 index 0000000000..dab306f45e --- /dev/null +++ b/src/pt/org/README.md @@ -0,0 +1 @@ +# Project diff --git a/src/pt/org/policies.md b/src/pt/org/policies.md new file mode 100644 index 0000000000..85a4ab524f --- /dev/null +++ b/src/pt/org/policies.md @@ -0,0 +1,61 @@ +# Policies + +## Versioning + +Sanic uses [calendar versioning](https://calver.org/), aka "calver". To be more specific, the pattern follows: + +``` +YY.MM.MICRO +``` + +Generally, versions are referred to in their ``YY.MM`` form. The `MICRO` number indicates an incremental patch version, starting at `0`. + +## Release Schedule + +There are four (4) scheduled releases per year: March, June, September, and December. Therefore, there are four (4) released versions per year: `YY.3`, `YY.6`, `YY.9`, and `YY.12`. + +This release schedule provides: + +- a predictable release cadence, +- relatively short development windows allowing features to be regularly released, +- controlled [deprecations](#deprecation), and +- consistent stability with a yearly LTS. + +We also use the yearly release cycle in conjunction with our governance model, covered by the [S.C.O.P.E.](./scope.md) + +### Long term support v Interim releases + +Sanic releases a long term support release (aka "LTS") once a year in December. The LTS releases receive bug fixes and security updates for **24 months**. Interim releases throughout the year occur every three months, and are supported until the subsequent release. + +| Version | LTS | Supported | +| ------- | ------------- | ----------------------- | +| 21.12 | until 2023-12 | :white_check_mark: | +| 21.9 | | :x: | +| 21.6 | | :x: | +| 21.3 | | :x: | +| 20.12 | until 2022-12 | :ballot_box_with_check: | +| 20.9 | | :x: | +| 20.6 | | :x: | +| 20.3 | | :x: | +| 19.12 | | :x: | +| 19.9 | | :x: | +| 19.6 | | :x: | +| 19.3 | | :x: | +| 18.12 | | :x: | +| 0.8.3 | | :x: | +| 0.7.0 | | :x: | +| 0.6.0 | | :x: | +| 0.5.4 | | :x: | +| 0.4.1 | | :x: | +| 0.3.1 | | :x: | +| 0.2.0 | | :x: | +| 0.1.9 | | :x: | + +:ballot_box_with_check: = security/bug fixes +:white_check_mark: = full support + +## Deprecation + +Before a feature is deprecated, or breaking changes are introduced into the API, it shall be publicized and shall appear with deprecation warnings through two release cycles. No deprecations shall be made in an LTS release. + +Breaking changes or feature removal may happen outside of these guidelines when absolutely warranted. These circumstances should be rare. For example, it might happen when no alternative is available to curtail a major security issue. diff --git a/src/pt/org/scope.md b/src/pt/org/scope.md new file mode 100644 index 0000000000..1af529361a --- /dev/null +++ b/src/pt/org/scope.md @@ -0,0 +1,264 @@ +--- +title: S.C.O.P.E +--- + + +Sanic Community Organization Policy E-manual +============================================ + +December 2019, version 1 + +Goals +----- + +To create a sustainable, community-driven organization around the Sanic projects that promote: (1) stability and predictability, (2) quick iteration and enhancement cycles, (3) engagement from outside contributors, (4) overall reliable software, and (5) a safe, rewarding environment for the community members. + +Overview +-------- + +This Policy is the governance model for the Sanic Community Organization (“SCO”). The SCO is a meritocratic, consensus-based community organization responsible for all projects adopted by it. Anyone with an interest in one of the projects can join the community, contribute to the community or projects, and participate in the decision making process. This document describes how that participation takes place and how to set about earning merit within the project community. + +Structure +--------- + +The SCO has multiple **projects**. Each project is represented by a single GitHub repository under the Sanic community umbrella. These projects are used by **users**, developed by **contributors**, governed by **core developers**, released by **release managers**, and ultimately overseen by a **steering council**. If this sounds similar to the Python project and PEP 8016 that is because it is intentionally designed that way. + +Roles and responsibilities +-------------------------- + +### Users + +Users are community members who have a need for the projects. They are the developers and personnel that download and install the packages. Users are the **most important** members of the community and without them the projects would have no purpose. Anyone can be a user and the licenses adopted by the projects shall be appropriate open source licenses. + +_The SCO asks its users to participate in the project and community as much as possible._ + +User contributions enable the project team to ensure that they are satisfying the needs of those users. Common user contributions include (but are not limited to): + +* evangelizing about the project (e.g. a link on a website and word-of-mouth awareness raising) +* informing developers of strengths and weaknesses from a new user perspective +* providing moral support (a ‘thank you’ goes a long way) +* providing financial support (the software is open source, but its developers need to eat) + +Users who continue to engage with the SCO, its projects, and its community will often become more and more involved. Such users may find themselves becoming contributors, as described in the next section. + +### Contributors + +Contributors are community members who contribute in concrete ways to one or more of the projects. Anyone can become a contributor and contributions can take many forms. Contributions and requirements are governed by each project separately by a contribution policy. + +There is **no expectation** of commitment to the project, **no specific skill requirements** and **no selection process**. + +In addition to their actions as users, contributors may also find themselves doing one or more of the following: + +* supporting new users (existing users are often the best people to support new users) +* reporting bugs +* identifying requirements +* providing graphics and web design +* Programming +* example use cases +* assisting with project infrastructure +* writing documentation +* fixing bugs +* adding features +* providing constructive opinions and engaging in community discourse + +Contributors engage with the projects through GitHub and the Community Forums. They submit changes to the projects itself via pull requests, which will be considered for inclusion in the project by the community at large. The Community Forums are the most appropriate place to ask for help when making that first contribution. + +Indeed one of the most important roles of a contributor may be to **simply engage in the community conversation**. Most decisions about the direction of a project are made by consensus. This is discussed in more detail below. In general, however, it is helpful for the health and direction of the projects for the contributors to **speak freely** (within the confines of the code of conduct) and **express their opinions and experiences** to help drive the consensus building. + +As contributors gain experience and familiarity with a project, their profile within, and commitment to, the community will increase. At some stage, they may find themselves being nominated for a core developer team. + +### Core Developer + +Each project under the SCO umbrella has its own team of core developers. They are the people in charge of that project. + +_What is a core developer?_ + +Core developers are community members who have shown that they are committed to the continued development of the project through ongoing engagement with the community. Being a core developer allows contributors to more easily carry on with their project related activities by giving them direct access to the project’s resources. They can make changes directly to the project repository without having to submit changes via pull requests from a fork. + +This does not mean that a core developer is free to do what they want. In fact, core developers have no more direct authority over the final release of a package than do contributors. While this honor does indicate a valued member of the community who has demonstrated a healthy respect for the project’s aims and objectives, their work continues to be reviewed by the community before acceptance in an official release. + +_What can a core developer do on a project?_ + +Each project might define this role slightly differently. However, the general usage of this designation is that an individual has risen to a level of trust within the community such that they now are given some control. This comes in the form of push rights to non-protected branches, and the ability to have a voice in the approval of pull requests. + +The projects employ various communication mechanisms to ensure that all contributions are reviewed by the community as a whole. This includes tools provided by GitHub, as well as the Community Forums. By the time a contributor is invited to become a core developer, they should be familiar with the various tools and workflows as a user and then as a contributor. + +_How to become a core developer?_ + +Anyone can become a core developer; there are no special requirements, other than to have shown a willingness and ability to positively participate in the project as a team player. + +Typically, a potential core developer will need to show that they have an understanding of the project, its objectives and its strategy. They will also have provided valuable contributions to the project over a period of time. However, there is **no technical or other skill** requirement for eligibility. + +New core developers can be **nominated by any existing core developer** at any time. At least twice a year (April and October) there will be a ballot process run by the Steering Council. Voting should be done by secret ballot. Each existing core developer for that project receives a number of votes equivalent to the number of nominees on the ballot. For example, if there are four nominees, then each existing core developer has four votes. The core developer may cast those votes however they choose, but may not vote for a single nominee more than once. A nominee must receive two-thirds approval from the number of cast ballots (not the number of eligible ballots). Once accepted by the core developers, it is the responsibility of the Steering Council to approve and finalize the nomination. The Steering Council does not have the right to determine whether a nominee is meritorious enough to receive the core developer title. However, they do retain the right to override a vote in cases where the health of the community would so require. + +Once the vote has been held, the aggregated voting results are published on the Community Forums. The nominee is entitled to request an explanation of any override against them. A nominee that fails to be admitted as a core developer may be nominated again in the future. + +It is important to recognize that being a core developer is a privilege, not a right. That privilege must be earned and once earned it can be removed by the Steering Council (see next section) in extreme circumstances. However, under normal circumstances the core developer title exists for as long as the individual wishes to continue engaging with the project and community. + +A committer who shows an above-average level of contribution to the project, particularly with respect to its strategic direction and long-term health, may be nominated to become a member of the Steering Council, or a Release Manager. This role is described below. + +_What are the rights and responsibilities of core developers?_ + +As discussed, the majority of decisions to be made are by consensus building. In certain circumstances where an issue has become more contentious, or a major decision needs to be made, the Release Manager or Steering Council may decide (or be required) to implement the RFC process, which is outlined in more detail below. + +It is also incumbent upon core developers to have a voice in the governance of the community. All core developers for all of the projects have the ability to be nominated to be on the Steering Council and vote in their elections. + +This Policy (the “SCOPE”) may only be changed under the authority of two-thirds of active core developers, except that in the first six (6) months after adoption, the core developers reserve the right to make changes under the authority of a simple majority of active core developers. + +_What if a core developer becomes inactive?_ + +It is hoped that all core developers participate and remain active on a regular basis in their projects. However, it is also understood that such commitments may not be realistic or possible from time to time. + +Therefore, the Steering Council has the duty to encourage participation and the responsibility to place core developers into an inactive status if they are no longer willing or capable to participate. The main purpose of this is **not to punish** a person for behavior, but to help the development process to continue for those that do remain active. + +To this end, a core developer that becomes “inactive” shall not have commit rights to a repository, and shall not participate in any votes. To be eligible to vote in an election, a core developer **must have been active** at the time of the previous scheduled project release. + +Inactive members may ask the Steering Council to reinstate their status at any time, and upon such request the Steering Council shall make the core developer active again. + +Individuals that know they will be unable to maintain their active status for a period are asked to be in communication with the Steering Council and declare themselves inactive if necessary. + +An “active” core developer is an individual that has participated in a meaningful way during the previous six months. Any further definition is within the discretion of the Steering Council. + +### Release Manager + +Core developers shall have access only to make commits and merges on non-protected branches. The “master” branch and other protected branches are controlled by the release management team for that project. Release managers shall be elected from the core development team by the core development team, and shall serve for a full release cycle. + +Each core developer team may decide how many release managers to have for each release cycle. It is highly encouraged that there be at least two release managers for a release cycle to help divide the responsibilities and not force too much effort upon a single person. However, there also should not be so many managers that their efforts are impeded. + +The main responsibilities of the release management team include: + +* push the development cycle forward by monitoring and facilitating technical discussions +* establish a release calendar and perform actions required to release packages +* approve pull requests to the master branch and other protected branches +* merge pull requests to the master branch and other protected branches + +The release managers **do not have the authority to veto or withhold a merge** of a pull request that otherwise meets contribution criteria and has been accepted by the community. It is not their responsibility to decide what should be developed, but rather that the decisions of the community are carried out and that the project is being moved forward. + +From time to time, a decision may need to be made that cannot be achieved through consensus. In that case, the release managers have the authority to call upon the removal of the decision to the RFC process. This should not occur regularly (unless required as discussed below), and its use should be discouraged in favor of the more communal consensus building strategy. + +Since not all projects have the same requirements, the specifics governing release managers on a project shall be set forth in an Appendix to this Policy, or in the project’s contribution guidelines. + +If necessary, the Steering Council has the right to remove a release manager that is derelict in their duties, or for other good cause. + +### Steering Council + +The Steering Council is the governing body consisting of those individuals identified as the “project owner” and having control of the resources and assets of the SCO. Their ultimate goal is to ensure the smooth operation of the projects by removing impediments, and assisting the members as needed. It is expected that they will be regular voices in the community. + +_What can the Steering Council do?_ + +The members of the Steering Council **do not individually have any more authority than any other core developer**, and shall not have any additional rights to make decisions, commits, merges, or the like on a project. + +However, as a body, the Steering Council has the following capacity: + +* accept, remand, and reject all RFCs +* enforce the community code of conduct +* administer community assets such as repositories, servers, forums, integration services, and the like (or, to delegate such authority to someone else) +* place core developers into inactive status where appropriate take any other enforcement measures afforded to it in this Policy, including, in extreme cases, removing core developers +* adopt or remove projects from the community umbrella + +It is highly encouraged that the Steering Council delegate its authority as much as possible, and where appropriate, to other willing community members. + +The Steering Council **does not have the authority** to change this Policy. + +_How many members are on the Steering Council?_ + +Four. + +While it seems like a committee with four votes may potentially end in a deadlock with no way to break a majority vote, the Steering Council is discouraged from voting as much as possible. Instead, it should try to work by consensus, and requires three consenting votes when it is necessary to vote on a matter. + +_How long do members serve on the Steering Council?_ + +A single term shall be for two calendar years starting in January. Terms shall be staggered so that each year there are two members continuing from the previous year’s council. + +Therefore, the inaugural vote shall have two positions available for a two year term, and two positions available for a one year term. + +There are no limits to the number of terms that can be served, and it is possible for an individual to serve consecutive terms. + +_Who runs the Steering Council?_ + +After the Steering Council is elected, the group shall collectively decide upon one person to act as the Chair. The Chair does not have any additional rights or authority over any other member of the Steering Council. + +The role of the Chair is merely as a coordinator and facilitator. The Chair is expected to ensure that all governance processes are adhered to. The position is more administrative and clerical, and is expected that the Chair sets agendas and coordinates discussion of the group. + +_How are council members elected?_ + +Once a year, **all eligible core developers** for each of the projects shall have the right to elect members to the Steering Council. + +Nominations shall be open from September 1 and shall close on September 30. After that, voting shall begin on October 1 and shall close on October 31. Every core developer active on the date of the June release of the Sanic Framework for that year shall be eligible to receive one vote per vacant seat on the Steering Council. For the sake of clarity, to be eligible to vote, a core developer **does not** need to be a core developer on Sanic Framework, but rather just have been active within their respective project on that date. + +The top recipients of votes shall be declared the winners. If there is any tie, it is highly encouraged that the tied nominees themselves resolve the dispute before a decision is made at random. + +In regards to the inaugural vote of the Steering Council, the top two vote-recipients shall serve for two years, and the next two vote-recipients shall assume the one-year seats. + +To be an eligible candidate for the Steering Council, the individual must have been a core developer in active status on at least one project for the previous twelve months. + +_What if there is a vacancy?_ + +If a vacancy on the Steering Council exists during a term, then the next highest vote-recipient in the previous election shall be offered to complete the remainder of the term. If one cannot be found this way, the Steering Council may decide the most appropriate course of action to fill the seat (whether by appointment, vote, or other means). + +If a member of the Steering Council becomes inactive, then that individual shall be removed from the Steering Council immediately and the seat shall become vacant. + +In extreme cases, the body of all core developers has the right to bring a vote to remove a member of the Steering Council for cause by a two-thirds majority of all eligible voting core developers. + +_How shall the Steering Council conduct its business?_ + +As much as possible, the Steering Council shall conduct its business and discussions in the open. Any member of the community should be allowed to enter the conversation with them. However, at times it may be necessary or appropriate for discussions to be held privately. Selecting the proper venue for conversations is part of the administrative duties of the Chair. + +While the specifics of how to operate are beyond the scope of the Policy, it is encouraged that the Steering Council attempt to meet at least one time per quarter in a “real-time” discussion. This could be achieved via video conferencing, live chatting, or other appropriate means. + +Support +------- + +All participants in the community are encouraged to provide support for users within the project management infrastructure. This support is provided as a way of growing the community. Those seeking support should recognize that all support activity within the project is voluntary and is therefore provided as and when time allows. A user requiring guaranteed response times or results should therefore seek to purchase a support contract from a community member. However, for those willing to engage with the project on its own terms, and willing to help support other users, the community support channels are ideal. + +Decision making process +----------------------- + +Decisions about the future of the projects are made through discussion with all members of the community, from the newest user to the most experienced member. Everyone has a voice. + +All non-sensitive project management discussion takes place on the community forums, or other designated channels. Occasionally, sensitive discussions may occur in private. + +In order to ensure that the project is not bogged down by endless discussion and continual voting, the project operates a policy of **lazy consensus**. This allows the majority of decisions to be made without resorting to a formal vote. For any **major decision** (as defined below), there is a separate Request for Comment (RFC) process. + +### Technical decisions + +Pull requests and technical decisions should generally fall into the following categories. + +* **Routine**: Documentation fixes, code changes that are for cleanup or additional testing. No functionality changes. +* **Minor**: Changes to the code base that either fix a bug, or introduce a trivial feature. No breaking changes. +* **Major**: Any change to the code base that breaks or deprecates existing API, alters operation in a non-trivial manner, or adds a significant feature. + +It is generally the responsibility of the release managers to make sure that changes to the repositories receive the proper authorization before merge. + +The release managers retain the authority to individually review and accept routine decisions that meet standards for code quality without additional input. + +### Lazy consensus + +Decision making (whether by the community or Steering Council) typically involves the following steps: + +* proposal +* discussion +* vote (if consensus is not reached through discussion) +* decision + +Any community member can make a proposal for consideration by the community. In order to initiate a discussion about a new idea, they should post a message on the appropriate channel on the Community forums, or submit a pull request implementing the idea on GitHub. This will prompt a review and, if necessary, a discussion of the idea. + +The goal of this review and discussion is to gain approval for the contribution. Since most people in the project community have a shared vision, there is often little need for discussion in order to reach consensus. + +In general, as long as nobody explicitly opposes a proposal or patch, it is recognized as having the support of the community. This is called lazy consensus; that is, those who have not stated their opinion explicitly have implicitly agreed to the implementation of the proposal. + +Lazy consensus is a very important concept within the SCO. It is this process that allows a large group of people to efficiently reach consensus, as someone with no objections to a proposal need not spend time stating their position, and others need not spend time reading such messages. + +For lazy consensus to be effective, it is necessary to allow an appropriate amount of time before assuming that there are no objections to the proposal. This is somewhat dependent upon the circumstances, but it is generally assumed that 72 hours is reasonable. This requirement ensures that everyone is given enough time to read, digest and respond to the proposal. This time period is chosen so as to be as inclusive as possible of all participants, regardless of their location and time commitments. The facilitators of discussion (whether it be the Chair or the Release Managers, where applicable) shall be charged with determining the proper length of time for such consensus to be reached. + +As discussed above regarding so-called routine decisions, the release managers have the right to make decisions within a shorter period of time. In such cases, lazy consensus shall be implied. + +### Request for Comment (RFC) + +The Steering Council shall be in charge of overseeing the RFC process. It shall be a process that remains open to debate to all members of the community, and shall allow for ample time to consider a proposal and for members to respond and engage in meaningful discussion. + +The final decision is vested with the Steering Council. However, it is strongly discouraged that the Steering Council adopt a decision that is contrary to any consensus that may exist in the community. From time to time this may happen if there is a conflict between consensus and the overall project and community goals. + +An RFC shall be initiated by submission to the Steering Council in the public manner as set forth by the Steering Council. Debate shall continue and be facilitated by the Steering Council in general, and the Chair specifically. + +In circumstances that the Steering Council feels it is appropriate, the RFC process may be waived in favor of lazy consensus. diff --git a/src/pt/plugins/README.md b/src/pt/plugins/README.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/plugins/sanic-ext/README.md b/src/pt/plugins/sanic-ext/README.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/plugins/sanic-ext/configuration.md b/src/pt/plugins/sanic-ext/configuration.md new file mode 100644 index 0000000000..5e0b7d2e53 --- /dev/null +++ b/src/pt/plugins/sanic-ext/configuration.md @@ -0,0 +1,239 @@ +# Configuration + +Sanic Extensions can be configured in all of the same ways that [you can configure Sanic](../../guide/deployment/configuration.md). That makes configuring Sanic Extensions very easy. + +```python +app = Sanic("MyApp") +app.config.OAS_URL_PREFIX = "/apidocs" +``` + +However, there are a few more configuration options that should be considered. + +## Manual `extend` + +---:1 +Even though Sanic Extensions will automatically attach to your application, you can manually choose `extend`. When you do that, you can pass all of the configuration values as a keyword arguments (lowercase). +:--: +```python +app = Sanic("MyApp") +app.extend(oas_url_prefix="/apidocs") +``` +:--- + +---:1 +Or, alternatively they could be passed all at once as a single `dict`. +:--: +```python +app = Sanic("MyApp") +app.extend(config={"oas_url_prefix": "/apidocs"}) +``` +:--- + +---:1 +Both of these solutions suffers from the fact that the names of the configuration settings are not discoverable by an IDE. Therefore, there is also a type annotated object that you can use. This should help the development experience. +:--: +```python +from sanic_ext import Config + +app = Sanic("MyApp") +app.extend(config=Config(oas_url_prefix="/apidocs")) +``` +:--- + +## Settings + +### `cors` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to enable CORS protection + +### `cors_allow_headers` + +- **Type**: `str` +- **Default**: `"*"` +- **Description**: Value of the header: `access-control-allow-headers` + +### `cors_always_send` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to always send the header: `access-control-allow-origin` + +### `cors_automatic_options` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to automatically generate `OPTIONS` endpoints for routes that do *not* already have one defined + +### `cors_expose_headers` + +- **Type**: `str` +- **Default**: `""` +- **Description**: Value of the header: `access-control-expose-headers` + +### `cors_max_age` + +- **Type**: `int` +- **Default**: `5` +- **Description**: Value of the header: `access-control-max-age` + +### `cors_methods` + +- **Type**: `str` +- **Default**: `""` +- **Description**: Value of the header: `access-control-access-control-allow-methods` + +### `cors_origins` + +- **Type**: `str` +- **Default**: `""` +- **Description**: Value of the header: `access-control-allow-origin` + +::: warning +Be very careful if you place `*` here. Do not do this unless you know what you are doing as it can be a security issue. +::: + +### `cors_send_wildcard` + +- **Type**: `bool` +- **Default**: `False` +- **Description**: Whether to send a wildcard origin instead of the incoming request origin + +### `cors_supports_credentials` + +- **Type**: `bool` +- **Default**: `False` +- **Description**: Value of the header: `access-control-allow-credentials` + +### `cors_vary_header` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to add the `vary` header + +### `http_all_methods` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Adds the HTTP `CONNECT` and `TRACE` methods as allowable + +### `http_auto_head` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Automatically adds `HEAD` handlers to any `GET` routes + +### `http_auto_options` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Automatically adds `OPTIONS` handlers to any routes without + +### `http_auto_trace` + +- **Type**: `bool` +- **Default**: `False` +- **Description**: Automatically adds `TRACE` handlers to any routes without + +### `oas` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to enable OpenAPI specification generation + +### `oas_autodoc` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to automatically extract OpenAPI details from the docstring of a route function + +### `oas_ignore_head` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: WHen `True`, it will not add `HEAD` endpoints into the OpenAPI specification + +### `oas_ignore_options` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: WHen `True`, it will not add `OPTIONS` endpoints into the OpenAPI specification + +### `oas_path_to_redoc_html` + +- **Type**: `Optional[str]` +- **Default**: `None` +- **Description**: Path to HTML file to override the existing Redoc HTML + +### `oas_path_to_swagger_html` + +- **Type**: `Optional[str]` +- **Default**: `None` +- **Description**: Path to HTML file to override the existing Swagger HTML + +### `oas_ui_default` + +- **Type**: `Optional[str]` +- **Default**: `"redoc"` +- **Description**: Which OAS documentation to serve on the bare `oas_url_prefix` endpoint; when `None` there will be no documentation at that location + +### `oas_ui_redoc` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to enable the Redoc UI + +### `oas_ui_swagger` + +- **Type**: `bool` +- **Default**: `True` +- **Description**: Whether to enable the Swagger UI + +### `oas_ui_swagger_version` + +- **Type**: `str` +- **Default**: `"4.1.0"` +- **Description**: Which Swagger version to use + +### `oas_uri_to_config` + +- **Type**: `str` +- **Default**: `"/swagger-config"` +- **Description**: Path to serve the Swagger configurtaion + +### `oas_uri_to_json` + +- **Type**: `str` +- **Default**: `"/openapi.json"` +- **Description**: Path to serve the OpenAPI JSON + +### `oas_uri_to_redoc` + +- **Type**: `str` +- **Default**: `"/redoc"` +- **Description**: Path to Redoc + +### `oas_uri_to_swagger` + +- **Type**: `str` +- **Default**: `"/swagger"` +- **Description**: Path to Swagger + +### `oas_url_prefix` + +- **Type**: `str` +- **Default**: `"/docs"` +- **Description**: URL prefix for the Blueprint that all of the OAS documentation witll attach to + +### `swagger_ui_configuration` + +- **Type**: `Dict[str, Any]` +- **Default**: `{"apisSorter": "alpha", "operationsSorter": "alpha", "docExpansion": "full"}` +- **Description**: The Swagger documentation to be served to the frontend + +### `trace_excluded_headers` + +- **Type**: `Sequence[str]` +- **Default**: `("authorization", "cookie")` +- **Description**: Which headers should be suppresed from responses to `TRACE` requests diff --git a/src/pt/plugins/sanic-ext/convenience.md b/src/pt/plugins/sanic-ext/convenience.md new file mode 100644 index 0000000000..34beba05a1 --- /dev/null +++ b/src/pt/plugins/sanic-ext/convenience.md @@ -0,0 +1,86 @@ +# Convenience + +## Fixed serializer + +---:1 + +Often when developing an application, there will be certain routes that always return the same sort of response. When this is the case, you can predefine the return serializer and on the endpoint, and then all that needs to be returned is the content. + +:--:1 + +```python +from sanic_ext import serializer + +@app.get("/") +@serializer(text) +async def hello_world(request, name: str): + if name.isnumeric(): + return "hello " * int(name) + return f"Hello, {name}" +``` + +:--- + + + +---:1 + +The `serializer` decorator also can add status codes. + +:--:1 +```python +from sanic_ext import serializer + +@app.post("/") +@serializer(text, status=202) +async def create_something(request): + ... +``` +:--- + +## Custom serializer + + +---:1 + +Using the `@serializer` decorator, you can also pass your own custom functions as long as they also return a valid type (`HTTPResonse`). + +:--:1 + +```python +def message(retval, request, action, status): + return json( + { + "request_id": str(request.id), + "action": action, + "message": retval, + }, + status=status, + ) + + +@app.post("/") +@serializer(message) +async def do_action(request, action: str): + return "This is a message" +``` + +:--- + +---:1 + +Now, returning just a string should return a nice serialized output. + +:--:1 + +```python +$ curl localhost:8000/eat_cookies -X POST +{ + "request_id": "ef81c45b-235c-46dd-9dbd-b550f8fa77f9", + "action": "eat_cookies", + "message": "This is a message" +} + +``` + +:--- diff --git a/src/pt/plugins/sanic-ext/getting-started.md b/src/pt/plugins/sanic-ext/getting-started.md new file mode 100644 index 0000000000..e93c617e04 --- /dev/null +++ b/src/pt/plugins/sanic-ext/getting-started.md @@ -0,0 +1,78 @@ +# Getting Started + +Sanic Extensions is an *officially supported* plugin developed, and maintained by the SCO. The primary goal of this project is to add additional features to help Web API and Web application development easier. + +## Features + +- Auto create `HEAD`, `OPTIONS`, and `TRACE` endpoints +- CORS protection +- Predefined, endpoint-specific response serializers +- Argument injection into route handlers +- OpenAPI documentation with Redoc and/or Swagger +- Request query arguments and body input validation + +## Minimum requirements + +- **Python**: 3.8+ +- **Sanic**: 21.9+ + +## Install + +The best method is to just install Sanic Extensions along with Sanic itself: + +```bash +pip install sanic[ext] +``` + +You can of course also just install it by itself. + +```bash +pip install sanic-ext +``` + +## Extend your application + +Out of the box, Sanic Extensions will enable a bunch of features for you. + +::: new NEW in v21.12 +---:1 +To setup Sanic Extensions (v21.12+), you need to do: **nothing**. If it is installed in the environment, it is setup and ready to go. + +This code is the Hello, world app in the [Sanic Getting Started page](../../guide/getting-started.md) _without any changes_. +:--:1 +```python +from sanic import Sanic +from sanic.response import text + +app = Sanic("MyHelloWorldApp") + +@app.get("/") +async def hello_world(request): + return text("Hello, world.") +``` +:--- +::: + +---:1 +**_OLD DEPRECATED SETUP_** + +In v21.9, the easiest way to get started is to instantiate it with `Extend`. + +If you look back at the Hello, world app in the [Sanic Getting Started page](../../guide/getting-started.md), you will see the only additions here are the two highlighted lines. +:--:1 + +```python{3,6} +from sanic import Sanic +from sanic.response import text +from sanic_ext import Extend + +app = Sanic("MyHelloWorldApp") +Extend(app) + +@app.get("/") +async def hello_world(request): + return text("Hello, world.") +``` +:--- + +Regardless of how it is setup, you should now be able to view the OpenAPI documentation and see some of the functionality in action: [http://localhost:8000/docs](http://localhost:8000/docs). diff --git a/src/pt/plugins/sanic-ext/http/README.md b/src/pt/plugins/sanic-ext/http/README.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/plugins/sanic-ext/http/cors.md b/src/pt/plugins/sanic-ext/http/cors.md new file mode 100644 index 0000000000..54b0b86493 --- /dev/null +++ b/src/pt/plugins/sanic-ext/http/cors.md @@ -0,0 +1,86 @@ +# CORS protection + +Cross-Origin Resource Sharing (aka CORS) is a *huge* topic by itself. The documentation here cannot go into enough detail about *what* it is. You are highly encouraged to do some research on your own to understand the security problem presented by it, and the theory behind the solutions. [MDN Web Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) are a great first step. + +In super brief terms, CORS protection is a framework that browsers use to facilitate how and when a web page can access information from another domain. It is extremely relevant to anyone building a single-page application. Often times your frontend might be on a domain like `https://portal.myapp.com`, but it needs to access the backend from `https://api.myapp.com`. + +The implementation here is heavily inspired by [`sanic-cors`](https://github.com/ashleysommer/sanic-cors), which is in turn based upon [`flask-cors`](https://github.com/corydolphin/flask-cors). It is therefore very likely that you can achieve a near drop-in replacement of `sanic-cors` with `sanic-ext`. + +## Basic implementation + +---:1 + +As shown in the example in the [auto-endpoints example](methods.md#options), Sanic Extensions will automatically enable CORS protection without further action. But, it does not offer too much out of the box. + +At a *bare minimum*, it is **highly** recommended that you set `config.CORS_ORIGINS` to the intended origin(s) that will be accessing the application. + +:--:1 +```python +from sanic import Sanic, text +from sanic_ext import Extend + +app = Sanic(__name__) +app.config.CORS_ORIGINS = "http://foobar.com,http://bar.com" +Extend(app) + +@app.get("/") +async def hello_world(request): + return text("Hello, world.") +``` + +``` +$ curl localhost:8000 -X OPTIONS -i +HTTP/1.1 204 No Content +allow: GET,HEAD,OPTIONS +access-control-allow-origin: http://foobar.com +connection: keep-alive +``` +:--- + +## Configuration + +The true power of CORS protection, however, comes into play once you start configuring it. Here is a table of all of the options. + +| Key | Type | Default| Description | +|--|--|--|--| +| `CORS_ALLOW_HEADERS` | `str` or `List[str]` | `"*"` | The list of headers that will appear in `access-control-allow-headers`. | +| `CORS_ALWAYS_SEND` | `bool` | `True` | When `True`, will always set a value for `access-control-allow-origin`. When `False`, will only set it if there is an `Origin` header. | +| `CORS_AUTOMATIC_OPTIONS` | `bool` | `True` | When the incoming preflight request is received, whether to automatically set values for `access-control-allow-headers`, `access-control-max-age`, and `access-control-allow-methods` headers. If `False` these values will only be set on routes that are decorated with the `@cors` decorator. | +| `CORS_EXPOSE_HEADERS` | `str` or `List[str]` | `""` | Specific list of headers to be set in `access-control-expose-headers` header. | +| `CORS_MAX_AGE` | `str`, `int`, `timedelta` | `0` | The maximum number of seconds the preflight response may be cached using the `access-control-max-age` header. A falsey value will cause the header to not be set. | +| `CORS_METHODS` | `str` or `List[str]` | `""` | The HTTP methods that the allowed origins can access, as set on the `access-control-allow-methods` header. | +| `CORS_ORIGINS` | `str`, `List[str]`, `re.Pattern` | `"*"` | The origins that are allowed to access the resource, as set on the `access-control-allow-origin` header. | +| `CORS_SEND_WILDCARD` | `bool` | `False` | If `True`, will send the wildcard `*` origin instead of the `origin` request header. | +| `CORS_SUPPORTS_CREDENTIALS` | `bool` | `False` | Whether to set the `access-control-allow-credentials` header. | +| `CORS_VARY_HEADER` | `bool` | `True` | Whether to add `vary` header, when appropriate. | + +*For the sake of brevity, where the above says `List[str]` any instance of a `list`, `set`, `frozenset`, or `tuple` will be acceptable. Alternatively, if the value is a `str`, it can be a comma delimited list.* + +## Route level overrides + +---:1 + +It may sometimes be necessary to override app-wide settings for a specific route. To allow for this, you can use the `@sanic_ext.cors()` decorator to set different route-specific values. + +The values that can be overridden with this decorator are: + +- `origins` +- `expose_headers` +- `allow_headers` +- `allow_methods` +- `supports_credentials` +- `max_age` + +:--:1 +```python +from sanic_ext import cors + +app.config.CORS_ORIGINS = "https://foo.com" + + +@app.get("/", host="bar.com") +@cors(origins="https://bar.com") +async def hello_world(request): + return text("Hello, world.") +``` +:--- diff --git a/src/pt/plugins/sanic-ext/http/methods.md b/src/pt/plugins/sanic-ext/http/methods.md new file mode 100644 index 0000000000..6efd5711b0 --- /dev/null +++ b/src/pt/plugins/sanic-ext/http/methods.md @@ -0,0 +1,137 @@ +# HTTP Methods + +## Auto-endpoints + +The default behavior is to automatically generate `HEAD` endpoints for all `GET` routes, and `OPTIONS` endpoints for all +routes. Additionally, there is the option to automatically generate `TRACE` endpoints. However, these are not enabled by +default. + +::::tabs + +:::tab HEAD + +- **Configuration**: `AUTO_HEAD` (default `True`) +- **MDN**: [Read more](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/HEAD) + +A `HEAD` request provides the headers and an otherwise identical response to what a `GET` request would provide. +However, it does not actually return the body. + +```python +@app.get("/") +async def hello_world(request): + return text("Hello, world.") +``` + +Given the above route definition, Sanic Extensions will enable `HEAD` responses, as seen here. + +``` +$ curl localhost:8000 --head +HTTP/1.1 200 OK +access-control-allow-origin: * +content-length: 13 +connection: keep-alive +content-type: text/plain; charset=utf-8 +``` + +::: + +:::tab OPTIONS + +- **Configuration**: `AUTO_OPTIONS` (default `True`) +- **MDN**: [Read more](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/OPTIONS) + +`OPTIONS` requests provide the recipient with details about how the client is allowed to communicate with a given +endpoint. + +```python +@app.get("/") +async def hello_world(request): + return text("Hello, world.") +``` + +Given the above route definition, Sanic Extensions will enable `OPTIONS` responses, as seen here. + +It is important to note that we also see `access-control-allow-origins` in this example. This is because +the [CORS protection](cors.md) is enabled by default. + +``` +$ curl localhost:8000 -X OPTIONS -i +HTTP/1.1 204 No Content +allow: GET,HEAD,OPTIONS +access-control-allow-origin: * +connection: keep-alive +``` + +::: tip Even though Sanic Extensions will setup these routes for you automatically, if you decide to manually create +an `@app.options` route, it will *not* be overridden. +::: + +:::tab TRACE + +- **Configuration**: `AUTO_TRACE` (default `False`) +- **MDN**: [Read more](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods/TRACE) + +By default, `TRACE` endpoints will **not** be automatically created. However, Sanic Extensions **will allow** you to +create them if you wanted. This is something that is not allowed in vanilla Sanic. + +```python +@app.route("/", methods=["trace"]) +async def handler(request): + ... +``` + +To enable auto-creation of these endpoints, you must first enable them when extending Sanic. + +```python +from sanic_ext import Extend, Config + +app.extend(config=Config(http_auto_trace=True)) +``` + +Now, assuming you have some endpoints setup, you can trace them as shown here: + +``` +$ curl localhost:8000 -X TRACE +TRACE / HTTP/1.1 +Host: localhost:9999 +User-Agent: curl/7.76.1 +Accept: */* +``` + +::: tip Setting up `AUTO_TRACE` can be super helpful, especially when your application is deployed behind a proxy since +it will help you determine how the proxy is behaving. +::: + +:::: + +## Additional method support + +Vanilla Sanic allows you to build endpoints with the following HTTP methods: + +- [GET](/en/guide/basics/routing.html#get) +- [POST](/en/guide/basics/routing.html#post) +- [PUT](/en/guide/basics/routing.html#put) +- [HEAD](/en/guide/basics/routing.html#head) +- [OPTIONS](/en/guide/basics/routing.html#options) +- [PATCH](/en/guide/basics/routing.html#patch) +- [DELETE](/en/guide/basics/routing.html#delete) + +See [MDN Web Docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods) for more. + +---:1 + +There are, however, two more "standard" HTTP methods: `TRACE` and `CONNECT`. Sanic Extensions will allow you to build +endpoints using these methods, which would otherwise not be allowed. + +It is worth pointing out that this will *NOT* enable convenience methods: `@app.trace` or `@app.connect`. You need to +use `@app.route` as shown in the example here. + +:--:1 + +```python +@app.route("/", methods=["trace", "connect"]) +async def handler(_): + return empty() +``` + +:--- diff --git a/src/pt/plugins/sanic-ext/injection.md b/src/pt/plugins/sanic-ext/injection.md new file mode 100644 index 0000000000..238ff0e842 --- /dev/null +++ b/src/pt/plugins/sanic-ext/injection.md @@ -0,0 +1,274 @@ +# Dependency Injection + +Dependency injection is a method to add arguments to a route handler based upon the defined function signature. Specifically, it looks at the **type annotations** of the arguments in the handler. This can be useful in a number of cases like: + +- Fetching an object based upon request headers (like the current session user) +- Recasting certain objects into a specific type +- Using the request object to prefetch data +- Auto inject services + +The `Extend` instance has two basic methods on it used for dependency injection: a lower level `add_dependency`, and a higher level `dependency`. + +**Lower level**: `app.ext.add_dependency(...)` + +- `type: Type,`: some unique class that will be the type of the oject +- `constructor: Optional[Callable[..., Any]],` (OPTIONAL): a function that will return that type + +**Higher level**: `app.ext.dependency(...)` + +- `obj: Any`: any object that you would like injected +- `name: Optional[str]`: some name that could alternately be used as a reference + +Let's explore some use cases here. + +::: warning +If you used dependency injection prior to v21.12, the lower level API method was called `injection`. It has since been renamed to `add_dependency` and starting in v21.12 `injection` is an alias for `add_dependency`. The `injection` method has been deprecated for removal in v22.6. +::: + +## Basic implementation + +The simplest use case would be simply to recast a value. + +---:1 + +This could be useful if you have a model that you want to generate based upon the matched path parameters. + +:--:1 + +```python +@dataclass +class IceCream: + flavor: str + + def __str__(self) -> str: + return f"{self.flavor.title()} (Yum!)" + + +app.ext.add_dependency(IceCream) + + +@app.get("/") +async def ice_cream(request, flavor: IceCream): + return text(f"You chose: {flavor}") +``` + +``` +$ curl localhost:8000/chocolate +You chose Chocolate (Yum!) +``` +:--- + +---:1 +This works by passing a keyword argument to the constructor of the `type` argument. The previous example is equivalent to this. +:--:1 +```python +flavor = IceCream(flavor="chocolate") +``` +:--- + +## Additional constructors + +---:1 + +Sometimes you may need to also pass a constructor. This could be a function, or perhaps even a classmethod that acts as a constructor. In this example, we are creating an injection that will call `Person.create` first. + +Also important to note on this example, we are actually injecting **two (2)** objects! It of course does not need to be this way, but we will inject objects based upon the function signature. + +:--:1 + +```python +@dataclass +class PersonID: + person_id: int + + +@dataclass +class Person: + person_id: PersonID + name: str + age: int + + @classmethod + async def create(cls, request: Request, person_id: int): + return cls(person_id=PersonID(person_id), name="noname", age=111) + + + +app.ext.add_dependency(Person, Person.create) +app.ext.add_dependency(PersonID) + +@app.get("/person/") +async def person_details( + request: Request, person_id: PersonID, person: Person +): + return text(f"{person_id}\n{person}") +``` + +``` +$ curl localhost:8000/person/123 +PersonID(person_id=123) +Person(person_id=PersonID(person_id=123), name='noname', age=111) +``` +:--- + +When a `constructor` is passed to `ext.add_dependency` (like in this example) that will be called. If not, then the object will be created by calling the `type`. A couple of important things to note about passing a `constructor`: + +1. A positional `request: Request` argument is expected. See the `Person.create` method above as an example. +1. All matched path parameters are injected as keyword arguments. +1. Dependencies can be chained and nested. Notice how in the previous example the `Person` dataclass has a `PersonID`? That means that `PersonID` will be called first, and that value is added to the keyword arguments when calling `Person.create`. + +## Objects from the `Request` + +---:1 + +Sometimes you may want to extract details from the request and preprocess them. You could, for example, cast the request JSON to a Python object, and then add some additional logic based upon DB queries. + +::: warning +If you plan to use this method, you should note that the injection actually happens *before* Sanic has had a chance to read the request body. The headers should already have been consumed. So, if you do want access to the body, you will need to manually consume as seen in this example. + +```python +await request.receive_body() +``` +::: + +This could be used in cases where you otherwise might: + +- use middleware to preprocess and add something to the `request.ctx` +- use decorators to preprocess and inject arguments into the request handler + +In this example, we are using the `Request` object in the `compule_profile` constructor to run a fake DB query to generate and return a `UserProfile` object. +:--:1 + +```python +@dataclass +class User: + name: str + + +@dataclass +class UserProfile: + user: User + age: int = field(default=0) + email: str = field(default="") + + def __json__(self): + return ujson.dumps( + { + "name": self.user.name, + "age": self.age, + "email": self.email, + } + ) + + +async def fake_request_to_db(body): + today = date.today() + email = f'{body["name"]}@something.com'.lower() + difference = today - date.fromisoformat(body["birthday"]) + age = int(difference.days / 365) + return UserProfile( + User(body["name"]), + age=age, + email=email, + ) + + +async def compile_profile(request: Request): + await request.receive_body() + profile = await fake_request_to_db(request.json) + return profile + + +app.ext.add_dependency(UserProfile, compile_profile) + + +@app.patch("/profile") +async def update_profile(request, profile: UserProfile): + return json(profile) +``` + +``` +$ curl localhost:8000/profile -X PATCH -d '{"name": "Alice", "birthday": "2000-01-01"}' +{ + "name":"Alice", + "age":21, + "email":"alice@something.com" +} +``` +:--- + +## Injecting services + +It is a common pattern to create things like database connection pools and store them on the `app.ctx` object. This makes them available throughout your application, which is certainly a convenience. One downside, however, is that you no longer have a typed object to work with. You can use dependency injections to fix this. First we will show the concept using the lower level `add_dependency` like we have been using in the previous examples. But, there is a better way using the higher level `dependency` method. + +---:1 + +### The lower level API using `add_dependency` + +This works very similar to the [last example](#objects-from-the-request) where the goal is the extract something from the `Request` object. In this example, a database object was created on the `app.ctx` instance, and is being returned in the dependency injection constructor. + +:--:1 + +```python +class FakeConnection: + async def execute(self, query: str, **arguments): + return "result" + + +@app.before_server_start +async def setup_db(app, _): + app.ctx.db_conn = FakeConnection() + app.ext.add_dependency(FakeConnection, get_db) + + +def get_db(request: Request): + return request.app.ctx.db_conn + + + + +@app.get("/") +async def handler(request, conn: FakeConnection): + response = await conn.execute("...") + return text(response) +``` +``` +$ curl localhost:8000/ +result +``` + +:--- + +---:1 + +### The higher level API using `dependency` + +Since we have an actual *object* that is available when adding the dependency injection, we can use the higher level `dependency` method. This will make the pattern much easier to write. + +This method should always be used when you want to inject something that exists throughout the lifetime of the application instance and is not request specific. It is very useful for services, third party clients, and connection pools since they are not request specific. + +:--:1 + +```python +class FakeConnection: + async def execute(self, query: str, **arguments): + return "result" + + +@app.before_server_start +async def setup_db(app, _): + db_conn = FakeConnection() + app.ext.dependency(db_conn) + + +@app.get("/") +async def handler(request, conn: FakeConnection): + response = await conn.execute("...") + return text(response) +``` +``` +$ curl localhost:8000/ +result +``` + +:--- diff --git a/src/pt/plugins/sanic-ext/openapi.md b/src/pt/plugins/sanic-ext/openapi.md new file mode 100644 index 0000000000..7e3314aa8c --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi.md @@ -0,0 +1,7 @@ +# Openapi + +- Adding documentation with decorators +- Documenting CBV +- Using autodoc +- Rendering docs with redoc/swagger +- Validation diff --git a/src/pt/plugins/sanic-ext/openapi/README.md b/src/pt/plugins/sanic-ext/openapi/README.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/plugins/sanic-ext/openapi/advanced.md b/src/pt/plugins/sanic-ext/openapi/advanced.md new file mode 100644 index 0000000000..b4b8c3e609 --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi/advanced.md @@ -0,0 +1,10 @@ +# Advanced + +_Documentation coming October 2021_ + +## CBV + +## Blueprints + + +## Components diff --git a/src/pt/plugins/sanic-ext/openapi/autodoc.md b/src/pt/plugins/sanic-ext/openapi/autodoc.md new file mode 100644 index 0000000000..c540cf909c --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi/autodoc.md @@ -0,0 +1,131 @@ +# Auto-documentation + +To make documenting endpoints easier, Sanic Extensions will use a function's docstring to populate your documentation. + +## Summary and description + +---:1 +A function's docstring will be used to create the summary and description. As you can see from this example here, the docstring has been parsed to use the first line as the summary, and the remainder of the string as the description. +:--:1 +```python +@app.get("/foo") +async def handler(request, something: str): + """This is a simple foo handler + + It is helpful to know that you could also use **markdown** inside your + docstrings. + + - one + - two + - three""" + return text(">>>") +``` +```json +"paths": { + "/foo": { + "get": { + "summary": "This is a simple foo handler", + "description": "It is helpful to know that you could also use **markdown** inside your
docstrings.

- one
- two
- three", + "responses": { + "default": { + "description": "OK" + } + }, + "operationId": "get_handler" + } + } +} +``` +:--- + +## Operation level YAML + +---:1 +You can expand upon this by adding valid OpenAPI YAML to the docstring. Simply add a line that contains `openapi:`, followed by your YAML. + +The `---` shown in the example is *not* necessary. It is just there to help visually identify the YAML as a distinct section of the docstring. +:--:1 +```python +@app.get("/foo") +async def handler(request, something: str): + """This is a simple foo handler + + Now we will add some more details + + openapi: + --- + operationId: fooDots + tags: + - one + - two + parameters: + - name: limit + in: query + description: How many items to return at one time (max 100) + required: false + schema: + type: integer + format: int32 + responses: + '200': + description: Just some dots + """ + return text("...") +``` +```json +"paths": { + "/foo": { + "get": { + "operationId": "fooDots", + "summary": "This is a simple foo handler", + "description": "Now we will add some more details", + "tags": [ + "one", + "two" + ], + "parameters": [ + { + "name": "limit", + "in": "query", + "description": "How many items to return at one time (max 100)", + "required": false, + "schema": { + "type": "integer", + "format": "int32" + } + } + ], + "responses": { + "200": { + "description": "Just some dots" + } + } + } + } +} +``` + +:--- + +::: tip +When both YAML documentation and decorators are used, it is the content from the decorators that will take priority when generating the documentation. +::: + +## Excluding docstrings + +---:1 +Sometimes a function may contain a docstring that is not meant to be consumed inside the documentation. + +**Option 1**: Globally turn off auto-documentation `app.config.OAS_AUTODOC = False` + +**Option 2**: Disable it for the single handler with the `@openapi.no_autodoc` decorator +:--:1 +```python +@app.get("/foo") +@openapi.no_autodoc +async def handler(request, something: str): + """This is a docstring about internal info only. Do not parse it. + """ + return text("...") +``` +:--- diff --git a/src/pt/plugins/sanic-ext/openapi/basic.md b/src/pt/plugins/sanic-ext/openapi/basic.md new file mode 100644 index 0000000000..5edc501dae --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi/basic.md @@ -0,0 +1,70 @@ +# Basics + +::: tip Side note +The OpenAPI implementation in Sanic Extensions is based upon the OAS3 implementation from [`sanic-openapi`](https://github.com/sanic-org/sanic-openapi). In fact, Sanic Extensions is in a large way the successor to that project, which entered maintenance mode upon the release of Sanic Extensions. If you were previously using OAS3 with `sanic-openapi` you should have an easy path to upgrading to Sanic Extensions. Unfortunately, this project does *NOT* support the OAS2 specification. +::: + +---:1 + +Out of the box, Sanic Extensions provides automatically generated API documentation using the [v3.0 OpenAPI specification](https://swagger.io/specification/). There is nothing special that you need to do + +:--:1 + +```python +from sanic import Sanic + +app = Sanic("MyApp") + +# Add all of your views +``` + +:--- + +After doing this, you will now have beautiful documentation already generated for you based upon your existing application: + +- [http://localhost:8000/docs](http://localhost:8000/docs) +- [http://localhost:8000/docs/redoc](http://localhost:8000/docs/redoc) +- [http://localhost:8000/docs/swagger](http://localhost:8000/docs/swagger) + +Checkout the [section on configuration](../configuration.md) to learn about changing the routes for the docs. You can also turn off one of the two UIs, and customize which UI will be available on the `/docs` route. + +---:1 + +Using [Redoc](https://github.com/Redocly/redoc) + +![Redoc](~@assets/images/sanic-ext-redoc.png) + + +:--:1 + +or [Swagger UI](https://github.com/swagger-api/swagger-ui) + +![Swagger UI](~@assets/images/sanic-ext-swagger.png) + + +:--- + +## Changing specification metadata + +---:1 +If you want to change any of the metada, you should use the `describe` method. + +In this example `dedent` is being used with the `description` argument to make multi-line strings a little cleaner. This is not necessary, you can pass any string value here. +:--:1 +```python +from textwrap import dedent + +app.ext.openapi.describe( + "Testing API", + version="1.2.3", + description=dedent( + """ + # Info + This is a description. It is a good place to add some _extra_ doccumentation. + + **MARKDOWN** is supported. + """ + ), +) +``` +:--- diff --git a/src/pt/plugins/sanic-ext/openapi/decorators.md b/src/pt/plugins/sanic-ext/openapi/decorators.md new file mode 100644 index 0000000000..81c2402033 --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi/decorators.md @@ -0,0 +1,456 @@ +# Decorators + +The primary mechanism for adding content to your schema is by decorating your endpoints. If you have +used `sanic-openapi` in the past, this should be familiar to you. The decorators and their arguments match closely +the [OAS v3.0 specification](https://swagger.io/specification/). + +---:1 + +All of the examples show will wrap around a route definition. When you are creating these, you should make sure that +your Sanic route decorator (`@app.route`, `@app.get`, etc) is the outermost decorator. That is to say that you should +put that first and then one or more of the below decorators after. + +:--:1 + +```python +from sanic_ext import openapi + + +@app.get("/path/to/") +@openapi.summary("This is a summary") +@openapi.description("This is a description") +async def handler(request, somethind: str): + ... +``` + +:--- + +---:1 + +You will also see a lot of the below examples reference a model object. For the sake of simplicity, the examples will +use `UserProfile` that will look like this. The point is that it can be any well-typed class. You could easily imagine +this being a `dataclass` or some other kind of model object. + +:--:1 + +```python +class UserProfile: + name: str + age: int + email: str +``` + +:--- + +## Definition decorator + +### `@openapi.definition` + +The `@openapi.definition` decorator allows you to define all parts of an operations on a path at once. It is an omnibums +decorator in that it has the same capabilities to create operation definitions as the rest of the decorators. Using +multiple field-specific decorators or a single decorator is a style choice for you the developer. + +The fields are purposely permissive in accepting multiple types to make it easiest for you to define your operation. + +**Arguments** + +| Field | Type | +| ------------- | --------------------------------------------------------------------------| +| `body` | **dict, RequestBody, *YourModel*** | +| `deprecated` | **bool** | +| `description` | **str** | +| `document` | **str, ExternalDocumentation** | +| `exclude` | **bool** | +| `operation` | **str** | +| `parameter` | **str, dict, Parameter, [str], [dict], [Parameter]** | +| `response` | **dict, Response, *YourModel*, [dict], [Response]** | +| `summary` | **str** | +| `tag` | **str, Tag, [str], [Tag]** | +| `secured` | **Dict[str, Any]** | + +**Examples** + +---:1 + +```python +@openapi.definition( + body=RequestBody(UserProfile, required=True), + summary="User profile update", + tag="one", + response=[Success, Response(Failure, status=400)], +) +``` + +:--:1 + +:--- + +*See below examples for more examples. Any of the values for the below decorators can be used in the corresponding +keyword argument.* + +## Field-specific decorators + +All the following decorators are based on `@openapi` + +::::tabs + +:::tab body + +**Arguments** + +| Field | Type | +| ----------- | ---------------------------------- | +| **content** | ***YourModel*, dict, RequestBody** | + +**Examples** + +---:1 + +```python +@openapi.body(UserProfile) +``` + +```python +@openapi.body({"application/json": UserProfile}) +``` + +```python +@openapi.body(RequestBody({"application/json": UserProfile})) +``` + +:--:1 + +```python +@openapi.body({"content": UserProfile}) +``` + +```python +@openapi.body(RequestBody(UserProfile)) +``` + +:--- + +::: + +:::tab deprecated + +**Arguments** + +*None* + +**Examples** + +---:1 + +```python +@openapi.deprecated() +``` + +:--:1 + +```python +@openapi.deprecated +``` + +:--- + +::: + +:::tab description + +**Arguments** + +| Field | Type | +| ------ | ------- | +| `text` | **str** | + +**Examples** + +---:1 + +```python +@openapi.description( + """This is a **description**. + +## You can use `markdown` + +- And +- make +- lists. +""" +) +``` + +:--:1 + +:--- + +::: + +:::tab document + +**Arguments** + +| Field | Type | +| ------------- | ------- | +| `url` | **str** | +| `description` | **str** | + +**Examples** + +---:1 + +```python +@openapi.document("http://example.com/docs") +``` + +:--:1 + +```python +@openapi.document(ExternalDocumentation("http://example.com/more")) +``` + +:--- + +::: + +:::tab exclude + +Can be used on route definitions like all of the other decorators, or can be called on a Blueprint + +**Arguments** + +| Field | Type | Default | +| ------ | ------------- | -------- | +| `flag` | **bool** | **True** | +| `bp` | **Blueprint** | | + +**Examples** + +---:1 + +```python +@openapi.exclude() +``` + +:--:1 + +```python +openapi.exclude(bp=some_blueprint) +``` + +:--- + +::: + +:::tab operation + +Sets the operation ID. + +**Arguments** + +| Field | Type | +| ------ | ------- | +| `name` | **str** | + +**Examples** + +---:1 + +```python +@openapi.operation("doNothing") +``` + +:--:1 + +:--- + +::: + +:::tab parameter + +**Arguments** + +| Field | Type | Default | +| ---------- | ----------------------------------------- | ----------- | +| `name` | **str** | | +| `schema` | ***type*** | **str** | +| `location` | **"query", "header", "path" or "cookie"** | **"query"** | + +**Examples** + +---:1 + +```python +@openapi.parameter("thing") +``` + +```python +@openapi.parameter(parameter=Parameter("foobar", deprecated=True)) +``` + +:--:1 + +```python +@openapi.parameter("Authorization", str, "header") +``` + +```python +@openapi.parameter("thing", required=True, allowEmptyValue=False) +``` + +:--- + +::: + +:::tab response + +**Arguments** + +If using a `Response` object, you should not pass any other arguments. + +| Field | Type | +| ------------- | ----------------------------- | +| `status` | **int** | +| `content` | ***type*, *YourModel*, dict** | +| `description` | **str** | +| `response` | **Response** | + +**Examples** + +---:1 + +```python +@openapi.response(200, str, "This is endpoint returns a string") +``` + +```python +@openapi.response(200, {"text/plain": str}, "...") +``` + +```python +@openapi.response(response=Response(UserProfile, description="...")) +``` + +```python +@openapi.response( + response=Response( + { + "application/json": UserProfile, + }, + description="...", + status=201, + ) +) +``` + +:--:1 + +```python +@openapi.response(200, UserProfile, "...") +``` + +```python +@openapi.response( + 200, + { + "application/json": UserProfile, + }, + "Description...", +) +``` + +:--- + +::: + +:::tab summary + +**Arguments** + +| Field | Type | +| ------ | ------- | +| `text` | **str** | + +**Examples** + +---:1 + +```python +@openapi.summary("This is an endpoint") +``` + +:--:1 + +:--- + +::: + +:::tab tag + +**Arguments** + +| Field | Type | +| ------- | ------------ | +| `*args` | **str, Tag** | + +**Examples** + +---:1 + +```python +@openapi.tag("foo") +``` + +:--:1 + +```python +@openapi.tag("foo", Tag("bar")) +``` + +:--- + +::: + +:::tab secured + +**Arguments** + +| Field | Type | +| ----------------- | ----------------------- | +| `*args, **kwargs` | **str, Dict[str, Any]** | + +**Examples** + +---:1 +```python +@openapi.secured() +``` +:--:1 +:--- + +---:1 +```python +@openapi.secured("foo") +``` +:--:1 +```python +@openapi.secured("token1", "token2") +``` +:--- + +---:1 +```python +@openapi.secured({"my_api_key": []}) +``` +:--:1 +```python +@openapi.secured(my_api_key=[]) +``` +:--- + +Do not forget to use `add_security_scheme`. See [security](./security.md) for more details. + +::: + +:::: diff --git a/src/pt/plugins/sanic-ext/openapi/security.md b/src/pt/plugins/sanic-ext/openapi/security.md new file mode 100644 index 0000000000..41723de611 --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi/security.md @@ -0,0 +1,91 @@ +# Security Schemes + +To document authentication schemes, there are two steps. + +_Security is only available starting in v21.12.2_ + +## Document the scheme + +---:1 +The first thing that you need to do is define one or more security schemes. The basic pattern will be to define it as: + +```python +add_security_scheme("", "") +``` + +The `type` should correspond to one of the allowed security schemes: `"apiKey"`, `"http"`, `"oauth2"`, `"openIdConnect"`. You can then pass appropriate keyword arguments as allowed by the specification. + +You should consult the [OpenAPI Specification](https://swagger.io/specification/) for details on what values are appropriate. +:--:1 +```python +app.ext.openapi.add_security_scheme("api_key", "apiKey") +app.ext.openapi.add_security_scheme( + "token", + "http", + scheme="bearer", + bearer_format="JWT", +) +app.ext.openapi.add_security_scheme("token2", "http") +app.ext.openapi.add_security_scheme( + "oldschool", + "http", + scheme="basic", +) +app.ext.openapi.add_security_scheme( + "oa2", + "oauth2", + flows={ + "implicit": { + "authorizationUrl": "http://example.com/auth", + "scopes": { + "on:two": "something", + "three:four": "something else", + "threefour": "something else...", + }, + } + }, +) +``` +:--- + +## Document the endpoints + +---:1 +There are two options, document _all_ endpoints. + +:--:1 +```python +app.ext.openapi.secured() +app.ext.openapi.secured("token") +``` +:--- + +---:1 +Or, document only specific routes. +:--:1 +```python +@app.route("/one") +async def handler1(request): + """ + openapi: + --- + security: + - foo: [] + """ + + +@app.route("/two") +@openapi.secured("foo") +@openapi.secured({"bar": []}) +@openapi.secured(baz=[]) +async def handler2(request): + ... + + +@app.route("/three") +@openapi.definition(secured="foo") +@openapi.definition(secured={"bar": []}) +async def handler3(request): + ... +``` +:--- diff --git a/src/pt/plugins/sanic-ext/openapi/ui.md b/src/pt/plugins/sanic-ext/openapi/ui.md new file mode 100644 index 0000000000..8651d52538 --- /dev/null +++ b/src/pt/plugins/sanic-ext/openapi/ui.md @@ -0,0 +1,26 @@ +# UI + +Sanic Extensions comes with both Redoc and Swagger interfaces. You have a choice to use one, or both of them. Out of the box, the following endpoints are setup for you, with the bare `/docs` displaying Redoc. + +- `/docs` +- `/docs/openapi.json` +- `/docs/redoc` +- `/docs/swagger` +- `/docs/openapi-config` + +## Config options + +| **Key** | **Type** | **Default** | **Desctiption** | +| -------------------------- | --------------- | ------------------- | ------------------------------------------------------------ | +| `OAS_IGNORE_HEAD` | `bool` | `True` | Whether to display `HEAD` endpoints. | +| `OAS_IGNORE_OPTIONS` | `bool` | `True` | Whether to display `OPTIONS` endpoints. | +| `OAS_PATH_TO_REDOC_HTML` | `Optional[str]` | `None` | Path to HTML to override the default Redoc HTML | +| `OAS_PATH_TO_SWAGGER_HTML` | `Optional[str]` | `None` | Path to HTML to override the default Swagger HTML | +| `OAS_UI_DEFAULT` | `Optional[str]` | `"redoc"` | Can be set to `redoc` or `swagger`. Controls which UI to display on the base route. If set to `None`, then the base route will not be setup. | +| `OAS_UI_REDOC` | `bool` | `True` | Whether to enable Redoc UI. | +| `OAS_UI_SWAGGER` | `bool` | `True` | Whether to enable Swagger UI. | +| `OAS_URI_TO_CONFIG` | `str` | `"/openapi-config"` | URI path to the OpenAPI config used by Swagger | +| `OAS_URI_TO_JSON` | `str` | `"/openapi.json"` | URI path to the JSON document. | +| `OAS_URI_TO_REDOC` | `str` | `"/redoc"` | URI path to Redoc. | +| `OAS_URI_TO_SWAGGER` | `str` | `"/swagger"` | URI path to Swagger. | +| `OAS_URL_PREFIX` | `str` | `"/docs"` | URL prefix to use for the Blueprint for OpenAPI docs. | diff --git a/src/pt/plugins/sanic-ext/validation.md b/src/pt/plugins/sanic-ext/validation.md new file mode 100644 index 0000000000..bec7ffda44 --- /dev/null +++ b/src/pt/plugins/sanic-ext/validation.md @@ -0,0 +1,127 @@ +# Validation + +One of the most commonly implemented features of a web application is user-input validation. For obvious reasons, this is not only a security issue, but also just plain good practice. You want to make sure your data conforms to expectations, and throw a `400` response when it does not. + +## Implementation + +### Validation with Dataclasses + +With the introduction of [Data Classes](https://docs.python.org/3/library/dataclasses.html), Python made it super simple to create objects that meet a defined schema. However, the standard library only supports type checking validation, **not** runtime validation. Sanic Extensions adds the ability to do runtime validations on incoming requests using `dataclasses`. + +---:1 + +First, define a model. + +:--:1 + +```python +@dataclass +class SearchParams: + q: str +``` + +:--- + +---:1 + +Then, attach it to your route + +:--:1 + +```python +from sanic_ext import validate + +@app.route("/search") +@validate(query=SearchParams) +async def handler(request, query: SearchParams): + return json(asdict(query)) +``` + +:--- + +---:1 + +You should now have validation on the incoming request. + +:--:1 + +``` +$ curl localhost:8000/search +⚠️ 400 — Bad Request +==================== +Invalid request body: SearchParams. Error: missing a required argument: 'q' +``` +``` +$ curl localhost:8000/search\?q=python +{"q":"python"} +``` + +:--- + +### Validation with Pydantic + +::: warning +Currently, only JSON body validation supports Pydantic models. +::: + +You can use Pydantic models also. + +---:1 + +First, define a model. + +:--:1 + +```python +class Person(BaseModel): + name: str + age: int +``` + +:--- + +---:1 + +Then, attach it to your route + +:--:1 + +```python +from sanic_ext import validate + +@app.post("/person") +@validate(json=Person) +async def handler(request, body: Person): + return json(body.dict()) +``` +:--- + +---:1 + +You should now have validation on the incoming request. + +:--:1 + +``` +$ curl localhost:8000/person -d '{"name": "Alice", "age": 21}' -X POST +{"name":"Alice","age":21} +``` + +:--- + +## What can be validated? + +The `validate` decorator can be used to validate incoming user date from three places: JSON body data (`request.json`), form body data (`request.form`), and query parameters (`request.args`). + +---:1 +As you might expect, you can attach your model using the keyword arguments of the decorator. + +:--:1 +```python +@validate( + json=ModelA, + query=ModelB, + form=ModelC, +) +``` +:--- diff --git a/src/pt/plugins/sanic-testing/README.md b/src/pt/plugins/sanic-testing/README.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/src/pt/plugins/sanic-testing/clients.md b/src/pt/plugins/sanic-testing/clients.md new file mode 100644 index 0000000000..a0fad19f7e --- /dev/null +++ b/src/pt/plugins/sanic-testing/clients.md @@ -0,0 +1,112 @@ +# Test Clients + +There are three different test clients available to you, each of them presents different capabilities. + +## Regular sync client: `SanicTestClient` + +The `SanicTestClient` runs an actual version of the Sanic Server on your local network to run its tests. Each time it calls an endpoint it will spin up a version of the application and bind it to a socket on the host OS. Then, it will use `httpx` to make calls directly to that application. + +This is the typical way that Sanic applications are tested. + +---:1 +Once installing Sanic Testing, the regular `SanicTestClient` can be used without further setup. This is because Sanic does the leg work for you under the hood. +:--: +```python +app.test_client.get("/path/to/endpoint") +``` +:--- + +---:1 +However, you may find it desirable to instantiate the client yourself. +:--: +```python +from sanic_testing.testing import SanicTestClient + +test_client = SanicTestClient(app) +test_client.get("/path/to/endpoint") +``` +:--- + +---:1 +A third option for starting the test client is to use the `TestManager`. This is a convenience object that sets up both the `SanicTestClient` and the `SanicASGITestClient`. + +:--: +```python +from sanic_testing import TestManager + +mgr = TestManager(app) +app.test_client.get("/path/to/endpoint") +# or +mgr.test_client.get("/path/to/endpoint") +``` +:--- + +You can make a request by using one of the following methods + +- `SanicTestClient.get` +- `SanicTestClient.post` +- `SanicTestClient.put` +- `SanicTestClient.patch` +- `SanicTestClient.delete` +- `SanicTestClient.options` +- `SanicTestClient.head` +- `SanicTestClient.websocket` +- `SanicTestClient.request` + +You can use these methods *almost* identically as you would when using `httpx`. Any argument that you would pass to `httpx` will be accepted, **with one caveat**: If you are using `test_client.request` and want to manually specify the HTTP method, you should use: `http_method`: + +```python +test_client.request("/path/to/endpoint", http_method="get") +``` + +## ASGI async client: `SanicASGITestClient` + +Unlike the `SanicTestClient` that spins up a server on every request, the `SanicASGITestClient` does not. Instead it makes use of the `httpx` library to execute Sanic as an ASGI application to reach inside and execute the route handlers. + +---:1 +This test client provides all of the same methods and generally works as the `SanicTestClient`. The only difference is that you will need to add an `await` to each call: +:--: +```python +await app.test_client.get("/path/to/endpoint") +``` +:--- + +The `SanicASGITestClient` can be used in the exact same three ways as the `SanicTestClient`. + +::: tip Note +The `SanicASGITestClient` does not need to only be used with ASGI applications. The same way that the `SanicTestClient` does not need to only test sync endpoints. Both of these clients are capable of testing *any* Sanic application. +::: + +## Persistent service client: `ReusableClient` + +This client works under a similar premise as the `SanicTestClient` in that it stands up an instance of your application and makes real HTTP requests to it. However, unlike the `SanicTestClient`, when using the `ReusableClient` you control the lifecycle of the application. + +That means that every request **does not** start a new web server. Instead you will start the server and stop it as needed and can make multiple requests to the same running instance. + +---:1 +Unlike the other two clients, you **must** instantiate this client for use: +:--: +```python +from sanic_testing.reusable import ReusableClient + +client = ReusableClient(app) +``` +:--- + + +---:1 +Once created, you will use the client inside of a context manager. Once outside of the scope of the manager, the server will shutdown. +:--: +```python +from sanic_testing.reusable import ReusableClient + +def test_multiple_endpoints_on_same_server(app): + client = ReusableClient(app) + with client: + _, response = client.get("/path/to/1") + assert response.status == 200 + + _, response = client.get("/path/to/2") + assert response.status == 200 +``` +:--- diff --git a/src/pt/plugins/sanic-testing/getting-started.md b/src/pt/plugins/sanic-testing/getting-started.md new file mode 100644 index 0000000000..e9f2c8c4e7 --- /dev/null +++ b/src/pt/plugins/sanic-testing/getting-started.md @@ -0,0 +1,83 @@ +# Getting Started + +Sanic Testing is the *official* testing client for Sanic. Its primary use is to power the tests of the Sanic project itself. However, it is also meant as an easy-to-use client for getting your API tests up and running quickly. + +## Minimum requirements + +- **Python**: 3.7+ +- **Sanic**: 21.3+ + +Versions of Sanic older than 21.3 have this module integrated into Sanic itself as `sanic.testing`. + +## Install + +Sanic Testing can be installed from PyPI: + +``` +pip install sanic-testing +``` + +## Basic Usage + +As long as the `sanic-testing` package is in the environment, there is nothing you need to do to start using it. + + +### Writing a sync test + +In order to use the test client, you just need to access the property `test_client` on your application instance: + +```python +import pytest +from sanic import Sanic, response + + +@pytest.fixture +def app(): + sanic_app = Sanic("TestSanic") + + @sanic_app.get("/") + def basic(request): + return response.text("foo") + + return sanic_app + +def test_basic_test_client(app): + request, response = app.test_client.get("/") + + assert request.method.lower() == "get" + assert response.body == b"foo" + assert response.status == 200 +``` + +### Writing an async test + +In order to use the async test client in `pytest`, you should install the `pytest-asyncio` plugin. + +``` +pip install pytest-asyncio +``` + +You can then create an async test and use the ASGI client: + +```python +import pytest +from sanic import Sanic, response + +@pytest.fixture +def app(): + sanic_app = Sanic(__name__) + + @sanic_app.get("/") + def basic(request): + return response.text("foo") + + return sanic_app + +@pytest.mark.asyncio +async def test_basic_asgi_client(app): + request, response = await app.asgi_client.get("/") + + assert request.method.lower() == "get" + assert response.body == b"foo" + assert response.status == 200 +```