Compare commits
	
		
			338 Commits
		
	
	
		
			v21.3.1
			...
			motd_addre
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 0489dd9aaf | ||
|   | 47215d4635 | ||
|   | 38ff9069f3 | ||
|   | 4dde4572ec | ||
|   | 31d14704cb | ||
|   | 6a89f4b2fe | ||
|   | 16256522f6 | ||
|   | 205795d1e8 | ||
|   | 9cbe1fb8ad | ||
|   | 31d7ba8f8c | ||
|   | dc3c4d1393 | ||
|   | 929d270569 | ||
|   | 93714df051 | ||
|   | 6e61eab872 | ||
|   | 6848ff24d8 | ||
|   | 666371bb92 | ||
|   | 4a2b82e42e | ||
|   | 5dd1623192 | ||
|   | 976da69e79 | ||
|   | 11a0b15194 | ||
|   | c21999a248 | ||
|   | c17230ef94 | ||
|   | 049983cb70 | ||
|   | e374409567 | ||
|   | 4068a0d83d | ||
|   | 70da5e9879 | ||
|   | f48506d620 | ||
|   | f2cc83c1ba | ||
|   | 273825dab6 | ||
|   | 9a7dafd531 | ||
|   | 50117d174c | ||
|   | af67801062 | ||
|   | 6eaab2a7e5 | ||
|   | d680af3709 | ||
|   | a8c2d77c91 | ||
|   | 6e1c787e5d | ||
|   | 932088e37e | ||
|   | 1a63b9bec0 | ||
|   | 61aa16f6ac | ||
|   | 71cd53b64e | ||
|   | 89188f5fc6 | ||
|   | a245ab3773 | ||
|   | ac1f56118a | ||
|   | 53820bc241 | ||
|   | 009954003c | ||
|   | 8f265b8169 | ||
|   | 5ee36fd933 | ||
|   | 08a81c81be | ||
|   | 5a0ed75171 | ||
|   | d62a92fac9 | ||
|   | 88c918e72f | ||
|   | c8aab8fb3d | ||
|   | ecacfd396b | ||
|   | 3c361e9852 | ||
|   | a5d7d03413 | ||
|   | 259e458847 | ||
|   | cb49c2b26d | ||
|   | dfc0704831 | ||
|   | d238995f1b | ||
|   | 6f5303e080 | ||
|   | 5e7f6998bd | ||
|   | c7a71cd00c | ||
|   | 9cb9e88678 | ||
|   | 30c53b6857 | ||
|   | 4ad8168bb0 | ||
|   | 28f5b3c301 | ||
|   | c573019e7f | ||
|   | 029f564032 | ||
|   | 2abe66b670 | ||
|   | 911485d52e | ||
|   | 4744a89c33 | ||
|   | f7040ccec8 | ||
|   | 518152d97e | ||
|   | 0e44e9cacb | ||
|   | bfb54b0969 | ||
|   | 154863d6c6 | ||
|   | a3ff0c13b7 | ||
|   | 95ee518aec | ||
|   | 71d3d87bcc | ||
|   | b276b91c21 | ||
|   | 064168f3c8 | ||
|   | db39e127bf | ||
|   | 13e9ab7ba9 | ||
|   | 92e7463721 | ||
|   | 8e720365c2 | ||
|   | d4041161c7 | ||
|   | f32437bf13 | ||
|   | 0909e94527 | ||
|   | aef2673c38 | ||
|   | 4c14910d5b | ||
|   | beae35f921 | ||
|   | ad4e526c77 | ||
|   | 4422d0c34d | ||
|   | ad9183d21d | ||
|   | d70636ba2e | ||
|   | da23f85675 | ||
|   | 3f4663b9f8 | ||
|   | 65d7447cf6 | ||
|   | 5369291c27 | ||
|   | 1c4925edf7 | ||
|   | 6b9edfd05c | ||
|   | 97f33f42df | ||
|   | 15a588a90c | ||
|   | 82421e7efc | ||
|   | f891995b48 | ||
|   | 5052321801 | ||
|   | 23ce4eaaa4 | ||
|   | 23a430c4ad | ||
|   | ec158ffa69 | ||
|   | 6e32270036 | ||
|   | 43ba381e7b | ||
|   | 16503319e5 | ||
|   | 389363ab71 | ||
|   | 7f894c45b3 | ||
|   | 4726cf1910 | ||
|   | d352a4155e | ||
|   | e5010286b4 | ||
|   | 358498db96 | ||
|   | e4999401ab | ||
|   | c8df0aa2cb | ||
|   | 5fb207176b | ||
|   | a12b560478 | ||
|   | 753ee992a6 | ||
|   | 09089b1bd3 | ||
|   | 7ddbe5e844 | ||
|   | ab5a7038af | ||
|   | 4f3c780dc3 | ||
|   | 71f7765a4c | ||
|   | 0392d1dcfc | ||
|   | 7827b1b41d | ||
|   | 8e9342e188 | ||
|   | 2f6f2bfa76 | ||
|   | dee09d7fff | ||
|   | 9cf38a0a83 | ||
|   | 3def3d3569 | ||
|   | e100a14fd4 | ||
|   | 2fa28f1711 | ||
|   | 9d415e4ec6 | ||
|   | 312ab298fd | ||
|   | 2fc21ad576 | ||
|   | 8f6c87c3d6 | ||
|   | 4429e76532 | ||
|   | e4be70bae8 | ||
|   | 13d5a44278 | ||
|   | aba333bfb6 | ||
|   | b59da498cc | ||
|   | 70382f21ba | ||
|   | 0e1bf89fad | ||
|   | 6c48c8b3ba | ||
|   | d1c5e8003b | ||
|   | ce926a34f2 | ||
|   | a744041e38 | ||
|   | 2f90a85df1 | ||
|   | a411bc06e3 | ||
|   | 1668e1532f | ||
|   | b87982769f | ||
|   | 65b53a5f3f | ||
|   | 49789b7841 | ||
|   | c249004c30 | ||
|   | 4ee2e57ec8 | ||
|   | 86ae5f981c | ||
|   | 2bfa65e0de | ||
|   | 293278bb08 | ||
|   | 5d683c6ea4 | ||
|   | 78b6723149 | ||
|   | 3a6cc7389c | ||
|   | cc97287f8e | ||
|   | 00218aa9f2 | ||
|   | 874718db94 | ||
|   | bb4474897f | ||
|   | 0cb342aef4 | ||
|   | 030987480c | ||
|   | f6fdc80b40 | ||
|   | 361c242473 | ||
|   | 32962d1e1c | ||
|   | 6e0a6871b5 | ||
|   | 0030425c8c | ||
|   | c9dbc8ed26 | ||
|   | 44b108b564 | ||
|   | 2a8e91052f | ||
|   | 0c9df02e66 | ||
|   | 7523e87937 | ||
|   | d4fb44e986 | ||
|   | 68b654d981 | ||
|   | 88bc6d8966 | ||
|   | ac388d644b | ||
|   | bb517ddcca | ||
|   | b8d991420b | ||
|   | 4a416e177a | ||
|   | 8dfa49b648 | ||
|   | 8b0eaa097c | ||
|   | 101151b419 | ||
|   | 4669036f45 | ||
|   | 9bf9067c99 | ||
|   | a7bc8b56ba | ||
|   | 371985d129 | ||
|   | 3eae00898d | ||
|   | dc3ccba527 | ||
|   | b91ffed010 | ||
|   | 8c07e388cd | ||
|   | 98ce4bdeb2 | ||
|   | 4659069350 | ||
|   | 080d41627a | ||
|   | d799c5f03c | ||
|   | abe062b371 | ||
|   | b5a00ac1ca | ||
|   | f282865362 | ||
|   | 377c2ada38 | ||
|   | 264453459e | ||
|   | 3d383d7b97 | ||
|   | c0cc26021b | ||
|   | 96c027bad5 | ||
|   | b2a1bc69f5 | ||
|   | 426742b3e2 | ||
|   | ab35121864 | ||
|   | cf3c205fa5 | ||
|   | 19f6544923 | ||
|   | f641830d26 | ||
|   | a8d55e180c | ||
|   | 55c36e0240 | ||
|   | 2c03eee329 | ||
|   | 65e28b8c22 | ||
|   | dfd33dd63d | ||
|   | 722a6db8d9 | ||
|   | 9c576c74db | ||
|   | 523db190a7 | ||
|   | 95631b9686 | ||
|   | 0860bfe1f1 | ||
|   | 85e7b712b9 | ||
|   | b731a6b48c | ||
|   | cde02b5936 | ||
|   | abeb8d0bc0 | ||
|   | 9a9f72ad64 | ||
|   | 392a497366 | ||
|   | 36e6a6c506 | ||
|   | a361b345ad | ||
|   | f5bd6e3b2f | ||
|   | 6c7df68c7c | ||
|   | 5b82884f8b | ||
|   | f0f81ec458 | ||
|   | 71cc30e5cd | ||
|   | 645310cff6 | ||
|   | 2f30b5748a | ||
|   | 5e1ef96934 | ||
|   | 57e98b62b3 | ||
|   | 3262878ebd | ||
|   | 5e12edbc38 | ||
|   | 50a606adee | ||
|   | f995612073 | ||
|   | bc08383acd | ||
|   | b83a1a184c | ||
|   | 59dd6814f8 | ||
|   | f7abf3db1b | ||
|   | cf1d2148ac | ||
|   | b5f2bd9b0e | ||
|   | ba2670e99c | ||
|   | 6ffc4d9756 | ||
|   | 595d2c76ac | ||
|   | d9796e9b1e | ||
|   | 404c5f9f9e | ||
|   | a937e08ef0 | ||
|   | ef4f058a6c | ||
|   | 69c5dde9bf | ||
|   | 945885d501 | ||
|   | 9d0b54c90d | ||
|   | 2e5c288fea | ||
|   | f32ef20b74 | ||
|   | e2eefaac55 | ||
|   | e1cfbf0fd9 | ||
|   | 08c5689441 | ||
|   | 8dbda247d6 | ||
|   | 71a631237d | ||
|   | e22ff3828b | ||
|   | b1b12e004e | ||
|   | 5308fec354 | ||
|   | 0ba57d4701 | ||
|   | 54ca6a6178 | ||
|   | 7dd4a78cf2 | ||
|   | 52ff49512a | ||
|   | 5a48b94089 | ||
|   | ba1c73d947 | ||
|   | 4732b6bdfa | ||
|   | a6e78b70ab | ||
|   | bb1174afc5 | ||
|   | df8abe9cfd | ||
|   | c3bca97ee1 | ||
|   | c3b6fa1bba | ||
|   | 94d496afe1 | ||
|   | 7b7a572f9b | ||
|   | 1b8cb742f9 | ||
|   | 3492d180a8 | ||
|   | 021da38373 | ||
|   | ac784759d5 | ||
|   | 36eda2cd62 | ||
|   | 08a4b3013f | ||
|   | 1dd0332e8b | ||
|   | a90877ac31 | ||
|   | 8b7ea27a48 | ||
|   | 8df80e276b | ||
|   | 30572c972d | ||
|   | 53da4dd091 | ||
|   | 108a4a99c7 | ||
|   | 7c180376d6 | ||
|   | f39b8b32f7 | ||
|   | c543d19f8a | ||
|   | 80fca9aef7 | ||
|   | 5bb9aa0c2c | ||
|   | 83c746ee57 | ||
|   | aff6604636 | ||
|   | 2c80571a8a | ||
|   | d964b552af | ||
|   | 48f8b37b74 | ||
|   | 141be0028d | ||
|   | a140c47195 | ||
|   | 0c3a8392f2 | ||
|   | 16875b1f41 | ||
|   | b1f31f2eeb | ||
|   | d16b9e5a02 | ||
|   | 680484bdc8 | ||
|   | 05cd44b5dd | ||
|   | ba374139f4 | ||
|   | 72a745bfd5 | ||
|   | 3a6fac7d59 | ||
|   | 28ba8e53df | ||
|   | 9b26358e63 | ||
|   | e21521f45c | ||
|   | 30479765cb | ||
|   | 53a571ec6c | ||
|   | ad97cac313 | ||
|   | 1a352ddf55 | ||
|   | 5ba43decf2 | ||
|   | 8f06d035cb | ||
|   | b716f48c84 | ||
|   | 42b1e7143e | ||
|   | eba7821a6d | ||
|   | 93a0246c03 | ||
|   | dfd1787a49 | ||
|   | 4998fd54c0 | 
| @@ -1,2 +0,0 @@ | ||||
| [tool.black] | ||||
| line-length = 79 | ||||
							
								
								
									
										12
									
								
								.coveragerc
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								.coveragerc
									
									
									
									
									
								
							| @@ -1,7 +1,15 @@ | ||||
| [run] | ||||
| branch = True | ||||
| source = sanic | ||||
| omit = site-packages, sanic/utils.py, sanic/__main__.py | ||||
| omit = | ||||
|     site-packages | ||||
|     sanic/__main__.py | ||||
|     sanic/server/legacy.py | ||||
|     sanic/compat.py | ||||
|     sanic/simple.py | ||||
|     sanic/utils.py | ||||
|     sanic/cli | ||||
|     sanic/pages | ||||
|  | ||||
| [html] | ||||
| directory = coverage | ||||
| @@ -13,3 +21,5 @@ exclude_lines = | ||||
|     noqa | ||||
|     NOQA | ||||
|     pragma: no cover | ||||
|     TYPE_CHECKING | ||||
| skip_empty = True | ||||
|   | ||||
							
								
								
									
										78
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,78 @@ | ||||
| name: 🐞 Bug report | ||||
| description: Create a report to help us improve | ||||
| labels: ["bug", "triage"] | ||||
| body: | ||||
|   - type: checkboxes | ||||
|     id: existing | ||||
|     attributes: | ||||
|       label: Is there an existing issue for this? | ||||
|       description: Please search to see if an issue already exists for the bug you encountered. | ||||
|       options: | ||||
|       - label: I have searched the existing issues | ||||
|         required: true | ||||
|   - type: textarea | ||||
|     id: description | ||||
|     attributes: | ||||
|       label: Describe the bug | ||||
|       description: A clear and concise description of what the bug is, make sure to paste any exceptions and tracebacks using markdown code-block syntax to make it easier to read. | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: code | ||||
|     attributes: | ||||
|       label: Code snippet | ||||
|       description: | | ||||
|           Relevant source code, make sure to remove what is not necessary. Please try and format your code so that it is easier to read. For example: | ||||
|  | ||||
|               ```python | ||||
|               from sanic import Sanic | ||||
|  | ||||
|               app = Sanic("Example") | ||||
|               ``` | ||||
|     validations: | ||||
|       required: false | ||||
|   - type: textarea | ||||
|     id: expected | ||||
|     attributes: | ||||
|       label: Expected Behavior | ||||
|       description: A concise description of what you expected to happen. | ||||
|     validations: | ||||
|       required: false | ||||
|   - type: dropdown | ||||
|     id: running | ||||
|     attributes: | ||||
|       label: How do you run Sanic? | ||||
|       options: | ||||
|         - Sanic CLI | ||||
|         - As a module | ||||
|         - As a script (`app.run` or `Sanic.serve`) | ||||
|         - ASGI | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: dropdown | ||||
|     id: os | ||||
|     attributes: | ||||
|       label: Operating System | ||||
|       description: What OS? | ||||
|       options: | ||||
|         - Linux | ||||
|         - MacOS | ||||
|         - Windows | ||||
|         - Other (tell us in the description) | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: input | ||||
|     id: version | ||||
|     attributes: | ||||
|       label: Sanic Version | ||||
|       description: Check startup logs or try `sanic --version` | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: additional | ||||
|     attributes: | ||||
|       label: Additional context | ||||
|       description: Add any other context about the problem here. | ||||
|     validations: | ||||
|       required: false | ||||
|  | ||||
							
								
								
									
										25
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| name: Bug report | ||||
| about: Create a report to help us improve | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Describe the bug** | ||||
| A clear and concise description of what the bug is, make sure to paste any exceptions and tracebacks. | ||||
|  | ||||
|  | ||||
| **Code snippet** | ||||
| Relevant source code, make sure to remove what is not necessary. | ||||
|  | ||||
|  | ||||
| **Expected behavior** | ||||
| A clear and concise description of what you expected to happen. | ||||
|  | ||||
|  | ||||
| **Environment (please complete the following information):** | ||||
|  - OS: [e.g. iOS] | ||||
|  - Version [e.g. 0.8.3] | ||||
|  | ||||
|  | ||||
| **Additional context** | ||||
| Add any other context about the problem here. | ||||
							
								
								
									
										5
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,8 @@ | ||||
| blank_issues_enabled: true | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
|   - name: Questions and Help | ||||
|     url: https://community.sanicframework.org/c/questions-and-help | ||||
|     about: Do you need help with Sanic? Ask your questions here. | ||||
|   - name: Discussion and Support | ||||
|     url: https://discord.gg/FARQzAEMAA | ||||
|     about: For live discussion and support, checkout the Sanic Discord server. | ||||
|   | ||||
							
								
								
									
										34
									
								
								.github/ISSUE_TEMPLATE/feature-request.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.github/ISSUE_TEMPLATE/feature-request.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| name: 🌟 Feature request | ||||
| description: Suggest an enhancement for Sanic | ||||
| labels: ["feature request"] | ||||
| body: | ||||
|   - type: checkboxes | ||||
|     id: existing | ||||
|     attributes: | ||||
|       label: Is there an existing issue for this? | ||||
|       description: Please search to see if an issue already exists for the enhancement you are proposing. | ||||
|       options: | ||||
|       - label: I have searched the existing issues | ||||
|         required: true | ||||
|   - type: textarea | ||||
|     id: description | ||||
|     attributes: | ||||
|       label: Is your feature request related to a problem? Please describe. | ||||
|       description: A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] | ||||
|     validations: | ||||
|       required: false | ||||
|   - type: textarea | ||||
|     id: code | ||||
|     attributes: | ||||
|       label: Describe the solution you'd like | ||||
|       description: A clear and concise description of what you want to happen. | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: additional | ||||
|     attributes: | ||||
|       label: Additional context | ||||
|       description: Add any other context about the problem here. | ||||
|     validations: | ||||
|       required: false | ||||
|  | ||||
							
								
								
									
										16
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,16 +0,0 @@ | ||||
| --- | ||||
| name: Feature request | ||||
| about: Suggest an idea for Sanic | ||||
|  | ||||
| --- | ||||
|  | ||||
| **Is your feature request related to a problem? Please describe.** | ||||
| A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] | ||||
|  | ||||
|  | ||||
| **Describe the solution you'd like** | ||||
| A clear and concise description of what you want to happen. | ||||
|  | ||||
|  | ||||
| **Additional context** | ||||
| Add any other context or sample code about the feature request here. | ||||
							
								
								
									
										33
									
								
								.github/ISSUE_TEMPLATE/rfc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								.github/ISSUE_TEMPLATE/rfc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| name: 💡 Request for Comments | ||||
| description: Open an RFC for discussion | ||||
| labels: ["RFC"] | ||||
| body: | ||||
|   - type: input | ||||
|     id: compare | ||||
|     attributes: | ||||
|       label: Link to code | ||||
|       description: If available, share a [comparison](https://github.com/sanic-org/sanic/compare) from a POC branch to main | ||||
|       placeholder: https://github.com/sanic-org/sanic/compare/main...some-new-branch | ||||
|     validations: | ||||
|       required: false | ||||
|   - type: textarea | ||||
|     id: proposal | ||||
|     attributes: | ||||
|       label: Proposal | ||||
|       description: A thorough discussion of the proposal discussing the problem it solves, potential code, use cases, and impacts | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: additional | ||||
|     attributes: | ||||
|       label: Additional context | ||||
|       description: Add any other context that is relevant | ||||
|     validations: | ||||
|       required: false | ||||
|   - type: checkboxes | ||||
|     id: breaking | ||||
|     attributes: | ||||
|       label: Is this a breaking change? | ||||
|       options: | ||||
|         - label: "Yes" | ||||
|           required: false | ||||
							
								
								
									
										45
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										45
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,27 +1,23 @@ | ||||
| # For most projects, this workflow file will not need changing; you simply need | ||||
| # to commit it to your repository. | ||||
| # | ||||
| # You may wish to alter this file to override the set of languages analyzed, | ||||
| # or to provide custom queries or build logic. | ||||
| # | ||||
| # ******** NOTE ******** | ||||
| # We have attempted to detect the languages in your repository. Please check | ||||
| # the `language` matrix defined below to confirm you have the correct set of | ||||
| # supported CodeQL languages. | ||||
| # | ||||
| name: "CodeQL" | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [ master ] | ||||
|     branches: | ||||
|       - main | ||||
|       - current-release | ||||
|       - "*LTS" | ||||
|   pull_request: | ||||
|     # The branches below must be a subset of the branches above | ||||
|     branches: [ master ] | ||||
|     branches: | ||||
|       - main | ||||
|       - current-release | ||||
|       - "*LTS" | ||||
|     types: [opened, synchronize, reopened, ready_for_review] | ||||
|   schedule: | ||||
|     - cron: '25 16 * * 0' | ||||
|  | ||||
| jobs: | ||||
|   analyze: | ||||
|     if: github.event.pull_request.draft == false | ||||
|     name: Analyze | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
| @@ -29,39 +25,18 @@ jobs: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         language: [ 'python' ] | ||||
|         # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python' ] | ||||
|         # Learn more: | ||||
|         # https://docs.github.com/en/free-pro-team@latest/github/finding-security-vulnerabilities-and-errors-in-your-code/configuring-code-scanning#changing-the-languages-that-are-analyzed | ||||
|  | ||||
|     steps: | ||||
|     - name: Checkout repository | ||||
|       uses: actions/checkout@v2 | ||||
|  | ||||
|     # Initializes the CodeQL tools for scanning. | ||||
|     - name: Initialize CodeQL | ||||
|       uses: github/codeql-action/init@v1 | ||||
|       with: | ||||
|         languages: ${{ matrix.language }} | ||||
|         # If you wish to specify custom queries, you can do so here or in a config file. | ||||
|         # By default, queries listed here will override any specified in a config file. | ||||
|         # Prefix the list here with "+" to use these queries and those in the config file. | ||||
|         # queries: ./path/to/local/query, your-org/your-repo/queries@main | ||||
|  | ||||
|     # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). | ||||
|     # If this step fails, then you should remove it and run the build manually (see below) | ||||
|     - name: Autobuild | ||||
|       uses: github/codeql-action/autobuild@v1 | ||||
|  | ||||
|     # ℹ️ Command-line programs to run using the OS shell. | ||||
|     # 📚 https://git.io/JvXDl | ||||
|  | ||||
|     # ✏️ If the Autobuild fails above, remove it and uncomment the following three lines | ||||
|     #    and modify them (or add more) to build your code if your project | ||||
|     #    uses a compiled language | ||||
|  | ||||
|     #- run: | | ||||
|     #   make bootstrap | ||||
|     #   make release | ||||
|  | ||||
|     - name: Perform CodeQL Analysis | ||||
|       uses: github/codeql-action/analyze@v1 | ||||
|   | ||||
							
								
								
									
										34
									
								
								.github/workflows/coverage.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.github/workflows/coverage.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| name: Coverage check | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
|       - current-release | ||||
|       - "*LTS" | ||||
|     tags: | ||||
|       - "!*" # Do not execute on tags | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - main | ||||
|       - current-release | ||||
|       - "*LTS" | ||||
|  | ||||
| jobs: | ||||
|   coverage: | ||||
|     name: Check coverage | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|  | ||||
|     steps: | ||||
|       - name: Run coverage | ||||
|         uses: sanic-org/simple-tox-action@v1 | ||||
|         with: | ||||
|           python-version: "3.11" | ||||
|           tox-env: coverage | ||||
|           ignore-errors: true | ||||
|       - name: Run Codecov | ||||
|         uses: codecov/codecov-action@v3 | ||||
|         with: | ||||
|           files: ./coverage.xml | ||||
|           fail_ci_if_error: false | ||||
							
								
								
									
										174
									
								
								.github/workflows/publish-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								.github/workflows/publish-release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | ||||
| name: Publish release | ||||
|  | ||||
| on: | ||||
|   release: | ||||
|     types: [created] | ||||
|  | ||||
| env: | ||||
|   IS_TEST: false | ||||
|   DOCKER_ORG_NAME: sanicframework | ||||
|   DOCKER_IMAGE_NAME: sanic | ||||
|   DOCKER_BASE_IMAGE_NAME: sanic-build | ||||
|   DOCKER_IMAGE_DOCKERFILE: ./docker/Dockerfile | ||||
|   DOCKER_BASE_IMAGE_DOCKERFILE: ./docker/Dockerfile-base | ||||
|  | ||||
| jobs: | ||||
|   generate_info: | ||||
|     name: Generate info | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       docker-tags: ${{ steps.generate_docker_info.outputs.tags }} | ||||
|       pypi-version: ${{ steps.parse_version_tag.outputs.pypi-version }} | ||||
|     steps: | ||||
|       - name: Parse version tag | ||||
|         id: parse_version_tag | ||||
|         env: | ||||
|           TAG_NAME: ${{ github.event.release.tag_name }} | ||||
|         run: | | ||||
|           tag_name="${{ env.TAG_NAME }}" | ||||
|  | ||||
|           if [[ ! "${tag_name}" =~ ^v([0-9]{2})\.([0-9]{1,2})\.([0-9]+)$ ]]; then | ||||
|             echo "::error::Tag name must be in the format vYY.MM.MICRO" | ||||
|             exit 1 | ||||
|           fi | ||||
|  | ||||
|           year_output="year=${BASH_REMATCH[1]}" | ||||
|           month_output="month=${BASH_REMATCH[2]}" | ||||
|           pypi_output="pypi-version=${tag_name#v}" | ||||
|  | ||||
|           echo "${year_output}" | ||||
|           echo "${month_output}" | ||||
|           echo "${pypi_output}" | ||||
|  | ||||
|           echo "${year_output}" >> $GITHUB_OUTPUT | ||||
|           echo "${month_output}" >> $GITHUB_OUTPUT | ||||
|           echo "${pypi_output}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Get latest release | ||||
|         id: get_latest_release | ||||
|         run: | | ||||
|           latest_tag=$( | ||||
|             curl -L \ | ||||
|               -H "Accept: application/vnd.github+json" \ | ||||
|               -H "Authorization: Bearer ${{ github.token }}" \ | ||||
|               -H "X-GitHub-Api-Version: 2022-11-28" \ | ||||
|               https://api.github.com/repos/${{ github.repository }}/releases/latest \ | ||||
|               | jq -r '.tag_name' | ||||
|           ) | ||||
|           echo "latest_tag=$latest_tag" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Generate Docker info | ||||
|         id: generate_docker_info | ||||
|         run: | | ||||
|           tag_year="${{ steps.parse_version_tag.outputs.year }}" | ||||
|           tag_month="${{ steps.parse_version_tag.outputs.month }}" | ||||
|           latest_tag="${{ steps.get_latest_release.outputs.latest_tag }}" | ||||
|           tag="${{ github.event.release.tag_name }}" | ||||
|  | ||||
|           tags="${tag_year}.${tag_month}" | ||||
|  | ||||
|           if [[ "${tag_month}" == "12" ]]; then | ||||
|             tags+=",LTS" | ||||
|             echo "::notice::Tag ${tag} is LTS version" | ||||
|           else | ||||
|             echo "::notice::Tag ${tag} is not LTS version" | ||||
|           fi | ||||
|  | ||||
|           if [[ "${latest_tag}" == "${{ github.event.release.tag_name }}" ]]; then | ||||
|             tags+=",latest" | ||||
|             echo "::notice::Tag ${tag} is marked as latest" | ||||
|           else | ||||
|             echo "::notice::Tag ${tag} is not marked as latest" | ||||
|           fi | ||||
|  | ||||
|           tags_output="tags=${tags}" | ||||
|  | ||||
|           echo "${tags_output}" | ||||
|           echo "${tags_output}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|   publish_package: | ||||
|     name: Build and publish package | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: generate_info | ||||
|     steps: | ||||
|     - name: Checkout repo | ||||
|       uses: actions/checkout@v3 | ||||
|  | ||||
|     - name: Setup Python | ||||
|       uses: actions/setup-python@v4 | ||||
|       with: | ||||
|         python-version: "3.11" | ||||
|  | ||||
|     - name: Install dependencies | ||||
|       run: pip install build twine | ||||
|  | ||||
|     - name: Update package version | ||||
|       run: | | ||||
|         echo "__version__ = \"${{ needs.generate_info.outputs.pypi-version }}\"" > sanic/__version__.py | ||||
|  | ||||
|     - name: Build a binary wheel and a source tarball | ||||
|       run: python -m build --sdist --wheel --outdir dist/ . | ||||
|  | ||||
|     - name: Publish to PyPi 🚀 | ||||
|       run: twine upload --non-interactive --disable-progress-bar dist/* | ||||
|       env: | ||||
|         TWINE_USERNAME: __token__ | ||||
|         TWINE_PASSWORD: ${{ env.IS_TEST == 'true' && secrets.SANIC_TEST_PYPI_API_TOKEN || secrets.SANIC_PYPI_API_TOKEN }} | ||||
|         TWINE_REPOSITORY: ${{ env.IS_TEST == 'true' && 'testpypi' || 'pypi' }} | ||||
|  | ||||
|   publish_docker: | ||||
|     name: Publish Docker / Python ${{ matrix.python-version }} | ||||
|     needs: [generate_info, publish_package] | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: true | ||||
|       matrix: | ||||
|         python-version: ["3.10", "3.11"] | ||||
|     steps: | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Set up Docker Buildx | ||||
|         uses: docker/setup-buildx-action@v2 | ||||
|  | ||||
|       - name: Login to Docker Hub | ||||
|         uses: docker/login-action@v2 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKER_ACCESS_USER }} | ||||
|           password: ${{ secrets.DOCKER_ACCESS_TOKEN }} | ||||
|  | ||||
|       - name: Build and push base image | ||||
|         uses: docker/build-push-action@v4 | ||||
|         with: | ||||
|           push: ${{ env.IS_TEST == 'false' }} | ||||
|           file: ${{ env.DOCKER_BASE_IMAGE_DOCKERFILE }} | ||||
|           tags: ${{ env.DOCKER_ORG_NAME }}/${{ env.DOCKER_BASE_IMAGE_NAME }}:${{ matrix.python-version }} | ||||
|           build-args: | | ||||
|             PYTHON_VERSION=${{ matrix.python-version }} | ||||
|  | ||||
|       - name: Parse tags for this Python version | ||||
|         id: parse_tags | ||||
|         run: | | ||||
|           IFS=',' read -ra tags <<< "${{ needs.generate_info.outputs.docker-tags }}" | ||||
|           tag_args="" | ||||
|  | ||||
|           for tag in "${tags[@]}"; do | ||||
|               tag_args+=",${{ env.DOCKER_ORG_NAME }}/${{ env.DOCKER_IMAGE_NAME }}:${tag}-py${{ matrix.python-version }}" | ||||
|           done | ||||
|  | ||||
|           tag_args_output="tag_args=${tag_args:1}" | ||||
|  | ||||
|           echo "${tag_args_output}" | ||||
|           echo "${tag_args_output}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Build and push Sanic image | ||||
|         uses: docker/build-push-action@v4 | ||||
|         with: | ||||
|           push: ${{ env.IS_TEST == 'false' }} | ||||
|           file: ${{ env.DOCKER_IMAGE_DOCKERFILE }} | ||||
|           tags: ${{ steps.parse_tags.outputs.tag_args }} | ||||
|           build-args: | | ||||
|             BASE_IMAGE_ORG=${{ env.DOCKER_ORG_NAME }} | ||||
|             BASE_IMAGE_NAME=${{ env.DOCKER_BASE_IMAGE_NAME }} | ||||
|             BASE_IMAGE_TAG=${{ matrix.python-version }} | ||||
|             SANIC_PYPI_VERSION=${{ needs.generate_info.outputs.pypi-version }} | ||||
							
								
								
									
										56
									
								
								.github/workflows/tests.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								.github/workflows/tests.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| name: Tests | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
|       - current-release | ||||
|       - "*LTS" | ||||
|     tags: | ||||
|       - "!*" | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - main | ||||
|       - current-release | ||||
|       - "*LTS" | ||||
|     types: [opened, synchronize, reopened, ready_for_review] | ||||
|  | ||||
| jobs: | ||||
|   run_tests: | ||||
|     name: "${{ matrix.config.platform == 'windows-latest' && 'Windows' || 'Linux' }} / Python ${{ matrix.config.python-version }} / tox -e ${{ matrix.config.tox-env }}" | ||||
|     if: github.event.pull_request.draft == false | ||||
|     runs-on: ${{ matrix.config.platform || 'ubuntu-latest' }} | ||||
|     strategy: | ||||
|       fail-fast: true | ||||
|       matrix: | ||||
|         config: | ||||
|           - { python-version: "3.8",  tox-env: security } | ||||
|           - { python-version: "3.9",  tox-env: security } | ||||
|           - { python-version: "3.10", tox-env: security } | ||||
|           - { python-version: "3.11", tox-env: security } | ||||
|           - { python-version: "3.10", tox-env: lint } | ||||
|           - { python-version: "3.10", tox-env: docs } | ||||
|           - { python-version: "3.8",  tox-env: type-checking } | ||||
|           - { python-version: "3.9",  tox-env: type-checking } | ||||
|           - { python-version: "3.10", tox-env: type-checking } | ||||
|           - { python-version: "3.11", tox-env: type-checking } | ||||
|           - { python-version: "3.8",  tox-env: py38,          max-attempts: 3 } | ||||
|           - { python-version: "3.8",  tox-env: py38-no-ext,   max-attempts: 3 } | ||||
|           - { python-version: "3.9",  tox-env: py39,          max-attempts: 3 } | ||||
|           - { python-version: "3.9",  tox-env: py39-no-ext,   max-attempts: 3 } | ||||
|           - { python-version: "3.10", tox-env: py310,         max-attempts: 3 } | ||||
|           - { python-version: "3.10", tox-env: py310-no-ext,  max-attempts: 3 } | ||||
|           - { python-version: "3.11", tox-env: py311,         max-attempts: 3 } | ||||
|           - { python-version: "3.11", tox-env: py311-no-ext,  max-attempts: 3 } | ||||
|           - { python-version: "3.8",  tox-env: py38-no-ext,   platform: windows-latest, ignore-errors: true } | ||||
|           - { python-version: "3.9",  tox-env: py39-no-ext,   platform: windows-latest, ignore-errors: true } | ||||
|           - { python-version: "3.10", tox-env: py310-no-ext,  platform: windows-latest, ignore-errors: true } | ||||
|           - { python-version: "3.11", tox-env: py310-no-ext,  platform: windows-latest, ignore-errors: true } | ||||
|     steps: | ||||
|       - name: Run tests | ||||
|         uses: sanic-org/simple-tox-action@v1 | ||||
|         with: | ||||
|           python-version: ${{ matrix.config.python-version }} | ||||
|           tox-env: ${{ matrix.config.tox-env }} | ||||
|           max-attempts: ${{ matrix.config.max-attempts || 1 }} | ||||
|           ignore-errors: ${{ matrix.config.ignore-errors || false }} | ||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -6,6 +6,7 @@ | ||||
| .coverage | ||||
| .coverage.* | ||||
| coverage | ||||
| coverage.xml | ||||
| .tox | ||||
| settings.py | ||||
| .idea/* | ||||
| @@ -18,3 +19,7 @@ build/* | ||||
| .DS_Store | ||||
| dist/* | ||||
| pip-wheel-metadata/ | ||||
| .pytest_cache/* | ||||
| .venv/* | ||||
| venv/* | ||||
| .vscode/* | ||||
|   | ||||
							
								
								
									
										94
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										94
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,94 +0,0 @@ | ||||
| sudo: false | ||||
| language: python | ||||
| cache: | ||||
|   directories: | ||||
|     - $HOME/.cache/pip | ||||
| matrix: | ||||
|   include: | ||||
|     - env: TOX_ENV=py37 | ||||
|       python: 3.7 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.7 with Extensions" | ||||
|     - env: TOX_ENV=py37-no-ext | ||||
|       python: 3.7 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.7 without Extensions" | ||||
|     - env: TOX_ENV=py38 | ||||
|       python: 3.8 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.8 with Extensions" | ||||
|     - env: TOX_ENV=py38-no-ext | ||||
|       python: 3.8 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.8 without Extensions" | ||||
|     - env: TOX_ENV=py39 | ||||
|       python: 3.9 | ||||
|       dist: bionic | ||||
|       sudo: true | ||||
|       name: "Python 3.9 with Extensions" | ||||
|     - env: TOX_ENV=py39-no-ext | ||||
|       python: 3.9 | ||||
|       dist: bionic | ||||
|       sudo: true | ||||
|       name: "Python 3.9 without Extensions" | ||||
|     - env: TOX_ENV=type-checking | ||||
|       python: 3.7 | ||||
|       name: "Python 3.7 Type checks" | ||||
|     - env: TOX_ENV=type-checking | ||||
|       python: 3.8 | ||||
|       name: "Python 3.8 Type checks" | ||||
|     - env: TOX_ENV=type-checking | ||||
|       python: 3.9 | ||||
|       dist: bionic | ||||
|       name: "Python 3.9 Type checks" | ||||
|     - env: TOX_ENV=security | ||||
|       python: 3.7 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.7 Bandit security scan" | ||||
|     - env: TOX_ENV=security | ||||
|       python: 3.8 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.8 Bandit security scan" | ||||
|     - env: TOX_ENV=security | ||||
|       python: 3.9 | ||||
|       dist: bionic | ||||
|       sudo: true | ||||
|       name: "Python 3.9 Bandit security scan" | ||||
|     - env: TOX_ENV=docs | ||||
|       python: 3.7 | ||||
|       dist: xenial | ||||
|       sudo: true | ||||
|       name: "Python 3.7 Documentation tests" | ||||
|     - env: TOX_ENV=pyNightly | ||||
|       python: "nightly" | ||||
|       name: "Python nightly with Extensions" | ||||
|     - env: TOX_ENV=pyNightly-no-ext | ||||
|       python: "nightly" | ||||
|       name: "Python nightly without Extensions" | ||||
|   allow_failures: | ||||
|     - env: TOX_ENV=pyNightly | ||||
|       python: "nightly" | ||||
|       name: "Python nightly with Extensions" | ||||
|     - env: TOX_ENV=pyNightly-no-ext | ||||
|       python: "nightly" | ||||
|       name: "Python nightly without Extensions" | ||||
| install: | ||||
|   - pip install -U tox | ||||
|   - pip install codecov | ||||
| script: travis_retry tox -e $TOX_ENV | ||||
| after_success: | ||||
|   - codecov | ||||
| deploy: | ||||
|   provider: pypi | ||||
|   user: brewmaster | ||||
|   password: | ||||
|     secure: "GoawLwmbtJOgKB6AJ0ZSYUUnNwIoonseHBxaAUH3zu79TS/Afrq+yB3lsVaMSG0CbyDgN4FrfD1phT1NzbvZ1VcLIOTDtCrmpQ1kLDw+zwgF40ab8sp8fPkKVHHHfCCs1mjltHIpxQa5lZTJcAs6Bpi/lbUWWwYxFzSV8pHw4W4hY09EHUd2o+evLTSVxaploetSt725DJUYKICUr2eAtCC11IDnIW4CzBJEx6krVV3uhzfTJW0Ls17x0c6sdZ9icMnV/G9xO/eQH6RIHe4xcrWJ6cmLDNKoGAkJp+BKr1CeVVg7Jw/MzPjvZKL2/ki6Beue1y6GUIy7lOS7jPVaOEhJ23b0zQwFcLMZw+Tt+E3v6QfHk+B/WBBBnM3zUZed9UI+QyW8+lqLLt39sQX0FO0P3eaDh8qTXtUuon2jTyFMMAMTFRTNpJmpAzuBH9yeMmDeALPTh0HphI+BkoUl5q1QbWFYjjnZMH2CatApxpLybt9A7rwm//PbOG0TSI93GEKNQ4w5DYryKTfwHzRBptNSephJSuxZYEfJsmUtas5es1D7Fe0PkyjxNNSU+eO+8wsTlitLUsJO4k0jAgy+cEKdU7YJ3J0GZVXocSkrNnUfd2hQPcJ3UtEJx3hLqqr8EM7EZBAasc1yGHh36NFetclzFY24YPih0G1+XurhTys=" | ||||
|   on: | ||||
|     tags: true | ||||
|   distributions: "sdist bdist_wheel" | ||||
							
								
								
									
										294
									
								
								CHANGELOG.rst
									
									
									
									
									
								
							
							
						
						
									
										294
									
								
								CHANGELOG.rst
									
									
									
									
									
								
							| @@ -1,8 +1,124 @@ | ||||
| .. note:: | ||||
|  | ||||
|   CHANGELOG files are maintained in ``./docs/sanic/releases``. To view the full CHANGELOG, please visit https://sanic.readthedocs.io/en/stable/sanic/changelog.html. | ||||
|  | ||||
|  | ||||
| Version 21.6.1 | ||||
| -------------- | ||||
|  | ||||
| **Bugfixes** | ||||
|  | ||||
|   * `#2178 <https://github.com/sanic-org/sanic/pull/2178>`_ | ||||
|     Update sanic-routing to allow for better splitting of complex URI templates | ||||
|   * `#2183 <https://github.com/sanic-org/sanic/pull/2183>`_ | ||||
|     Proper handling of chunked request bodies to resolve phantom 503 in logs | ||||
|   * `#2181 <https://github.com/sanic-org/sanic/pull/2181>`_ | ||||
|     Resolve regression in exception logging | ||||
|   * `#2201 <https://github.com/sanic-org/sanic/pull/2201>`_ | ||||
|     Cleanup request info in pipelined requests | ||||
|  | ||||
| Version 21.6.0 | ||||
| -------------- | ||||
|  | ||||
| **Features** | ||||
|  | ||||
|   * `#2094 <https://github.com/sanic-org/sanic/pull/2094>`_ | ||||
|     Add ``response.eof()`` method for closing a stream in a handler | ||||
|   * `#2097 <https://github.com/sanic-org/sanic/pull/2097>`_ | ||||
|     Allow case-insensitive HTTP Upgrade header | ||||
|   * `#2104 <https://github.com/sanic-org/sanic/pull/2104>`_ | ||||
|     Explicit usage of CIMultiDict getters | ||||
|   * `#2109 <https://github.com/sanic-org/sanic/pull/2109>`_ | ||||
|     Consistent use of error loggers | ||||
|   * `#2114 <https://github.com/sanic-org/sanic/pull/2114>`_ | ||||
|     New ``client_ip`` access of connection info instance | ||||
|   * `#2119 <https://github.com/sanic-org/sanic/pull/2119>`_ | ||||
|     Alternatate classes on instantiation for ``Config`` and ``Sanic.ctx`` | ||||
|   * `#2133 <https://github.com/sanic-org/sanic/pull/2133>`_ | ||||
|     Implement new version of AST router | ||||
|  | ||||
|       * Proper differentiation between ``alpha`` and ``string`` param types | ||||
|       * Adds a ``slug`` param type, example: ``<foo:slug>`` | ||||
|       * Deprecates ``<foo:string>`` in favor of ``<foo:str>`` | ||||
|       * Deprecates ``<foo:number>`` in favor of ``<foo:float>`` | ||||
|       * Adds a ``route.uri`` accessor | ||||
|   * `#2136 <https://github.com/sanic-org/sanic/pull/2136>`_ | ||||
|     CLI improvements with new optional params | ||||
|   * `#2137 <https://github.com/sanic-org/sanic/pull/2137>`_ | ||||
|     Add ``version_prefix`` to URL builders | ||||
|   * `#2140 <https://github.com/sanic-org/sanic/pull/2140>`_ | ||||
|     Event autoregistration with ``EVENT_AUTOREGISTER`` | ||||
|   * `#2146 <https://github.com/sanic-org/sanic/pull/2146>`_, `#2147 <https://github.com/sanic-org/sanic/pull/2147>`_ | ||||
|     Require stricter names on  ``Sanic()`` and ``Blueprint()`` | ||||
|   * `#2150 <https://github.com/sanic-org/sanic/pull/2150>`_ | ||||
|     Infinitely reusable and nestable ``Blueprint`` and ``BlueprintGroup`` | ||||
|   * `#2154 <https://github.com/sanic-org/sanic/pull/2154>`_ | ||||
|     Upgrade ``websockets`` dependency to min version | ||||
|   * `#2155 <https://github.com/sanic-org/sanic/pull/2155>`_ | ||||
|     Allow for maximum header sizes to be increased: ``REQUEST_MAX_HEADER_SIZE`` | ||||
|   * `#2157 <https://github.com/sanic-org/sanic/pull/2157>`_ | ||||
|     Allow app factory pattern in CLI | ||||
|   * `#2165 <https://github.com/sanic-org/sanic/pull/2165>`_ | ||||
|     Change HTTP methods to enums | ||||
|   * `#2167 <https://github.com/sanic-org/sanic/pull/2167>`_ | ||||
|     Allow auto-reloading on additional directories | ||||
|   * `#2168 <https://github.com/sanic-org/sanic/pull/2168>`_ | ||||
|     Add simple HTTP server to CLI | ||||
|   * `#2170 <https://github.com/sanic-org/sanic/pull/2170>`_ | ||||
|     Additional methods for attaching ``HTTPMethodView`` | ||||
|  | ||||
| **Bugfixes** | ||||
|  | ||||
|   * `#2091 <https://github.com/sanic-org/sanic/pull/2091>`_ | ||||
|     Fix ``UserWarning`` in ASGI mode for missing ``__slots__`` | ||||
|   * `#2099 <https://github.com/sanic-org/sanic/pull/2099>`_ | ||||
|     Fix static request handler logging exception on 404 | ||||
|   * `#2110 <https://github.com/sanic-org/sanic/pull/2110>`_ | ||||
|     Fix request.args.pop removes parameters inconsistently | ||||
|   * `#2107 <https://github.com/sanic-org/sanic/pull/2107>`_ | ||||
|     Fix type hinting for load_env | ||||
|   * `#2127 <https://github.com/sanic-org/sanic/pull/2127>`_ | ||||
|     Make sure ASGI ws subprotocols is a list | ||||
|   * `#2128 <https://github.com/sanic-org/sanic/pull/2128>`_ | ||||
|     Fix issue where Blueprint exception handlers do not consistently route to proper handler | ||||
|  | ||||
|  | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * `#2156 <https://github.com/sanic-org/sanic/pull/2156>`_ | ||||
|     Remove config value ``REQUEST_BUFFER_QUEUE_SIZE`` | ||||
|   * `#2170 <https://github.com/sanic-org/sanic/pull/2170>`_ | ||||
|     ``CompositionView`` deprecated and marked for removal in 21.12 | ||||
|   * `#2172 <https://github.com/sanic-org/sanic/pull/2170>`_ | ||||
|     Deprecate StreamingHTTPResponse | ||||
|  | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * `#2149 <https://github.com/sanic-org/sanic/pull/2149>`_ | ||||
|     Remove Travis CI in favor of GitHub Actions | ||||
|  | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * `#2164 <https://github.com/sanic-org/sanic/pull/2164>`_ | ||||
|     Fix typo in documentation | ||||
|   * `#2100 <https://github.com/sanic-org/sanic/pull/2100>`_ | ||||
|     Remove documentation for non-existent arguments | ||||
|  | ||||
| Version 21.3.2 | ||||
| -------------- | ||||
|  | ||||
| **Bugfixes** | ||||
|  | ||||
|   * `#2081 <https://github.com/sanic-org/sanic/pull/2081>`_ | ||||
|     Disable response timeout on websocket connections | ||||
|  | ||||
|   * `#2085 <https://github.com/sanic-org/sanic/pull/2085>`_ | ||||
|     Make sure that blueprints with no slash is maintained when applied | ||||
|  | ||||
| Version 21.3.1 | ||||
| -------------- | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * `#2076 <https://github.com/sanic-org/sanic/pull/2076>`_ | ||||
|     Static files inside subfolders are not accessible (404) | ||||
| @@ -12,8 +128,7 @@ Version 21.3.0 | ||||
|  | ||||
| `Release Notes <https://sanicframework.org/en/guide/release-notes/v21.3.html>`_ | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1876 <https://github.com/sanic-org/sanic/pull/1876>`_ | ||||
| @@ -66,8 +181,7 @@ Features | ||||
|     `#2063 <https://github.com/sanic-org/sanic/pull/2063>`_ | ||||
|     App and connection level context objects | ||||
|  | ||||
| Bugfixes and issues resolved | ||||
| **************************** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * Resolve `#1420 <https://github.com/sanic-org/sanic/pull/1420>`_ | ||||
|     ``url_for`` where ``strict_slashes`` are on for a path ending in ``/`` | ||||
| @@ -97,8 +211,7 @@ Bugfixes and issues resolved | ||||
|     `#2001 <https://github.com/sanic-org/sanic/pull/2001>`_ | ||||
|     Raise ValueError when cookie max-age is not an integer | ||||
|  | ||||
| Deprecations and Removals | ||||
| ************************* | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * | ||||
|     `#2007 <https://github.com/sanic-org/sanic/pull/2007>`_ | ||||
| @@ -117,8 +230,7 @@ Deprecations and Removals | ||||
|   * ``Request.endpoint`` deprecated in favor of ``Request.name`` | ||||
|   * handler type name prefixes removed (static, websocket, etc) | ||||
|  | ||||
| Developer infrastructure | ||||
| ************************ | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * | ||||
|     `#1995 <https://github.com/sanic-org/sanic/pull/1995>`_ | ||||
| @@ -136,8 +248,7 @@ Developer infrastructure | ||||
|     `#2049 <https://github.com/sanic-org/sanic/pull/2049>`_ | ||||
|     Updated setup.py to use ``find_packages`` | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * | ||||
|     `#1218 <https://github.com/sanic-org/sanic/pull/1218>`_ | ||||
| @@ -159,8 +270,7 @@ Improved Documentation | ||||
|     `#2052 <https://github.com/sanic-org/sanic/pull/2052>`_ | ||||
|     Fix some examples and docs | ||||
|  | ||||
| Miscellaneous | ||||
| ************* | ||||
| **Miscellaneous** | ||||
|  | ||||
|   * ``Request.route`` property | ||||
|   * Better websocket subprotocols support | ||||
| @@ -206,8 +316,7 @@ Miscellaneous | ||||
| Version 20.12.3 | ||||
| --------------- | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#2021 <https://github.com/sanic-org/sanic/pull/2021>`_ | ||||
| @@ -216,8 +325,7 @@ Bugfixes | ||||
| Version 20.12.2 | ||||
| --------------- | ||||
|  | ||||
| Dependencies | ||||
| ************ | ||||
| **Dependencies** | ||||
|  | ||||
|   * | ||||
|     `#2026 <https://github.com/sanic-org/sanic/pull/2026>`_ | ||||
| @@ -230,8 +338,7 @@ Dependencies | ||||
| Version 19.12.5 | ||||
| --------------- | ||||
|  | ||||
| Dependencies | ||||
| ************ | ||||
| **Dependencies** | ||||
|  | ||||
|   * | ||||
|     `#2025 <https://github.com/sanic-org/sanic/pull/2025>`_ | ||||
| @@ -244,19 +351,12 @@ Dependencies | ||||
| Version 20.12.0 | ||||
| --------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1993 <https://github.com/sanic-org/sanic/pull/1993>`_ | ||||
|     Add disable app registry | ||||
|  | ||||
| Version 20.12.0 | ||||
| --------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
|  | ||||
|   * | ||||
|     `#1945 <https://github.com/sanic-org/sanic/pull/1945>`_ | ||||
|     Static route more verbose if file not found | ||||
| @@ -293,22 +393,19 @@ Features | ||||
|     `#1979 <https://github.com/sanic-org/sanic/pull/1979>`_ | ||||
|     Add app registry and Sanic class level app retrieval | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1965 <https://github.com/sanic-org/sanic/pull/1965>`_ | ||||
|     Fix Chunked Transport-Encoding in ASGI streaming response | ||||
|  | ||||
| Deprecations and Removals | ||||
| ************************* | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * | ||||
|     `#1981 <https://github.com/sanic-org/sanic/pull/1981>`_ | ||||
|     Cleanup and remove deprecated code | ||||
|  | ||||
| Developer infrastructure | ||||
| ************************ | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * | ||||
|     `#1956 <https://github.com/sanic-org/sanic/pull/1956>`_ | ||||
| @@ -322,8 +419,7 @@ Developer infrastructure | ||||
|     `#1986 <https://github.com/sanic-org/sanic/pull/1986>`_ | ||||
|     Update tox requirements | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * | ||||
|     `#1951 <https://github.com/sanic-org/sanic/pull/1951>`_ | ||||
| @@ -341,8 +437,7 @@ Improved Documentation | ||||
| Version 20.9.1 | ||||
| --------------- | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1954 <https://github.com/sanic-org/sanic/pull/1954>`_ | ||||
| @@ -355,8 +450,7 @@ Bugfixes | ||||
| Version 19.12.3 | ||||
| --------------- | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1959 <https://github.com/sanic-org/sanic/pull/1959>`_ | ||||
| @@ -367,8 +461,7 @@ Version 20.9.0 | ||||
| --------------- | ||||
|  | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1887 <https://github.com/sanic-org/sanic/pull/1887>`_ | ||||
| @@ -395,22 +488,19 @@ Features | ||||
|     `#1937 <https://github.com/sanic-org/sanic/pull/1937>`_ | ||||
|     Added auto, text, and json fallback error handlers (in v21.3, the default will change form html to auto) | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1897 <https://github.com/sanic-org/sanic/pull/1897>`_ | ||||
|     Resolves exception from unread bytes in stream | ||||
|  | ||||
| Deprecations and Removals | ||||
| ************************* | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * | ||||
|     `#1903 <https://github.com/sanic-org/sanic/pull/1903>`_ | ||||
|     config.from_envar, config.from_pyfile, and config.from_object are deprecated and set to be removed in v21.3 | ||||
|  | ||||
| Developer infrastructure | ||||
| ************************ | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * | ||||
|     `#1890 <https://github.com/sanic-org/sanic/pull/1890>`_, | ||||
| @@ -425,8 +515,7 @@ Developer infrastructure | ||||
|     `#1924 <https://github.com/sanic-org/sanic/pull/1924>`_ | ||||
|     Adding --strict-markers for pytest | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * | ||||
|     `#1922 <https://github.com/sanic-org/sanic/pull/1922>`_ | ||||
| @@ -436,8 +525,7 @@ Improved Documentation | ||||
| Version 20.6.3 | ||||
| --------------- | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1884 <https://github.com/sanic-org/sanic/pull/1884>`_ | ||||
| @@ -447,8 +535,7 @@ Bugfixes | ||||
| Version 20.6.2 | ||||
| --------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1641 <https://github.com/sanic-org/sanic/pull/1641>`_ | ||||
| @@ -458,8 +545,7 @@ Features | ||||
| Version 20.6.1 | ||||
| --------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1760 <https://github.com/sanic-org/sanic/pull/1760>`_ | ||||
| @@ -473,8 +559,7 @@ Features | ||||
|     `#1880 <https://github.com/sanic-org/sanic/pull/1880>`_ | ||||
|     Add handler names for websockets for url_for usage | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1776 <https://github.com/sanic-org/sanic/pull/1776>`_ | ||||
| @@ -496,15 +581,13 @@ Bugfixes | ||||
|     `#1853 <https://github.com/sanic-org/sanic/pull/1853>`_ | ||||
|     Fix pickle error when attempting to pickle an application which contains websocket routes | ||||
|  | ||||
| Deprecations and Removals | ||||
| ************************* | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * | ||||
|     `#1739 <https://github.com/sanic-org/sanic/pull/1739>`_ | ||||
|     Deprecate body_bytes to merge into body | ||||
|  | ||||
| Developer infrastructure | ||||
| ************************ | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * | ||||
|     `#1852 <https://github.com/sanic-org/sanic/pull/1852>`_ | ||||
| @@ -519,8 +602,7 @@ Developer infrastructure | ||||
|     Wrap run()'s "protocol" type annotation in Optional[] | ||||
|  | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * | ||||
|     `#1846 <https://github.com/sanic-org/sanic/pull/1846>`_ | ||||
| @@ -534,14 +616,13 @@ Improved Documentation | ||||
| Version 20.6.0 | ||||
| --------------- | ||||
|  | ||||
| *Released, but unintentionally ommitting PR #1880, so was replaced by 20.6.1* | ||||
| *Released, but unintentionally omitting PR #1880, so was replaced by 20.6.1* | ||||
|  | ||||
|  | ||||
| Version 20.3.0 | ||||
| --------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1762 <https://github.com/sanic-org/sanic/pull/1762>`_ | ||||
| @@ -572,8 +653,7 @@ Features | ||||
|     `#1820 <https://github.com/sanic-org/sanic/pull/1820>`_ | ||||
|     Do not set content-type and content-length headers in exceptions | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1748 <https://github.com/sanic-org/sanic/pull/1748>`_ | ||||
| @@ -591,8 +671,7 @@ Bugfixes | ||||
|     `#1808 <https://github.com/sanic-org/sanic/pull/1808>`_ | ||||
|      Fix Ctrl+C and tests on Windows | ||||
|  | ||||
| Deprecations and Removals | ||||
| ************************* | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * | ||||
|     `#1800 <https://github.com/sanic-org/sanic/pull/1800>`_ | ||||
| @@ -610,8 +689,7 @@ Deprecations and Removals | ||||
|     `#1818 <https://github.com/sanic-org/sanic/pull/1818>`_ | ||||
|     Complete deprecation of ``app.remove_route`` and ``request.raw_args`` | ||||
|  | ||||
| Dependencies | ||||
| ************ | ||||
| **Dependencies** | ||||
|  | ||||
|   * | ||||
|     `#1794 <https://github.com/sanic-org/sanic/pull/1794>`_ | ||||
| @@ -621,15 +699,13 @@ Dependencies | ||||
|     `#1806 <https://github.com/sanic-org/sanic/pull/1806>`_ | ||||
|     Import ``ASGIDispatch`` from top-level ``httpx`` (from third-party deprecation) | ||||
|  | ||||
| Developer infrastructure | ||||
| ************************ | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * | ||||
|     `#1833 <https://github.com/sanic-org/sanic/pull/1833>`_ | ||||
|     Resolve broken documentation builds | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * | ||||
|     `#1755 <https://github.com/sanic-org/sanic/pull/1755>`_ | ||||
| @@ -671,8 +747,7 @@ Improved Documentation | ||||
| Version 19.12.0 | ||||
| --------------- | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
| - Fix blueprint middleware application | ||||
|  | ||||
| @@ -691,8 +766,7 @@ Bugfixes | ||||
|   due to an `AttributeError`. This fix makes the availability of `SERVER_NAME` on our `app.config` an optional behavior. (`#1707 <https://github.com/sanic-org/sanic/issues/1707>`__) | ||||
|  | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
| - Move docs from MD to RST | ||||
|  | ||||
| @@ -706,8 +780,7 @@ Improved Documentation | ||||
| Version 19.6.3 | ||||
| -------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
| - Enable Towncrier Support | ||||
|  | ||||
| @@ -715,8 +788,7 @@ Features | ||||
|   of generating and managing change logs as part of each of pull requests. (`#1631 <https://github.com/sanic-org/sanic/issues/1631>`__) | ||||
|  | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
| - Documentation infrastructure changes | ||||
|  | ||||
| @@ -729,8 +801,7 @@ Improved Documentation | ||||
| Version 19.6.2 | ||||
| -------------- | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1562 <https://github.com/sanic-org/sanic/pull/1562>`_ | ||||
| @@ -746,8 +817,7 @@ Features | ||||
|     Add Configure support from object string | ||||
|  | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|   * | ||||
|     `#1587 <https://github.com/sanic-org/sanic/pull/1587>`_ | ||||
| @@ -765,8 +835,7 @@ Bugfixes | ||||
|     `#1594 <https://github.com/sanic-org/sanic/pull/1594>`_ | ||||
|     Strict Slashes behavior fix | ||||
|  | ||||
| Deprecations and Removals | ||||
| ************************* | ||||
| **Deprecations and Removals** | ||||
|  | ||||
|   * | ||||
|     `#1544 <https://github.com/sanic-org/sanic/pull/1544>`_ | ||||
| @@ -790,8 +859,7 @@ Deprecations and Removals | ||||
| Version 19.3 | ||||
| ------------ | ||||
|  | ||||
| Features | ||||
| ******** | ||||
| **Features** | ||||
|  | ||||
|   * | ||||
|     `#1497 <https://github.com/sanic-org/sanic/pull/1497>`_ | ||||
| @@ -859,8 +927,7 @@ Features | ||||
|  | ||||
|     This is a breaking change. | ||||
|  | ||||
| Bugfixes | ||||
| ******** | ||||
| **Bugfixes** | ||||
|  | ||||
|  | ||||
|   * | ||||
| @@ -896,8 +963,7 @@ Bugfixes | ||||
|     This allows the access log to be disabled for example when running via | ||||
|     gunicorn. | ||||
|  | ||||
| Developer infrastructure | ||||
| ************************ | ||||
| **Developer infrastructure** | ||||
|  | ||||
|   * `#1529 <https://github.com/sanic-org/sanic/pull/1529>`_ Update project PyPI credentials | ||||
|   * `#1515 <https://github.com/sanic-org/sanic/pull/1515>`_ fix linter issue causing travis build failures (fix #1514) | ||||
| @@ -905,8 +971,7 @@ Developer infrastructure | ||||
|   * `#1478 <https://github.com/sanic-org/sanic/pull/1478>`_ Upgrade setuptools version and use native docutils in doc build | ||||
|   * `#1464 <https://github.com/sanic-org/sanic/pull/1464>`_ Upgrade pytest, and fix caplog unit tests | ||||
|  | ||||
| Improved Documentation | ||||
| ********************** | ||||
| **Improved Documentation** | ||||
|  | ||||
|   * `#1516 <https://github.com/sanic-org/sanic/pull/1516>`_ Fix typo at the exception documentation | ||||
|   * `#1510 <https://github.com/sanic-org/sanic/pull/1510>`_ fix typo in Asyncio example | ||||
| @@ -967,21 +1032,19 @@ Version 18.12 | ||||
|   * Fix Range header handling for static files (#1402) | ||||
|   * Fix the logger and make it work (#1397) | ||||
|   * Fix type pikcle->pickle in multiprocessing test | ||||
|   * Fix pickling blueprints Change the string passed in the "name" section of the namedtuples in Blueprint to match the name of the Blueprint module attribute name. This allows blueprints to be pickled and unpickled, without errors, which is a requirment of running Sanic in multiprocessing mode in Windows. Added a test for pickling and unpickling blueprints Added a test for pickling and unpickling sanic itself Added a test for enabling multiprocessing on an app with a blueprint (only useful to catch this bug if the tests are run on Windows). | ||||
|   * Fix pickling blueprints Change the string passed in the "name" section of the namedtuples in Blueprint to match the name of the Blueprint module attribute name. This allows blueprints to be pickled and unpickled, without errors, which is a requirement of running Sanic in multiprocessing mode in Windows. Added a test for pickling and unpickling blueprints Added a test for pickling and unpickling sanic itself Added a test for enabling multiprocessing on an app with a blueprint (only useful to catch this bug if the tests are run on Windows). | ||||
|   * Fix document for logging | ||||
|  | ||||
| Version 0.8 | ||||
| ----------- | ||||
|  | ||||
| 0.8.3 | ||||
| ***** | ||||
| **0.8.3** | ||||
|  | ||||
| * Changes: | ||||
|  | ||||
|   * Ownership changed to org 'sanic-org' | ||||
|  | ||||
| 0.8.0 | ||||
| ***** | ||||
| **0.8.0** | ||||
|  | ||||
| * Changes: | ||||
|  | ||||
| @@ -1006,7 +1069,7 @@ Version 0.8 | ||||
|   * Content-length header on 204/304 responses (Arnulfo Solís) | ||||
|   * Extend WebSocketProtocol arguments and add docs (Bob Olde Hampsink, yunstanford) | ||||
|   * Update development status from pre-alpha to beta (Maksim Anisenkov) | ||||
|   * KeepAlive Timout log level changed to debug (Arnulfo Solís) | ||||
|   * KeepAlive Timeout log level changed to debug (Arnulfo Solís) | ||||
|   * Pin pytest to 3.3.2 because of pytest-dev/pytest#3170 (Maksim Aniskenov) | ||||
|   * Install Python 3.5 and 3.6 on docker container for tests (Shahin Azad) | ||||
|   * Add support for blueprint groups and nesting (Elias Tarhini) | ||||
| @@ -1061,19 +1124,16 @@ Version 0.1 | ||||
| ----------- | ||||
|  | ||||
|  | ||||
| 0.1.7 | ||||
| ***** | ||||
| **0.1.7** | ||||
|  | ||||
|   * Reversed static url and directory arguments to meet spec | ||||
|  | ||||
| 0.1.6 | ||||
| ***** | ||||
| **0.1.6** | ||||
|  | ||||
|   * Static files | ||||
|   * Lazy Cookie Loading | ||||
|  | ||||
| 0.1.5 | ||||
| ***** | ||||
| **0.1.5** | ||||
|  | ||||
|   * Cookies | ||||
|   * Blueprint listeners and ordering | ||||
| @@ -1081,23 +1141,19 @@ Version 0.1 | ||||
|   * Fix: Incomplete file reads on medium+ sized post requests | ||||
|   * Breaking: after_start and before_stop now pass sanic as their first argument | ||||
|  | ||||
| 0.1.4 | ||||
| ***** | ||||
| **0.1.4** | ||||
|  | ||||
|   * Multiprocessing | ||||
|  | ||||
| 0.1.3 | ||||
| ***** | ||||
| **0.1.3** | ||||
|  | ||||
|   * Blueprint support | ||||
|   * Faster Response processing | ||||
|  | ||||
| 0.1.1 - 0.1.2 | ||||
| ************* | ||||
| **0.1.1 - 0.1.2** | ||||
|  | ||||
|   * Struggling to update pypi via CI | ||||
|  | ||||
| 0.1.0 | ||||
| ***** | ||||
| **0.1.0** | ||||
|  | ||||
|   * Released to public | ||||
|   | ||||
| @@ -55,7 +55,7 @@ further defined and clarified by project maintainers. | ||||
| ## Enforcement | ||||
| 
 | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||
| reported by contacting the project team at sanic-maintainers@googlegroups.com. All | ||||
| reported by contacting the project team at adam@sanicframework.org. All | ||||
| complaints will be reviewed and investigated and will result in a response that | ||||
| is deemed necessary and appropriate to the circumstances. The project team is | ||||
| obligated to maintain confidentiality with regard to the reporter of an incident. | ||||
| @@ -19,7 +19,7 @@ a virtual environment already set up, then run: | ||||
|  | ||||
| .. code-block:: bash | ||||
|  | ||||
|    pip3 install -e . ".[dev]" | ||||
|    pip install -e ".[dev]" | ||||
|  | ||||
| Dependency Changes | ||||
| ------------------ | ||||
| @@ -71,9 +71,9 @@ To execute only unittests, run ``tox`` with environment like so: | ||||
|  | ||||
| .. code-block:: bash | ||||
|  | ||||
|    tox -e py36 -v -- tests/test_config.py | ||||
|    # or | ||||
|    tox -e py37 -v -- tests/test_config.py | ||||
|    # or | ||||
|    tox -e py310 -v -- tests/test_config.py | ||||
|  | ||||
| Run lint checks | ||||
| --------------- | ||||
| @@ -87,7 +87,7 @@ Permform ``flake8``\ , ``black`` and ``isort`` checks. | ||||
|    tox -e lint | ||||
|  | ||||
| Run type annotation checks | ||||
| --------------- | ||||
| -------------------------- | ||||
|  | ||||
| ``tox`` environment -> ``[testenv:type-checking]`` | ||||
|  | ||||
| @@ -140,6 +140,7 @@ To maintain the code consistency, Sanic uses following tools. | ||||
| #. `isort <https://github.com/timothycrosley/isort>`_ | ||||
| #. `black <https://github.com/python/black>`_ | ||||
| #. `flake8 <https://github.com/PyCQA/flake8>`_ | ||||
| #. `slotscheck <https://github.com/ariebovenberg/slotscheck>`_ | ||||
|  | ||||
| isort | ||||
| ***** | ||||
| @@ -167,7 +168,13 @@ flake8 | ||||
| #. pycodestyle | ||||
| #. Ned Batchelder's McCabe script | ||||
|  | ||||
| ``isort``\ , ``black`` and ``flake8`` checks are performed during ``tox`` lint checks. | ||||
| slotscheck | ||||
| ********** | ||||
|  | ||||
| ``slotscheck`` ensures that there are no problems with ``__slots__`` | ||||
| (e.g. overlaps, or missing slots in base classes). | ||||
|  | ||||
| ``isort``\ , ``black``\ , ``flake8`` and ``slotscheck`` checks are performed during ``tox`` lint checks. | ||||
|  | ||||
| The **easiest** way to make your code conform is to run the following before committing. | ||||
|  | ||||
|   | ||||
							
								
								
									
										12
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								Makefile
									
									
									
									
									
								
							| @@ -49,6 +49,9 @@ test: clean | ||||
| test-coverage: clean | ||||
| 	python setup.py test --pytest-args="--cov sanic --cov-report term --cov-append " | ||||
|  | ||||
| view-coverage: | ||||
| 	sanic ./coverage --simple | ||||
|  | ||||
| install: | ||||
| 	python setup.py install | ||||
|  | ||||
| @@ -63,15 +66,15 @@ ifdef include_tests | ||||
| 	isort -rc sanic tests | ||||
| else | ||||
| 	$(info Sorting Imports) | ||||
| 	isort -rc sanic tests  --profile=black | ||||
| 	isort -rc sanic tests | ||||
| endif | ||||
| endif | ||||
|  | ||||
| black: | ||||
| 	black --config ./.black.toml sanic tests | ||||
| 	black sanic tests | ||||
|  | ||||
| isort: | ||||
| 	isort sanic tests --profile=black | ||||
| 	isort sanic tests | ||||
|  | ||||
| pretty: black isort | ||||
|  | ||||
| @@ -85,8 +88,7 @@ docs-test: docs-clean | ||||
| 	cd docs && make dummy | ||||
|  | ||||
| docs-serve: | ||||
| 	# python -m http.server --directory=./docs/_build/html 9999 | ||||
| 	sphinx-autobuild docs docs/_build/html --port 9999 --watch ./sanic | ||||
| 	sphinx-autobuild docs docs/_build/html --port 9999 --watch ./ | ||||
|  | ||||
| changelog: | ||||
| 	python scripts/changelog.py | ||||
|   | ||||
							
								
								
									
										47
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										47
									
								
								README.rst
									
									
									
									
									
								
							| @@ -11,7 +11,7 @@ Sanic | Build fast. Run fast. | ||||
|     :stub-columns: 1 | ||||
|  | ||||
|     * - Build | ||||
|       - | |Build Status| |AppVeyor Build Status| |Codecov| | ||||
|       - | |Tests| | ||||
|     * - Docs | ||||
|       - | |UserGuide| |Documentation| | ||||
|     * - Package | ||||
| @@ -19,7 +19,7 @@ Sanic | Build fast. Run fast. | ||||
|     * - Support | ||||
|       - | |Forums| |Discord| |Awesome| | ||||
|     * - Stats | ||||
|       - | |Downloads| |WkDownloads| |Conda downloads| | ||||
|       - | |Monthly Downloads| |Weekly Downloads| |Conda downloads| | ||||
|  | ||||
| .. |UserGuide| image:: https://img.shields.io/badge/user%20guide-sanic-ff0068 | ||||
|    :target: https://sanicframework.org/ | ||||
| @@ -27,12 +27,8 @@ Sanic | Build fast. Run fast. | ||||
|    :target: https://community.sanicframework.org/ | ||||
| .. |Discord| image:: https://img.shields.io/discord/812221182594121728?logo=discord | ||||
|    :target: https://discord.gg/FARQzAEMAA | ||||
| .. |Codecov| image:: https://codecov.io/gh/sanic-org/sanic/branch/master/graph/badge.svg | ||||
|     :target: https://codecov.io/gh/sanic-org/sanic | ||||
| .. |Build Status| image:: https://travis-ci.com/sanic-org/sanic.svg?branch=master | ||||
|    :target: https://travis-ci.com/sanic-org/sanic | ||||
| .. |AppVeyor Build Status| image:: https://ci.appveyor.com/api/projects/status/d8pt3ids0ynexi8c/branch/master?svg=true | ||||
|    :target: https://ci.appveyor.com/project/sanic-org/sanic | ||||
| .. |Tests| image:: https://github.com/sanic-org/sanic/actions/workflows/tests.yml/badge.svg?branch=main | ||||
|    :target: https://github.com/sanic-org/sanic/actions/workflows/tests.yml | ||||
| .. |Documentation| image:: https://readthedocs.org/projects/sanic/badge/?version=latest | ||||
|    :target: http://sanic.readthedocs.io/en/latest/?badge=latest | ||||
| .. |PyPI| image:: https://img.shields.io/pypi/v/sanic.svg | ||||
| @@ -50,21 +46,25 @@ Sanic | Build fast. Run fast. | ||||
| .. |Awesome| image:: https://cdn.rawgit.com/sindresorhus/awesome/d7305f38d29fed78fa85652e3a63e154dd8e8829/media/badge.svg | ||||
|     :alt: Awesome Sanic List | ||||
|     :target: https://github.com/mekicha/awesome-sanic | ||||
| .. |Downloads| image:: https://pepy.tech/badge/sanic/month | ||||
| .. |Monthly Downloads| image:: https://img.shields.io/pypi/dm/sanic.svg | ||||
|     :alt: Downloads | ||||
|     :target: https://pepy.tech/project/sanic | ||||
| .. |WkDownloads| image:: https://pepy.tech/badge/sanic/week | ||||
| .. |Weekly Downloads| image:: https://img.shields.io/pypi/dw/sanic.svg | ||||
|     :alt: Downloads | ||||
|     :target: https://pepy.tech/project/sanic | ||||
| .. |Conda downloads| image:: https://img.shields.io/conda/dn/conda-forge/sanic.svg | ||||
|     :alt: Downloads | ||||
|     :target: https://anaconda.org/conda-forge/sanic | ||||
| .. |Linode| image:: https://www.linode.com/wp-content/uploads/2021/01/Linode-Logo-Black.svg | ||||
|     :alt: Linode | ||||
|     :target: https://www.linode.com | ||||
|     :width: 200px | ||||
|  | ||||
| .. end-badges | ||||
|  | ||||
| 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. | ||||
| Sanic is a **Python 3.8+** 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. | ||||
|  | ||||
| Sanic is also ASGI compliant, so you can deploy it with an `alternative ASGI webserver <https://sanic.readthedocs.io/en/latest/sanic/deploying.html#running-via-asgi>`_. | ||||
| Sanic is also ASGI compliant, so you can deploy it with an `alternative ASGI webserver <https://sanicframework.org/en/guide/deployment/running.html#asgi>`_. | ||||
|  | ||||
| `Source code on GitHub <https://github.com/sanic-org/sanic/>`_ | `Help and discussion board <https://community.sanicframework.org/>`_ | `User Guide <https://sanicframework.org>`_ | `Chat on Discord <https://discord.gg/FARQzAEMAA>`_ | ||||
|  | ||||
| @@ -75,17 +75,11 @@ The goal of the project is to provide a simple way to get up and running a highl | ||||
| Sponsor | ||||
| ------- | ||||
|  | ||||
| |Try CodeStream| | ||||
| Check out `open collective <https://opencollective.com/sanic-org>`_ to learn more about helping to fund Sanic. | ||||
|  | ||||
| .. |Try CodeStream| image:: https://alt-images.codestream.com/codestream_logo_sanicorg.png | ||||
|    :target: https://codestream.com/?utm_source=github&utm_campaign=sanicorg&utm_medium=banner | ||||
|    :alt: Try CodeStream | ||||
| Thanks to `Linode <https://www.linode.com>`_ for their contribution towards the development and community of Sanic. | ||||
|  | ||||
| Manage pull requests and conduct code reviews in your IDE with full source-tree context. Comment on any line, not just the diffs. Use jump-to-definition, your favorite keybindings, and code intelligence with more of your workflow. | ||||
|  | ||||
| `Learn More <https://codestream.com/?utm_source=github&utm_campaign=sanicorg&utm_medium=banner>`_ | ||||
|  | ||||
| Thank you to our sponsor. Check out `open collective <https://opencollective.com/sanic-org>`_ to learn more about helping to fund Sanic. | ||||
| |Linode| | ||||
|  | ||||
| Installation | ||||
| ------------ | ||||
| @@ -106,9 +100,6 @@ Installation | ||||
|   If you are running on a clean install of Fedora 28 or above, please make sure you have the ``redhat-rpm-config`` package installed in case if you want to | ||||
|   use ``sanic`` with ``ujson`` dependency. | ||||
|  | ||||
| .. note:: | ||||
|  | ||||
|   Windows support is currently "experimental" and on a best-effort basis. Multiple workers are also not currently supported on Windows (see `Issue #1517 <https://github.com/sanic-org/sanic/issues/1517>`_), but setting ``workers=1`` should launch the server successfully. | ||||
|  | ||||
| Hello World Example | ||||
| ------------------- | ||||
| @@ -118,7 +109,7 @@ Hello World Example | ||||
|     from sanic import Sanic | ||||
|     from sanic.response import json | ||||
|  | ||||
|     app = Sanic("My Hello, world app") | ||||
|     app = Sanic("my-hello-world-app") | ||||
|  | ||||
|     @app.route('/') | ||||
|     async def test(request): | ||||
| @@ -148,17 +139,17 @@ And, we can verify it is working: ``curl localhost:8000 -i`` | ||||
|  | ||||
| **Now, let's go build something fast!** | ||||
|  | ||||
| Minimum Python version is 3.7. If you need Python 3.6 support, please use v20.12LTS. | ||||
| Minimum Python version is 3.8. If you need Python 3.7 support, please use v22.12LTS. | ||||
|  | ||||
| Documentation | ||||
| ------------- | ||||
|  | ||||
| `User Guide <https://sanicframework.org>`__ and `API Documentation <http://sanic.readthedocs.io/>`__. | ||||
| `User Guide <https://sanic.dev>`__ and `API Documentation <http://sanic.readthedocs.io/>`__. | ||||
|  | ||||
| Changelog | ||||
| --------- | ||||
|  | ||||
| `Release Changelogs <https://github.com/sanic-org/sanic/blob/master/CHANGELOG.rst>`__. | ||||
| `Release Changelogs <https://sanic.readthedocs.io/en/stable/sanic/changelog.html>`__. | ||||
|  | ||||
|  | ||||
| Questions and Discussion | ||||
|   | ||||
							
								
								
									
										53
									
								
								SECURITY.md
									
									
									
									
									
								
							
							
						
						
									
										53
									
								
								SECURITY.md
									
									
									
									
									
								
							| @@ -4,31 +4,42 @@ | ||||
|  | ||||
| Sanic releases long term support release once a year in December. LTS releases receive bug and security updates for **24 months**. Interim releases throughout the year occur every three months, and are supported until the subsequent interim release. | ||||
|  | ||||
| | Version | LTS           | Supported          | | ||||
| | ------- | ------------- | ------------------ | | ||||
| | 20.12   | until 2022-12 | :heavy_check_mark: | | ||||
| | 20.9    |               | :x:                | | ||||
| | 20.6    |               | :x:                | | ||||
| | 20.3    |               | :x:                | | ||||
| | 19.12   | until 2021-12 | :white_check_mark: | | ||||
| | 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:                | | ||||
|  | ||||
| :white_check_mark: = security/bug fixes | ||||
| :heavy_check_mark: = full support | ||||
| | Version | LTS           | Supported               | | ||||
| | ------- | ------------- | ----------------------- | | ||||
| | 22.12   | until 2024-12 | :white_check_mark:      | | ||||
| | 22.9    |               | :x:                     | | ||||
| | 22.6    |               | :x:                     | | ||||
| | 22.3    |               | :x:                     | | ||||
| | 21.12   | until 2023-12 | :ballot_box_with_check: | | ||||
| | 21.9    |               | :x:                     | | ||||
| | 21.6    |               | :x:                     | | ||||
| | 21.3    |               | :x:                     | | ||||
| | 20.12   |               | :x:                     | | ||||
| | 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 | ||||
|  | ||||
| ## Reporting a Vulnerability | ||||
|  | ||||
| If you discover a security vulnerability, we ask that you **do not** create an issue on GitHub. Instead, please [send a message to the core-devs](https://community.sanicframework.org/g/core-devs) on the community forums. Once logged in, you can send a message to the core-devs by clicking the message button. | ||||
|  | ||||
| Alternatively, you can send a private message to Adam Hopkins on Discord. Find him on the [Sanic discord server](https://discord.gg/FARQzAEMAA). | ||||
|  | ||||
| This will help to not publicize the issue until the team can address it and resolve it. | ||||
|   | ||||
							
								
								
									
										32
									
								
								codecov.yml
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								codecov.yml
									
									
									
									
									
								
							| @@ -1,14 +1,28 @@ | ||||
| codecov: | ||||
|   require_ci_to_pass: no | ||||
| coverage: | ||||
|   precision: 3 | ||||
|   round: nearest | ||||
|   status: | ||||
|     project: | ||||
|       default: | ||||
|         target: auto | ||||
|         threshold: 0.5% | ||||
|     patch: | ||||
|       default: | ||||
|         target: auto | ||||
|         threshold: 0.75% | ||||
|         threshold: 0.75 | ||||
|         informational: true | ||||
|     project: | ||||
|       default: | ||||
|         target: auto | ||||
|         threshold: 0.5 | ||||
|   precision: 3 | ||||
| codecov: | ||||
|   require_ci_to_pass: false | ||||
| ignore: | ||||
|   - "sanic/__main__.py" | ||||
|   - "sanic/compat.py" | ||||
|   - "sanic/simple.py" | ||||
|   - "sanic/utils.py" | ||||
|   - "sanic/cli/" | ||||
|   - "sanic/pages/" | ||||
|   - ".github/" | ||||
|   - "changelogs/" | ||||
|   - "docker/" | ||||
|   - "docs/" | ||||
|   - "examples/" | ||||
|   - "scripts/" | ||||
|   - "tests/" | ||||
|   | ||||
| @@ -1,28 +1,13 @@ | ||||
| FROM alpine:3.7 | ||||
| ARG BASE_IMAGE_ORG | ||||
| ARG BASE_IMAGE_NAME | ||||
| ARG BASE_IMAGE_TAG | ||||
|  | ||||
| RUN apk add --no-cache --update \ | ||||
|         curl \ | ||||
|         bash \ | ||||
|         build-base \ | ||||
|         ca-certificates \ | ||||
|         git \ | ||||
|         bzip2-dev \ | ||||
|         linux-headers \ | ||||
|         ncurses-dev \ | ||||
|         openssl \ | ||||
|         openssl-dev \ | ||||
|         readline-dev \ | ||||
|         sqlite-dev | ||||
| FROM ${BASE_IMAGE_ORG}/${BASE_IMAGE_NAME}:${BASE_IMAGE_TAG} | ||||
|  | ||||
| RUN apk update | ||||
| RUN update-ca-certificates | ||||
| RUN rm -rf /var/cache/apk/* | ||||
|  | ||||
| ENV PYENV_ROOT="/root/.pyenv" | ||||
| ENV PATH="$PYENV_ROOT/bin:$PATH" | ||||
| ARG SANIC_PYPI_VERSION | ||||
|  | ||||
| ADD . /app | ||||
| WORKDIR /app | ||||
|  | ||||
| RUN /app/docker/bin/install_python.sh 3.5.4 3.6.4 | ||||
|  | ||||
| ENTRYPOINT ["./docker/bin/entrypoint.sh"] | ||||
| RUN pip install -U pip && pip install sanic==${SANIC_PYPI_VERSION} | ||||
| RUN apk del build-base | ||||
|   | ||||
							
								
								
									
										9
									
								
								docker/Dockerfile-base
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								docker/Dockerfile-base
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| ARG PYTHON_VERSION | ||||
|  | ||||
| FROM python:${PYTHON_VERSION}-alpine | ||||
| RUN apk update | ||||
| RUN apk add --no-cache --update build-base \ | ||||
|         ca-certificates \ | ||||
|         openssl | ||||
| RUN update-ca-certificates | ||||
| RUN rm -rf /var/cache/apk/* | ||||
| @@ -1,11 +0,0 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| eval "$(pyenv init -)" | ||||
| eval "$(pyenv virtualenv-init -)" | ||||
| source /root/.pyenv/completions/pyenv.bash | ||||
|  | ||||
| pip install tox | ||||
|  | ||||
| exec $@ | ||||
|  | ||||
| @@ -1,17 +0,0 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| export CFLAGS='-O2' | ||||
| export EXTRA_CFLAGS="-DTHREAD_STACK_SIZE=0x100000" | ||||
|  | ||||
| curl -L https://raw.githubusercontent.com/pyenv/pyenv-installer/master/bin/pyenv-installer | bash | ||||
| eval "$(pyenv init -)" | ||||
|  | ||||
| for ver in $@ | ||||
| do | ||||
|     pyenv install $ver | ||||
| done | ||||
|  | ||||
| pyenv global $@ | ||||
| pip install --upgrade pip | ||||
| pyenv rehash | ||||
							
								
								
									
										9
									
								
								docs/_static/custom.css
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								docs/_static/custom.css
									
									
									
									
										vendored
									
									
								
							| @@ -2,3 +2,12 @@ | ||||
| .wy-nav-top { | ||||
|   background: #444444; | ||||
| } | ||||
|  | ||||
| #changelog section { | ||||
|   padding-left: 3rem; | ||||
| } | ||||
|  | ||||
| #changelog section h2, | ||||
| #changelog section h3 { | ||||
|   margin-left: -3rem; | ||||
| } | ||||
|   | ||||
							
								
								
									
										24
									
								
								docs/conf.py
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								docs/conf.py
									
									
									
									
									
								
							| @@ -10,10 +10,8 @@ | ||||
| import os | ||||
| import sys | ||||
|  | ||||
| # Add support for auto-doc | ||||
| import recommonmark | ||||
|  | ||||
| from recommonmark.transform import AutoStructify | ||||
| # Add support for auto-doc | ||||
|  | ||||
|  | ||||
| # Ensure that sanic is present in the path, to allow sphinx-apidoc to | ||||
| @@ -26,7 +24,11 @@ import sanic | ||||
|  | ||||
| # -- General configuration ------------------------------------------------ | ||||
|  | ||||
| extensions = ["sphinx.ext.autodoc", "recommonmark"] | ||||
| extensions = [ | ||||
|     "sphinx.ext.autodoc", | ||||
|     "m2r2", | ||||
|     "enum_tools.autoenum", | ||||
| ] | ||||
|  | ||||
| templates_path = ["_templates"] | ||||
|  | ||||
| @@ -162,20 +164,6 @@ autodoc_default_options = { | ||||
|     "member-order": "groupwise", | ||||
| } | ||||
|  | ||||
|  | ||||
| # app setup hook | ||||
| def setup(app): | ||||
|     app.add_config_value( | ||||
|         "recommonmark_config", | ||||
|         { | ||||
|             "enable_eval_rst": True, | ||||
|             "enable_auto_doc_ref": False, | ||||
|         }, | ||||
|         True, | ||||
|     ) | ||||
|     app.add_transform(AutoStructify) | ||||
|  | ||||
|  | ||||
| html_theme_options = { | ||||
|     "style_external_links": False, | ||||
| } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ API | ||||
| === | ||||
|  | ||||
| .. toctree:: | ||||
|    :maxdepth: 2 | ||||
|    :maxdepth: 3 | ||||
|  | ||||
|    👥 User Guide <https://sanicframework.org/guide/> | ||||
|    sanic/api_reference | ||||
|   | ||||
							
								
								
									
										33
									
								
								docs/sanic/api/app.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								docs/sanic/api/app.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| Application | ||||
| =========== | ||||
|  | ||||
| sanic.app | ||||
| --------- | ||||
|  | ||||
| .. automodule:: sanic.app | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|     :inherited-members: | ||||
|  | ||||
| sanic.config | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.config | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.application.constants | ||||
| --------------------------- | ||||
|  | ||||
| .. automodule:: sanic.application.constants | ||||
|     :exclude-members: StrEnum | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|     :inherited-members: | ||||
|  | ||||
| sanic.application.state | ||||
| ----------------------- | ||||
|  | ||||
| .. automodule:: sanic.application.state | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
							
								
								
									
										17
									
								
								docs/sanic/api/blueprints.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								docs/sanic/api/blueprints.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| Blueprints | ||||
| ========== | ||||
|  | ||||
| sanic.blueprints | ||||
| ---------------- | ||||
|  | ||||
| .. automodule:: sanic.blueprints | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|     :inherited-members: | ||||
|  | ||||
| sanic.blueprint_group | ||||
| --------------------- | ||||
|  | ||||
| .. automodule:: sanic.blueprint_group | ||||
|     :members: | ||||
|     :special-members: | ||||
							
								
								
									
										48
									
								
								docs/sanic/api/core.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								docs/sanic/api/core.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| Core | ||||
| ==== | ||||
|  | ||||
| sanic.cookies | ||||
| ------------- | ||||
|  | ||||
| .. automodule:: sanic.cookies | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
|  | ||||
| sanic.handlers | ||||
| -------------- | ||||
|  | ||||
| .. automodule:: sanic.handlers | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
|  | ||||
| sanic.headers | ||||
| -------------- | ||||
|  | ||||
| .. automodule:: sanic.headers | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
|  | ||||
| sanic.request | ||||
| ------------- | ||||
|  | ||||
| .. automodule:: sanic.request | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.response | ||||
| -------------- | ||||
|  | ||||
| .. automodule:: sanic.response | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
|  | ||||
| sanic.views | ||||
| ----------- | ||||
|  | ||||
| .. automodule:: sanic.views | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
							
								
								
									
										16
									
								
								docs/sanic/api/exceptions.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								docs/sanic/api/exceptions.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| Exceptions | ||||
| ========== | ||||
|  | ||||
| sanic.errorpages | ||||
| ---------------- | ||||
|  | ||||
| .. automodule:: sanic.errorpages | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.exceptions | ||||
| ---------------- | ||||
|  | ||||
| .. automodule:: sanic.exceptions | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
							
								
								
									
										18
									
								
								docs/sanic/api/router.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								docs/sanic/api/router.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| Routing | ||||
| ======= | ||||
|  | ||||
| sanic_routing models | ||||
| -------------------- | ||||
|  | ||||
| .. autoclass:: sanic_routing.route::Route | ||||
|     :members: | ||||
|  | ||||
| .. autoclass:: sanic_routing.group::RouteGroup | ||||
|     :members: | ||||
|  | ||||
| sanic.router | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.router | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
							
								
								
									
										18
									
								
								docs/sanic/api/server.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								docs/sanic/api/server.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| Sanic Server | ||||
| ============ | ||||
|  | ||||
| sanic.http | ||||
| ---------- | ||||
|  | ||||
| .. automodule:: sanic.http | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
|  | ||||
| sanic.server | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.server | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
							
								
								
									
										16
									
								
								docs/sanic/api/utility.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								docs/sanic/api/utility.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| Utility | ||||
| ======= | ||||
|  | ||||
| sanic.compat | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.compat | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.log | ||||
| --------- | ||||
|  | ||||
| .. automodule:: sanic.log | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
| @@ -1,132 +1,13 @@ | ||||
| 📑 API Reference | ||||
| ================ | ||||
|  | ||||
| sanic.app | ||||
| --------- | ||||
| .. toctree:: | ||||
|    :maxdepth: 2 | ||||
|  | ||||
| .. automodule:: sanic.app | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|     :inherited-members: | ||||
|  | ||||
| sanic.blueprints | ||||
| ---------------- | ||||
|  | ||||
| .. automodule:: sanic.blueprints | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|     :inherited-members: | ||||
|  | ||||
| sanic.blueprint_group | ||||
| --------------------- | ||||
|  | ||||
| .. automodule:: sanic.blueprint_group | ||||
|     :members: | ||||
|     :special-members: | ||||
|  | ||||
|  | ||||
| sanic.compat | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.compat | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.config | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.config | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.cookies | ||||
| ------------- | ||||
|  | ||||
| .. automodule:: sanic.cookies | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.errorpages | ||||
| ---------------- | ||||
|  | ||||
| .. automodule:: sanic.errorpages | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.exceptions | ||||
| ---------------- | ||||
|  | ||||
| .. automodule:: sanic.exceptions | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.handlers | ||||
| -------------- | ||||
|  | ||||
| .. automodule:: sanic.handlers | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.http | ||||
| ---------- | ||||
|  | ||||
| .. automodule:: sanic.http | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.log | ||||
| --------- | ||||
|  | ||||
| .. automodule:: sanic.log | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.request | ||||
| ------------- | ||||
|  | ||||
| .. automodule:: sanic.request | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.response | ||||
| -------------- | ||||
|  | ||||
| .. automodule:: sanic.response | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.router | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.router | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.server | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.server | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
|  | ||||
| sanic.views | ||||
| ----------- | ||||
|  | ||||
| .. automodule:: sanic.views | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.websocket | ||||
| --------------- | ||||
|  | ||||
| .. automodule:: sanic.websocket | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|  | ||||
| sanic.worker | ||||
| ------------ | ||||
|  | ||||
| .. automodule:: sanic.worker | ||||
|     :members: | ||||
|     :show-inheritance: | ||||
|    api/app | ||||
|    api/blueprints | ||||
|    api/core | ||||
|    api/exceptions | ||||
|    api/router | ||||
|    api/server | ||||
|    api/utility | ||||
|   | ||||
| @@ -1,4 +1,16 @@ | ||||
| 📜 Changelog | ||||
| ============ | ||||
|  | ||||
| | 🔶 Current release | ||||
| | 🔷 In support release | ||||
| | | ||||
|  | ||||
| .. mdinclude:: ./releases/23/23.6.md | ||||
| .. mdinclude:: ./releases/23/23.3.md | ||||
| .. mdinclude:: ./releases/22/22.12.md | ||||
| .. mdinclude:: ./releases/22/22.9.md | ||||
| .. mdinclude:: ./releases/22/22.6.md | ||||
| .. mdinclude:: ./releases/22/22.3.md | ||||
| .. mdinclude:: ./releases/21/21.12.md | ||||
| .. mdinclude:: ./releases/21/21.9.md | ||||
| .. include:: ../../CHANGELOG.rst | ||||
|   | ||||
							
								
								
									
										66
									
								
								docs/sanic/releases/21/21.12.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								docs/sanic/releases/21/21.12.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| ## Version 21.12.1 🔷 | ||||
|  | ||||
| _Current LTS version_ | ||||
|  | ||||
| - [#2349](https://github.com/sanic-org/sanic/pull/2349) Only display MOTD on startup | ||||
| - [#2354](https://github.com/sanic-org/sanic/pull/2354) Ignore name argument in Python 3.7 | ||||
| - [#2355](https://github.com/sanic-org/sanic/pull/2355) Add config.update support for all config values | ||||
|  | ||||
| ## Version 21.12.0 🔹 | ||||
|  | ||||
| ### Features | ||||
| - [#2260](https://github.com/sanic-org/sanic/pull/2260) Allow early Blueprint registrations to still apply later added objects | ||||
| - [#2262](https://github.com/sanic-org/sanic/pull/2262) Noisy exceptions - force logging of all exceptions | ||||
| - [#2264](https://github.com/sanic-org/sanic/pull/2264) Optional `uvloop` by configuration | ||||
| - [#2270](https://github.com/sanic-org/sanic/pull/2270) Vhost support using multiple TLS certificates | ||||
| - [#2277](https://github.com/sanic-org/sanic/pull/2277) Change signal routing for increased consistency | ||||
|     - *BREAKING CHANGE*: If you were manually routing signals there is a breaking change. The signal router's `get` is no longer 100% determinative. There is now an additional step to loop thru the returned signals for proper matching on the requirements. If signals are being dispatched using `app.dispatch` or `bp.dispatch`, there is no change. | ||||
| - [#2290](https://github.com/sanic-org/sanic/pull/2290) Add contextual exceptions | ||||
| - [#2291](https://github.com/sanic-org/sanic/pull/2291) Increase join concat performance  | ||||
| - [#2295](https://github.com/sanic-org/sanic/pull/2295), [#2316](https://github.com/sanic-org/sanic/pull/2316), [#2331](https://github.com/sanic-org/sanic/pull/2331) Restructure of CLI and application state with new displays and more command parity with `app.run` | ||||
| - [#2302](https://github.com/sanic-org/sanic/pull/2302) Add route context at definition time | ||||
| - [#2304](https://github.com/sanic-org/sanic/pull/2304) Named tasks and new API for managing background tasks | ||||
| - [#2307](https://github.com/sanic-org/sanic/pull/2307) On app auto-reload, provide insight of changed files | ||||
| - [#2308](https://github.com/sanic-org/sanic/pull/2308) Auto extend application with [Sanic Extensions](https://sanicframework.org/en/plugins/sanic-ext/getting-started.html) if it is installed, and provide first class support for accessing the extensions | ||||
| - [#2309](https://github.com/sanic-org/sanic/pull/2309) Builtin signals changed to `Enum` | ||||
| - [#2313](https://github.com/sanic-org/sanic/pull/2313) Support additional config implementation use case | ||||
| - [#2321](https://github.com/sanic-org/sanic/pull/2321) Refactor environment variable hydration logic | ||||
| - [#2327](https://github.com/sanic-org/sanic/pull/2327) Prevent sending multiple or mixed responses on a single request | ||||
| - [#2330](https://github.com/sanic-org/sanic/pull/2330) Custom type casting on environment variables | ||||
| - [#2332](https://github.com/sanic-org/sanic/pull/2332) Make all deprecation notices consistent | ||||
| - [#2335](https://github.com/sanic-org/sanic/pull/2335) Allow underscore to start instance names | ||||
|  | ||||
| ### Bugfixes | ||||
| - [#2273](https://github.com/sanic-org/sanic/pull/2273) Replace assignation by typing for `websocket_handshake` | ||||
| - [#2285](https://github.com/sanic-org/sanic/pull/2285) Fix IPv6 display in startup logs | ||||
| - [#2299](https://github.com/sanic-org/sanic/pull/2299) Dispatch `http.lifecyle.response` from exception handler | ||||
|  | ||||
| ### Deprecations and Removals | ||||
| - [#2306](https://github.com/sanic-org/sanic/pull/2306) Removal of deprecated items | ||||
|     - `Sanic` and `Blueprint` may no longer have arbitrary properties attached to them | ||||
|     - `Sanic` and `Blueprint` forced to have compliant names | ||||
|         - alphanumeric + `_` + `-` | ||||
|         - must start with letter or `_` | ||||
|     - `load_env` keyword argument of `Sanic` | ||||
|     - `sanic.exceptions.abort` | ||||
|     - `sanic.views.CompositionView` | ||||
|     - `sanic.response.StreamingHTTPResponse` | ||||
|         - *NOTE:* the `stream()` response method (where you pass a callable streaming function) has been deprecated and will be removed in v22.6. You should upgrade all streaming responses to the new style: https://sanicframework.org/en/guide/advanced/streaming.html#response-streaming | ||||
| - [#2320](https://github.com/sanic-org/sanic/pull/2320) Remove app instance from Config for error handler setting | ||||
|  | ||||
| ### Developer infrastructure | ||||
| - [#2251](https://github.com/sanic-org/sanic/pull/2251) Change dev install command | ||||
| - [#2286](https://github.com/sanic-org/sanic/pull/2286) Change codeclimate complexity threshold from 5 to 10 | ||||
| - [#2287](https://github.com/sanic-org/sanic/pull/2287) Update host test function names so they are not overwritten | ||||
| - [#2292](https://github.com/sanic-org/sanic/pull/2292) Fail CI on error | ||||
| - [#2311](https://github.com/sanic-org/sanic/pull/2311), [#2324](https://github.com/sanic-org/sanic/pull/2324) Do not run tests for draft PRs | ||||
| - [#2336](https://github.com/sanic-org/sanic/pull/2336) Remove paths from coverage checks | ||||
| - [#2338](https://github.com/sanic-org/sanic/pull/2338) Cleanup ports on tests | ||||
|  | ||||
| ### Improved Documentation | ||||
| - [#2269](https://github.com/sanic-org/sanic/pull/2269), [#2329](https://github.com/sanic-org/sanic/pull/2329), [#2333](https://github.com/sanic-org/sanic/pull/2333) Cleanup typos and fix language | ||||
|  | ||||
| ### Miscellaneous | ||||
| - [#2257](https://github.com/sanic-org/sanic/pull/2257), [#2294](https://github.com/sanic-org/sanic/pull/2294), [#2341](https://github.com/sanic-org/sanic/pull/2341) Add Python 3.10 support | ||||
| - [#2279](https://github.com/sanic-org/sanic/pull/2279), [#2317](https://github.com/sanic-org/sanic/pull/2317), [#2322](https://github.com/sanic-org/sanic/pull/2322) Add/correct missing type annotations | ||||
| - [#2305](https://github.com/sanic-org/sanic/pull/2305) Fix examples to use modern implementations | ||||
							
								
								
									
										50
									
								
								docs/sanic/releases/21/21.9.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								docs/sanic/releases/21/21.9.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| ## Version 21.9.3 | ||||
| *Rerelease of v21.9.2 with some cleanup* | ||||
|  | ||||
| ## Version 21.9.2 | ||||
| - [#2268](https://github.com/sanic-org/sanic/pull/2268) Make HTTP connections start in IDLE stage, avoiding delays and error messages | ||||
| - [#2310](https://github.com/sanic-org/sanic/pull/2310) More consistent config setting with post-FALLBACK_ERROR_FORMAT apply | ||||
|  | ||||
| ## Version 21.9.1 | ||||
| - [#2259](https://github.com/sanic-org/sanic/pull/2259) Allow non-conforming ErrorHandlers | ||||
|  | ||||
| ## Version 21.9.0 | ||||
|  | ||||
| ### Features | ||||
| - [#2158](https://github.com/sanic-org/sanic/pull/2158), [#2248](https://github.com/sanic-org/sanic/pull/2248) Complete overhaul of I/O to websockets | ||||
| - [#2160](https://github.com/sanic-org/sanic/pull/2160) Add new 17 signals into server and request lifecycles | ||||
| - [#2162](https://github.com/sanic-org/sanic/pull/2162) Smarter `auto` fallback formatting upon exception | ||||
| - [#2184](https://github.com/sanic-org/sanic/pull/2184) Introduce implementation for copying a Blueprint | ||||
| - [#2200](https://github.com/sanic-org/sanic/pull/2200) Accept header parsing | ||||
| - [#2207](https://github.com/sanic-org/sanic/pull/2207) Log remote address if available | ||||
| - [#2209](https://github.com/sanic-org/sanic/pull/2209) Add convenience methods to BP groups | ||||
| - [#2216](https://github.com/sanic-org/sanic/pull/2216) Add default messages to SanicExceptions | ||||
| - [#2225](https://github.com/sanic-org/sanic/pull/2225) Type annotation convenience for annotated handlers with path parameters | ||||
| - [#2236](https://github.com/sanic-org/sanic/pull/2236) Allow Falsey (but not-None) responses from route handlers | ||||
| - [#2238](https://github.com/sanic-org/sanic/pull/2238) Add `exception` decorator to Blueprint Groups | ||||
| - [#2244](https://github.com/sanic-org/sanic/pull/2244) Explicit static directive for serving file or dir (ex: `static(..., resource_type="file")`) | ||||
| - [#2245](https://github.com/sanic-org/sanic/pull/2245) Close HTTP loop when connection task cancelled | ||||
|  | ||||
| ### Bugfixes | ||||
| - [#2188](https://github.com/sanic-org/sanic/pull/2188) Fix the handling of the end of a chunked request | ||||
| - [#2195](https://github.com/sanic-org/sanic/pull/2195) Resolve unexpected error handling on static requests | ||||
| - [#2208](https://github.com/sanic-org/sanic/pull/2208) Make blueprint-based exceptions attach and trigger in a more intuitive manner | ||||
| - [#2211](https://github.com/sanic-org/sanic/pull/2211) Fixed for handling exceptions of asgi app call | ||||
| - [#2213](https://github.com/sanic-org/sanic/pull/2213) Fix bug where ws exceptions not being logged | ||||
| - [#2231](https://github.com/sanic-org/sanic/pull/2231) Cleaner closing of tasks by using `abort()` in strategic places to avoid dangling sockets | ||||
| - [#2247](https://github.com/sanic-org/sanic/pull/2247) Fix logging of auto-reload status in debug mode | ||||
| - [#2246](https://github.com/sanic-org/sanic/pull/2246) Account for BP with exception handler but no routes | ||||
|  | ||||
| ### Developer infrastructure   | ||||
| - [#2194](https://github.com/sanic-org/sanic/pull/2194) HTTP unit tests with raw client | ||||
| - [#2199](https://github.com/sanic-org/sanic/pull/2199) Switch to codeclimate | ||||
| - [#2214](https://github.com/sanic-org/sanic/pull/2214) Try Reopening Windows Tests | ||||
| - [#2229](https://github.com/sanic-org/sanic/pull/2229) Refactor `HttpProtocol` into a base class | ||||
| - [#2230](https://github.com/sanic-org/sanic/pull/2230) Refactor `server.py` into multi-file module | ||||
|  | ||||
| ### Miscellaneous | ||||
| - [#2173](https://github.com/sanic-org/sanic/pull/2173) Remove Duplicated Dependencies and PEP 517 Support  | ||||
| - [#2193](https://github.com/sanic-org/sanic/pull/2193), [#2196](https://github.com/sanic-org/sanic/pull/2196), [#2217](https://github.com/sanic-org/sanic/pull/2217) Type annotation changes | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										55
									
								
								docs/sanic/releases/22/22.12.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								docs/sanic/releases/22/22.12.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| ## Version 22.12.0 🔷 | ||||
|  | ||||
| _Current version_ | ||||
|  | ||||
| ### Features | ||||
|  | ||||
| - [#2569](https://github.com/sanic-org/sanic/pull/2569) Add `JSONResponse` class with some convenient methods when updating a response object | ||||
| - [#2598](https://github.com/sanic-org/sanic/pull/2598) Change `uvloop` requirement to `>=0.15.0` | ||||
| - [#2609](https://github.com/sanic-org/sanic/pull/2609) Add compatibility with `websockets` v11.0 | ||||
| - [#2610](https://github.com/sanic-org/sanic/pull/2610) Kill server early on worker error | ||||
|     - Raise deadlock timeout to 30s | ||||
| - [#2617](https://github.com/sanic-org/sanic/pull/2617) Scale number of running server workers | ||||
| - [#2621](https://github.com/sanic-org/sanic/pull/2621) [#2634](https://github.com/sanic-org/sanic/pull/2634) Send `SIGKILL` on subsequent `ctrl+c` to force worker exit | ||||
| - [#2622](https://github.com/sanic-org/sanic/pull/2622) Add API to restart all workers from the multiplexer | ||||
| - [#2624](https://github.com/sanic-org/sanic/pull/2624) Default to `spawn` for all subprocesses unless specifically set: | ||||
|     ```python | ||||
|     from sanic import Sanic | ||||
|      | ||||
|     Sanic.start_method = "fork" | ||||
|     ``` | ||||
| - [#2625](https://github.com/sanic-org/sanic/pull/2625) Filename normalisation of form-data/multipart file uploads | ||||
| - [#2626](https://github.com/sanic-org/sanic/pull/2626) Move to HTTP Inspector: | ||||
|     - Remote access to inspect running Sanic instances | ||||
|     - TLS support for encrypted calls to Inspector | ||||
|     - Authentication to Inspector with API key | ||||
|     - Ability to extend Inspector with custom commands | ||||
| - [#2632](https://github.com/sanic-org/sanic/pull/2632) Control order of restart operations | ||||
| - [#2633](https://github.com/sanic-org/sanic/pull/2633) Move reload interval to class variable | ||||
| - [#2636](https://github.com/sanic-org/sanic/pull/2636) Add `priority` to `register_middleware` method | ||||
| - [#2639](https://github.com/sanic-org/sanic/pull/2639) Add `unquote` to `add_route` method | ||||
| - [#2640](https://github.com/sanic-org/sanic/pull/2640) ASGI websockets to receive `text` or `bytes` | ||||
|  | ||||
|  | ||||
| ### Bugfixes | ||||
|  | ||||
| - [#2607](https://github.com/sanic-org/sanic/pull/2607) Force socket shutdown before close to allow rebinding | ||||
| - [#2590](https://github.com/sanic-org/sanic/pull/2590) Use actual `StrEnum` in Python 3.11+ | ||||
| - [#2615](https://github.com/sanic-org/sanic/pull/2615) Ensure middleware executes only once per request timeout | ||||
| - [#2627](https://github.com/sanic-org/sanic/pull/2627) Crash ASGI application on lifespan failure | ||||
| - [#2635](https://github.com/sanic-org/sanic/pull/2635) Resolve error with low-level server creation on Windows | ||||
|  | ||||
|  | ||||
| ### Deprecations and Removals | ||||
|  | ||||
| - [#2608](https://github.com/sanic-org/sanic/pull/2608) [#2630](https://github.com/sanic-org/sanic/pull/2630) Signal conditions and triggers saved on `signal.extra`  | ||||
| - [#2626](https://github.com/sanic-org/sanic/pull/2626) Move to HTTP Inspector | ||||
|     - 🚨 *BREAKING CHANGE*: Moves the Inspector to a Sanic app from a simple TCP socket with a custom protocol | ||||
|     - *DEPRECATE*: The `--inspect*` commands have been deprecated in favor of `inspect ...` commands | ||||
| - [#2628](https://github.com/sanic-org/sanic/pull/2628) Replace deprecated `distutils.strtobool` | ||||
|  | ||||
|  | ||||
| ### Developer infrastructure | ||||
|  | ||||
| - [#2612](https://github.com/sanic-org/sanic/pull/2612) Add CI testing for Python 3.11 | ||||
|  | ||||
							
								
								
									
										52
									
								
								docs/sanic/releases/22/22.3.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								docs/sanic/releases/22/22.3.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| ## Version 22.3.0 | ||||
|  | ||||
| ### Features | ||||
| - [#2347](https://github.com/sanic-org/sanic/pull/2347) API for multi-application server | ||||
|     - 🚨 *BREAKING CHANGE*: The old `sanic.worker.GunicornWorker` has been **removed**. To run Sanic with `gunicorn`, you should use it thru `uvicorn` [as described in their docs](https://www.uvicorn.org/#running-with-gunicorn). | ||||
|     - 🧁 *SIDE EFFECT*: Named background tasks are now supported, even in Python 3.7 | ||||
| - [#2357](https://github.com/sanic-org/sanic/pull/2357) Parse `Authorization` header as `Request.credentials` | ||||
| - [#2361](https://github.com/sanic-org/sanic/pull/2361) Add config option to skip `Touchup` step in application startup | ||||
| - [#2372](https://github.com/sanic-org/sanic/pull/2372) Updates to CLI help messaging | ||||
| - [#2382](https://github.com/sanic-org/sanic/pull/2382) Downgrade warnings to backwater debug messages  | ||||
| - [#2396](https://github.com/sanic-org/sanic/pull/2396) Allow for `multidict` v0.6 | ||||
| - [#2401](https://github.com/sanic-org/sanic/pull/2401) Upgrade CLI catching for alternative application run types | ||||
| - [#2402](https://github.com/sanic-org/sanic/pull/2402) Conditionally inject CLI arguments into factory | ||||
| - [#2413](https://github.com/sanic-org/sanic/pull/2413) Add new start and stop event listeners to reloader process | ||||
| - [#2414](https://github.com/sanic-org/sanic/pull/2414) Remove loop as required listener arg | ||||
| - [#2415](https://github.com/sanic-org/sanic/pull/2415) Better exception for bad URL parsing | ||||
| - [sanic-routing#47](https://github.com/sanic-org/sanic-routing/pull/47) Add a new extention parameter type: `<file:ext>`, `<file:ext=jpg>`, `<file:ext=jpg|png|gif|svg>`, `<file=int:ext>`, `<file=int:ext=jpg|png|gif|svg>`, `<file=float:ext=tar.gz>` | ||||
|     - 👶 *BETA FEATURE*: This feature will not work with `path` type matching, and is being released as a beta feature only. | ||||
| - [sanic-routing#57](https://github.com/sanic-org/sanic-routing/pull/57) Change `register_pattern` to accept a `str` or `Pattern` | ||||
| - [sanic-routing#58](https://github.com/sanic-org/sanic-routing/pull/58) Default matching on non-empty strings only, and new `strorempty` pattern type | ||||
|     - 🚨 *BREAKING CHANGE*: Previously a route with a dynamic string parameter (`/<foo>` or `/<foo:str>`) 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: `/<foo:strorempty>`. | ||||
|  | ||||
| ### Bugfixes | ||||
| - [#2373](https://github.com/sanic-org/sanic/pull/2373) Remove `error_logger` on websockets | ||||
| - [#2381](https://github.com/sanic-org/sanic/pull/2381) Fix newly assigned `None` in task registry | ||||
| - [sanic-routing#52](https://github.com/sanic-org/sanic-routing/pull/52) Add type casting to regex route matching | ||||
| - [sanic-routing#60](https://github.com/sanic-org/sanic-routing/pull/60) Add requirements check on regex routes (this resolves, for example, multiple static directories with differing `host` values) | ||||
|  | ||||
| ### Deprecations and Removals | ||||
| - [#2362](https://github.com/sanic-org/sanic/pull/2362) 22.3 Deprecations and changes | ||||
|     1. `debug=True` and `--debug` do _NOT_ automatically run `auto_reload` | ||||
|     2. Default error render is with plain text (browsers still get HTML by default because `auto` looks at headers) | ||||
|     3. `config` is required for `ErrorHandler.finalize` | ||||
|     4. `ErrorHandler.lookup` requires two positional args | ||||
|     5. Unused websocket protocol args removed | ||||
| - [#2344](https://github.com/sanic-org/sanic/pull/2344) Deprecate loading of lowercase environment variables | ||||
|  | ||||
| ### Developer infrastructure | ||||
| - [#2363](https://github.com/sanic-org/sanic/pull/2363) Revert code coverage back to Codecov | ||||
| - [#2405](https://github.com/sanic-org/sanic/pull/2405) Upgrade tests for `sanic-routing` changes | ||||
| - [sanic-testing#35](https://github.com/sanic-org/sanic-testing/pull/35) Allow for httpx v0.22 | ||||
|  | ||||
| ### Improved Documentation | ||||
| - [#2350](https://github.com/sanic-org/sanic/pull/2350) Fix link in README for ASGI | ||||
| - [#2398](https://github.com/sanic-org/sanic/pull/2398) Document middleware on_request and on_response | ||||
| - [#2409](https://github.com/sanic-org/sanic/pull/2409) Add missing documentation for `Request.respond` | ||||
|  | ||||
| ### Miscellaneous | ||||
| - [#2376](https://github.com/sanic-org/sanic/pull/2376) Fix typing for `ListenerMixin.listener` | ||||
| - [#2383](https://github.com/sanic-org/sanic/pull/2383) Clear deprecation warning in `asyncio.wait` | ||||
| - [#2387](https://github.com/sanic-org/sanic/pull/2387) Cleanup `__slots__` implementations | ||||
| - [#2390](https://github.com/sanic-org/sanic/pull/2390) Clear deprecation warning in `asyncio.get_event_loop` | ||||
							
								
								
									
										54
									
								
								docs/sanic/releases/22/22.6.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								docs/sanic/releases/22/22.6.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| ## Version 22.6.2 | ||||
|  | ||||
| ### Bugfixes | ||||
|  | ||||
| - [#2522](https://github.com/sanic-org/sanic/pull/2522) Always show server location in ASGI | ||||
|  | ||||
| ## Version 22.6.1 | ||||
|  | ||||
| ### Bugfixes | ||||
|  | ||||
| - [#2477](https://github.com/sanic-org/sanic/pull/2477) Sanic static directory fails when folder name ends with ".." | ||||
|  | ||||
|  | ||||
| ## Version 22.6.0 | ||||
|  | ||||
| ### Features | ||||
| - [#2378](https://github.com/sanic-org/sanic/pull/2378) Introduce HTTP/3 and autogeneration of TLS certificates in `DEBUG` mode | ||||
|     - 👶 *EARLY RELEASE FEATURE*: Serving Sanic over HTTP/3 is an early release feature. It does not yet fully cover the HTTP/3 spec, but instead aims for feature parity with Sanic's existing HTTP/1.1 server. Websockets, WebTransport, push responses are examples of some features not yet implemented. | ||||
|     - 📦 *EXTRA REQUIREMENT*: Not all HTTP clients are capable of interfacing with HTTP/3 servers. You may need to install a [HTTP/3 capable client](https://curl.se/docs/http3.html). | ||||
|     - 📦 *EXTRA REQUIREMENT*: In order to use TLS autogeneration, you must install either [mkcert](https://github.com/FiloSottile/mkcert) or [trustme](https://github.com/python-trio/trustme). | ||||
| - [#2416](https://github.com/sanic-org/sanic/pull/2416) Add message to `task.cancel` | ||||
| - [#2420](https://github.com/sanic-org/sanic/pull/2420) Add exception aliases for more consistent naming with standard HTTP response types (`BadRequest`, `MethodNotAllowed`, `RangeNotSatisfiable`) | ||||
| - [#2432](https://github.com/sanic-org/sanic/pull/2432) Expose ASGI `scope` as a property on the `Request` object | ||||
| - [#2438](https://github.com/sanic-org/sanic/pull/2438) Easier access to websocket class for annotation: `from sanic import Websocket` | ||||
| - [#2439](https://github.com/sanic-org/sanic/pull/2439) New API for reading form values with options: `Request.get_form`  | ||||
| - [#2445](https://github.com/sanic-org/sanic/pull/2445) Add custom `loads` function | ||||
| - [#2447](https://github.com/sanic-org/sanic/pull/2447), [#2486](https://github.com/sanic-org/sanic/pull/2486) Improved API to support setting cache control headers | ||||
| - [#2453](https://github.com/sanic-org/sanic/pull/2453) Move verbosity filtering to logger | ||||
| - [#2475](https://github.com/sanic-org/sanic/pull/2475) Expose getter for current request using `Request.get_current()` | ||||
|  | ||||
| ### Bugfixes | ||||
| - [#2448](https://github.com/sanic-org/sanic/pull/2448) Fix to allow running with `pythonw.exe` or places where there is no `sys.stdout` | ||||
| - [#2451](https://github.com/sanic-org/sanic/pull/2451) Trigger `http.lifecycle.request` signal in ASGI mode | ||||
| - [#2455](https://github.com/sanic-org/sanic/pull/2455) Resolve typing of stacked route definitions | ||||
| - [#2463](https://github.com/sanic-org/sanic/pull/2463) Properly catch websocket CancelledError in websocket handler in Python 3.7 | ||||
|  | ||||
| ### Deprecations and Removals | ||||
| - [#2487](https://github.com/sanic-org/sanic/pull/2487) v22.6 deprecations and changes | ||||
|     1. Optional application registry | ||||
|     1. Execution of custom handlers after some part of response was sent | ||||
|     1. Configuring fallback handlers on the `ErrorHandler` | ||||
|     1. Custom `LOGO` setting | ||||
|     1. `sanic.response.stream` | ||||
|     1. `AsyncioServer.init` | ||||
|  | ||||
| ### Developer infrastructure | ||||
| - [#2449](https://github.com/sanic-org/sanic/pull/2449) Clean up `black` and `isort` config | ||||
| - [#2479](https://github.com/sanic-org/sanic/pull/2479) Fix some flappy tests | ||||
|  | ||||
| ### Improved Documentation | ||||
| - [#2461](https://github.com/sanic-org/sanic/pull/2461) Update example to match current application naming standards | ||||
| - [#2466](https://github.com/sanic-org/sanic/pull/2466) Better type annotation for `Extend` | ||||
| - [#2485](https://github.com/sanic-org/sanic/pull/2485) Improved help messages in CLI | ||||
|  | ||||
							
								
								
									
										74
									
								
								docs/sanic/releases/22/22.9.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								docs/sanic/releases/22/22.9.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
| ## Version 22.9.1 | ||||
|  | ||||
| ### Features | ||||
|  | ||||
| - [#2585](https://github.com/sanic-org/sanic/pull/2585) Improved error message when no applications have been registered | ||||
|  | ||||
|  | ||||
| ### Bugfixes | ||||
|  | ||||
| - [#2578](https://github.com/sanic-org/sanic/pull/2578) Add certificate loader for in process certificate creation | ||||
| - [#2591](https://github.com/sanic-org/sanic/pull/2591) Do not use sentinel identity for `spawn` compatibility | ||||
| - [#2592](https://github.com/sanic-org/sanic/pull/2592) Fix properties in nested blueprint groups | ||||
| - [#2595](https://github.com/sanic-org/sanic/pull/2595) Introduce sleep interval on new worker reloader | ||||
|  | ||||
|  | ||||
| ### Deprecations and Removals | ||||
|  | ||||
|  | ||||
| ### Developer infrastructure | ||||
|  | ||||
| - [#2588](https://github.com/sanic-org/sanic/pull/2588) Markdown templates on issue forms | ||||
|  | ||||
|  | ||||
| ### Improved Documentation | ||||
|  | ||||
| - [#2556](https://github.com/sanic-org/sanic/pull/2556) v22.9 documentation | ||||
| - [#2582](https://github.com/sanic-org/sanic/pull/2582) Cleanup documentation on Windows support | ||||
|  | ||||
|  | ||||
| ## Version 22.9.0 | ||||
|  | ||||
| ### Features | ||||
|  | ||||
| - [#2445](https://github.com/sanic-org/sanic/pull/2445) Add custom loads function  | ||||
| - [#2490](https://github.com/sanic-org/sanic/pull/2490) Make `WebsocketImplProtocol` async iterable | ||||
| - [#2499](https://github.com/sanic-org/sanic/pull/2499) Sanic Server WorkerManager refactor | ||||
| - [#2506](https://github.com/sanic-org/sanic/pull/2506) Use `pathlib` for path resolution (for static file serving) | ||||
| - [#2508](https://github.com/sanic-org/sanic/pull/2508) Use `path.parts` instead of `match` (for static file serving) | ||||
| - [#2513](https://github.com/sanic-org/sanic/pull/2513) Better request cancel handling | ||||
| - [#2516](https://github.com/sanic-org/sanic/pull/2516) Add request properties for HTTP method info: | ||||
|     - `request.is_safe` | ||||
|     - `request.is_idempotent` | ||||
|     - `request.is_cacheable` | ||||
|     - *See* [MDN docs](https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods) *for more information about when these apply* | ||||
| - [#2522](https://github.com/sanic-org/sanic/pull/2522) Always show server location in ASGI | ||||
| - [#2526](https://github.com/sanic-org/sanic/pull/2526) Cache control support for static files for returning 304 when appropriate | ||||
| - [#2533](https://github.com/sanic-org/sanic/pull/2533) Refactor `_static_request_handler` | ||||
| - [#2540](https://github.com/sanic-org/sanic/pull/2540) Add signals before and after handler execution | ||||
|     - `http.handler.before` | ||||
|     - `http.handler.after` | ||||
| - [#2542](https://github.com/sanic-org/sanic/pull/2542) Add *[redacted]* to CLI :) | ||||
| - [#2546](https://github.com/sanic-org/sanic/pull/2546) Add deprecation warning filter | ||||
| - [#2550](https://github.com/sanic-org/sanic/pull/2550) Middleware priority and performance enhancements | ||||
|  | ||||
| ### Bugfixes | ||||
|  | ||||
| - [#2495](https://github.com/sanic-org/sanic/pull/2495) Prevent directory traversion with static files | ||||
| - [#2515](https://github.com/sanic-org/sanic/pull/2515) Do not apply double slash to paths in certain static dirs in Blueprints | ||||
|  | ||||
| ### Deprecations and Removals | ||||
|  | ||||
| - [#2525](https://github.com/sanic-org/sanic/pull/2525) Warn on duplicate route names, will be prevented outright in v23.3 | ||||
| - [#2537](https://github.com/sanic-org/sanic/pull/2537) Raise warning and deprecation notice on duplicate exceptions, will be prevented outright in v23.3 | ||||
|  | ||||
| ### Developer infrastructure | ||||
|  | ||||
| - [#2504](https://github.com/sanic-org/sanic/pull/2504) Cleanup test suite | ||||
| - [#2505](https://github.com/sanic-org/sanic/pull/2505) Replace Unsupported Python Version Number from the Contributing Doc | ||||
| - [#2530](https://github.com/sanic-org/sanic/pull/2530) Do not include tests folder in installed package resolver | ||||
|  | ||||
| ### Improved Documentation | ||||
|  | ||||
| - [#2502](https://github.com/sanic-org/sanic/pull/2502) Fix a few typos | ||||
| - [#2517](https://github.com/sanic-org/sanic/pull/2517) [#2536](https://github.com/sanic-org/sanic/pull/2536) Add some type hints | ||||
							
								
								
									
										53
									
								
								docs/sanic/releases/23/23.3.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								docs/sanic/releases/23/23.3.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| ## Version 23.3.0 | ||||
|  | ||||
| ### Features | ||||
| - [#2545](https://github.com/sanic-org/sanic/pull/2545) Standardize init of exceptions for more consistent control of HTTP responses using exceptions | ||||
| - [#2606](https://github.com/sanic-org/sanic/pull/2606) Decode headers as UTF-8 also in ASGI | ||||
| - [#2646](https://github.com/sanic-org/sanic/pull/2646) Separate ASGI request and lifespan callables | ||||
| - [#2659](https://github.com/sanic-org/sanic/pull/2659) Use ``FALLBACK_ERROR_FORMAT`` for handlers that return ``empty()`` | ||||
| - [#2662](https://github.com/sanic-org/sanic/pull/2662) Add basic file browser (HTML page) and auto-index serving | ||||
| - [#2667](https://github.com/sanic-org/sanic/pull/2667) Nicer traceback formatting (HTML page) | ||||
| - [#2668](https://github.com/sanic-org/sanic/pull/2668) Smarter error page rendering format selection; more reliant upon header and "common sense" defaults | ||||
| - [#2680](https://github.com/sanic-org/sanic/pull/2680) Check the status of socket before shutting down with ``SHUT_RDWR`` | ||||
| - [#2687](https://github.com/sanic-org/sanic/pull/2687) Refresh ``Request.accept`` functionality to be more performant and spec-compliant | ||||
| - [#2696](https://github.com/sanic-org/sanic/pull/2696) Add header accessors as properties | ||||
|     ``` | ||||
|     Example-Field: Foo, Bar | ||||
|     Example-Field: Baz | ||||
|     ``` | ||||
|     ```python | ||||
|     request.headers.example_field == "Foo, Bar,Baz" | ||||
|     ``` | ||||
| - [#2700](https://github.com/sanic-org/sanic/pull/2700) Simpler CLI targets | ||||
|  | ||||
|     ```sh | ||||
|     $ sanic path.to.module:app          # global app instance | ||||
|     $ sanic path.to.module:create_app   # factory pattern | ||||
|     $ sanic ./path/to/directory/        # simple serve | ||||
|     ``` | ||||
| - [#2701](https://github.com/sanic-org/sanic/pull/2701) API to define a number of workers in managed processes | ||||
| - [#2704](https://github.com/sanic-org/sanic/pull/2704) Add convenience for dynamic changes to routing | ||||
| - [#2706](https://github.com/sanic-org/sanic/pull/2706) Add convenience methods for cookie creation and deletion | ||||
|      | ||||
|     ```python | ||||
|     response = text("...") | ||||
|     response.add_cookie("test", "It worked!", domain=".yummy-yummy-cookie.com") | ||||
|     ``` | ||||
| - [#2707](https://github.com/sanic-org/sanic/pull/2707) Simplified ``parse_content_header`` escaping to be RFC-compliant and remove outdated FF hack | ||||
| - [#2710](https://github.com/sanic-org/sanic/pull/2710) Stricter charset handling and escaping of request URLs | ||||
| - [#2711](https://github.com/sanic-org/sanic/pull/2711) Consume body on ``DELETE`` by default | ||||
| - [#2719](https://github.com/sanic-org/sanic/pull/2719) Allow ``password`` to be passed to TLS context | ||||
| - [#2720](https://github.com/sanic-org/sanic/pull/2720) Skip middleware on ``RequestCancelled`` | ||||
| - [#2721](https://github.com/sanic-org/sanic/pull/2721) Change access logging format to ``%s`` | ||||
| - [#2722](https://github.com/sanic-org/sanic/pull/2722) Add ``CertLoader`` as application option for directly controlling ``SSLContext`` objects | ||||
| - [#2725](https://github.com/sanic-org/sanic/pull/2725) Worker sync state tolerance on race condition | ||||
|  | ||||
| ### Bugfixes | ||||
| - [#2651](https://github.com/sanic-org/sanic/pull/2651) ASGI websocket to pass thru bytes as is | ||||
| - [#2697](https://github.com/sanic-org/sanic/pull/2697) Fix comparison between datetime aware and naive in ``file`` when using ``If-Modified-Since`` | ||||
|  | ||||
| ### Deprecations and Removals | ||||
| - [#2666](https://github.com/sanic-org/sanic/pull/2666) Remove deprecated ``__blueprintname__`` property | ||||
|  | ||||
| ### Improved Documentation | ||||
| - [#2712](https://github.com/sanic-org/sanic/pull/2712) Improved example using ``'https'`` to create the redirect | ||||
							
								
								
									
										33
									
								
								docs/sanic/releases/23/23.6.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								docs/sanic/releases/23/23.6.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| ## Version 23.6.0  🔶 | ||||
|  | ||||
| ### Features | ||||
| - [#2670](https://github.com/sanic-org/sanic/pull/2670) Increase `KEEP_ALIVE_TIMEOUT` default to 120 seconds | ||||
| - [#2716](https://github.com/sanic-org/sanic/pull/2716) Adding allow route overwrite option in blueprint | ||||
| - [#2724](https://github.com/sanic-org/sanic/pull/2724) and [#2792](https://github.com/sanic-org/sanic/pull/2792) Add a new exception signal for ALL exceptions raised anywhere in application | ||||
| - [#2727](https://github.com/sanic-org/sanic/pull/2727) Add name prefixing to BP groups | ||||
| - [#2754](https://github.com/sanic-org/sanic/pull/2754) Update request type on middleware types | ||||
| - [#2770](https://github.com/sanic-org/sanic/pull/2770) Better exception message on startup time application induced import error | ||||
| - [#2776](https://github.com/sanic-org/sanic/pull/2776) Set multiprocessing start method early | ||||
| - [#2785](https://github.com/sanic-org/sanic/pull/2785) Add custom typing to config and ctx objects | ||||
| - [#2790](https://github.com/sanic-org/sanic/pull/2790) Add `request.client_ip` | ||||
|  | ||||
| ### Bugfixes | ||||
| - [#2728](https://github.com/sanic-org/sanic/pull/2728) Fix traversals for intended results | ||||
| - [#2729](https://github.com/sanic-org/sanic/pull/2729) Handle case when headers argument of ResponseStream constructor is None | ||||
| - [#2737](https://github.com/sanic-org/sanic/pull/2737) Fix type annotation for `JSONREsponse` default content type | ||||
| - [#2740](https://github.com/sanic-org/sanic/pull/2740) Use Sanic's serializer for JSON responses in the Inspector | ||||
| - [#2760](https://github.com/sanic-org/sanic/pull/2760) Support for `Request.get_current` in ASGI mode | ||||
| - [#2773](https://github.com/sanic-org/sanic/pull/2773) Alow Blueprint routes to explicitly define error_format | ||||
| - [#2774](https://github.com/sanic-org/sanic/pull/2774) Resolve headers on different renderers | ||||
| - [#2782](https://github.com/sanic-org/sanic/pull/2782) Resolve pypy compatibility issues | ||||
|  | ||||
| ### Deprecations and Removals | ||||
| - [#2777](https://github.com/sanic-org/sanic/pull/2777) Remove Python 3.7 support | ||||
|  | ||||
| ### Developer infrastructure | ||||
| - [#2766](https://github.com/sanic-org/sanic/pull/2766) Unpin setuptools version | ||||
| - [#2779](https://github.com/sanic-org/sanic/pull/2779) Run keep alive tests in loop to get available port | ||||
|  | ||||
| ### Improved Documentation | ||||
| - [#2741](https://github.com/sanic-org/sanic/pull/2741) Better documentation examples about running Sanic | ||||
| From that list, the items to highlight in the release notes: | ||||
| @@ -4,12 +4,14 @@ import asyncio | ||||
|  | ||||
| from sanic import Sanic | ||||
|  | ||||
| app = Sanic() | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| async def notify_server_started_after_five_seconds(): | ||||
|     await asyncio.sleep(5) | ||||
|     print('Server successfully started!') | ||||
|     print("Server successfully started!") | ||||
|  | ||||
|  | ||||
| app.add_task(notify_server_started_after_five_seconds()) | ||||
|  | ||||
|   | ||||
| @@ -1,30 +1,29 @@ | ||||
| from sanic import Sanic | ||||
| from sanic.response import text | ||||
| from random import randint | ||||
|  | ||||
| app = Sanic() | ||||
| from sanic import Sanic | ||||
| from sanic.response import text | ||||
|  | ||||
|  | ||||
| @app.middleware('request') | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.middleware("request") | ||||
| def append_request(request): | ||||
|     # Add new key with random value | ||||
|     request['num'] = randint(0, 100) | ||||
|     request.ctx.num = randint(0, 100) | ||||
|  | ||||
|  | ||||
| @app.get('/pop') | ||||
| @app.get("/pop") | ||||
| def pop_handler(request): | ||||
|     # Pop key from request object | ||||
|     num = request.pop('num') | ||||
|     return text(num) | ||||
|     return text(request.ctx.num) | ||||
|  | ||||
|  | ||||
| @app.get('/key_exist') | ||||
| @app.get("/key_exist") | ||||
| def key_exist_handler(request): | ||||
|     # Check the key is exist or not | ||||
|     if 'num' in request: | ||||
|         return text('num exist in request') | ||||
|     if hasattr(request.ctx, "num"): | ||||
|         return text("num exist in request") | ||||
|  | ||||
|     return text('num does not exist in reqeust') | ||||
|     return text("num does not exist in request") | ||||
|  | ||||
|  | ||||
| app.run(host="0.0.0.0", port=8000, debug=True) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| from sanic import Sanic | ||||
| from functools import wraps | ||||
|  | ||||
| from sanic import Sanic | ||||
| from sanic.response import json | ||||
|  | ||||
| app = Sanic() | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| def check_request_for_authorization_status(request): | ||||
| @@ -27,14 +29,16 @@ def authorized(f): | ||||
|             return response | ||||
|         else: | ||||
|             # the user is not authorized. | ||||
|             return json({'status': 'not_authorized'}, 403) | ||||
|             return json({"status": "not_authorized"}, 403) | ||||
|  | ||||
|     return decorated_function | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| @authorized | ||||
| async def test(request): | ||||
|     return json({'status': 'authorized'}) | ||||
|     return json({"status": "authorized"}) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -1,43 +1,54 @@ | ||||
| from sanic import Sanic, Blueprint | ||||
| from sanic import Blueprint, Sanic | ||||
| from sanic.response import text | ||||
| ''' | ||||
| Demonstrates that blueprint request middleware are executed in the order they  | ||||
|  | ||||
|  | ||||
| """ | ||||
| Demonstrates that blueprint request middleware are executed in the order they | ||||
| are added. And blueprint response middleware are executed in _reverse_ order. | ||||
| On a valid request, it should print "1 2 3 6 5 4" to terminal | ||||
| ''' | ||||
| """ | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
| bp = Blueprint("bp_"+__name__) | ||||
| bp = Blueprint("bp_example") | ||||
|  | ||||
| @bp.middleware('request') | ||||
|  | ||||
| @bp.on_request | ||||
| def request_middleware_1(request): | ||||
|     print('1') | ||||
|     print("1") | ||||
|  | ||||
| @bp.middleware('request') | ||||
|  | ||||
| @bp.on_request | ||||
| def request_middleware_2(request): | ||||
|     print('2') | ||||
|     print("2") | ||||
|  | ||||
| @bp.middleware('request') | ||||
|  | ||||
| @bp.on_request | ||||
| def request_middleware_3(request): | ||||
|     print('3') | ||||
|     print("3") | ||||
|  | ||||
| @bp.middleware('response') | ||||
|  | ||||
| @bp.on_response | ||||
| def resp_middleware_4(request, response): | ||||
|     print('4') | ||||
|     print("4") | ||||
|  | ||||
| @bp.middleware('response') | ||||
|  | ||||
| @bp.on_response | ||||
| def resp_middleware_5(request, response): | ||||
|     print('5') | ||||
|     print("5") | ||||
|  | ||||
| @bp.middleware('response') | ||||
|  | ||||
| @bp.on_response | ||||
| def resp_middleware_6(request, response): | ||||
|     print('6') | ||||
|     print("6") | ||||
|  | ||||
| @bp.route('/') | ||||
|  | ||||
| @bp.route("/") | ||||
| def pop_handler(request): | ||||
|     return text('hello world') | ||||
|     return text("hello world") | ||||
|  | ||||
| app.blueprint(bp, url_prefix='/bp') | ||||
|  | ||||
| app.run(host="0.0.0.0", port=8000, debug=True, auto_reload=False) | ||||
| app.blueprint(bp, url_prefix="/bp") | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True, auto_reload=False) | ||||
|   | ||||
| @@ -1,10 +1,11 @@ | ||||
| from sanic import Blueprint, Sanic | ||||
| from sanic.response import file, json | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| blueprint = Blueprint("name", url_prefix="/my_blueprint") | ||||
| blueprint2 = Blueprint("name2", url_prefix="/my_blueprint2") | ||||
| blueprint3 = Blueprint("name3", url_prefix="/my_blueprint3") | ||||
|  | ||||
| app = Sanic("Example") | ||||
| blueprint = Blueprint("bp_example", url_prefix="/my_blueprint") | ||||
| blueprint2 = Blueprint("bp_example2", url_prefix="/my_blueprint2") | ||||
| blueprint3 = Blueprint("bp_example3", url_prefix="/my_blueprint3") | ||||
|  | ||||
|  | ||||
| @blueprint.route("/foo") | ||||
| @@ -36,4 +37,5 @@ app.blueprint(blueprint) | ||||
| app.blueprint(blueprint2) | ||||
| app.blueprint(blueprint3) | ||||
|  | ||||
| app.run(host="0.0.0.0", port=9999, debug=True) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=9999, debug=True) | ||||
|   | ||||
| @@ -2,17 +2,21 @@ from asyncio import sleep | ||||
|  | ||||
| from sanic import Sanic, response | ||||
|  | ||||
| app = Sanic(__name__, strict_slashes=True) | ||||
|  | ||||
| app = Sanic("DelayedResponseApp", strict_slashes=True) | ||||
| app.config.AUTO_EXTEND = False | ||||
|  | ||||
|  | ||||
| @app.get("/") | ||||
| async def handler(request): | ||||
|     return response.redirect("/sleep/3") | ||||
|  | ||||
| @app.get("/sleep/<t:number>") | ||||
|  | ||||
| @app.get("/sleep/<t:float>") | ||||
| async def handler2(request, t=0.3): | ||||
|     await sleep(t) | ||||
|     return response.text(f"Slept {t:.1f} seconds.\n") | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -7,8 +7,10 @@ and pass in an instance of it when we create our Sanic instance. Inside this | ||||
| class' default handler, we can do anything including sending exceptions to | ||||
| an external service. | ||||
| """ | ||||
| from sanic.handlers import ErrorHandler | ||||
| from sanic.exceptions import SanicException | ||||
| from sanic.handlers import ErrorHandler | ||||
|  | ||||
|  | ||||
| """ | ||||
| Imports and code relevant for our CustomHandler class | ||||
| (Ordinarily this would be in a separate file) | ||||
| @@ -16,7 +18,6 @@ Imports and code relevant for our CustomHandler class | ||||
|  | ||||
|  | ||||
| class CustomHandler(ErrorHandler): | ||||
|  | ||||
|     def default(self, request, exception): | ||||
|         # Here, we have access to the exception object | ||||
|         # and can do anything with it (log, send to external service, etc) | ||||
| @@ -38,17 +39,17 @@ server's error_handler to an instance of our CustomHandler | ||||
|  | ||||
| from sanic import Sanic | ||||
|  | ||||
| app = Sanic(__name__) | ||||
|  | ||||
| handler = CustomHandler() | ||||
| app.error_handler = handler | ||||
| app = Sanic("Example", error_handler=handler) | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| async def test(request): | ||||
|     # Here, something occurs which causes an unexpected exception | ||||
|     # This exception will flow to our custom handler. | ||||
|     raise SanicException('You Broke It!') | ||||
|     raise SanicException("You Broke It!") | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
| from sanic import Sanic, response | ||||
| 
 | ||||
| app = Sanic(__name__) | ||||
| 
 | ||||
| app = Sanic("Example") | ||||
| 
 | ||||
| 
 | ||||
| @app.route("/") | ||||
| @@ -9,5 +9,5 @@ async def test(request): | ||||
|     return response.json({"test": True}) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
| @@ -1,4 +1,6 @@ | ||||
| from sanic import Sanic, response, text | ||||
| from sanic.handlers import ErrorHandler | ||||
| from sanic.server.async_server import AsyncioServer | ||||
|  | ||||
|  | ||||
| HTTP_PORT = 9999 | ||||
| @@ -27,25 +29,45 @@ def proxy(request, path): | ||||
|         path=path, | ||||
|         _server=https.config.SERVER_NAME, | ||||
|         _external=True, | ||||
|         _scheme="http", | ||||
|         _scheme="https", | ||||
|     ) | ||||
|     return response.redirect(url) | ||||
|  | ||||
|  | ||||
| @https.listener("main_process_start") | ||||
| @https.main_process_start | ||||
| async def start(app, _): | ||||
|     global http | ||||
|     app.http_server = await http.create_server( | ||||
|     http_server = await http.create_server( | ||||
|         port=HTTP_PORT, return_asyncio_server=True | ||||
|     ) | ||||
|     app.http_server.after_start() | ||||
|     app.add_task(runner(http, http_server)) | ||||
|     app.ctx.http_server = http_server | ||||
|     app.ctx.http = http | ||||
|  | ||||
|  | ||||
| @https.listener("main_process_stop") | ||||
| @https.main_process_stop | ||||
| async def stop(app, _): | ||||
|     app.http_server.before_stop() | ||||
|     await app.http_server.close() | ||||
|     app.http_server.after_stop() | ||||
|     await app.ctx.http_server.before_stop() | ||||
|     await app.ctx.http_server.close() | ||||
|     for connection in app.ctx.http_server.connections: | ||||
|         connection.close_if_idle() | ||||
|     await app.ctx.http_server.after_stop() | ||||
|     app.ctx.http = False | ||||
|  | ||||
|  | ||||
| https.run(port=HTTPS_PORT, debug=True) | ||||
| async def runner(app: Sanic, app_server: AsyncioServer): | ||||
|     app.is_running = True | ||||
|     try: | ||||
|         app.signalize() | ||||
|         app.finalize() | ||||
|         ErrorHandler.finalize(app.error_handler) | ||||
|         app_server.init = True | ||||
|  | ||||
|         await app_server.before_start() | ||||
|         await app_server.after_start() | ||||
|         await app_server.serve_forever() | ||||
|     finally: | ||||
|         app.is_running = False | ||||
|         app.is_stopping = True | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     https.run(port=HTTPS_PORT, debug=True) | ||||
|   | ||||
| @@ -1,26 +1,30 @@ | ||||
| import asyncio | ||||
|  | ||||
| import httpx | ||||
|  | ||||
| from sanic import Sanic | ||||
| from sanic.response import json | ||||
|  | ||||
| import asyncio | ||||
| import aiohttp | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
| sem = None | ||||
|  | ||||
|  | ||||
| @app.listener('before_server_start') | ||||
| def init(sanic, loop): | ||||
| @app.before_server_start | ||||
| def init(sanic, _): | ||||
|     global sem | ||||
|     concurrency_per_worker = 4 | ||||
|     sem = asyncio.Semaphore(concurrency_per_worker, loop=loop) | ||||
|     sem = asyncio.Semaphore(concurrency_per_worker) | ||||
|  | ||||
|  | ||||
| async def bounded_fetch(session, url): | ||||
|     """ | ||||
|     Use session object to perform 'get' request on url | ||||
|     """ | ||||
|     async with sem, session.get(url) as response: | ||||
|         return await response.json() | ||||
|     async with sem: | ||||
|         response = await session.get(url) | ||||
|         return response.json() | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| @@ -28,11 +32,12 @@ async def test(request): | ||||
|     """ | ||||
|     Download and serve example JSON | ||||
|     """ | ||||
|     url = "https://api.github.com/repos/channelcat/sanic" | ||||
|     url = "https://api.github.com/repos/sanic-org/sanic" | ||||
|  | ||||
|     async with aiohttp.ClientSession() as session: | ||||
|     async with httpx.AsyncClient() as session: | ||||
|         response = await bounded_fetch(session, url) | ||||
|         return json(response) | ||||
|  | ||||
|  | ||||
| app.run(host="0.0.0.0", port=8000, workers=2) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, workers=2) | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import logging | ||||
|  | ||||
| import aiotask_context as context | ||||
| from contextvars import ContextVar | ||||
|  | ||||
| from sanic import Sanic, response | ||||
|  | ||||
| @@ -11,8 +11,8 @@ log = logging.getLogger(__name__) | ||||
| class RequestIdFilter(logging.Filter): | ||||
|     def filter(self, record): | ||||
|         try: | ||||
|             record.request_id = context.get("X-Request-ID") | ||||
|         except ValueError: | ||||
|             record.request_id = app.ctx.request_id.get(None) or "n/a" | ||||
|         except AttributeError: | ||||
|             record.request_id = "n/a" | ||||
|         return True | ||||
|  | ||||
| @@ -44,13 +44,12 @@ LOG_SETTINGS = { | ||||
| } | ||||
|  | ||||
|  | ||||
| app = Sanic(__name__, log_config=LOG_SETTINGS) | ||||
| app = Sanic("Example", log_config=LOG_SETTINGS) | ||||
|  | ||||
|  | ||||
| @app.on_request | ||||
| async def set_request_id(request): | ||||
|     request_id = request.id | ||||
|     context.set("X-Request-ID", request_id) | ||||
|     request.app.ctx.request_id.set(request.id) | ||||
|     log.info(f"Setting {request.id=}") | ||||
|  | ||||
|  | ||||
| @@ -61,14 +60,14 @@ async def set_request_header(request, response): | ||||
|  | ||||
| @app.route("/") | ||||
| async def test(request): | ||||
|     log.debug("X-Request-ID: %s", context.get("X-Request-ID")) | ||||
|     log.debug("X-Request-ID: %s", request.id) | ||||
|     log.info("Hello from test!") | ||||
|     return response.json({"test": True}) | ||||
|  | ||||
|  | ||||
| @app.before_server_start | ||||
| def setup(app, loop): | ||||
|     loop.set_task_factory(context.task_factory) | ||||
|     app.ctx.request_id = ContextVar("request_id") | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import logging | ||||
| import socket | ||||
|  | ||||
| from os import getenv | ||||
| from platform import node | ||||
| from uuid import getnode as get_mac | ||||
| @@ -7,10 +8,11 @@ from uuid import getnode as get_mac | ||||
| from logdna import LogDNAHandler | ||||
|  | ||||
| from sanic import Sanic | ||||
| from sanic.response import json | ||||
| from sanic.request import Request | ||||
| from sanic.response import json | ||||
|  | ||||
| log = logging.getLogger('logdna') | ||||
|  | ||||
| log = logging.getLogger("logdna") | ||||
| log.setLevel(logging.INFO) | ||||
|  | ||||
|  | ||||
| @@ -30,16 +32,18 @@ logdna_options = { | ||||
|     "index_meta": True, | ||||
|     "hostname": node(), | ||||
|     "ip": get_my_ip_address(), | ||||
|     "mac": get_mac_address() | ||||
|     "mac": get_mac_address(), | ||||
| } | ||||
|  | ||||
| logdna_handler = LogDNAHandler(getenv("LOGDNA_API_KEY"), options=logdna_options) | ||||
| logdna_handler = LogDNAHandler( | ||||
|     getenv("LOGDNA_API_KEY"), options=logdna_options | ||||
| ) | ||||
|  | ||||
| logdna = logging.getLogger(__name__) | ||||
| logdna.setLevel(logging.INFO) | ||||
| logdna.addHandler(logdna_handler) | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.middleware | ||||
| @@ -49,13 +53,8 @@ def log_request(request: Request): | ||||
|  | ||||
| @app.route("/") | ||||
| def default(request): | ||||
|     return json({ | ||||
|         "response": "I was here" | ||||
|     }) | ||||
|     return json({"response": "I was here"}) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run( | ||||
|         host="0.0.0.0", | ||||
|         port=getenv("PORT", 8080) | ||||
|     ) | ||||
|     app.run(host="0.0.0.0", port=getenv("PORT", 8080)) | ||||
|   | ||||
| @@ -2,27 +2,29 @@ | ||||
| Modify header or status in response | ||||
| """ | ||||
|  | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| from sanic import Sanic, response | ||||
|  | ||||
|  | ||||
| @app.route('/') | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| def handle_request(request): | ||||
|     return response.json( | ||||
|         {'message': 'Hello world!'}, | ||||
|         headers={'X-Served-By': 'sanic'}, | ||||
|         status=200 | ||||
|         {"message": "Hello world!"}, | ||||
|         headers={"X-Served-By": "sanic"}, | ||||
|         status=200, | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @app.route('/unauthorized') | ||||
| @app.route("/unauthorized") | ||||
| def handle_request(request): | ||||
|     return response.json( | ||||
|         {'message': 'You are not authorized'}, | ||||
|         headers={'X-Served-By': 'sanic'}, | ||||
|         status=404 | ||||
|         {"message": "You are not authorized"}, | ||||
|         headers={"X-Served-By": "sanic"}, | ||||
|         status=404, | ||||
|     ) | ||||
|  | ||||
| app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|   | ||||
| @@ -20,4 +20,5 @@ def test(request): | ||||
|     return text("hey") | ||||
|  | ||||
|  | ||||
| app.run(host="0.0.0.0", port=8000) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -32,7 +32,7 @@ def test_port(worker_id): | ||||
|  | ||||
| @pytest.fixture(scope="session") | ||||
| def app(): | ||||
|     app = Sanic() | ||||
|     app = Sanic("Example") | ||||
|  | ||||
|     @app.route("/") | ||||
|     async def index(request): | ||||
|   | ||||
| @@ -8,7 +8,6 @@ from sanic.handlers import ErrorHandler | ||||
|  | ||||
|  | ||||
| class RaygunExceptionReporter(ErrorHandler): | ||||
|  | ||||
|     def __init__(self, raygun_api_key=None): | ||||
|         super().__init__() | ||||
|         if raygun_api_key is None: | ||||
| @@ -22,16 +21,13 @@ class RaygunExceptionReporter(ErrorHandler): | ||||
|  | ||||
|  | ||||
| raygun_error_reporter = RaygunExceptionReporter() | ||||
| app = Sanic(__name__, error_handler=raygun_error_reporter) | ||||
| app = Sanic("Example", error_handler=raygun_error_reporter) | ||||
|  | ||||
|  | ||||
| @app.route("/raise") | ||||
| async def test(request): | ||||
|     raise SanicException('You Broke It!') | ||||
|     raise SanicException("You Broke It!") | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     app.run( | ||||
|         host="0.0.0.0", | ||||
|         port=getenv("PORT", 8080) | ||||
|     ) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=getenv("PORT", 8080)) | ||||
|   | ||||
| @@ -1,18 +1,18 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
| from sanic import Sanic, response | ||||
|  | ||||
| app = Sanic(__name__) | ||||
|  | ||||
|      | ||||
| @app.route('/') | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| def handle_request(request): | ||||
|     return response.redirect('/redirect') | ||||
|     return response.redirect("/redirect") | ||||
|  | ||||
|  | ||||
| @app.route('/redirect') | ||||
| @app.route("/redirect") | ||||
| async def test(request): | ||||
|     return response.json({"Redirected": True}) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -6,5 +6,5 @@ data = "" | ||||
| for i in range(1, 250000): | ||||
|     data += str(i) | ||||
|  | ||||
| r = requests.post('http://0.0.0.0:8000/stream', data=data) | ||||
| r = requests.post("http://0.0.0.0:8000/stream", data=data) | ||||
| print(r.text) | ||||
|   | ||||
| @@ -1,65 +1,63 @@ | ||||
| from sanic import Sanic | ||||
| from sanic.views import CompositionView | ||||
| from sanic.views import HTTPMethodView | ||||
| from sanic.views import stream as stream_decorator | ||||
| from sanic.blueprints import Blueprint | ||||
| from sanic.response import stream, text | ||||
| from sanic.views import HTTPMethodView | ||||
| from sanic.views import stream as stream_decorator | ||||
|  | ||||
| bp = Blueprint('blueprint_request_stream') | ||||
| app = Sanic('request_stream') | ||||
|  | ||||
| bp = Blueprint("bp_example") | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| class SimpleView(HTTPMethodView): | ||||
|  | ||||
|     @stream_decorator | ||||
|     async def post(self, request): | ||||
|         result = '' | ||||
|         result = "" | ||||
|         while True: | ||||
|             body = await request.stream.get() | ||||
|             if body is None: | ||||
|                 break | ||||
|             result += body.decode('utf-8') | ||||
|             result += body.decode("utf-8") | ||||
|         return text(result) | ||||
|  | ||||
|  | ||||
| @app.post('/stream', stream=True) | ||||
| @app.post("/stream", stream=True) | ||||
| async def handler(request): | ||||
|     async def streaming(response): | ||||
|         while True: | ||||
|             body = await request.stream.get() | ||||
|             if body is None: | ||||
|                 break | ||||
|             body = body.decode('utf-8').replace('1', 'A') | ||||
|             body = body.decode("utf-8").replace("1", "A") | ||||
|             await response.write(body) | ||||
|  | ||||
|     return stream(streaming) | ||||
|  | ||||
|  | ||||
| @bp.put('/bp_stream', stream=True) | ||||
| @bp.put("/bp_stream", stream=True) | ||||
| async def bp_handler(request): | ||||
|     result = '' | ||||
|     result = "" | ||||
|     while True: | ||||
|         body = await request.stream.get() | ||||
|         if body is None: | ||||
|             break | ||||
|         result += body.decode('utf-8').replace('1', 'A') | ||||
|         result += body.decode("utf-8").replace("1", "A") | ||||
|     return text(result) | ||||
|  | ||||
|  | ||||
| async def post_handler(request): | ||||
|     result = '' | ||||
|     result = "" | ||||
|     while True: | ||||
|         body = await request.stream.get() | ||||
|         if body is None: | ||||
|             break | ||||
|         result += body.decode('utf-8') | ||||
|         result += body.decode("utf-8") | ||||
|     return text(result) | ||||
|  | ||||
|  | ||||
| app.blueprint(bp) | ||||
| app.add_route(SimpleView.as_view(), '/method_view') | ||||
| view = CompositionView() | ||||
| view.add(['POST'], post_handler, stream=True) | ||||
| app.add_route(view, '/composition_view') | ||||
| app.add_route(SimpleView.as_view(), "/method_view") | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     app.run(host='0.0.0.0', port=8000) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -1,21 +1,24 @@ | ||||
| import asyncio | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
|  | ||||
| from sanic import Sanic, response | ||||
| from sanic.config import Config | ||||
| from sanic.exceptions import RequestTimeout | ||||
|  | ||||
|  | ||||
| Config.REQUEST_TIMEOUT = 1 | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route('/') | ||||
| @app.route("/") | ||||
| async def test(request): | ||||
|     await asyncio.sleep(3) | ||||
|     return response.text('Hello, world!') | ||||
|     return response.text("Hello, world!") | ||||
|  | ||||
|  | ||||
| @app.exception(RequestTimeout) | ||||
| def timeout(request, exception): | ||||
|     return response.text('RequestTimeout from error_handler.', 408) | ||||
|     return response.text("RequestTimeout from error_handler.", 408) | ||||
|  | ||||
| app.run(host='0.0.0.0', port=8000) | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -1,21 +1,22 @@ | ||||
| from os import getenv | ||||
|  | ||||
| import rollbar | ||||
|  | ||||
| from sanic.handlers import ErrorHandler | ||||
| from sanic import Sanic | ||||
| from sanic.exceptions import SanicException | ||||
| from os import getenv | ||||
| from sanic.handlers import ErrorHandler | ||||
|  | ||||
|  | ||||
| rollbar.init(getenv("ROLLBAR_API_KEY")) | ||||
|  | ||||
|  | ||||
| class RollbarExceptionHandler(ErrorHandler): | ||||
|  | ||||
|     def default(self, request, exception): | ||||
|         rollbar.report_message(str(exception)) | ||||
|         return super().default(request, exception) | ||||
|  | ||||
|  | ||||
| app = Sanic(__name__, error_handler=RollbarExceptionHandler()) | ||||
| app = Sanic("Example", error_handler=RollbarExceptionHandler()) | ||||
|  | ||||
|  | ||||
| @app.route("/raise") | ||||
| @@ -24,7 +25,4 @@ def create_error(request): | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run( | ||||
|         host="0.0.0.0", | ||||
|         port=getenv("PORT", 8080) | ||||
|     ) | ||||
|     app.run(host="0.0.0.0", port=getenv("PORT", 8080)) | ||||
|   | ||||
| @@ -11,7 +11,7 @@ from pathlib import Path | ||||
| from sanic import Sanic, response | ||||
|  | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/text") | ||||
| @@ -59,31 +59,31 @@ async def handler_stream(request): | ||||
|     return response.stream(body) | ||||
|  | ||||
|  | ||||
| @app.listener("before_server_start") | ||||
| @app.before_server_start | ||||
| async def listener_before_server_start(*args, **kwargs): | ||||
|     print("before_server_start") | ||||
|  | ||||
|  | ||||
| @app.listener("after_server_start") | ||||
| @app.after_server_start | ||||
| async def listener_after_server_start(*args, **kwargs): | ||||
|     print("after_server_start") | ||||
|  | ||||
|  | ||||
| @app.listener("before_server_stop") | ||||
| @app.before_server_stop | ||||
| async def listener_before_server_stop(*args, **kwargs): | ||||
|     print("before_server_stop") | ||||
|  | ||||
|  | ||||
| @app.listener("after_server_stop") | ||||
| @app.after_server_stop | ||||
| async def listener_after_server_stop(*args, **kwargs): | ||||
|     print("after_server_stop") | ||||
|  | ||||
|  | ||||
| @app.middleware("request") | ||||
| @app.on_request | ||||
| async def print_on_request(request): | ||||
|     print("print_on_request") | ||||
|  | ||||
|  | ||||
| @app.middleware("response") | ||||
| @app.on_response | ||||
| async def print_on_response(request, response): | ||||
|     print("print_on_response") | ||||
|   | ||||
| @@ -1,22 +1,30 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
| from signal import signal, SIGINT | ||||
| import asyncio | ||||
|  | ||||
| import uvloop | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| from sanic import Sanic, response | ||||
|  | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| async def test(request): | ||||
|     return response.json({"answer": "42"}) | ||||
|  | ||||
| asyncio.set_event_loop(uvloop.new_event_loop()) | ||||
| server = app.create_server(host="0.0.0.0", port=8000, return_asyncio_server=True) | ||||
| loop = asyncio.get_event_loop() | ||||
| task = asyncio.ensure_future(server) | ||||
| signal(SIGINT, lambda s, f: loop.stop()) | ||||
| try: | ||||
|     loop.run_forever() | ||||
| except: | ||||
|     loop.stop() | ||||
|  | ||||
| async def main(): | ||||
|     server = await app.create_server( | ||||
|         port=8000, host="0.0.0.0", return_asyncio_server=True | ||||
|     ) | ||||
|  | ||||
|     if server is None: | ||||
|         return | ||||
|  | ||||
|     await server.startup() | ||||
|     await server.serve_forever() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     asyncio.set_event_loop(uvloop.new_event_loop()) | ||||
|     asyncio.run(main()) | ||||
|   | ||||
| @@ -1,38 +1,68 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
| from signal import signal, SIGINT | ||||
| import asyncio | ||||
|  | ||||
| from signal import SIGINT, signal | ||||
|  | ||||
| import uvloop | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| from sanic import Sanic, response | ||||
| from sanic.server import AsyncioServer | ||||
|  | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.before_server_start | ||||
| async def before_server_start(app, loop): | ||||
|     print("Async Server starting") | ||||
|  | ||||
|  | ||||
| @app.after_server_start | ||||
| async def after_server_start(app, loop): | ||||
|     print("Async Server started") | ||||
|  | ||||
|  | ||||
| @app.before_server_stop | ||||
| async def before_server_stop(app, loop): | ||||
|     print("Async Server stopping") | ||||
|  | ||||
|  | ||||
| @app.after_server_stop | ||||
| async def after_server_stop(app, loop): | ||||
|     print("Async Server stopped") | ||||
|  | ||||
| @app.listener('after_server_start') | ||||
| async def after_start_test(app, loop): | ||||
|     print("Async Server Started!") | ||||
|  | ||||
| @app.route("/") | ||||
| async def test(request): | ||||
|     return response.json({"answer": "42"}) | ||||
|  | ||||
| asyncio.set_event_loop(uvloop.new_event_loop()) | ||||
| serv_coro = app.create_server(host="0.0.0.0", port=8000, return_asyncio_server=True) | ||||
| loop = asyncio.get_event_loop() | ||||
| serv_task = asyncio.ensure_future(serv_coro, loop=loop) | ||||
| signal(SIGINT, lambda s, f: loop.stop()) | ||||
| server = loop.run_until_complete(serv_task) | ||||
| server.after_start() | ||||
| try: | ||||
|     loop.run_forever() | ||||
| except KeyboardInterrupt as e: | ||||
|     loop.stop() | ||||
| finally: | ||||
|     server.before_stop() | ||||
| if __name__ == "__main__": | ||||
|     asyncio.set_event_loop(uvloop.new_event_loop()) | ||||
|     serv_coro = app.create_server( | ||||
|         host="0.0.0.0", port=8000, return_asyncio_server=True | ||||
|     ) | ||||
|     loop = asyncio.get_event_loop() | ||||
|     serv_task = asyncio.ensure_future(serv_coro, loop=loop) | ||||
|     signal(SIGINT, lambda s, f: loop.stop()) | ||||
|     server: AsyncioServer = loop.run_until_complete(serv_task) | ||||
|     loop.run_until_complete(server.startup()) | ||||
|  | ||||
|     # Wait for server to close | ||||
|     close_task = server.close() | ||||
|     loop.run_until_complete(close_task) | ||||
|     # When using app.run(), this actually triggers before the serv_coro. | ||||
|     # But, in this example, we are using the convenience method, even if it is | ||||
|     # out of order. | ||||
|     loop.run_until_complete(server.before_start()) | ||||
|     loop.run_until_complete(server.after_start()) | ||||
|     try: | ||||
|         loop.run_forever() | ||||
|     except KeyboardInterrupt: | ||||
|         loop.stop() | ||||
|     finally: | ||||
|         loop.run_until_complete(server.before_stop()) | ||||
|  | ||||
|     # Complete all tasks on the loop | ||||
|     for connection in server.connections: | ||||
|         connection.close_if_idle() | ||||
|     server.after_stop() | ||||
|         # Wait for server to close | ||||
|         close_task = server.close() | ||||
|         loop.run_until_complete(close_task) | ||||
|  | ||||
|         # Complete all tasks on the loop | ||||
|         for connection in server.connections: | ||||
|             connection.close_if_idle() | ||||
|         loop.run_until_complete(server.after_stop()) | ||||
|   | ||||
| @@ -6,20 +6,19 @@ from sentry_sdk.integrations.sanic import SanicIntegration | ||||
| from sanic import Sanic | ||||
| from sanic.response import json | ||||
|  | ||||
|  | ||||
| sentry_init( | ||||
|     dsn=getenv("SENTRY_DSN"), | ||||
|     integrations=[SanicIntegration()], | ||||
| ) | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| # noinspection PyUnusedLocal | ||||
| @app.route("/working") | ||||
| async def working_path(request): | ||||
|     return json({ | ||||
|         "response": "Working API Response" | ||||
|     }) | ||||
|     return json({"response": "Working API Response"}) | ||||
|  | ||||
|  | ||||
| # noinspection PyUnusedLocal | ||||
| @@ -28,8 +27,5 @@ async def raise_error(request): | ||||
|     raise Exception("Testing Sentry Integration") | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     app.run( | ||||
|         host="0.0.0.0", | ||||
|         port=getenv("PORT", 8080) | ||||
|     ) | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=getenv("PORT", 8080)) | ||||
|   | ||||
| @@ -1,42 +1,41 @@ | ||||
| from sanic import Sanic | ||||
| from sanic.views import HTTPMethodView | ||||
| from sanic.response import text | ||||
| from sanic.views import HTTPMethodView | ||||
|  | ||||
| app = Sanic('some_name') | ||||
|  | ||||
| app = Sanic("some_name") | ||||
|  | ||||
|  | ||||
| class SimpleView(HTTPMethodView): | ||||
|  | ||||
|     def get(self, request): | ||||
|         return text('I am get method') | ||||
|         return text("I am get method") | ||||
|  | ||||
|     def post(self, request): | ||||
|         return text('I am post method') | ||||
|         return text("I am post method") | ||||
|  | ||||
|     def put(self, request): | ||||
|         return text('I am put method') | ||||
|         return text("I am put method") | ||||
|  | ||||
|     def patch(self, request): | ||||
|         return text('I am patch method') | ||||
|         return text("I am patch method") | ||||
|  | ||||
|     def delete(self, request): | ||||
|         return text('I am delete method') | ||||
|         return text("I am delete method") | ||||
|  | ||||
|  | ||||
| class SimpleAsyncView(HTTPMethodView): | ||||
|  | ||||
|     async def get(self, request): | ||||
|         return text('I am async get method') | ||||
|         return text("I am async get method") | ||||
|  | ||||
|     async def post(self, request): | ||||
|         return text('I am async post method') | ||||
|         return text("I am async post method") | ||||
|  | ||||
|     async def put(self, request): | ||||
|         return text('I am async put method') | ||||
|         return text("I am async put method") | ||||
|  | ||||
|  | ||||
| app.add_route(SimpleView.as_view(), '/') | ||||
| app.add_route(SimpleAsyncView.as_view(), '/async') | ||||
| app.add_route(SimpleView.as_view(), "/") | ||||
| app.add_route(SimpleAsyncView.as_view(), "/async") | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| from sanic import Sanic | ||||
|  | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
|  | ||||
| app.static("/", "./static") | ||||
|   | ||||
| @@ -1,13 +1,14 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response as res | ||||
|  | ||||
| app = Sanic(__name__) | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| async def test(req): | ||||
|     return res.text("I\'m a teapot", status=418) | ||||
|     return res.text("I'm a teapot", status=418) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| import os | ||||
|  | ||||
| from sanic import Sanic | ||||
| from sanic.log import logger as log | ||||
| from sanic import response | ||||
| from sanic import Sanic, response | ||||
| from sanic.exceptions import ServerError | ||||
| from sanic.log import logger as log | ||||
|  | ||||
| app = Sanic(__name__) | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| @@ -13,7 +13,7 @@ async def test_async(request): | ||||
|     return response.json({"test": True}) | ||||
|  | ||||
|  | ||||
| @app.route("/sync", methods=['GET', 'POST']) | ||||
| @app.route("/sync", methods=["GET", "POST"]) | ||||
| def test_sync(request): | ||||
|     return response.json({"test": True}) | ||||
|  | ||||
| @@ -31,6 +31,7 @@ def exception(request): | ||||
| @app.route("/await") | ||||
| async def test_await(request): | ||||
|     import asyncio | ||||
|  | ||||
|     await asyncio.sleep(5) | ||||
|     return response.text("I'm feeling sleepy") | ||||
|  | ||||
| @@ -42,8 +43,10 @@ async def test_file(request): | ||||
|  | ||||
| @app.route("/file_stream") | ||||
| async def test_file_stream(request): | ||||
|     return await response.file_stream(os.path.abspath("setup.py"), | ||||
|                                       chunk_size=1024) | ||||
|     return await response.file_stream( | ||||
|         os.path.abspath("setup.py"), chunk_size=1024 | ||||
|     ) | ||||
|  | ||||
|  | ||||
| # ----------------------------------------------- # | ||||
| # Exceptions | ||||
| @@ -52,14 +55,17 @@ async def test_file_stream(request): | ||||
|  | ||||
| @app.exception(ServerError) | ||||
| async def test(request, exception): | ||||
|     return response.json({"exception": "{}".format(exception), "status": exception.status_code}, | ||||
|                          status=exception.status_code) | ||||
|     return response.json( | ||||
|         {"exception": str(exception), "status": exception.status_code}, | ||||
|         status=exception.status_code, | ||||
|     ) | ||||
|  | ||||
|  | ||||
| # ----------------------------------------------- # | ||||
| # Read from request | ||||
| # ----------------------------------------------- # | ||||
|  | ||||
|  | ||||
| @app.route("/json") | ||||
| def post_json(request): | ||||
|     return response.json({"received": True, "message": request.json}) | ||||
| @@ -67,38 +73,51 @@ def post_json(request): | ||||
|  | ||||
| @app.route("/form") | ||||
| def post_form_json(request): | ||||
|     return response.json({"received": True, "form_data": request.form, "test": request.form.get('test')}) | ||||
|     return response.json( | ||||
|         { | ||||
|             "received": True, | ||||
|             "form_data": request.form, | ||||
|             "test": request.form.get("test"), | ||||
|         } | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @app.route("/query_string") | ||||
| def query_string(request): | ||||
|     return response.json({"parsed": True, "args": request.args, "url": request.url, | ||||
|                           "query_string": request.query_string}) | ||||
|     return response.json( | ||||
|         { | ||||
|             "parsed": True, | ||||
|             "args": request.args, | ||||
|             "url": request.url, | ||||
|             "query_string": request.query_string, | ||||
|         } | ||||
|     ) | ||||
|  | ||||
|  | ||||
| # ----------------------------------------------- # | ||||
| # Run Server | ||||
| # ----------------------------------------------- # | ||||
|  | ||||
| @app.listener('before_server_start') | ||||
|  | ||||
| @app.before_server_start | ||||
| def before_start(app, loop): | ||||
|     log.info("SERVER STARTING") | ||||
|  | ||||
|  | ||||
| @app.listener('after_server_start') | ||||
| @app.after_server_start | ||||
| def after_start(app, loop): | ||||
|     log.info("OH OH OH OH OHHHHHHHH") | ||||
|  | ||||
|  | ||||
| @app.listener('before_server_stop') | ||||
| @app.before_server_stop | ||||
| def before_stop(app, loop): | ||||
|     log.info("SERVER STOPPING") | ||||
|  | ||||
|  | ||||
| @app.listener('after_server_stop') | ||||
| @app.after_server_stop | ||||
| def after_stop(app, loop): | ||||
|     log.info("TRIED EVERYTHING") | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|   | ||||
| @@ -1,23 +1,13 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
| import socket | ||||
| import os | ||||
| from sanic import Sanic, response | ||||
|  | ||||
| app = Sanic(__name__) | ||||
|  | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/test") | ||||
| async def test(request): | ||||
|     return response.text("OK") | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     server_address = './uds_socket' | ||||
|     # Make sure the socket does not already exist | ||||
|     try: | ||||
|             os.unlink(server_address) | ||||
|     except OSError: | ||||
|             if os.path.exists(server_address): | ||||
|                         raise | ||||
|     sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) | ||||
|     sock.bind(server_address) | ||||
|     app.run(sock=sock) | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(unix="./uds_socket") | ||||
|   | ||||
| @@ -1,20 +1,21 @@ | ||||
| from sanic import Sanic | ||||
| from sanic import response | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| from sanic import Sanic, response | ||||
|  | ||||
|  | ||||
| @app.route('/') | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| async def index(request): | ||||
|     # generate a URL for the endpoint `post_handler` | ||||
|     url = app.url_for('post_handler', post_id=5) | ||||
|     url = app.url_for("post_handler", post_id=5) | ||||
|     # the URL is `/posts/5`, redirect to it | ||||
|     return response.redirect(url) | ||||
|  | ||||
|  | ||||
| @app.route('/posts/<post_id>') | ||||
| @app.route("/posts/<post_id>") | ||||
| async def post_handler(request, post_id): | ||||
|     return response.text('Post - {}'.format(post_id)) | ||||
|      | ||||
| if __name__ == '__main__': | ||||
|     return response.text("Post - {}".format(post_id)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|   | ||||
| @@ -8,7 +8,9 @@ app = Sanic(name="blue-print-group-version-example") | ||||
| bp1 = Blueprint(name="ultron", url_prefix="/ultron") | ||||
| bp2 = Blueprint(name="vision", url_prefix="/vision", strict_slashes=None) | ||||
|  | ||||
| bpg = Blueprint.group([bp1, bp2], url_prefix="/sentient/robot", version=1, strict_slashes=True) | ||||
| bpg = Blueprint.group( | ||||
|     bp1, bp2, url_prefix="/sentient/robot", version=1, strict_slashes=True | ||||
| ) | ||||
|  | ||||
|  | ||||
| @bp1.get("/name") | ||||
| @@ -31,5 +33,5 @@ async def bp2_revised_name(request): | ||||
|  | ||||
| app.blueprint(bpg) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000) | ||||
|   | ||||
| @@ -8,7 +8,7 @@ from sanic.blueprints import Blueprint | ||||
| # curl -H "Host: bp.example.com" localhost:8000/question | ||||
| # curl -H "Host: bp.example.com" localhost:8000/answer | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| app = Sanic("Example") | ||||
| bp = Blueprint("bp", host="bp.example.com") | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,24 +1,27 @@ | ||||
| from sanic import Sanic | ||||
| from sanic.response import file | ||||
|  | ||||
| app = Sanic(__name__) | ||||
| from sanic.response import redirect | ||||
|  | ||||
|  | ||||
| @app.route('/') | ||||
| async def index(request): | ||||
|     return await file('websocket.html') | ||||
| app = Sanic("Example") | ||||
|  | ||||
|  | ||||
| @app.websocket('/feed') | ||||
| app.static("index.html", "websocket.html") | ||||
|  | ||||
|  | ||||
| @app.route("/") | ||||
| def index(request): | ||||
|     return redirect("index.html") | ||||
|  | ||||
|  | ||||
| @app.websocket("/feed") | ||||
| async def feed(request, ws): | ||||
|     while True: | ||||
|         data = 'hello!' | ||||
|         print('Sending: ' + data) | ||||
|         data = "hello!" | ||||
|         print("Sending: " + data) | ||||
|         await ws.send(data) | ||||
|         data = await ws.recv() | ||||
|         print('Received: ' + data) | ||||
|         print("Received: " + data) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     app.run(host="0.0.0.0", port=8000, debug=True) | ||||
|  | ||||
|   | ||||
							
								
								
									
										29
									
								
								pyproject.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								pyproject.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| [build-system] | ||||
| requires = ["setuptools", "wheel"] | ||||
| build-backend = "setuptools.build_meta" | ||||
|  | ||||
| [tool.black] | ||||
| line-length = 79 | ||||
|  | ||||
| [tool.isort] | ||||
| atomic = true | ||||
| default_section = "THIRDPARTY" | ||||
| include_trailing_comma = true | ||||
| known_first_party = "sanic" | ||||
| known_third_party = "pytest" | ||||
| line_length = 79 | ||||
| lines_after_imports = 2 | ||||
| lines_between_types = 1 | ||||
| multi_line_output = 3 | ||||
| profile = "black" | ||||
|  | ||||
| [[tool.mypy.overrides]] | ||||
| module = [ | ||||
|     "httptools.*", | ||||
|     "trustme.*", | ||||
|     "sanic_routing.*", | ||||
|     "aioquic.*", | ||||
|     "html5tagger.*", | ||||
|     "tracerite.*", | ||||
| ] | ||||
| ignore_missing_imports = true | ||||
| @@ -6,4 +6,4 @@ python: | ||||
|         path: . | ||||
|         extra_requirements: | ||||
|             - docs | ||||
|    system_packages: true | ||||
|    system_packages: true | ||||
|   | ||||
| @@ -1,17 +1,86 @@ | ||||
| from types import SimpleNamespace | ||||
|  | ||||
| from typing_extensions import TypeAlias | ||||
|  | ||||
| from sanic.__version__ import __version__ | ||||
| from sanic.app import Sanic | ||||
| from sanic.blueprints import Blueprint | ||||
| from sanic.config import Config | ||||
| from sanic.constants import HTTPMethod | ||||
| from sanic.exceptions import ( | ||||
|     BadRequest, | ||||
|     ExpectationFailed, | ||||
|     FileNotFound, | ||||
|     Forbidden, | ||||
|     HeaderNotFound, | ||||
|     InternalServerError, | ||||
|     InvalidHeader, | ||||
|     MethodNotAllowed, | ||||
|     NotFound, | ||||
|     RangeNotSatisfiable, | ||||
|     SanicException, | ||||
|     ServerError, | ||||
|     ServiceUnavailable, | ||||
|     Unauthorized, | ||||
| ) | ||||
| from sanic.request import Request | ||||
| from sanic.response import HTTPResponse, html, json, text | ||||
| from sanic.response import ( | ||||
|     HTTPResponse, | ||||
|     empty, | ||||
|     file, | ||||
|     html, | ||||
|     json, | ||||
|     raw, | ||||
|     redirect, | ||||
|     text, | ||||
| ) | ||||
| from sanic.server.websockets.impl import WebsocketImplProtocol as Websocket | ||||
|  | ||||
|  | ||||
| DefaultSanic: TypeAlias = "Sanic[Config, SimpleNamespace]" | ||||
| """ | ||||
| A type alias for a Sanic app with a default config and namespace. | ||||
| """ | ||||
|  | ||||
| DefaultRequest: TypeAlias = Request[DefaultSanic, SimpleNamespace] | ||||
| """ | ||||
| A type alias for a request with a default Sanic app and namespace. | ||||
| """ | ||||
|  | ||||
| __all__ = ( | ||||
|     "__version__", | ||||
|     # Common objects | ||||
|     "Sanic", | ||||
|     "Config", | ||||
|     "Blueprint", | ||||
|     "HTTPMethod", | ||||
|     "HTTPResponse", | ||||
|     "Request", | ||||
|     "Websocket", | ||||
|     # Common types | ||||
|     "DefaultSanic", | ||||
|     "DefaultRequest", | ||||
|     # Common exceptions | ||||
|     "BadRequest", | ||||
|     "ExpectationFailed", | ||||
|     "FileNotFound", | ||||
|     "Forbidden", | ||||
|     "HeaderNotFound", | ||||
|     "InternalServerError", | ||||
|     "InvalidHeader", | ||||
|     "MethodNotAllowed", | ||||
|     "NotFound", | ||||
|     "RangeNotSatisfiable", | ||||
|     "SanicException", | ||||
|     "ServerError", | ||||
|     "ServiceUnavailable", | ||||
|     "Unauthorized", | ||||
|     # Common response methods | ||||
|     "empty", | ||||
|     "file", | ||||
|     "html", | ||||
|     "json", | ||||
|     "raw", | ||||
|     "redirect", | ||||
|     "text", | ||||
| ) | ||||
|   | ||||
| @@ -1,131 +1,15 @@ | ||||
| import os | ||||
| import sys | ||||
|  | ||||
| from argparse import ArgumentParser, RawDescriptionHelpFormatter | ||||
| from importlib import import_module | ||||
| from typing import Any, Dict, Optional | ||||
|  | ||||
| from sanic import __version__ | ||||
| from sanic.app import Sanic | ||||
| from sanic.config import BASE_LOGO | ||||
| from sanic.log import logger | ||||
| from sanic.cli.app import SanicCLI | ||||
| from sanic.compat import OS_IS_WINDOWS, enable_windows_color_support | ||||
|  | ||||
|  | ||||
| class SanicArgumentParser(ArgumentParser): | ||||
|     def add_bool_arguments(self, *args, **kwargs): | ||||
|         group = self.add_mutually_exclusive_group() | ||||
|         group.add_argument(*args, action="store_true", **kwargs) | ||||
|         kwargs["help"] = "no " + kwargs["help"] | ||||
|         group.add_argument( | ||||
|             "--no-" + args[0][2:], *args[1:], action="store_false", **kwargs | ||||
|         ) | ||||
| if OS_IS_WINDOWS: | ||||
|     enable_windows_color_support() | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     parser = SanicArgumentParser( | ||||
|         prog="sanic", | ||||
|         description=BASE_LOGO, | ||||
|         formatter_class=RawDescriptionHelpFormatter, | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "-H", | ||||
|         "--host", | ||||
|         dest="host", | ||||
|         type=str, | ||||
|         default="127.0.0.1", | ||||
|         help="host address [default 127.0.0.1]", | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "-p", | ||||
|         "--port", | ||||
|         dest="port", | ||||
|         type=int, | ||||
|         default=8000, | ||||
|         help="port to serve on [default 8000]", | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "-u", | ||||
|         "--unix", | ||||
|         dest="unix", | ||||
|         type=str, | ||||
|         default="", | ||||
|         help="location of unix socket", | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "--cert", dest="cert", type=str, help="location of certificate for SSL" | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "--key", dest="key", type=str, help="location of keyfile for SSL." | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "-w", | ||||
|         "--workers", | ||||
|         dest="workers", | ||||
|         type=int, | ||||
|         default=1, | ||||
|         help="number of worker processes [default 1]", | ||||
|     ) | ||||
|     parser.add_argument("--debug", dest="debug", action="store_true") | ||||
|     parser.add_bool_arguments( | ||||
|         "--access-logs", dest="access_log", help="display access logs" | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "-v", | ||||
|         "--version", | ||||
|         action="version", | ||||
|         version=f"Sanic {__version__}", | ||||
|     ) | ||||
|     parser.add_argument( | ||||
|         "module", help="path to your Sanic app. Example: path.to.server:app" | ||||
|     ) | ||||
|     args = parser.parse_args() | ||||
|  | ||||
|     try: | ||||
|         module_path = os.path.abspath(os.getcwd()) | ||||
|         if module_path not in sys.path: | ||||
|             sys.path.append(module_path) | ||||
|  | ||||
|         if ":" in args.module: | ||||
|             module_name, app_name = args.module.rsplit(":", 1) | ||||
|         else: | ||||
|             module_parts = args.module.split(".") | ||||
|             module_name = ".".join(module_parts[:-1]) | ||||
|             app_name = module_parts[-1] | ||||
|  | ||||
|         module = import_module(module_name) | ||||
|         app = getattr(module, app_name, None) | ||||
|         app_name = type(app).__name__ | ||||
|  | ||||
|         if not isinstance(app, Sanic): | ||||
|             raise ValueError( | ||||
|                 f"Module is not a Sanic app, it is a {app_name}.  " | ||||
|                 f"Perhaps you meant {args.module}.app?" | ||||
|             ) | ||||
|         if args.cert is not None or args.key is not None: | ||||
|             ssl = { | ||||
|                 "cert": args.cert, | ||||
|                 "key": args.key, | ||||
|             }  # type: Optional[Dict[str, Any]] | ||||
|         else: | ||||
|             ssl = None | ||||
|  | ||||
|         app.run( | ||||
|             host=args.host, | ||||
|             port=args.port, | ||||
|             unix=args.unix, | ||||
|             workers=args.workers, | ||||
|             debug=args.debug, | ||||
|             access_log=args.access_log, | ||||
|             ssl=ssl, | ||||
|         ) | ||||
|     except ImportError as e: | ||||
|         logger.error( | ||||
|             f"No module named {e.name} found.\n" | ||||
|             f"  Example File: project/sanic_server.py -> app\n" | ||||
|             f"  Example Module: project.sanic_server.app" | ||||
|         ) | ||||
|     except ValueError: | ||||
|         logger.exception("Failed to run app") | ||||
| def main(args=None): | ||||
|     cli = SanicCLI() | ||||
|     cli.attach() | ||||
|     cli.run(args) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| __version__ = "21.3.1" | ||||
| __version__ = "23.6.0" | ||||
|   | ||||
							
								
								
									
										1672
									
								
								sanic/app.py
									
									
									
									
									
								
							
							
						
						
									
										1672
									
								
								sanic/app.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										0
									
								
								sanic/application/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								sanic/application/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										32
									
								
								sanic/application/constants.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								sanic/application/constants.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| from enum import Enum, IntEnum, auto | ||||
|  | ||||
|  | ||||
| class StrEnum(str, Enum):  # no cov | ||||
|     def _generate_next_value_(name: str, *args) -> str:  # type: ignore | ||||
|         return name.lower() | ||||
|  | ||||
|     def __eq__(self, value: object) -> bool: | ||||
|         value = str(value).upper() | ||||
|         return super().__eq__(value) | ||||
|  | ||||
|     def __hash__(self) -> int: | ||||
|         return hash(self.value) | ||||
|  | ||||
|     def __str__(self) -> str: | ||||
|         return self.value | ||||
|  | ||||
|  | ||||
| class Server(StrEnum): | ||||
|     SANIC = auto() | ||||
|     ASGI = auto() | ||||
|  | ||||
|  | ||||
| class Mode(StrEnum): | ||||
|     PRODUCTION = auto() | ||||
|     DEBUG = auto() | ||||
|  | ||||
|  | ||||
| class ServerStage(IntEnum): | ||||
|     STOPPED = auto() | ||||
|     PARTIAL = auto() | ||||
|     SERVING = auto() | ||||
							
								
								
									
										34
									
								
								sanic/application/ext.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								sanic/application/ext.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| from contextlib import suppress | ||||
| from importlib import import_module | ||||
| from typing import TYPE_CHECKING | ||||
|  | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from sanic import Sanic | ||||
|  | ||||
|  | ||||
| def setup_ext(app: Sanic, *, fail: bool = False, **kwargs): | ||||
|     if not app.config.AUTO_EXTEND: | ||||
|         return | ||||
|  | ||||
|     sanic_ext = None | ||||
|     with suppress(ModuleNotFoundError): | ||||
|         sanic_ext = import_module("sanic_ext") | ||||
|  | ||||
|     if not sanic_ext:  # no cov | ||||
|         if fail: | ||||
|             raise RuntimeError( | ||||
|                 "Sanic Extensions is not installed. You can add it to your " | ||||
|                 "environment using:\n$ pip install sanic[ext]\nor\n$ pip " | ||||
|                 "install sanic-ext" | ||||
|             ) | ||||
|  | ||||
|         return | ||||
|  | ||||
|     if not getattr(app, "_ext", None): | ||||
|         Ext = getattr(sanic_ext, "Extend") | ||||
|         app._ext = Ext(app, **kwargs) | ||||
|  | ||||
|         return app.ext | ||||
							
								
								
									
										61
									
								
								sanic/application/logo.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								sanic/application/logo.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| import re | ||||
| import sys | ||||
|  | ||||
| from os import environ | ||||
|  | ||||
| from sanic.helpers import is_atty | ||||
|  | ||||
|  | ||||
| BASE_LOGO = """ | ||||
|  | ||||
|                  Sanic | ||||
|          Build Fast. Run Fast. | ||||
|  | ||||
| """ | ||||
| COFFEE_LOGO = """\033[48;2;255;13;104m                     \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m     ▄████████▄      \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m    ██       ██▀▀▄   \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m    ███████████  █   \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m    ███████████▄▄▀   \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m     ▀███████▀       \033[0m | ||||
| \033[48;2;255;13;104m                     \033[0m | ||||
| Dark roast. No sugar.""" | ||||
|  | ||||
| COLOR_LOGO = """\033[48;2;255;13;104m                     \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m    ▄███ █████ ██    \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m   ██                \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m    ▀███████ ███▄    \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m                ██   \033[0m | ||||
| \033[38;2;255;255;255;48;2;255;13;104m   ████ ████████▀    \033[0m | ||||
| \033[48;2;255;13;104m                     \033[0m | ||||
| Build Fast. Run Fast.""" | ||||
|  | ||||
| FULL_COLOR_LOGO = """ | ||||
|  | ||||
| \033[38;2;255;13;104m  ▄███ █████ ██ \033[0m     ▄█▄      ██       █   █   ▄██████████ | ||||
| \033[38;2;255;13;104m ██             \033[0m    █   █     █ ██     █   █  ██ | ||||
| \033[38;2;255;13;104m  ▀███████ ███▄ \033[0m   ▀     █    █   ██   ▄   █  ██ | ||||
| \033[38;2;255;13;104m              ██\033[0m  █████████   █     ██ █   █  ▄▄ | ||||
| \033[38;2;255;13;104m ████ ████████▀ \033[0m █         █  █       ██   █   ▀██ ███████ | ||||
|  | ||||
| """  # noqa | ||||
|  | ||||
| SVG_LOGO_SIMPLE = """<svg id=logo-simple viewBox="0 0 964 279"><desc>Sanic</desc><path d="M107 222c9-2 10-20 1-22s-20-2-30-2-17 7-16 14 6 10 15 10h30zm115-1c16-2 30-11 35-23s6-24 2-33-6-14-15-20-24-11-38-10c-7 3-10 13-5 19s17-1 24 4 15 14 13 24-5 15-14 18-50 0-74 0h-17c-6 4-10 15-4 20s16 2 23 3zM251 83q9-1 9-7 0-15-10-16h-13c-10 6-10 20 0 22zM147 60c-4 0-10 3-11 11s5 13 10 12 42 0 67 0c5-3 7-10 6-15s-4-8-9-8zm-33 1c-8 0-16 0-24 3s-20 10-25 20-6 24-4 36 15 22 26 27 78 8 94 3c4-4 4-12 0-18s-69 8-93-10c-8-7-9-23 0-30s12-10 20-10 12 2 16-3 1-15-5-18z" fill="#ff0d68"/><path d="M676 74c0-14-18-9-20 0s0 30 0 39 20 9 20 2zm-297-10c-12 2-15 12-23 23l-41 58H340l22-30c8-12 23-13 30-4s20 24 24 38-10 10-17 10l-68 2q-17 1-48 30c-7 6-10 20 0 24s15-8 20-13 20 -20 58-21h50 c20 2 33 9 52 30 8 10 24-4 16-13L384 65q-3-2-5-1zm131 0c-10 1-12 12-11 20v96c1 10-3 23 5 32s20-5 17-15c0-23-3-46 2-67 5-12 22-14 32-5l103 87c7 5 19 1 18-9v-64c-3-10-20-9-21 2s-20 22-30 13l-97-80c-5-4-10-10-18-10zM701 76v128c2 10 15 12 20 4s0-102 0-124s-20-18-20-7z M850 63c-35 0-69-2-86 15s-20 60-13 66 13 8 16 0 1-10 1-27 12-26 20-32 66-5 85-5 31 4 31-10-18-7-54-7M764 159c-6-2-15-2-16 12s19 37 33 43 23 8 25-4-4-11-11-14q-9-3-22-18c-4-7-3-16-10-19zM828 196c-4 0-8 1-10 5s-4 12 0 15 8 2 12 2h60c5 0 10-2 12-6 3-7-1-16-8-16z" fill="#1f1f1f"/></svg>"""  # noqa | ||||
|  | ||||
| ansi_pattern = re.compile(r"\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])") | ||||
|  | ||||
|  | ||||
| def get_logo(full=False, coffee=False): | ||||
|     logo = ( | ||||
|         (FULL_COLOR_LOGO if full else (COFFEE_LOGO if coffee else COLOR_LOGO)) | ||||
|         if is_atty() | ||||
|         else BASE_LOGO | ||||
|     ) | ||||
|  | ||||
|     if ( | ||||
|         sys.platform == "darwin" | ||||
|         and environ.get("TERM_PROGRAM") == "Apple_Terminal" | ||||
|     ): | ||||
|         logo = ansi_pattern.sub("", logo) | ||||
|  | ||||
|     return logo | ||||
							
								
								
									
										145
									
								
								sanic/application/motd.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								sanic/application/motd.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| from abc import ABC, abstractmethod | ||||
| from shutil import get_terminal_size | ||||
| from textwrap import indent, wrap | ||||
| from typing import Dict, Optional | ||||
|  | ||||
| from sanic import __version__ | ||||
| from sanic.helpers import is_atty | ||||
| from sanic.log import logger | ||||
|  | ||||
|  | ||||
| class MOTD(ABC): | ||||
|     def __init__( | ||||
|         self, | ||||
|         logo: Optional[str], | ||||
|         serve_location: str, | ||||
|         data: Dict[str, str], | ||||
|         extra: Dict[str, str], | ||||
|     ) -> None: | ||||
|         self.logo = logo | ||||
|         self.serve_location = serve_location | ||||
|         self.data = data | ||||
|         self.extra = extra | ||||
|         self.key_width = 0 | ||||
|         self.value_width = 0 | ||||
|  | ||||
|     @abstractmethod | ||||
|     def display(self): | ||||
|         ...  # noqa | ||||
|  | ||||
|     @classmethod | ||||
|     def output( | ||||
|         cls, | ||||
|         logo: Optional[str], | ||||
|         serve_location: str, | ||||
|         data: Dict[str, str], | ||||
|         extra: Dict[str, str], | ||||
|     ) -> None: | ||||
|         motd_class = MOTDTTY if is_atty() else MOTDBasic | ||||
|         motd_class(logo, serve_location, data, extra).display() | ||||
|  | ||||
|  | ||||
| class MOTDBasic(MOTD): | ||||
|     def display(self): | ||||
|         if self.logo: | ||||
|             logger.debug(self.logo) | ||||
|         lines = [f"Sanic v{__version__}"] | ||||
|         if self.serve_location: | ||||
|             lines.append(f"Goin' Fast @ {self.serve_location}") | ||||
|         lines += [ | ||||
|             *(f"{key}: {value}" for key, value in self.data.items()), | ||||
|             *(f"{key}: {value}" for key, value in self.extra.items()), | ||||
|         ] | ||||
|         for line in lines: | ||||
|             logger.info(line) | ||||
|  | ||||
|  | ||||
| class MOTDTTY(MOTD): | ||||
|     def __init__(self, *args, **kwargs) -> None: | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.set_variables() | ||||
|  | ||||
|     def set_variables(self):  # no  cov | ||||
|         fallback = (108, 24) | ||||
|         terminal_width = max( | ||||
|             get_terminal_size(fallback=fallback).columns, fallback[0] | ||||
|         ) | ||||
|         self.max_value_width = terminal_width - fallback[0] + 36 | ||||
|  | ||||
|         self.key_width = 4 | ||||
|         self.value_width = self.max_value_width | ||||
|         if self.data: | ||||
|             self.key_width = max(map(len, self.data.keys())) | ||||
|             self.value_width = min( | ||||
|                 max(map(len, self.data.values())), self.max_value_width | ||||
|             ) | ||||
|         self.logo_lines = self.logo.split("\n") if self.logo else [] | ||||
|         self.logo_line_length = 24 | ||||
|         self.centering_length = ( | ||||
|             self.key_width + self.value_width + 2 + self.logo_line_length | ||||
|         ) | ||||
|         self.display_length = self.key_width + self.value_width + 2 | ||||
|  | ||||
|     def display(self, version=True, action="Goin' Fast", out=None): | ||||
|         if not out: | ||||
|             out = logger.info | ||||
|         header = "Sanic" | ||||
|         if version: | ||||
|             header += f" v{__version__}" | ||||
|         header = header.center(self.centering_length) | ||||
|         running = ( | ||||
|             f"{action} @ {self.serve_location}" if self.serve_location else "" | ||||
|         ).center(self.centering_length) | ||||
|         length = len(header) + 2 - self.logo_line_length | ||||
|         first_filler = "─" * (self.logo_line_length - 1) | ||||
|         second_filler = "─" * length | ||||
|         display_filler = "─" * (self.display_length + 2) | ||||
|         lines = [ | ||||
|             f"\n┌{first_filler}─{second_filler}┐", | ||||
|             f"│ {header} │", | ||||
|             f"│ {running} │", | ||||
|             f"├{first_filler}┬{second_filler}┤", | ||||
|         ] | ||||
|  | ||||
|         self._render_data(lines, self.data, 0) | ||||
|         if self.extra: | ||||
|             logo_part = self._get_logo_part(len(lines) - 4) | ||||
|             lines.append(f"| {logo_part} ├{display_filler}┤") | ||||
|             self._render_data(lines, self.extra, len(lines) - 4) | ||||
|  | ||||
|         self._render_fill(lines) | ||||
|  | ||||
|         lines.append(f"└{first_filler}┴{second_filler}┘\n") | ||||
|         out(indent("\n".join(lines), "  ")) | ||||
|  | ||||
|     def _render_data(self, lines, data, start): | ||||
|         offset = 0 | ||||
|         for idx, (key, value) in enumerate(data.items(), start=start): | ||||
|             key = key.rjust(self.key_width) | ||||
|  | ||||
|             wrapped = wrap(value, self.max_value_width, break_on_hyphens=False) | ||||
|             for wrap_index, part in enumerate(wrapped): | ||||
|                 part = part.ljust(self.value_width) | ||||
|                 logo_part = self._get_logo_part(idx + offset + wrap_index) | ||||
|                 display = ( | ||||
|                     f"{key}: {part}" | ||||
|                     if wrap_index == 0 | ||||
|                     else (" " * len(key) + f"  {part}") | ||||
|                 ) | ||||
|                 lines.append(f"│ {logo_part} │ {display} │") | ||||
|                 if wrap_index: | ||||
|                     offset += 1 | ||||
|  | ||||
|     def _render_fill(self, lines): | ||||
|         filler = " " * self.display_length | ||||
|         idx = len(lines) - 5 | ||||
|         for i in range(1, len(self.logo_lines) - idx): | ||||
|             logo_part = self.logo_lines[idx + i] | ||||
|             lines.append(f"│ {logo_part} │ {filler} │") | ||||
|  | ||||
|     def _get_logo_part(self, idx): | ||||
|         try: | ||||
|             logo_part = self.logo_lines[idx] | ||||
|         except IndexError: | ||||
|             logo_part = " " * (self.logo_line_length - 3) | ||||
|         return logo_part | ||||
							
								
								
									
										86
									
								
								sanic/application/spinner.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								sanic/application/spinner.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
|  | ||||
| from contextlib import contextmanager | ||||
| from queue import Queue | ||||
| from threading import Thread | ||||
|  | ||||
|  | ||||
| if os.name == "nt":  # noqa | ||||
|     import ctypes  # noqa | ||||
|  | ||||
|     class _CursorInfo(ctypes.Structure): | ||||
|         _fields_ = [("size", ctypes.c_int), ("visible", ctypes.c_byte)] | ||||
|  | ||||
|  | ||||
| class Spinner:  # noqa | ||||
|     def __init__(self, message: str) -> None: | ||||
|         self.message = message | ||||
|         self.queue: Queue[int] = Queue() | ||||
|         self.spinner = self.cursor() | ||||
|         self.thread = Thread(target=self.run) | ||||
|  | ||||
|     def start(self): | ||||
|         self.queue.put(1) | ||||
|         self.thread.start() | ||||
|         self.hide() | ||||
|  | ||||
|     def run(self): | ||||
|         while self.queue.get(): | ||||
|             output = f"\r{self.message} [{next(self.spinner)}]" | ||||
|             sys.stdout.write(output) | ||||
|             sys.stdout.flush() | ||||
|             time.sleep(0.1) | ||||
|             self.queue.put(1) | ||||
|  | ||||
|     def stop(self): | ||||
|         self.queue.put(0) | ||||
|         self.thread.join() | ||||
|         self.show() | ||||
|  | ||||
|     @staticmethod | ||||
|     def cursor(): | ||||
|         while True: | ||||
|             for cursor in "|/-\\": | ||||
|                 yield cursor | ||||
|  | ||||
|     @staticmethod | ||||
|     def hide(): | ||||
|         if os.name == "nt": | ||||
|             ci = _CursorInfo() | ||||
|             handle = ctypes.windll.kernel32.GetStdHandle(-11) | ||||
|             ctypes.windll.kernel32.GetConsoleCursorInfo( | ||||
|                 handle, ctypes.byref(ci) | ||||
|             ) | ||||
|             ci.visible = False | ||||
|             ctypes.windll.kernel32.SetConsoleCursorInfo( | ||||
|                 handle, ctypes.byref(ci) | ||||
|             ) | ||||
|         elif os.name == "posix": | ||||
|             sys.stdout.write("\033[?25l") | ||||
|             sys.stdout.flush() | ||||
|  | ||||
|     @staticmethod | ||||
|     def show(): | ||||
|         if os.name == "nt": | ||||
|             ci = _CursorInfo() | ||||
|             handle = ctypes.windll.kernel32.GetStdHandle(-11) | ||||
|             ctypes.windll.kernel32.GetConsoleCursorInfo( | ||||
|                 handle, ctypes.byref(ci) | ||||
|             ) | ||||
|             ci.visible = True | ||||
|             ctypes.windll.kernel32.SetConsoleCursorInfo( | ||||
|                 handle, ctypes.byref(ci) | ||||
|             ) | ||||
|         elif os.name == "posix": | ||||
|             sys.stdout.write("\033[?25h") | ||||
|             sys.stdout.flush() | ||||
|  | ||||
|  | ||||
| @contextmanager | ||||
| def loading(message: str = "Loading"):  # noqa | ||||
|     spinner = Spinner(message) | ||||
|     spinner.start() | ||||
|     yield | ||||
|     spinner.stop() | ||||
							
								
								
									
										91
									
								
								sanic/application/state.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								sanic/application/state.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| import logging | ||||
|  | ||||
| from dataclasses import dataclass, field | ||||
| from pathlib import Path | ||||
| from socket import socket | ||||
| from ssl import SSLContext | ||||
| from typing import TYPE_CHECKING, Any, Dict, List, Optional, Set, Union | ||||
|  | ||||
| from sanic.application.constants import Mode, Server, ServerStage | ||||
| from sanic.log import VerbosityFilter, logger | ||||
| from sanic.server.async_server import AsyncioServer | ||||
|  | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from sanic import Sanic | ||||
|  | ||||
|  | ||||
| @dataclass | ||||
| class ApplicationServerInfo: | ||||
|     settings: Dict[str, Any] | ||||
|     stage: ServerStage = field(default=ServerStage.STOPPED) | ||||
|     server: Optional[AsyncioServer] = field(default=None) | ||||
|  | ||||
|  | ||||
| @dataclass | ||||
| class ApplicationState: | ||||
|     app: Sanic | ||||
|     asgi: bool = field(default=False) | ||||
|     coffee: bool = field(default=False) | ||||
|     fast: bool = field(default=False) | ||||
|     host: str = field(default="") | ||||
|     port: int = field(default=0) | ||||
|     ssl: Optional[SSLContext] = field(default=None) | ||||
|     sock: Optional[socket] = field(default=None) | ||||
|     unix: Optional[str] = field(default=None) | ||||
|     mode: Mode = field(default=Mode.PRODUCTION) | ||||
|     reload_dirs: Set[Path] = field(default_factory=set) | ||||
|     auto_reload: bool = field(default=False) | ||||
|     server: Server = field(default=Server.SANIC) | ||||
|     is_running: bool = field(default=False) | ||||
|     is_started: bool = field(default=False) | ||||
|     is_stopping: bool = field(default=False) | ||||
|     verbosity: int = field(default=0) | ||||
|     workers: int = field(default=0) | ||||
|     primary: bool = field(default=True) | ||||
|     server_info: List[ApplicationServerInfo] = field(default_factory=list) | ||||
|  | ||||
|     # This property relates to the ApplicationState instance and should | ||||
|     # not be changed except in the __post_init__ method | ||||
|     _init: bool = field(default=False) | ||||
|  | ||||
|     def __post_init__(self) -> None: | ||||
|         self._init = True | ||||
|  | ||||
|     def __setattr__(self, name: str, value: Any) -> None: | ||||
|         if self._init and name == "_init": | ||||
|             raise RuntimeError( | ||||
|                 "Cannot change the value of _init after instantiation" | ||||
|             ) | ||||
|         super().__setattr__(name, value) | ||||
|         if self._init and hasattr(self, f"set_{name}"): | ||||
|             getattr(self, f"set_{name}")(value) | ||||
|  | ||||
|     def set_mode(self, value: Union[str, Mode]): | ||||
|         if hasattr(self.app, "error_handler"): | ||||
|             self.app.error_handler.debug = self.app.debug | ||||
|         if getattr(self.app, "configure_logging", False) and self.app.debug: | ||||
|             logger.setLevel(logging.DEBUG) | ||||
|  | ||||
|     def set_verbosity(self, value: int): | ||||
|         VerbosityFilter.verbosity = value | ||||
|  | ||||
|     @property | ||||
|     def is_debug(self): | ||||
|         return self.mode is Mode.DEBUG | ||||
|  | ||||
|     @property | ||||
|     def stage(self) -> ServerStage: | ||||
|         if not self.server_info: | ||||
|             return ServerStage.STOPPED | ||||
|  | ||||
|         if all(info.stage is ServerStage.SERVING for info in self.server_info): | ||||
|             return ServerStage.SERVING | ||||
|         elif any( | ||||
|             info.stage is ServerStage.SERVING for info in self.server_info | ||||
|         ): | ||||
|             return ServerStage.PARTIAL | ||||
|  | ||||
|         return ServerStage.STOPPED | ||||
							
								
								
									
										245
									
								
								sanic/asgi.py
									
									
									
									
									
								
							
							
						
						
									
										245
									
								
								sanic/asgi.py
									
									
									
									
									
								
							| @@ -1,36 +1,49 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| import warnings | ||||
|  | ||||
| from inspect import isawaitable | ||||
| from typing import Optional | ||||
| from urllib.parse import quote | ||||
|  | ||||
| import sanic.app  # noqa | ||||
| from typing import TYPE_CHECKING, Optional | ||||
|  | ||||
| from sanic.compat import Header | ||||
| from sanic.exceptions import ServerError | ||||
| from sanic.exceptions import BadRequest, ServerError | ||||
| from sanic.helpers import Default | ||||
| from sanic.http import Stage | ||||
| from sanic.log import error_logger, logger | ||||
| from sanic.models.asgi import ASGIReceive, ASGIScope, ASGISend, MockTransport | ||||
| from sanic.request import Request | ||||
| from sanic.response import BaseHTTPResponse | ||||
| from sanic.server import ConnInfo | ||||
| from sanic.websocket import WebSocketConnection | ||||
| from sanic.server.websockets.connection import WebSocketConnection | ||||
|  | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from sanic import Sanic | ||||
|  | ||||
|  | ||||
| class Lifespan: | ||||
|     def __init__(self, asgi_app: "ASGIApp") -> None: | ||||
|         self.asgi_app = asgi_app | ||||
|     def __init__( | ||||
|         self, sanic_app, scope: ASGIScope, receive: ASGIReceive, send: ASGISend | ||||
|     ) -> None: | ||||
|         self.sanic_app = sanic_app | ||||
|         self.scope = scope | ||||
|         self.receive = receive | ||||
|         self.send = send | ||||
|  | ||||
|         if "before_server_start" in self.asgi_app.sanic_app.listeners: | ||||
|             warnings.warn( | ||||
|         if "server.init.before" in self.sanic_app.signal_router.name_index: | ||||
|             logger.debug( | ||||
|                 'You have set a listener for "before_server_start" ' | ||||
|                 "in ASGI mode. " | ||||
|                 "It will be executed as early as possible, but not before " | ||||
|                 "the ASGI server is started." | ||||
|                 "the ASGI server is started.", | ||||
|                 extra={"verbosity": 1}, | ||||
|             ) | ||||
|         if "after_server_stop" in self.asgi_app.sanic_app.listeners: | ||||
|             warnings.warn( | ||||
|         if "server.shutdown.after" in self.sanic_app.signal_router.name_index: | ||||
|             logger.debug( | ||||
|                 'You have set a listener for "after_server_stop" ' | ||||
|                 "in ASGI mode. " | ||||
|                 "It will be executed as late as possible, but not after " | ||||
|                 "the ASGI server is stopped." | ||||
|                 "the ASGI server is stopped.", | ||||
|                 extra={"verbosity": 1}, | ||||
|             ) | ||||
|  | ||||
|     async def startup(self) -> None: | ||||
| @@ -42,19 +55,16 @@ class Lifespan: | ||||
|         in sequence since the ASGI lifespan protocol only supports a single | ||||
|         startup event. | ||||
|         """ | ||||
|         self.asgi_app.sanic_app.router.finalize() | ||||
|         if self.asgi_app.sanic_app.signal_router.routes: | ||||
|             self.asgi_app.sanic_app.signal_router.finalize() | ||||
|         listeners = self.asgi_app.sanic_app.listeners.get( | ||||
|             "before_server_start", [] | ||||
|         ) + self.asgi_app.sanic_app.listeners.get("after_server_start", []) | ||||
|         await self.sanic_app._startup() | ||||
|         await self.sanic_app._server_event("init", "before") | ||||
|         await self.sanic_app._server_event("init", "after") | ||||
|  | ||||
|         for handler in listeners: | ||||
|             response = handler( | ||||
|                 self.asgi_app.sanic_app, self.asgi_app.sanic_app.loop | ||||
|         if not isinstance(self.sanic_app.config.USE_UVLOOP, Default): | ||||
|             warnings.warn( | ||||
|                 "You have set the USE_UVLOOP configuration option, but Sanic " | ||||
|                 "cannot control the event loop when running in ASGI mode." | ||||
|                 "This option will be ignored." | ||||
|             ) | ||||
|             if response and isawaitable(response): | ||||
|                 await response | ||||
|  | ||||
|     async def shutdown(self) -> None: | ||||
|         """ | ||||
| @@ -65,97 +75,117 @@ class Lifespan: | ||||
|         in sequence since the ASGI lifespan protocol only supports a single | ||||
|         shutdown event. | ||||
|         """ | ||||
|         listeners = self.asgi_app.sanic_app.listeners.get( | ||||
|             "before_server_stop", [] | ||||
|         ) + self.asgi_app.sanic_app.listeners.get("after_server_stop", []) | ||||
|         await self.sanic_app._server_event("shutdown", "before") | ||||
|         await self.sanic_app._server_event("shutdown", "after") | ||||
|  | ||||
|         for handler in listeners: | ||||
|             response = handler( | ||||
|                 self.asgi_app.sanic_app, self.asgi_app.sanic_app.loop | ||||
|             ) | ||||
|             if response and isawaitable(response): | ||||
|                 await response | ||||
|  | ||||
|     async def __call__( | ||||
|         self, scope: ASGIScope, receive: ASGIReceive, send: ASGISend | ||||
|     ) -> None: | ||||
|         message = await receive() | ||||
|         if message["type"] == "lifespan.startup": | ||||
|             await self.startup() | ||||
|             await send({"type": "lifespan.startup.complete"}) | ||||
|  | ||||
|         message = await receive() | ||||
|         if message["type"] == "lifespan.shutdown": | ||||
|             await self.shutdown() | ||||
|             await send({"type": "lifespan.shutdown.complete"}) | ||||
|     async def __call__(self) -> None: | ||||
|         while True: | ||||
|             message = await self.receive() | ||||
|             if message["type"] == "lifespan.startup": | ||||
|                 try: | ||||
|                     await self.startup() | ||||
|                 except Exception as e: | ||||
|                     error_logger.exception(e) | ||||
|                     await self.send( | ||||
|                         {"type": "lifespan.startup.failed", "message": str(e)} | ||||
|                     ) | ||||
|                 else: | ||||
|                     await self.send({"type": "lifespan.startup.complete"}) | ||||
|             elif message["type"] == "lifespan.shutdown": | ||||
|                 try: | ||||
|                     await self.shutdown() | ||||
|                 except Exception as e: | ||||
|                     error_logger.exception(e) | ||||
|                     await self.send( | ||||
|                         {"type": "lifespan.shutdown.failed", "message": str(e)} | ||||
|                     ) | ||||
|                 else: | ||||
|                     await self.send({"type": "lifespan.shutdown.complete"}) | ||||
|                 return | ||||
|  | ||||
|  | ||||
| class ASGIApp: | ||||
|     sanic_app: "sanic.app.Sanic" | ||||
|     sanic_app: Sanic | ||||
|     request: Request | ||||
|     transport: MockTransport | ||||
|     lifespan: Lifespan | ||||
|     ws: Optional[WebSocketConnection] | ||||
|  | ||||
|     def __init__(self) -> None: | ||||
|         self.ws = None | ||||
|     stage: Stage | ||||
|     response: Optional[BaseHTTPResponse] | ||||
|  | ||||
|     @classmethod | ||||
|     async def create( | ||||
|         cls, sanic_app, scope: ASGIScope, receive: ASGIReceive, send: ASGISend | ||||
|     ) -> "ASGIApp": | ||||
|         cls, | ||||
|         sanic_app: Sanic, | ||||
|         scope: ASGIScope, | ||||
|         receive: ASGIReceive, | ||||
|         send: ASGISend, | ||||
|     ) -> ASGIApp: | ||||
|         instance = cls() | ||||
|         instance.ws = None | ||||
|         instance.sanic_app = sanic_app | ||||
|         instance.transport = MockTransport(scope, receive, send) | ||||
|         instance.transport.loop = sanic_app.loop | ||||
|         instance.stage = Stage.IDLE | ||||
|         instance.response = None | ||||
|         instance.sanic_app.state.is_started = True | ||||
|         setattr(instance.transport, "add_task", sanic_app.loop.create_task) | ||||
|  | ||||
|         headers = Header( | ||||
|             [ | ||||
|                 (key.decode("latin-1"), value.decode("latin-1")) | ||||
|                 for key, value in scope.get("headers", []) | ||||
|             ] | ||||
|         ) | ||||
|         instance.lifespan = Lifespan(instance) | ||||
|         try: | ||||
|             headers = Header( | ||||
|                 [ | ||||
|                     ( | ||||
|                         key.decode("ASCII"), | ||||
|                         value.decode(errors="surrogateescape"), | ||||
|                     ) | ||||
|                     for key, value in scope.get("headers", []) | ||||
|                 ] | ||||
|             ) | ||||
|         except UnicodeDecodeError: | ||||
|             raise BadRequest( | ||||
|                 "Header names can only contain US-ASCII characters" | ||||
|             ) | ||||
|  | ||||
|         if scope["type"] == "lifespan": | ||||
|             await instance.lifespan(scope, receive, send) | ||||
|         if scope["type"] == "http": | ||||
|             version = scope["http_version"] | ||||
|             method = scope["method"] | ||||
|         elif scope["type"] == "websocket": | ||||
|             version = "1.1" | ||||
|             method = "GET" | ||||
|  | ||||
|             instance.ws = instance.transport.create_websocket_connection( | ||||
|                 send, receive | ||||
|             ) | ||||
|         else: | ||||
|             path = ( | ||||
|                 scope["path"][1:] | ||||
|                 if scope["path"].startswith("/") | ||||
|                 else scope["path"] | ||||
|             ) | ||||
|             url = "/".join([scope.get("root_path", ""), quote(path)]) | ||||
|             url_bytes = url.encode("latin-1") | ||||
|             url_bytes += b"?" + scope["query_string"] | ||||
|             raise ServerError("Received unknown ASGI scope") | ||||
|  | ||||
|             if scope["type"] == "http": | ||||
|                 version = scope["http_version"] | ||||
|                 method = scope["method"] | ||||
|             elif scope["type"] == "websocket": | ||||
|                 version = "1.1" | ||||
|                 method = "GET" | ||||
|         url_bytes, query = scope["raw_path"], scope["query_string"] | ||||
|         if query: | ||||
|             # httpx ASGI client sends query string as part of raw_path | ||||
|             url_bytes = url_bytes.split(b"?", 1)[0] | ||||
|             # All servers send them separately | ||||
|             url_bytes = b"%b?%b" % (url_bytes, query) | ||||
|  | ||||
|                 instance.ws = instance.transport.create_websocket_connection( | ||||
|                     send, receive | ||||
|                 ) | ||||
|                 await instance.ws.accept() | ||||
|             else: | ||||
|                 raise ServerError("Received unknown ASGI scope") | ||||
|         request_class = sanic_app.request_class or Request | ||||
|         instance.request = request_class( | ||||
|             url_bytes, | ||||
|             headers, | ||||
|             version, | ||||
|             method, | ||||
|             instance.transport, | ||||
|             sanic_app, | ||||
|         ) | ||||
|         request_class._current.set(instance.request) | ||||
|         instance.request.stream = instance  # type: ignore | ||||
|         instance.request_body = True | ||||
|         instance.request.conn_info = ConnInfo(instance.transport) | ||||
|  | ||||
|             request_class = sanic_app.request_class or Request | ||||
|             instance.request = request_class( | ||||
|                 url_bytes, | ||||
|                 headers, | ||||
|                 version, | ||||
|                 method, | ||||
|                 instance.transport, | ||||
|                 sanic_app, | ||||
|             ) | ||||
|             instance.request.stream = instance | ||||
|             instance.request_body = True | ||||
|             instance.request.conn_info = ConnInfo(instance.transport) | ||||
|         await instance.sanic_app.dispatch( | ||||
|             "http.lifecycle.request", | ||||
|             inline=True, | ||||
|             context={"request": instance.request}, | ||||
|             fail_not_found=False, | ||||
|         ) | ||||
|  | ||||
|         return instance | ||||
|  | ||||
| @@ -163,11 +193,15 @@ class ASGIApp: | ||||
|         """ | ||||
|         Read and stream the body in chunks from an incoming ASGI message. | ||||
|         """ | ||||
|         if self.stage is Stage.IDLE: | ||||
|             self.stage = Stage.REQUEST | ||||
|         message = await self.transport.receive() | ||||
|         body = message.get("body", b"") | ||||
|         if not message.get("more_body", False): | ||||
|             self.request_body = False | ||||
|             return None | ||||
|         return message.get("body", b"") | ||||
|             if not body: | ||||
|                 return None | ||||
|         return body | ||||
|  | ||||
|     async def __aiter__(self): | ||||
|         while self.request_body: | ||||
| @@ -175,11 +209,17 @@ class ASGIApp: | ||||
|             if data: | ||||
|                 yield data | ||||
|  | ||||
|     def respond(self, response): | ||||
|     def respond(self, response: BaseHTTPResponse): | ||||
|         if self.stage is not Stage.HANDLER: | ||||
|             self.stage = Stage.FAILED | ||||
|             raise RuntimeError("Response already started") | ||||
|         if self.response is not None: | ||||
|             self.response.stream = None | ||||
|         response.stream, self.response = self, response | ||||
|         return response | ||||
|  | ||||
|     async def send(self, data, end_stream): | ||||
|         self.stage = Stage.IDLE if end_stream else Stage.RESPONSE | ||||
|         if self.response: | ||||
|             response, self.response = self.response, None | ||||
|             await self.transport.send( | ||||
| @@ -206,4 +246,11 @@ class ASGIApp: | ||||
|         """ | ||||
|         Handle the incoming request. | ||||
|         """ | ||||
|         await self.sanic_app.handle_request(self.request) | ||||
|         try: | ||||
|             self.stage = Stage.HANDLER | ||||
|             await self.sanic_app.handle_request(self.request) | ||||
|         except Exception as e: | ||||
|             try: | ||||
|                 await self.sanic_app.handle_exception(self.request, e) | ||||
|             except Exception as exc: | ||||
|                 await self.sanic_app.handle_exception(self.request, exc, False) | ||||
|   | ||||
| @@ -1,41 +0,0 @@ | ||||
| from typing import Any, Tuple | ||||
| from warnings import warn | ||||
|  | ||||
| from sanic.mixins.exceptions import ExceptionMixin | ||||
| from sanic.mixins.listeners import ListenerMixin | ||||
| from sanic.mixins.middleware import MiddlewareMixin | ||||
| from sanic.mixins.routes import RouteMixin | ||||
| from sanic.mixins.signals import SignalMixin | ||||
|  | ||||
|  | ||||
| class BaseSanic( | ||||
|     RouteMixin, | ||||
|     MiddlewareMixin, | ||||
|     ListenerMixin, | ||||
|     ExceptionMixin, | ||||
|     SignalMixin, | ||||
| ): | ||||
|     __fake_slots__: Tuple[str, ...] | ||||
|  | ||||
|     def __init__(self, *args, **kwargs) -> None: | ||||
|         for base in BaseSanic.__bases__: | ||||
|             base.__init__(self, *args, **kwargs)  # type: ignore | ||||
|  | ||||
|     def __str__(self) -> str: | ||||
|         return f"<{self.__class__.__name__} {self.name}>" | ||||
|  | ||||
|     def __repr__(self) -> str: | ||||
|         return f'{self.__class__.__name__}(name="{self.name}")' | ||||
|  | ||||
|     def __setattr__(self, name: str, value: Any) -> None: | ||||
|         # This is a temporary compat layer so we can raise a warning until | ||||
|         # setting attributes on the app instance can be removed and deprecated | ||||
|         # with a proper implementation of __slots__ | ||||
|         if name not in self.__fake_slots__: | ||||
|             warn( | ||||
|                 f"Setting variables on {self.__class__.__name__} instances is " | ||||
|                 "deprecated and will be removed in version 21.9. You should " | ||||
|                 f"change your {self.__class__.__name__} instance to use " | ||||
|                 f"instance.ctx.{name} instead." | ||||
|             ) | ||||
|         super().__setattr__(name, value) | ||||
							
								
								
									
										0
									
								
								sanic/base/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								sanic/base/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										6
									
								
								sanic/base/meta.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								sanic/base/meta.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| class SanicMeta(type): | ||||
|     @classmethod | ||||
|     def __prepare__(metaclass, name, bases, **kwds): | ||||
|         cls = super().__prepare__(metaclass, name, bases, **kwds) | ||||
|         cls["__slots__"] = () | ||||
|         return cls | ||||
							
								
								
									
										67
									
								
								sanic/base/root.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								sanic/base/root.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| import re | ||||
|  | ||||
| from typing import Any, Optional | ||||
|  | ||||
| from sanic.base.meta import SanicMeta | ||||
| from sanic.exceptions import SanicException | ||||
| from sanic.mixins.exceptions import ExceptionMixin | ||||
| from sanic.mixins.listeners import ListenerMixin | ||||
| from sanic.mixins.middleware import MiddlewareMixin | ||||
| from sanic.mixins.routes import RouteMixin | ||||
| from sanic.mixins.signals import SignalMixin | ||||
| from sanic.mixins.static import StaticMixin | ||||
|  | ||||
|  | ||||
| VALID_NAME = re.compile(r"^[a-zA-Z_][a-zA-Z0-9_\-]*$") | ||||
|  | ||||
|  | ||||
| class BaseSanic( | ||||
|     RouteMixin, | ||||
|     StaticMixin, | ||||
|     MiddlewareMixin, | ||||
|     ListenerMixin, | ||||
|     ExceptionMixin, | ||||
|     SignalMixin, | ||||
|     metaclass=SanicMeta, | ||||
| ): | ||||
|     __slots__ = ("name",) | ||||
|  | ||||
|     def __init__( | ||||
|         self, name: Optional[str] = None, *args: Any, **kwargs: Any | ||||
|     ) -> None: | ||||
|         class_name = self.__class__.__name__ | ||||
|  | ||||
|         if name is None: | ||||
|             raise SanicException( | ||||
|                 f"{class_name} instance cannot be unnamed. " | ||||
|                 "Please use Sanic(name='your_application_name') instead.", | ||||
|             ) | ||||
|  | ||||
|         if not VALID_NAME.match(name): | ||||
|             raise SanicException( | ||||
|                 f"{class_name} instance named '{name}' uses an invalid " | ||||
|                 "format. Names must begin with a character and may only " | ||||
|                 "contain alphanumeric characters, _, or -." | ||||
|             ) | ||||
|  | ||||
|         self.name = name | ||||
|  | ||||
|         for base in BaseSanic.__bases__: | ||||
|             base.__init__(self, *args, **kwargs)  # type: ignore | ||||
|  | ||||
|     def __str__(self) -> str: | ||||
|         return f"<{self.__class__.__name__} {self.name}>" | ||||
|  | ||||
|     def __repr__(self) -> str: | ||||
|         return f'{self.__class__.__name__}(name="{self.name}")' | ||||
|  | ||||
|     def __setattr__(self, name: str, value: Any) -> None: | ||||
|         try: | ||||
|             super().__setattr__(name, value) | ||||
|         except AttributeError as e: | ||||
|             raise AttributeError( | ||||
|                 f"Setting variables on {self.__class__.__name__} instances is " | ||||
|                 "not allowed. You should change your " | ||||
|                 f"{self.__class__.__name__} instance to use " | ||||
|                 f"instance.ctx.{name} instead.", | ||||
|             ) from e | ||||
| @@ -1,7 +1,12 @@ | ||||
| from collections.abc import MutableSequence | ||||
| from typing import List, Optional, Union | ||||
| from __future__ import annotations | ||||
|  | ||||
| import sanic | ||||
| from collections.abc import MutableSequence | ||||
| from functools import partial | ||||
| from typing import TYPE_CHECKING, List, Optional, Union | ||||
|  | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from sanic.blueprints import Blueprint | ||||
|  | ||||
|  | ||||
| class BlueprintGroup(MutableSequence): | ||||
| @@ -54,9 +59,23 @@ class BlueprintGroup(MutableSequence): | ||||
|         app.blueprint(bpg) | ||||
|     """ | ||||
|  | ||||
|     __slots__ = ("_blueprints", "_url_prefix", "_version", "_strict_slashes") | ||||
|     __slots__ = ( | ||||
|         "_blueprints", | ||||
|         "_url_prefix", | ||||
|         "_version", | ||||
|         "_strict_slashes", | ||||
|         "_version_prefix", | ||||
|         "_name_prefix", | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, url_prefix=None, version=None, strict_slashes=None): | ||||
|     def __init__( | ||||
|         self, | ||||
|         url_prefix: Optional[str] = None, | ||||
|         version: Optional[Union[int, str, float]] = None, | ||||
|         strict_slashes: Optional[bool] = None, | ||||
|         version_prefix: str = "/v", | ||||
|         name_prefix: Optional[str] = "", | ||||
|     ): | ||||
|         """ | ||||
|         Create a new Blueprint Group | ||||
|  | ||||
| @@ -65,13 +84,15 @@ class BlueprintGroup(MutableSequence): | ||||
|             inherited by each of the Blueprint | ||||
|         :param strict_slashes: URL Strict slash behavior indicator | ||||
|         """ | ||||
|         self._blueprints = [] | ||||
|         self._blueprints: List[Blueprint] = [] | ||||
|         self._url_prefix = url_prefix | ||||
|         self._version = version | ||||
|         self._version_prefix = version_prefix | ||||
|         self._strict_slashes = strict_slashes | ||||
|         self._name_prefix = name_prefix | ||||
|  | ||||
|     @property | ||||
|     def url_prefix(self) -> str: | ||||
|     def url_prefix(self) -> Optional[Union[int, str, float]]: | ||||
|         """ | ||||
|         Retrieve the URL prefix being used for the Current Blueprint Group | ||||
|  | ||||
| @@ -80,7 +101,7 @@ class BlueprintGroup(MutableSequence): | ||||
|         return self._url_prefix | ||||
|  | ||||
|     @property | ||||
|     def blueprints(self) -> List["sanic.Blueprint"]: | ||||
|     def blueprints(self) -> List[Blueprint]: | ||||
|         """ | ||||
|         Retrieve a list of all the available blueprints under this group. | ||||
|  | ||||
| @@ -107,6 +128,24 @@ class BlueprintGroup(MutableSequence): | ||||
|         """ | ||||
|         return self._strict_slashes | ||||
|  | ||||
|     @property | ||||
|     def version_prefix(self) -> str: | ||||
|         """ | ||||
|         Version prefix; defaults to ``/v`` | ||||
|  | ||||
|         :return: str | ||||
|         """ | ||||
|         return self._version_prefix | ||||
|  | ||||
|     @property | ||||
|     def name_prefix(self) -> Optional[str]: | ||||
|         """ | ||||
|         Name prefix for the blueprint group | ||||
|  | ||||
|         :return: str | ||||
|         """ | ||||
|         return self._name_prefix | ||||
|  | ||||
|     def __iter__(self): | ||||
|         """ | ||||
|         Tun the class Blueprint Group into an Iterable item | ||||
| @@ -161,34 +200,37 @@ class BlueprintGroup(MutableSequence): | ||||
|         """ | ||||
|         return len(self._blueprints) | ||||
|  | ||||
|     def _sanitize_blueprint(self, bp: "sanic.Blueprint") -> "sanic.Blueprint": | ||||
|         """ | ||||
|         Sanitize the Blueprint Entity to override the Version and strict slash | ||||
|         behaviors as required. | ||||
|  | ||||
|         :param bp: Sanic Blueprint entity Object | ||||
|         :return: Modified Blueprint | ||||
|         """ | ||||
|         if self._url_prefix: | ||||
|             merged_prefix = "/".join( | ||||
|                 u.strip("/") for u in [self._url_prefix, bp.url_prefix or ""] | ||||
|             ).rstrip("/") | ||||
|             bp.url_prefix = f"/{merged_prefix}" | ||||
|         for _attr in ["version", "strict_slashes"]: | ||||
|             if getattr(bp, _attr) is None: | ||||
|                 setattr(bp, _attr, getattr(self, _attr)) | ||||
|         return bp | ||||
|  | ||||
|     def append(self, value: "sanic.Blueprint") -> None: | ||||
|     def append(self, value: Blueprint) -> None: | ||||
|         """ | ||||
|         The Abstract class `MutableSequence` leverages this append method to | ||||
|         perform the `BlueprintGroup.append` operation. | ||||
|         :param value: New `Blueprint` object. | ||||
|         :return: None | ||||
|         """ | ||||
|         self._blueprints.append(self._sanitize_blueprint(bp=value)) | ||||
|         self._blueprints.append(value) | ||||
|  | ||||
|     def insert(self, index: int, item: "sanic.Blueprint") -> None: | ||||
|     def exception(self, *exceptions, **kwargs): | ||||
|         """ | ||||
|         A decorator that can be used to implement a global exception handler | ||||
|         for all the Blueprints that belong to this Blueprint Group. | ||||
|  | ||||
|         In case of nested Blueprint Groups, the same handler is applied | ||||
|         across each of the Blueprints recursively. | ||||
|  | ||||
|         :param args: List of Python exceptions to be caught by the handler | ||||
|         :param kwargs: Additional optional arguments to be passed to the | ||||
|             exception handler | ||||
|         :return: a decorated method to handle global exceptions for any | ||||
|             blueprint registered under this group. | ||||
|         """ | ||||
|  | ||||
|         def register_exception_handler_for_blueprints(fn): | ||||
|             for blueprint in self.blueprints: | ||||
|                 blueprint.exception(*exceptions, **kwargs)(fn) | ||||
|  | ||||
|         return register_exception_handler_for_blueprints | ||||
|  | ||||
|     def insert(self, index: int, item: Blueprint) -> None: | ||||
|         """ | ||||
|         The Abstract class `MutableSequence` leverages this insert method to | ||||
|         perform the `BlueprintGroup.append` operation. | ||||
| @@ -197,7 +239,7 @@ class BlueprintGroup(MutableSequence): | ||||
|         :param item: New `Blueprint` object. | ||||
|         :return: None | ||||
|         """ | ||||
|         self._blueprints.insert(index, self._sanitize_blueprint(item)) | ||||
|         self._blueprints.insert(index, item) | ||||
|  | ||||
|     def middleware(self, *args, **kwargs): | ||||
|         """ | ||||
| @@ -221,3 +263,15 @@ class BlueprintGroup(MutableSequence): | ||||
|             args = list(args)[1:] | ||||
|             return register_middleware_for_blueprints(fn) | ||||
|         return register_middleware_for_blueprints | ||||
|  | ||||
|     def on_request(self, middleware=None): | ||||
|         if callable(middleware): | ||||
|             return self.middleware(middleware, "request") | ||||
|         else: | ||||
|             return partial(self.middleware, attach_to="request") | ||||
|  | ||||
|     def on_response(self, middleware=None): | ||||
|         if callable(middleware): | ||||
|             return self.middleware(middleware, "response") | ||||
|         else: | ||||
|             return partial(self.middleware, attach_to="response") | ||||
|   | ||||
| @@ -3,15 +3,31 @@ from __future__ import annotations | ||||
| import asyncio | ||||
|  | ||||
| from collections import defaultdict | ||||
| from copy import deepcopy | ||||
| from functools import wraps | ||||
| from inspect import isfunction | ||||
| from itertools import chain | ||||
| from types import SimpleNamespace | ||||
| from typing import TYPE_CHECKING, Dict, Iterable, List, Optional, Set, Union | ||||
| from typing import ( | ||||
|     TYPE_CHECKING, | ||||
|     Any, | ||||
|     Dict, | ||||
|     Iterable, | ||||
|     List, | ||||
|     Optional, | ||||
|     Sequence, | ||||
|     Set, | ||||
|     Tuple, | ||||
|     Union, | ||||
| ) | ||||
|  | ||||
| from sanic_routing.exceptions import NotFound  # type: ignore | ||||
| from sanic_routing.route import Route  # type: ignore | ||||
| from sanic_routing.exceptions import NotFound | ||||
| from sanic_routing.route import Route | ||||
|  | ||||
| from sanic.base import BaseSanic | ||||
| from sanic.base.root import BaseSanic | ||||
| from sanic.blueprint_group import BlueprintGroup | ||||
| from sanic.exceptions import SanicException | ||||
| from sanic.helpers import Default, _default | ||||
| from sanic.models.futures import FutureRoute, FutureStatic | ||||
| from sanic.models.handler_types import ( | ||||
|     ListenerType, | ||||
| @@ -21,7 +37,33 @@ from sanic.models.handler_types import ( | ||||
|  | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from sanic import Sanic  # noqa | ||||
|     from sanic import Sanic | ||||
|  | ||||
|  | ||||
| def lazy(func, as_decorator=True): | ||||
|     @wraps(func) | ||||
|     def decorator(bp, *args, **kwargs): | ||||
|         nonlocal as_decorator | ||||
|         kwargs["apply"] = False | ||||
|         pass_handler = None | ||||
|  | ||||
|         if args and isfunction(args[0]): | ||||
|             as_decorator = False | ||||
|  | ||||
|         def wrapper(handler): | ||||
|             future = func(bp, *args, **kwargs) | ||||
|             if as_decorator: | ||||
|                 future = future(handler) | ||||
|  | ||||
|             if bp.registered: | ||||
|                 for app in bp.apps: | ||||
|                     bp.register(app, {}) | ||||
|  | ||||
|             return future | ||||
|  | ||||
|         return wrapper if as_decorator else wrapper(pass_handler) | ||||
|  | ||||
|     return decorator | ||||
|  | ||||
|  | ||||
| class Blueprint(BaseSanic): | ||||
| @@ -37,13 +79,13 @@ class Blueprint(BaseSanic): | ||||
|  | ||||
|     :param name: unique name of the blueprint | ||||
|     :param url_prefix: URL to be prefixed before all route URLs | ||||
|     :param host: IP Address of FQDN for the sanic server to use. | ||||
|     :param host: IP Address or FQDN for the sanic server to use. | ||||
|     :param version: Blueprint Version | ||||
|     :param strict_slashes: Enforce the API urls are requested with a | ||||
|         training */* | ||||
|         trailing */* | ||||
|     """ | ||||
|  | ||||
|     __fake_slots__ = ( | ||||
|     __slots__ = ( | ||||
|         "_apps", | ||||
|         "_future_routes", | ||||
|         "_future_statics", | ||||
| @@ -51,17 +93,19 @@ class Blueprint(BaseSanic): | ||||
|         "_future_listeners", | ||||
|         "_future_exceptions", | ||||
|         "_future_signals", | ||||
|         "_allow_route_overwrite", | ||||
|         "copied_from", | ||||
|         "ctx", | ||||
|         "exceptions", | ||||
|         "host", | ||||
|         "listeners", | ||||
|         "middlewares", | ||||
|         "name", | ||||
|         "routes", | ||||
|         "statics", | ||||
|         "strict_slashes", | ||||
|         "url_prefix", | ||||
|         "version", | ||||
|         "version_prefix", | ||||
|         "websocket_routes", | ||||
|     ) | ||||
|  | ||||
| @@ -69,25 +113,25 @@ class Blueprint(BaseSanic): | ||||
|         self, | ||||
|         name: str, | ||||
|         url_prefix: Optional[str] = None, | ||||
|         host: Optional[str] = None, | ||||
|         version: Optional[int] = None, | ||||
|         host: Optional[Union[List[str], str]] = None, | ||||
|         version: Optional[Union[int, str, float]] = None, | ||||
|         strict_slashes: Optional[bool] = None, | ||||
|         version_prefix: str = "/v", | ||||
|     ): | ||||
|         super().__init__() | ||||
|  | ||||
|         self._apps: Set[Sanic] = set() | ||||
|         super().__init__(name=name) | ||||
|         self.reset() | ||||
|         self._allow_route_overwrite = False | ||||
|         self.copied_from = "" | ||||
|         self.ctx = SimpleNamespace() | ||||
|         self.exceptions: List[RouteHandler] = [] | ||||
|         self.host = host | ||||
|         self.listeners: Dict[str, List[ListenerType]] = {} | ||||
|         self.middlewares: List[MiddlewareType] = [] | ||||
|         self.name = name | ||||
|         self.routes: List[Route] = [] | ||||
|         self.statics: List[RouteHandler] = [] | ||||
|         self.strict_slashes = strict_slashes | ||||
|         self.url_prefix = url_prefix | ||||
|         self.url_prefix = ( | ||||
|             url_prefix[:-1] | ||||
|             if url_prefix and url_prefix.endswith("/") | ||||
|             else url_prefix | ||||
|         ) | ||||
|         self.version = version | ||||
|         self.websocket_routes: List[Route] = [] | ||||
|         self.version_prefix = version_prefix | ||||
|  | ||||
|     def __repr__(self) -> str: | ||||
|         args = ", ".join( | ||||
| @@ -114,32 +158,106 @@ class Blueprint(BaseSanic): | ||||
|             ) | ||||
|         return self._apps | ||||
|  | ||||
|     def route(self, *args, **kwargs): | ||||
|         kwargs["apply"] = False | ||||
|         return super().route(*args, **kwargs) | ||||
|     @property | ||||
|     def registered(self) -> bool: | ||||
|         return bool(self._apps) | ||||
|  | ||||
|     def static(self, *args, **kwargs): | ||||
|         kwargs["apply"] = False | ||||
|         return super().static(*args, **kwargs) | ||||
|     exception = lazy(BaseSanic.exception) | ||||
|     listener = lazy(BaseSanic.listener) | ||||
|     middleware = lazy(BaseSanic.middleware) | ||||
|     route = lazy(BaseSanic.route) | ||||
|     signal = lazy(BaseSanic.signal) | ||||
|     static = lazy(BaseSanic.static, as_decorator=False) | ||||
|  | ||||
|     def middleware(self, *args, **kwargs): | ||||
|         kwargs["apply"] = False | ||||
|         return super().middleware(*args, **kwargs) | ||||
|     def reset(self): | ||||
|         self._apps: Set[Sanic] = set() | ||||
|         self._allow_route_overwrite = False | ||||
|         self.exceptions: List[RouteHandler] = [] | ||||
|         self.listeners: Dict[str, List[ListenerType[Any]]] = {} | ||||
|         self.middlewares: List[MiddlewareType] = [] | ||||
|         self.routes: List[Route] = [] | ||||
|         self.statics: List[RouteHandler] = [] | ||||
|         self.websocket_routes: List[Route] = [] | ||||
|  | ||||
|     def listener(self, *args, **kwargs): | ||||
|         kwargs["apply"] = False | ||||
|         return super().listener(*args, **kwargs) | ||||
|     def copy( | ||||
|         self, | ||||
|         name: str, | ||||
|         url_prefix: Optional[Union[str, Default]] = _default, | ||||
|         version: Optional[Union[int, str, float, Default]] = _default, | ||||
|         version_prefix: Union[str, Default] = _default, | ||||
|         allow_route_overwrite: Union[bool, Default] = _default, | ||||
|         strict_slashes: Optional[Union[bool, Default]] = _default, | ||||
|         with_registration: bool = True, | ||||
|         with_ctx: bool = False, | ||||
|     ): | ||||
|         """ | ||||
|         Copy a blueprint instance with some optional parameters to | ||||
|         override the values of attributes in the old instance. | ||||
|  | ||||
|     def exception(self, *args, **kwargs): | ||||
|         kwargs["apply"] = False | ||||
|         return super().exception(*args, **kwargs) | ||||
|         :param name: unique name of the blueprint | ||||
|         :param url_prefix: URL to be prefixed before all route URLs | ||||
|         :param version: Blueprint Version | ||||
|         :param version_prefix: the prefix of the version number shown in the | ||||
|             URL. | ||||
|         :param strict_slashes: Enforce the API urls are requested with a | ||||
|             trailing */* | ||||
|         :param with_registration: whether register new blueprint instance with | ||||
|             sanic apps that were registered with the old instance or not. | ||||
|         :param with_ctx: whether ``ctx`` will be copied or not. | ||||
|         """ | ||||
|  | ||||
|     def signal(self, event: str, *args, **kwargs): | ||||
|         kwargs["apply"] = False | ||||
|         return super().signal(event, *args, **kwargs) | ||||
|         attrs_backup = { | ||||
|             "_apps": self._apps, | ||||
|             "routes": self.routes, | ||||
|             "websocket_routes": self.websocket_routes, | ||||
|             "middlewares": self.middlewares, | ||||
|             "exceptions": self.exceptions, | ||||
|             "listeners": self.listeners, | ||||
|             "statics": self.statics, | ||||
|         } | ||||
|  | ||||
|         self.reset() | ||||
|         new_bp = deepcopy(self) | ||||
|         new_bp.name = name | ||||
|         new_bp.copied_from = self.name | ||||
|  | ||||
|         if not isinstance(url_prefix, Default): | ||||
|             new_bp.url_prefix = url_prefix | ||||
|         if not isinstance(version, Default): | ||||
|             new_bp.version = version | ||||
|         if not isinstance(strict_slashes, Default): | ||||
|             new_bp.strict_slashes = strict_slashes | ||||
|         if not isinstance(version_prefix, Default): | ||||
|             new_bp.version_prefix = version_prefix | ||||
|         if not isinstance(allow_route_overwrite, Default): | ||||
|             new_bp._allow_route_overwrite = allow_route_overwrite | ||||
|  | ||||
|         for key, value in attrs_backup.items(): | ||||
|             setattr(self, key, value) | ||||
|  | ||||
|         if with_registration and self._apps: | ||||
|             if new_bp._future_statics: | ||||
|                 raise SanicException( | ||||
|                     "Static routes registered with the old blueprint instance," | ||||
|                     " cannot be registered again." | ||||
|                 ) | ||||
|             for app in self._apps: | ||||
|                 app.blueprint(new_bp) | ||||
|  | ||||
|         if not with_ctx: | ||||
|             new_bp.ctx = SimpleNamespace() | ||||
|  | ||||
|         return new_bp | ||||
|  | ||||
|     @staticmethod | ||||
|     def group(*blueprints, url_prefix="", version=None, strict_slashes=None): | ||||
|     def group( | ||||
|         *blueprints: Union[Blueprint, BlueprintGroup], | ||||
|         url_prefix: Optional[str] = None, | ||||
|         version: Optional[Union[int, str, float]] = None, | ||||
|         strict_slashes: Optional[bool] = None, | ||||
|         version_prefix: str = "/v", | ||||
|         name_prefix: Optional[str] = "", | ||||
|     ) -> BlueprintGroup: | ||||
|         """ | ||||
|         Create a list of blueprints, optionally grouping them under a | ||||
|         general URL prefix. | ||||
| @@ -156,8 +274,6 @@ class Blueprint(BaseSanic): | ||||
|             for i in nested: | ||||
|                 if isinstance(i, (list, tuple)): | ||||
|                     yield from chain(i) | ||||
|                 elif isinstance(i, BlueprintGroup): | ||||
|                     yield from i.blueprints | ||||
|                 else: | ||||
|                     yield i | ||||
|  | ||||
| @@ -165,6 +281,8 @@ class Blueprint(BaseSanic): | ||||
|             url_prefix=url_prefix, | ||||
|             version=version, | ||||
|             strict_slashes=strict_slashes, | ||||
|             version_prefix=version_prefix, | ||||
|             name_prefix=name_prefix, | ||||
|         ) | ||||
|         for bp in chain(blueprints): | ||||
|             bps.append(bp) | ||||
| @@ -182,45 +300,89 @@ class Blueprint(BaseSanic): | ||||
|  | ||||
|         self._apps.add(app) | ||||
|         url_prefix = options.get("url_prefix", self.url_prefix) | ||||
|         opt_version = options.get("version", None) | ||||
|         opt_strict_slashes = options.get("strict_slashes", None) | ||||
|         opt_version_prefix = options.get("version_prefix", self.version_prefix) | ||||
|         opt_name_prefix = options.get("name_prefix", None) | ||||
|         error_format = options.get( | ||||
|             "error_format", app.config.FALLBACK_ERROR_FORMAT | ||||
|         ) | ||||
|  | ||||
|         routes = [] | ||||
|         middleware = [] | ||||
|         exception_handlers = [] | ||||
|         listeners = defaultdict(list) | ||||
|         registered = set() | ||||
|  | ||||
|         # Routes | ||||
|         for future in self._future_routes: | ||||
|             # attach the blueprint name to the handler so that it can be | ||||
|             # prefixed properly in the router | ||||
|             future.handler.__blueprintname__ = self.name | ||||
|             # Prepend the blueprint URI prefix if available | ||||
|             uri = url_prefix + future.uri if url_prefix else future.uri | ||||
|             uri = self._setup_uri(future.uri, url_prefix) | ||||
|  | ||||
|             strict_slashes = ( | ||||
|                 self.strict_slashes | ||||
|                 if future.strict_slashes is None | ||||
|                 and self.strict_slashes is not None | ||||
|                 else future.strict_slashes | ||||
|             route_error_format = ( | ||||
|                 future.error_format if future.error_format else error_format | ||||
|             ) | ||||
|             name = app._generate_name(future.name) | ||||
|  | ||||
|             version_prefix = self.version_prefix | ||||
|             for prefix in ( | ||||
|                 future.version_prefix, | ||||
|                 opt_version_prefix, | ||||
|             ): | ||||
|                 if prefix and prefix != "/v": | ||||
|                     version_prefix = prefix | ||||
|                     break | ||||
|  | ||||
|             version = self._extract_value( | ||||
|                 future.version, opt_version, self.version | ||||
|             ) | ||||
|             strict_slashes = self._extract_value( | ||||
|                 future.strict_slashes, opt_strict_slashes, self.strict_slashes | ||||
|             ) | ||||
|  | ||||
|             name = future.name | ||||
|             if opt_name_prefix: | ||||
|                 name = f"{opt_name_prefix}_{future.name}" | ||||
|             name = app._generate_name(name) | ||||
|             host = future.host or self.host | ||||
|             if isinstance(host, list): | ||||
|                 host = tuple(host) | ||||
|  | ||||
|             apply_route = FutureRoute( | ||||
|                 future.handler, | ||||
|                 uri[1:] if uri.startswith("//") else uri, | ||||
|                 uri, | ||||
|                 future.methods, | ||||
|                 future.host or self.host, | ||||
|                 host, | ||||
|                 strict_slashes, | ||||
|                 future.stream, | ||||
|                 future.version or self.version, | ||||
|                 version, | ||||
|                 name, | ||||
|                 future.ignore_body, | ||||
|                 future.websocket, | ||||
|                 future.subprotocols, | ||||
|                 future.unquote, | ||||
|                 future.static, | ||||
|                 version_prefix, | ||||
|                 route_error_format, | ||||
|                 future.route_context, | ||||
|             ) | ||||
|  | ||||
|             route = app._apply_route(apply_route) | ||||
|             if (self, apply_route) in app._future_registry: | ||||
|                 continue | ||||
|  | ||||
|             registered.add(apply_route) | ||||
|             route = app._apply_route( | ||||
|                 apply_route, overwrite=self._allow_route_overwrite | ||||
|             ) | ||||
|  | ||||
|             # If it is a copied BP, then make sure all of the names of routes | ||||
|             # matchup with the new BP name | ||||
|             if self.copied_from: | ||||
|                 for r in route: | ||||
|                     r.name = r.name.replace(self.copied_from, self.name) | ||||
|                     r.extra.ident = r.extra.ident.replace( | ||||
|                         self.copied_from, self.name | ||||
|                     ) | ||||
|  | ||||
|             operation = ( | ||||
|                 routes.extend if isinstance(route, list) else routes.append | ||||
|             ) | ||||
| @@ -229,43 +391,71 @@ class Blueprint(BaseSanic): | ||||
|         # Static Files | ||||
|         for future in self._future_statics: | ||||
|             # Prepend the blueprint URI prefix if available | ||||
|             uri = url_prefix + future.uri if url_prefix else future.uri | ||||
|             uri = self._setup_uri(future.uri, url_prefix) | ||||
|             apply_route = FutureStatic(uri, *future[1:]) | ||||
|  | ||||
|             if (self, apply_route) in app._future_registry: | ||||
|                 continue | ||||
|  | ||||
|             registered.add(apply_route) | ||||
|             route = app._apply_static(apply_route) | ||||
|             routes.append(route) | ||||
|  | ||||
|         route_names = [route.name for route in routes if route] | ||||
|  | ||||
|         # Middleware | ||||
|         if route_names: | ||||
|             # Middleware | ||||
|             for future in self._future_middleware: | ||||
|                 if (self, future) in app._future_registry: | ||||
|                     continue | ||||
|                 middleware.append(app._apply_middleware(future, route_names)) | ||||
|  | ||||
|         # Exceptions | ||||
|         for future in self._future_exceptions: | ||||
|             exception_handlers.append(app._apply_exception_handler(future)) | ||||
|             # Exceptions | ||||
|             for future in self._future_exceptions: | ||||
|                 if (self, future) in app._future_registry: | ||||
|                     continue | ||||
|                 exception_handlers.append( | ||||
|                     app._apply_exception_handler(future, route_names) | ||||
|                 ) | ||||
|  | ||||
|         # Event listeners | ||||
|         for listener in self._future_listeners: | ||||
|             listeners[listener.event].append(app._apply_listener(listener)) | ||||
|         for future in self._future_listeners: | ||||
|             if (self, future) in app._future_registry: | ||||
|                 continue | ||||
|             listeners[future.event].append(app._apply_listener(future)) | ||||
|  | ||||
|         for signal in self._future_signals: | ||||
|             signal.condition.update({"blueprint": self.name}) | ||||
|             app._apply_signal(signal) | ||||
|         # Signals | ||||
|         for future in self._future_signals: | ||||
|             if (self, future) in app._future_registry: | ||||
|                 continue | ||||
|             future.condition.update({"__blueprint__": self.name}) | ||||
|             # Force exclusive to be False | ||||
|             app._apply_signal(tuple((*future[:-1], False))) | ||||
|  | ||||
|         self.routes = [route for route in routes if isinstance(route, Route)] | ||||
|  | ||||
|         # Deprecate these in 21.6 | ||||
|         self.websocket_routes = [ | ||||
|             route for route in self.routes if route.ctx.websocket | ||||
|         self.routes += [route for route in routes if isinstance(route, Route)] | ||||
|         self.websocket_routes += [ | ||||
|             route for route in self.routes if route.extra.websocket | ||||
|         ] | ||||
|         self.middlewares = middleware | ||||
|         self.exceptions = exception_handlers | ||||
|         self.listeners = dict(listeners) | ||||
|         self.middlewares += middleware | ||||
|         self.exceptions += exception_handlers | ||||
|         self.listeners.update(dict(listeners)) | ||||
|  | ||||
|         if self.registered: | ||||
|             self.register_futures( | ||||
|                 self.apps, | ||||
|                 self, | ||||
|                 chain( | ||||
|                     registered, | ||||
|                     self._future_middleware, | ||||
|                     self._future_exceptions, | ||||
|                     self._future_listeners, | ||||
|                     self._future_signals, | ||||
|                 ), | ||||
|             ) | ||||
|  | ||||
|     async def dispatch(self, *args, **kwargs): | ||||
|         condition = kwargs.pop("condition", {}) | ||||
|         condition.update({"blueprint": self.name}) | ||||
|         condition.update({"__blueprint__": self.name}) | ||||
|         kwargs["condition"] = condition | ||||
|         await asyncio.gather( | ||||
|             *[app.dispatch(*args, **kwargs) for app in self.apps] | ||||
| @@ -280,7 +470,35 @@ class Blueprint(BaseSanic): | ||||
|             events.add(signal.ctx.event) | ||||
|  | ||||
|         return asyncio.wait( | ||||
|             [event.wait() for event in events], | ||||
|             [asyncio.create_task(event.wait()) for event in events], | ||||
|             return_when=asyncio.FIRST_COMPLETED, | ||||
|             timeout=timeout, | ||||
|         ) | ||||
|  | ||||
|     @staticmethod | ||||
|     def _extract_value(*values): | ||||
|         value = values[-1] | ||||
|         for v in values: | ||||
|             if v is not None: | ||||
|                 value = v | ||||
|                 break | ||||
|         return value | ||||
|  | ||||
|     @staticmethod | ||||
|     def _setup_uri(base: str, prefix: Optional[str]): | ||||
|         uri = base | ||||
|         if prefix: | ||||
|             uri = prefix | ||||
|             if base.startswith("/") and prefix.endswith("/"): | ||||
|                 uri += base[1:] | ||||
|             else: | ||||
|                 uri += base | ||||
|  | ||||
|         return uri[1:] if uri.startswith("//") else uri | ||||
|  | ||||
|     @staticmethod | ||||
|     def register_futures( | ||||
|         apps: Set[Sanic], bp: Blueprint, futures: Sequence[Tuple[Any, ...]] | ||||
|     ): | ||||
|         for app in apps: | ||||
|             app._future_registry.update(set((bp, item) for item in futures)) | ||||
|   | ||||
							
								
								
									
										0
									
								
								sanic/cli/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								sanic/cli/__init__.py
									
									
									
									
									
										Normal file
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user