diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml new file mode 100644 index 00000000..1113fa80 --- /dev/null +++ b/.github/workflows/codeql-analysis.yml @@ -0,0 +1,40 @@ +name: "CodeQL" + +on: + push: + branches: + - main + - "*LTS" + pull_request: + branches: + - main + - "*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 + + strategy: + fail-fast: false + matrix: + language: [ 'python' ] + + steps: + - name: Checkout repository + uses: actions/checkout@v2 + + - name: Initialize CodeQL + uses: github/codeql-action/init@v1 + with: + languages: ${{ matrix.language }} + + - name: Autobuild + uses: github/codeql-action/autobuild@v1 + + - name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v1 diff --git a/.github/workflows/coverage.yml b/.github/workflows/coverage.yml new file mode 100644 index 00000000..9b5834fa --- /dev/null +++ b/.github/workflows/coverage.yml @@ -0,0 +1,37 @@ +name: Coverage check +on: + push: + branches: + - main + - "*LTS" + tags: + - "!*" # Do not execute on tags + pull_request: + types: [opened, synchronize, reopened, ready_for_review] +jobs: + test: + if: github.event.pull_request.draft == false + runs-on: ${{ matrix.os }} + strategy: + matrix: + python-version: [3.9] + os: [ubuntu-latest] + fail-fast: false + + steps: + - uses: actions/checkout@v2 + + - uses: actions/setup-python@v1 + with: + python-version: ${{ matrix.python-version }} + + - name: Install dependencies 🔨 + run: | + python -m pip install --upgrade pip + pip install tox + - uses: paambaati/codeclimate-action@v2.5.3 + if: always() + env: + CC_TEST_REPORTER_ID: ${{ secrets.CODECLIMATE }} + with: + coverageCommand: tox -e coverage diff --git a/.github/workflows/on-demand.yml b/.github/workflows/on-demand.yml new file mode 100644 index 00000000..1aa721f3 --- /dev/null +++ b/.github/workflows/on-demand.yml @@ -0,0 +1,39 @@ +name: On Demand Task +on: + workflow_dispatch: + inputs: + python-version: + description: 'Version of Python to use for running Test' + required: false + default: "3.8" + tox-env: + description: 'Test Environment to Run' + required: true + default: '' + os: + description: 'Operating System to Run Test on' + required: false + default: ubuntu-latest +jobs: + onDemand: + name: tox-${{ matrix.config.tox-env }}-on-${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: ["${{ github.event.inputs.os}}"] + config: + - { tox-env: "${{ github.event.inputs.tox-env }}", py-version: "${{ github.event.inputs.python-version }}"} + steps: + - name: Checkout Repository + uses: actions/checkout@v2 + + - name: Run tests + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.py-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" + experimental-ignore-error: "yes" diff --git a/.github/workflows/pr-bandit.yml b/.github/workflows/pr-bandit.yml new file mode 100644 index 00000000..2bd70204 --- /dev/null +++ b/.github/workflows/pr-bandit.yml @@ -0,0 +1,36 @@ +name: Security Analysis +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + bandit: + if: github.event.pull_request.draft == false + name: type-check-${{ matrix.config.python-version }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [ubuntu-latest] + config: + - { python-version: 3.7, tox-env: security} + - { python-version: 3.8, tox-env: security} + - { python-version: 3.9, tox-env: security} + - { python-version: "3.10", tox-env: security} + steps: + - name: Checkout the repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Linter Checks + id: linter-check + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" diff --git a/.github/workflows/pr-docs.yml b/.github/workflows/pr-docs.yml new file mode 100644 index 00000000..8479aef5 --- /dev/null +++ b/.github/workflows/pr-docs.yml @@ -0,0 +1,32 @@ +name: Document Linter +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + docsLinter: + if: github.event.pull_request.draft == false + name: Lint Documentation + runs-on: ubuntu-latest + strategy: + matrix: + config: + - {python-version: "3.8", tox-env: "docs"} + fail-fast: false + + + steps: + - name: Checkout repository + uses: actions/checkout@v2 + + - name: Run Document Linter + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" diff --git a/.github/workflows/pr-linter.yml b/.github/workflows/pr-linter.yml new file mode 100644 index 00000000..11ad9d29 --- /dev/null +++ b/.github/workflows/pr-linter.yml @@ -0,0 +1,33 @@ +name: Linter Checks +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + linter: + if: github.event.pull_request.draft == false + name: lint + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [ubuntu-latest] + config: + - { python-version: 3.8, tox-env: lint} + steps: + - name: Checkout the repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Linter Checks + id: linter-check + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" diff --git a/.github/workflows/pr-python-pypy.yml b/.github/workflows/pr-python-pypy.yml new file mode 100644 index 00000000..c1a6c7cf --- /dev/null +++ b/.github/workflows/pr-python-pypy.yml @@ -0,0 +1,41 @@ +name: Python PyPy Tests +on: + workflow_dispatch: + inputs: + tox-env: + description: "Tox Env to run on the PyPy Infra" + required: false + default: "pypy37" + pypy-version: + description: "Version of PyPy to use" + required: false + default: "pypy-3.7" +jobs: + testPyPy: + name: ut-${{ matrix.config.tox-env }}-${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + # os: [ubuntu-latest, macos-latest] + os: [ubuntu-latest] + config: + - { + python-version: "${{ github.event.inputs.pypy-version }}", + tox-env: "${{ github.event.inputs.tox-env }}", + } + steps: + - name: Checkout the Repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Unit Tests + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" + experimental-ignore-error: "true" + command-timeout: "600000" diff --git a/.github/workflows/pr-python37.yml b/.github/workflows/pr-python37.yml new file mode 100644 index 00000000..c0051d33 --- /dev/null +++ b/.github/workflows/pr-python37.yml @@ -0,0 +1,35 @@ +name: Python 3.7 Tests +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + testPy37: + if: github.event.pull_request.draft == false + name: ut-${{ matrix.config.tox-env }}-${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: true + matrix: + # os: [ubuntu-latest, macos-latest] + os: [ubuntu-latest] + config: + - { python-version: 3.7, tox-env: py37 } + - { python-version: 3.7, tox-env: py37-no-ext } + steps: + - name: Checkout the Repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Unit Tests + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" + test-failure-retry: "3" diff --git a/.github/workflows/pr-python38.yml b/.github/workflows/pr-python38.yml new file mode 100644 index 00000000..09e93f3f --- /dev/null +++ b/.github/workflows/pr-python38.yml @@ -0,0 +1,35 @@ +name: Python 3.8 Tests +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + testPy38: + if: github.event.pull_request.draft == false + name: ut-${{ matrix.config.tox-env }}-${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: true + matrix: + # os: [ubuntu-latest, macos-latest] + os: [ubuntu-latest] + config: + - { python-version: 3.8, tox-env: py38 } + - { python-version: 3.8, tox-env: py38-no-ext } + steps: + - name: Checkout the Repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Unit Tests + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" + test-failure-retry: "3" diff --git a/.github/workflows/pr-python39.yml b/.github/workflows/pr-python39.yml new file mode 100644 index 00000000..ff479459 --- /dev/null +++ b/.github/workflows/pr-python39.yml @@ -0,0 +1,47 @@ +name: Python 3.9 Tests +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + testPy39: + if: github.event.pull_request.draft == false + name: ut-${{ matrix.config.tox-env }}-${{ matrix.os }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: true + matrix: + # os: [ubuntu-latest, macos-latest] + os: [ubuntu-latest] + config: + - { + python-version: 3.9, + tox-env: py39, + ignore-error-flake: "false", + command-timeout: "0", + } + - { + python-version: 3.9, + tox-env: py39-no-ext, + ignore-error-flake: "true", + command-timeout: "600000", + } + steps: + - name: Checkout the Repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Unit Tests + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }},-vv=''" + experimental-ignore-error: "${{ matrix.config.ignore-error-flake }}" + command-timeout: "${{ matrix.config.command-timeout }}" + test-failure-retry: "3" diff --git a/.github/workflows/pr-type-check.yml b/.github/workflows/pr-type-check.yml new file mode 100644 index 00000000..c071fdee --- /dev/null +++ b/.github/workflows/pr-type-check.yml @@ -0,0 +1,35 @@ +name: Typing Checks +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + typeChecking: + if: github.event.pull_request.draft == false + name: type-check-${{ matrix.config.python-version }} + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [ubuntu-latest] + config: + # - { python-version: 3.7, tox-env: type-checking} + - { python-version: 3.8, tox-env: type-checking} + - { python-version: 3.9, tox-env: type-checking} + steps: + - name: Checkout the repository + uses: actions/checkout@v2 + id: checkout-branch + + - name: Run Linter Checks + id: linter-check + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" diff --git a/.github/workflows/pr-windows.yml b/.github/workflows/pr-windows.yml new file mode 100644 index 00000000..b789f2b6 --- /dev/null +++ b/.github/workflows/pr-windows.yml @@ -0,0 +1,37 @@ +name: Run Unit Tests on Windows +on: + pull_request: + branches: + - main + - "*LTS" + types: [opened, synchronize, reopened, ready_for_review] + +jobs: + testsOnWindows: + if: github.event.pull_request.draft == false + name: ut-${{ matrix.config.tox-env }} + runs-on: windows-latest + strategy: + fail-fast: false + matrix: + config: + - { python-version: 3.7, tox-env: py37-no-ext } + - { python-version: 3.8, tox-env: py38-no-ext } + - { python-version: 3.9, tox-env: py39-no-ext } + - { python-version: pypy-3.7, tox-env: pypy37-no-ext } + + steps: + - name: Checkout Repository + uses: actions/checkout@v2 + + - name: Run Unit Tests + uses: ahopkins/custom-actions@pip-extra-args + with: + python-version: ${{ matrix.config.python-version }} + test-infra-tool: tox + test-infra-version: latest + action: tests + test-additional-args: "-e=${{ matrix.config.tox-env }}" + experimental-ignore-error: "true" + command-timeout: "600000" + pip-extra-args: "--user" diff --git a/.github/workflows/publish-images.yml b/.github/workflows/publish-images.yml new file mode 100644 index 00000000..8c78f96c --- /dev/null +++ b/.github/workflows/publish-images.yml @@ -0,0 +1,48 @@ +name: Publish Docker Images +on: + workflow_run: + workflows: + - 'Publish Artifacts' + types: + - completed + +jobs: + publishDockerImages: + name: Docker Image Build [${{ matrix.python-version }}] + runs-on: ubuntu-latest + + strategy: + fail-fast: true + matrix: + python-version: ["3.7", "3.8", "3.9"] + + steps: + - name: Checkout repository + uses: actions/checkout@v2 + + - name: Build Latest Base images for ${{ matrix.python-version }} + uses: harshanarayana/custom-actions@main + with: + docker-image-base-name: sanicframework/sanic-build + ignore-python-setup: 'true' + dockerfile-base-dir: './docker' + action: 'image-publish' + docker-image-tag: "${{ matrix.python-version }}" + docker-file-suffix: "base" + docker-build-args: "PYTHON_VERSION=${{ matrix.python-version }}" + registry-auth-user: ${{ secrets.DOCKER_ACCESS_USER }} + registry-auth-password: ${{ secrets.DOCKER_ACCESS_TOKEN }} + push-images: 'true' + + - name: Publish Sanic Docker Image for ${{ matrix.python-version }} + uses: harshanarayana/custom-actions@main + with: + docker-image-base-name: sanicframework/sanic + ignore-python-setup: 'true' + dockerfile-base-dir: './docker' + action: 'image-publish' + docker-build-args: "BASE_IMAGE_TAG=${{ matrix.python-version }}" + docker-image-prefix: "${{ matrix.python-version }}" + registry-auth-user: ${{ secrets.DOCKER_ACCESS_USER }} + registry-auth-password: ${{ secrets.DOCKER_ACCESS_TOKEN }} + push-images: 'true' diff --git a/.github/workflows/publish-package.yml b/.github/workflows/publish-package.yml new file mode 100644 index 00000000..faa3b7e6 --- /dev/null +++ b/.github/workflows/publish-package.yml @@ -0,0 +1,28 @@ +name: Publish Artifacts +on: + release: + types: [created] + +jobs: + publishPythonPackage: + name: Publishing Sanic Release Artifacts + runs-on: ubuntu-latest + + strategy: + fail-fast: true + matrix: + python-version: ["3.8"] + + steps: + - name: Checkout Repository + uses: actions/checkout@v2 + + - name: Publish Python Package + uses: harshanarayana/custom-actions@main + with: + python-version: ${{ matrix.python-version }} + package-infra-name: "twine" + pypi-user: __token__ + pypi-access-token: ${{ secrets.PYPI_ACCESS_TOKEN }} + action: "package-publish" + pypi-verify-metadata: "true" diff --git a/CHANGELOG.rst b/CHANGELOG.rst index cb828d00..41f30c07 100644 --- a/CHANGELOG.rst +++ b/CHANGELOG.rst @@ -1,3 +1,13 @@ +Version 20.12.5 +=============== + +Bugfixes +******** + + * + `#2366 `_ + websocket dependency for websockets 9.1 security fix + Version 20.12.0 =============== diff --git a/sanic/__version__.py b/sanic/__version__.py index 7e720e33..43f6a244 100644 --- a/sanic/__version__.py +++ b/sanic/__version__.py @@ -1 +1 @@ -__version__ = "20.12.4" +__version__ = "20.12.5" diff --git a/sanic/compat.py b/sanic/compat.py index f6244426..48c18a40 100644 --- a/sanic/compat.py +++ b/sanic/compat.py @@ -20,7 +20,6 @@ if use_trio: def stat_async(path): return Path(path).stat() - else: from aiofiles import open as aio_open # type: ignore from aiofiles.os import stat as stat_async # type: ignore # noqa: F401 diff --git a/sanic/request.py b/sanic/request.py index 95b64d34..092264f1 100644 --- a/sanic/request.py +++ b/sanic/request.py @@ -50,7 +50,7 @@ class StreamBuffer: self._queue = asyncio.Queue(buffer_size) async def read(self): - """ Stop reading when gets None """ + """Stop reading when gets None""" payload = await self._queue.get() self._queue.task_done() return payload diff --git a/setup.py b/setup.py index ab5c0f3a..0cc5f595 100644 --- a/setup.py +++ b/setup.py @@ -88,7 +88,7 @@ requirements = [ uvloop, ujson, "aiofiles>=0.6.0", - "websockets>=8.1,<9.0", + "websockets>=8.1,<=9.1", "multidict>=5.0,<6.0", "httpx==0.15.4", ] diff --git a/tests/test_asgi.py b/tests/test_asgi.py index 0c728493..a6d0287f 100644 --- a/tests/test_asgi.py +++ b/tests/test_asgi.py @@ -1,6 +1,3 @@ -import asyncio -import sys - from collections import deque, namedtuple import pytest @@ -82,14 +79,6 @@ def test_listeners_triggered(app): with pytest.warns(UserWarning): server.run() - all_tasks = ( - asyncio.Task.all_tasks() - if sys.version_info < (3, 7) - else asyncio.all_tasks(asyncio.get_event_loop()) - ) - for task in all_tasks: - task.cancel() - assert before_server_start assert after_server_start assert before_server_stop @@ -132,14 +121,6 @@ def test_listeners_triggered_async(app): with pytest.warns(UserWarning): server.run() - all_tasks = ( - asyncio.Task.all_tasks() - if sys.version_info < (3, 7) - else asyncio.all_tasks(asyncio.get_event_loop()) - ) - for task in all_tasks: - task.cancel() - assert before_server_start assert after_server_start assert before_server_stop diff --git a/tests/test_config.py b/tests/test_config.py index 7c232d75..754d29c3 100644 --- a/tests/test_config.py +++ b/tests/test_config.py @@ -13,7 +13,7 @@ from sanic.exceptions import PyFileError @contextmanager def temp_path(): - """ a simple cross platform replacement for NamedTemporaryFile """ + """a simple cross platform replacement for NamedTemporaryFile""" with TemporaryDirectory() as td: yield Path(td, "file") diff --git a/tests/test_logging.py b/tests/test_logging.py index faa83571..95236be4 100644 --- a/tests/test_logging.py +++ b/tests/test_logging.py @@ -102,7 +102,7 @@ def test_logging_pass_customer_logconfig(): @pytest.mark.parametrize("debug", (True, False)) def test_log_connection_lost(app, debug, monkeypatch): - """ Should not log Connection lost exception on non debug """ + """Should not log Connection lost exception on non debug""" stream = StringIO() root = logging.getLogger("sanic.root") root.addHandler(logging.StreamHandler(stream)) diff --git a/tests/test_requests.py b/tests/test_requests.py index 7a47c3f9..63ab59d4 100644 --- a/tests/test_requests.py +++ b/tests/test_requests.py @@ -289,18 +289,18 @@ def test_query_string(app): assert request.args.getlist("test1") == ["1"] assert request.args.get("test3", default="My value") == "My value" + def test_popped_stays_popped(app): @app.route("/") async def handler(request): return text("OK") - request, response = app.test_client.get( - "/", params=[("test1", "1")] - ) + request, response = app.test_client.get("/", params=[("test1", "1")]) assert request.args.pop("test1") == ["1"] assert "test1" not in request.args + @pytest.mark.asyncio async def test_query_string_asgi(app): @app.route("/") diff --git a/tox.ini b/tox.ini index 2110bd2b..6f5c0adc 100644 --- a/tox.ini +++ b/tox.ini @@ -20,7 +20,7 @@ deps = beautifulsoup4 gunicorn==20.0.4 uvicorn - websockets>=8.1,<9.0 + websockets>=8.1,<=9.1 commands = pytest {posargs:tests --cov sanic} - coverage combine --append