Skip to content

Commit ae78b8d

Browse files
committed
Add linting both as a workflow and locally using pre-commit hooks
1 parent 4be9fc3 commit ae78b8d

17 files changed

+201
-21
lines changed

.dockerignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -137,4 +137,4 @@ dmypy.json
137137
# Pyre type checker
138138
.pyre/
139139

140-
postgres/
140+
postgres/

.github/workflows/lint-and-test.yml

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
name: Lint & Test
2+
3+
on:
4+
push:
5+
branches:
6+
- rewrite
7+
pull_request:
8+
9+
10+
jobs:
11+
lint-and-test:
12+
runs-on: ubuntu-latest
13+
env:
14+
# Hide the graphical elements from pipenv's output
15+
PIPENV_HIDE_EMOJIS: 1
16+
PIPENV_NOSPIN: 1
17+
18+
steps:
19+
- name: Checkout repository
20+
uses: actions/checkout@v2
21+
22+
- name: Set up Python3.8
23+
uses: actions/setup-python@v2
24+
with:
25+
python-version: 3.8
26+
27+
- name: Install dependencies
28+
run: |
29+
pip install pipenv
30+
pipenv install --dev --system --deploy
31+
32+
- name: Run pre-commit hooks.
33+
run: SKIP=flake8; pre-commit run --all-files
34+
35+
# Thanks pydis!
36+
# Run flake8 and have it format the linting errors in the format of
37+
# the GitHub Workflow command to register error annotations. This
38+
# means that our flake8 output is automatically added as an error
39+
# annotation to both the run result and in the "Files" tab of a
40+
# pull request.
41+
#
42+
# Format used:
43+
# ::error file={filename},line={line},col={col}::{message}
44+
- name: Run flake8
45+
run: "flake8 \
46+
--format='::error file=%(path)s,line=%(row)d,col=%(col)d::\
47+
[flake8] %(code)s: %(text)s'"
48+
49+
- name: Run pytest
50+
run: |
51+
pytest

.gitignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -137,4 +137,4 @@ dmypy.json
137137
# Pyre type checker
138138
.pyre/
139139

140-
postgres/
140+
postgres/

.pre-commit-config.yaml

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
repos:
2+
- repo: local
3+
hooks:
4+
- id: black
5+
name: black
6+
entry: black --check .
7+
language: system
8+
types: [python]
9+
10+
- id: flake8
11+
name: Flake8
12+
entry: flake8
13+
language: system
14+
types: [python]
15+
require_serial: true
16+
17+
- repo: https://github.com/pre-commit/pre-commit-hooks
18+
rev: v2.5.0
19+
hooks:
20+
- id: check-json
21+
- id: check-yaml
22+
- id: end-of-file-fixer
23+
- id: check-merge-conflict
24+
- id: mixed-line-ending
25+
args: [--fix=lf]
26+
- id: trailing-whitespace
27+
args: [--markdown-linebreak-ext=md]

Dockerfile

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,4 +12,4 @@ RUN pip install -U pipenv
1212
RUN pipenv install --system --deploy
1313

1414
# Run the API.
15-
CMD python launch.py runserver --host 0.0.0.0 --port 5000 --initdb
15+
CMD python launch.py runserver --host 0.0.0.0 --port 5000 --initdb

LICENCE

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,4 +18,4 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
1818
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1919
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
2020
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21-
SOFTWARE.
21+
SOFTWARE.

Pipfile

+3
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ postdb = "*"
1515
pyjwt = "*"
1616
aiohttp = "*"
1717
uvloop = {markers = "platform_system == 'linux'", version = "*"}
18+
pytest-asyncio = "*"
1819

1920
[requires]
2021
python_version = "3.8"
@@ -23,3 +24,5 @@ python_version = "3.8"
2324
allow_prereleases = true
2425

2526
[scripts]
27+
precommit = "pre-commit install"
28+
lint = "pre-commit run --all-files"

Pipfile.lock

+72-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

api/__init__.py

+2
Original file line numberDiff line numberDiff line change
@@ -1 +1,3 @@
11
from .app import app
2+
3+
__all__ = ("app",)

api/app.py

+7-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
from quart import Quart, exceptions, jsonify, request
1+
from quart import Quart, exceptions, jsonify
22
from datetime import datetime, date
33
from aiohttp import ClientSession
44
from typing import Any, Optional
@@ -58,6 +58,12 @@ async def startup(self) -> None:
5858
app = cors(app, allow_origin="*") # TODO: Restrict the origin(s) in production.
5959

6060

61+
@app.route("/")
62+
async def index():
63+
"""Index endpoint used for testing."""
64+
return jsonify({"status": "OK"})
65+
66+
6167
@app.errorhandler(500)
6268
async def error_500(error: BaseException):
6369
"""

api/models/__init__.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,8 @@
1+
from typing import List, Type
12
from postDB import Model
2-
from typing import List
33

44

55
from .user import User
66

77

8-
models_ordered: List[Model] = []
8+
models_ordered: List[Type[Model]] = [User]

api/models/user.py

+11-11
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,17 @@ class User(Model):
3131
avatar = Column(types.String(), nullable=True)
3232
type = Column(types.String(), default="USER")
3333

34+
@classmethod
35+
async def fetch(cls, id: Union[str, int]) -> Optional["User"]:
36+
"""Fetch a user with the given ID."""
37+
query = "SELECT * FROM users WHERE id = $1"
38+
user = await cls.pool.fetchrow(query, int(id))
39+
40+
if user is not None:
41+
user = cls(**user)
42+
43+
return user
44+
3445
@classmethod
3546
async def create(
3647
cls,
@@ -61,17 +72,6 @@ async def create(
6172

6273
return cls(**record)
6374

64-
@classmethod
65-
async def fetch(cls, id: Union[str, int]) -> Optional["User"]:
66-
"""Fetch a user with the given ID."""
67-
query = "SELECT * FROM users WHERE id = $1"
68-
user = await cls.pool.fetchrow(query, int(id))
69-
70-
if user is not None:
71-
user = cls(**user)
72-
73-
return user
74-
7575
@property
7676
def created_at(self) -> datetime:
7777
"""Returns """

launch.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@
1717

1818

1919
try:
20-
import uvloop # noqa
20+
import uvloop
2121
except ModuleNotFoundError:
2222
loop = asyncio.new_event_loop()
2323
else:
@@ -118,7 +118,7 @@ async def safe_create_tables(verbose: bool = False) -> None:
118118

119119
for model in models_ordered:
120120
await model.create_table(verbose=verbose)
121-
log.info("Created table %s" % type(model).__tablename__)
121+
log.info("Created table %s" % model.__tablename__)
122122

123123

124124
@app.cli.command(name="initdb")

tests/__init__.py

Whitespace-only changes.

tests/test_index.py

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
from api import app as quart_app
2+
3+
from quart.testing import QuartClient
4+
import pytest
5+
6+
7+
@pytest.fixture(name="app")
8+
def _test_app() -> QuartClient:
9+
return quart_app.test_client()
10+
11+
12+
@pytest.mark.asyncio
13+
async def test_index(app: QuartClient):
14+
response = await app.get("/")
15+
assert response.status_code == 200

tox.ini

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
[flake8]
2+
max-line-length=120
3+
import-order-style=pycharm

utils/__init__.py

+2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
11
from .middleware import TokenAuthMiddleware
22
from .time import snowflake_time
33
from .request import Request
4+
5+
__all__ = ("TokenAuthMiddleware", "snowflake_time", "Request")

0 commit comments

Comments
 (0)