From 849423f880dc77a17cb19610d2ff902f873b3f45 Mon Sep 17 00:00:00 2001 From: Berry den Hartog <38954346+berrydenhartog@users.noreply.github.com> Date: Thu, 6 Jun 2024 12:32:39 +0000 Subject: [PATCH] Add database schema initialization This PR ALso improves the connectivity of the database during tests and it improves the debugging capabilitys --- .env.test | 24 ---- .gitignore | 1 + .pre-commit-config.yaml | 3 +- .vscode/launch.json | 17 ++- Dockerfile | 7 +- compose.yml | 15 +- database/init-user-db.sh | 3 +- docker-entrypoint.sh | 51 +++++++ poetry.lock | 110 +++++++-------- .env => prod.env | 9 +- pyproject.toml | 5 +- script/build | 5 + script/format | 2 +- script/lint | 3 +- script/test | 8 +- tad/api/deps.py | 14 ++ tad/api/routes/deps.py | 15 -- tad/api/routes/pages.py | 2 +- tad/api/routes/root.py | 9 +- tad/api/routes/tasks.py | 2 +- tad/core/config.py | 69 ++++++---- tad/core/db.py | 63 +++++++-- tad/core/log.py | 2 +- tad/core/types.py | 3 +- tad/main.py | 34 ++--- .../versions/006c480a1920_a_message.py | 36 ----- ...68e4_create_status_user_and_task_table.py} | 34 +++-- tad/services/storage.py | 6 +- tad/services/tasks.py | 5 +- tad/site/templates/default_layout.jinja | 1 + tests/api/routes/test_pages.py | 14 +- tests/api/routes/test_root.py | 12 +- tests/api/routes/test_static.py | 7 +- tests/api/routes/test_status.py | 26 ++-- tests/api/routes/test_tasks_move.py | 14 +- tests/conftest.py | 129 ++++++++++-------- tests/constants.py | 39 ++++++ tests/core/test_config.py | 47 +++---- tests/core/test_db.py | 68 +++++++-- tests/core/test_log.py | 16 --- tests/database_test_utils.py | 117 +++------------- tests/e2e/test_move_task.py | 28 ++-- tests/repositories/test_statuses.py | 53 ++++--- tests/repositories/test_tasks.py | 76 +++++------ tests/services/test_storage.py | 28 ++-- 45 files changed, 641 insertions(+), 591 deletions(-) delete mode 100644 .env.test create mode 100755 docker-entrypoint.sh rename .env => prod.env (80%) create mode 100755 script/build create mode 100644 tad/api/deps.py delete mode 100644 tad/api/routes/deps.py delete mode 100644 tad/migrations/versions/006c480a1920_a_message.py rename tad/migrations/versions/{eb2eed884ae9_a_message.py => b62dbd9468e4_create_status_user_and_task_table.py} (70%) create mode 100644 tests/constants.py diff --git a/.env.test b/.env.test deleted file mode 100644 index 4b36cdf8c..000000000 --- a/.env.test +++ /dev/null @@ -1,24 +0,0 @@ -# Domain -DOMAIN=localhost - -# Environment: local, staging, production -ENVIRONMENT=local -PROJECT_NAME="TAD" - -# TAD backend -BACKEND_CORS_ORIGINS="http://localhost,https://localhost,http://127.0.0.1,https://127.0.0.1" -SECRET_KEY=changethis -APP_DATABASE_SCHEME="sqlite" -APP_DATABASE_USER=tad -APP_DATABASE_DB=tad -APP_DATABASE_PASSWORD=changethis - -# Postgres database -POSTGRES_SERVER=db -POSTGRES_PORT=5432 -POSTGRES_DB=postgres -POSTGRES_USER=postgres -POSTGRES_PASSWORD=changethis - -# Database viewer -PGADMIN_DEFAULT_PASSWORD=changethis diff --git a/.gitignore b/.gitignore index b2a562978..c94f0c1a6 100644 --- a/.gitignore +++ b/.gitignore @@ -35,3 +35,4 @@ __pypackages__/ # tad tool tad.log* database.sqlite3 +output/ diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 8f4cb9b7f..83340105f 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -5,10 +5,11 @@ repos: rev: v4.6.0 hooks: - id: end-of-file-fixer - exclude: ^tad/static/vendor/* + exclude: ^tad/static/vendor/.* - id: trailing-whitespace - id: check-yaml - id: check-json + - id: check-added-large-files - id: check-merge-conflict - id: check-toml diff --git a/.vscode/launch.json b/.vscode/launch.json index 00ed195bc..593742573 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -7,12 +7,19 @@ "request": "launch", "module": "uvicorn", "justMyCode": false, - "args": [ "--log-level", "warning" ,"tad.main:app"], + "args": [ + "--log-level", + "warning", + "tad.main:app" + ], "cwd": "${workspaceFolder}/", "env": { - "PYTHONPATH": "${workspaceFolder}" - }, - "envFile": "${workspaceFolder}/.env.test" + "PYTHONPATH": "${workspaceFolder}", + "DEBUG": "True", + "AUTO_CREATE_SCHEMA": "True", + "ENVIRONMENT": "demo", + "LOGGING_LEVEL": "DEBUG" + } }, { "name": "Project: tests", @@ -20,7 +27,7 @@ "request": "launch", "module": "pytest", "cwd": "${workspaceFolder}", - "justMyCode": true, + "justMyCode": false, "args": [] } ] diff --git a/Dockerfile b/Dockerfile index 51faf57df..8fc3c6ffa 100644 --- a/Dockerfile +++ b/Dockerfile @@ -61,10 +61,13 @@ USER tad COPY --chown=root:root --chmod=755 ./tad /app/tad COPY --chown=root:root --chmod=755 alembic.ini /app/alembic.ini -COPY --chown=root:root --chmod=755 .env /app/.env +COPY --chown=root:root --chmod=755 prod.env /app/.env COPY --chown=root:root --chmod=755 LICENSE /app/LICENSE +COPY --chown=tad:tad --chmod=755 docker-entrypoint.sh /app/docker-entrypoint.sh ENV PYTHONPATH=/app/ WORKDIR /app/ -CMD ["python", "-m", "uvicorn", "--host", "0.0.0.0", "tad.main:app", "--log-level", "warning" ] +ENV PATH="/app/:$PATH" + +CMD [ "docker-entrypoint.sh" ] diff --git a/compose.yml b/compose.yml index 4f4a93c4f..d1c7a372a 100644 --- a/compose.yml +++ b/compose.yml @@ -9,10 +9,10 @@ services: db: condition: service_healthy env_file: - - path: .env + - path: prod.env required: true environment: - - POSTGRES_PASSWORD=${POSTGRES_PASSWORD:?Variable not set} + - ENVIRONMENT=demo ports: - 8000:8000 healthcheck: @@ -25,12 +25,10 @@ services: - app-db-data:/var/lib/postgresql/data/pgdata - ./database/:/docker-entrypoint-initdb.d/:cached env_file: - - path: .env + - path: prod.env required: true environment: - PGDATA=/var/lib/postgresql/data/pgdata - - POSTGRES_PASSWORD=${POSTGRES_PASSWORD:?Variable not set} - - SECRET_KEY=${SECRET_KEY:?Variable not set} healthcheck: test: ["CMD", "pg_isready", "-q", "-d", "tad", "-U", "tad"] @@ -40,16 +38,15 @@ services: ports: - 8080:8080 environment: - - PGADMIN_DEFAULT_EMAIL=${PGADMIN_DEFAULT_EMAIL:-tad@minbzk.nl} - - PGADMIN_DEFAULT_PASSWORD=${PGADMIN_DEFAULT_PASSWORD:?Variable not set} - PGADMIN_LISTEN_PORT=${PGADMIN_LISTEN_PORT:-8080} + env_file: + - path: prod.env + required: true depends_on: db: condition: service_healthy healthcheck: test: ["CMD", "wget", "-O", "-", "http://localhost:8080/misc/ping"] -#TODO(berry): Traefik - volumes: app-db-data: diff --git a/database/init-user-db.sh b/database/init-user-db.sh index b13ccff2f..57a7e55a0 100755 --- a/database/init-user-db.sh +++ b/database/init-user-db.sh @@ -4,6 +4,5 @@ set -e # todo(berry): make user and database variables psql -v ON_ERROR_STOP=1 --username "$POSTGRES_USER" --dbname "$POSTGRES_DB" <<-EOSQL CREATE USER tad WITH PASSWORD 'changethis'; - CREATE DATABASE tad; - GRANT ALL PRIVILEGES ON DATABASE tad TO tad; + CREATE DATABASE tad OWNER tad; EOSQL diff --git a/docker-entrypoint.sh b/docker-entrypoint.sh new file mode 100755 index 000000000..fa5698e36 --- /dev/null +++ b/docker-entrypoint.sh @@ -0,0 +1,51 @@ +#!/usr/bin/env bash + +DATABASE_MIGRATE="" +HOST="0.0.0.0" +LOGLEVEL="warning" +PORT="8000" + +while getopts "dh:l:p:" opt; do + case $opt in + d) + DATABASE_MIGRATE="True" + ;; + h) + HOST=$OPTARG + ;; + l) + LOGLEVEL=$OPTARG + ;; + p) + PORT=$OPTARG + ;; + :) + echo "Option -${OPTARG} requires an argument." + exit 1 + ;; + + ?) + echo "Invalid option: $OPTARG" + + echo "Usage: docker-entrypoint.sh [-d] [-h host] [-l loglevel]" + exit 1 + ;; + esac +done + +echo "DATABASE_MIGRATE: $DATABASE_MIGRATE" +echo "HOST: $HOST" +echo "LOGLEVEL: $LOGLEVEL" +echo "PORT: $PORT" + + +if [ -z $DATABASE_MIGRATE ]; then + echo "Upgrading database" + if ! alembic upgrade head; then + echo "Failed to upgrade database" + exit 1 + fi +fi + +echo "Starting server" +python -m uvicorn --host "$HOST" tad.main:app --port "$PORT" --log-level "$LOGLEVEL" diff --git a/poetry.lock b/poetry.lock index 58726fe6a..1393833f1 100644 --- a/poetry.lock +++ b/poetry.lock @@ -761,68 +761,68 @@ files = [ [[package]] name = "orjson" -version = "3.10.3" +version = "3.10.4" description = "Fast, correct Python JSON library supporting dataclasses, datetimes, and numpy" optional = false python-versions = ">=3.8" files = [ - {file = "orjson-3.10.3-cp310-cp310-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:9fb6c3f9f5490a3eb4ddd46fc1b6eadb0d6fc16fb3f07320149c3286a1409dd8"}, - {file = "orjson-3.10.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:252124b198662eee80428f1af8c63f7ff077c88723fe206a25df8dc57a57b1fa"}, - {file = "orjson-3.10.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9f3e87733823089a338ef9bbf363ef4de45e5c599a9bf50a7a9b82e86d0228da"}, - {file = "orjson-3.10.3-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c8334c0d87103bb9fbbe59b78129f1f40d1d1e8355bbed2ca71853af15fa4ed3"}, - {file = "orjson-3.10.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1952c03439e4dce23482ac846e7961f9d4ec62086eb98ae76d97bd41d72644d7"}, - {file = "orjson-3.10.3-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:c0403ed9c706dcd2809f1600ed18f4aae50be263bd7112e54b50e2c2bc3ebd6d"}, - {file = "orjson-3.10.3-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:382e52aa4270a037d41f325e7d1dfa395b7de0c367800b6f337d8157367bf3a7"}, - {file = "orjson-3.10.3-cp310-none-win32.whl", hash = "sha256:be2aab54313752c04f2cbaab4515291ef5af8c2256ce22abc007f89f42f49109"}, - {file = "orjson-3.10.3-cp310-none-win_amd64.whl", hash = "sha256:416b195f78ae461601893f482287cee1e3059ec49b4f99479aedf22a20b1098b"}, - {file = "orjson-3.10.3-cp311-cp311-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:73100d9abbbe730331f2242c1fc0bcb46a3ea3b4ae3348847e5a141265479700"}, - {file = "orjson-3.10.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:544a12eee96e3ab828dbfcb4d5a0023aa971b27143a1d35dc214c176fdfb29b3"}, - {file = "orjson-3.10.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:520de5e2ef0b4ae546bea25129d6c7c74edb43fc6cf5213f511a927f2b28148b"}, - {file = "orjson-3.10.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ccaa0a401fc02e8828a5bedfd80f8cd389d24f65e5ca3954d72c6582495b4bcf"}, - {file = "orjson-3.10.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9a7bc9e8bc11bac40f905640acd41cbeaa87209e7e1f57ade386da658092dc16"}, - {file = "orjson-3.10.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:3582b34b70543a1ed6944aca75e219e1192661a63da4d039d088a09c67543b08"}, - {file = "orjson-3.10.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:1c23dfa91481de880890d17aa7b91d586a4746a4c2aa9a145bebdbaf233768d5"}, - {file = "orjson-3.10.3-cp311-none-win32.whl", hash = "sha256:1770e2a0eae728b050705206d84eda8b074b65ee835e7f85c919f5705b006c9b"}, - {file = "orjson-3.10.3-cp311-none-win_amd64.whl", hash = "sha256:93433b3c1f852660eb5abdc1f4dd0ced2be031ba30900433223b28ee0140cde5"}, - {file = "orjson-3.10.3-cp312-cp312-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:a39aa73e53bec8d410875683bfa3a8edf61e5a1c7bb4014f65f81d36467ea098"}, - {file = "orjson-3.10.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0943a96b3fa09bee1afdfccc2cb236c9c64715afa375b2af296c73d91c23eab2"}, - {file = "orjson-3.10.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e852baafceff8da3c9defae29414cc8513a1586ad93e45f27b89a639c68e8176"}, - {file = "orjson-3.10.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18566beb5acd76f3769c1d1a7ec06cdb81edc4d55d2765fb677e3eaa10fa99e0"}, - {file = "orjson-3.10.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1bd2218d5a3aa43060efe649ec564ebedec8ce6ae0a43654b81376216d5ebd42"}, - {file = "orjson-3.10.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:cf20465e74c6e17a104ecf01bf8cd3b7b252565b4ccee4548f18b012ff2f8069"}, - {file = "orjson-3.10.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:ba7f67aa7f983c4345eeda16054a4677289011a478ca947cd69c0a86ea45e534"}, - {file = "orjson-3.10.3-cp312-none-win32.whl", hash = "sha256:17e0713fc159abc261eea0f4feda611d32eabc35708b74bef6ad44f6c78d5ea0"}, - {file = "orjson-3.10.3-cp312-none-win_amd64.whl", hash = "sha256:4c895383b1ec42b017dd2c75ae8a5b862fc489006afde06f14afbdd0309b2af0"}, - {file = "orjson-3.10.3-cp38-cp38-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:be2719e5041e9fb76c8c2c06b9600fe8e8584e6980061ff88dcbc2691a16d20d"}, - {file = "orjson-3.10.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0175a5798bdc878956099f5c54b9837cb62cfbf5d0b86ba6d77e43861bcec2"}, - {file = "orjson-3.10.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:978be58a68ade24f1af7758626806e13cff7748a677faf95fbb298359aa1e20d"}, - {file = "orjson-3.10.3-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:16bda83b5c61586f6f788333d3cf3ed19015e3b9019188c56983b5a299210eb5"}, - {file = "orjson-3.10.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4ad1f26bea425041e0a1adad34630c4825a9e3adec49079b1fb6ac8d36f8b754"}, - {file = "orjson-3.10.3-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:9e253498bee561fe85d6325ba55ff2ff08fb5e7184cd6a4d7754133bd19c9195"}, - {file = "orjson-3.10.3-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:0a62f9968bab8a676a164263e485f30a0b748255ee2f4ae49a0224be95f4532b"}, - {file = "orjson-3.10.3-cp38-none-win32.whl", hash = "sha256:8d0b84403d287d4bfa9bf7d1dc298d5c1c5d9f444f3737929a66f2fe4fb8f134"}, - {file = "orjson-3.10.3-cp38-none-win_amd64.whl", hash = "sha256:8bc7a4df90da5d535e18157220d7915780d07198b54f4de0110eca6b6c11e290"}, - {file = "orjson-3.10.3-cp39-cp39-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:9059d15c30e675a58fdcd6f95465c1522b8426e092de9fff20edebfdc15e1cb0"}, - {file = "orjson-3.10.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8d40c7f7938c9c2b934b297412c067936d0b54e4b8ab916fd1a9eb8f54c02294"}, - {file = "orjson-3.10.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d4a654ec1de8fdaae1d80d55cee65893cb06494e124681ab335218be6a0691e7"}, - {file = "orjson-3.10.3-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:831c6ef73f9aa53c5f40ae8f949ff7681b38eaddb6904aab89dca4d85099cb78"}, - {file = "orjson-3.10.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:99b880d7e34542db89f48d14ddecbd26f06838b12427d5a25d71baceb5ba119d"}, - {file = "orjson-3.10.3-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:2e5e176c994ce4bd434d7aafb9ecc893c15f347d3d2bbd8e7ce0b63071c52e25"}, - {file = "orjson-3.10.3-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:b69a58a37dab856491bf2d3bbf259775fdce262b727f96aafbda359cb1d114d8"}, - {file = "orjson-3.10.3-cp39-none-win32.whl", hash = "sha256:b8d4d1a6868cde356f1402c8faeb50d62cee765a1f7ffcfd6de732ab0581e063"}, - {file = "orjson-3.10.3-cp39-none-win_amd64.whl", hash = "sha256:5102f50c5fc46d94f2033fe00d392588564378260d64377aec702f21a7a22912"}, - {file = "orjson-3.10.3.tar.gz", hash = "sha256:2b166507acae7ba2f7c315dcf185a9111ad5e992ac81f2d507aac39193c2c818"}, + {file = "orjson-3.10.4-cp310-cp310-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:afca963f19ca60c7aedadea9979f769139127288dd58ccf3f7c5e8e6dc62cabf"}, + {file = "orjson-3.10.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42b112eff36ba7ccc7a9d6b87e17b9d6bde4312d05e3ddf66bf5662481dee846"}, + {file = "orjson-3.10.4-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:02b192eaba048b1039eca9a0cef67863bd5623042f5c441889a9957121d97e14"}, + {file = "orjson-3.10.4-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:827c3d0e4fc44242c82bfdb1a773235b8c0575afee99a9fa9a8ce920c14e440f"}, + {file = "orjson-3.10.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ca8ec09724f10ec209244caeb1f9f428b6bb03f2eda9ed5e2c4dd7f2b7fabd44"}, + {file = "orjson-3.10.4-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:8eaa5d531a8fde11993cbcb27e9acf7d9c457ba301adccb7fa3a021bfecab46c"}, + {file = "orjson-3.10.4-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:e112aa7fc4ea67367ec5e86c39a6bb6c5719eddc8f999087b1759e765ddaf2d4"}, + {file = "orjson-3.10.4-cp310-none-win32.whl", hash = "sha256:1538844fb88446c42da3889f8c4ecce95a630b5a5ba18ecdfe5aea596f4dff21"}, + {file = "orjson-3.10.4-cp310-none-win_amd64.whl", hash = "sha256:de02811903a2e434127fba5389c3cc90f689542339a6e52e691ab7f693407b5a"}, + {file = "orjson-3.10.4-cp311-cp311-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:358afaec75de7237dfea08e6b1b25d226e33a1e3b6dc154fc99eb697f24a1ffa"}, + {file = "orjson-3.10.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bb4e292c3198ab3d93e5f877301d2746be4ca0ba2d9c513da5e10eb90e19ff52"}, + {file = "orjson-3.10.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5c39e57cf6323a39238490092985d5d198a7da4a3be013cc891a33fef13a536e"}, + {file = "orjson-3.10.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f86df433fc01361ff9270ad27455ce1ad43cd05e46de7152ca6adb405a16b2f6"}, + {file = "orjson-3.10.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0c9966276a2c97e93e6cbe8286537f88b2a071827514f0d9d47a0aefa77db458"}, + {file = "orjson-3.10.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:c499a14155a1f5a1e16e0cd31f6cf6f93965ac60a0822bc8340e7e2d3dac1108"}, + {file = "orjson-3.10.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:3087023ce904a327c29487eb7e1f2c060070e8dbb9a3991b8e7952a9c6e62f38"}, + {file = "orjson-3.10.4-cp311-none-win32.whl", hash = "sha256:f965893244fe348b59e5ce560693e6dd03368d577ce26849b5d261ce31c70101"}, + {file = "orjson-3.10.4-cp311-none-win_amd64.whl", hash = "sha256:c212f06fad6aa6ce85d5665e91a83b866579f29441a47d3865c57329c0857357"}, + {file = "orjson-3.10.4-cp312-cp312-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:d0965a8b0131959833ca8a65af60285995d57ced0de2fd8f16fc03235975d238"}, + {file = "orjson-3.10.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:27b64695d9f2aef3ae15a0522e370ec95c946aaea7f2c97a1582a62b3bdd9169"}, + {file = "orjson-3.10.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:867d882ddee6a20be4c8b03ae3d2b0333894d53ad632d32bd9b8123649577171"}, + {file = "orjson-3.10.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a0667458f8a8ceb6dee5c08fec0b46195f92c474cbbec71dca2a6b7fd5b67b8d"}, + {file = "orjson-3.10.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a3eac9befc4eaec1d1ff3bba6210576be4945332dde194525601c5ddb5c060d3"}, + {file = "orjson-3.10.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:4343245443552eae240a33047a6d1bcac7a754ad4b1c57318173c54d7efb9aea"}, + {file = "orjson-3.10.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:30153e269eea43e98918d4d462a36a7065031d9246407dfff2579a4e457515c1"}, + {file = "orjson-3.10.4-cp312-none-win32.whl", hash = "sha256:1a7d092ee043abf3db19c2183115e80676495c9911843fdb3ebd48ca7b73079e"}, + {file = "orjson-3.10.4-cp312-none-win_amd64.whl", hash = "sha256:07a2adbeb8b9efe6d68fc557685954a1f19d9e33f5cc018ae1a89e96647c1b65"}, + {file = "orjson-3.10.4-cp38-cp38-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:f5a746f3d908bce1a1e347b9ca89864047533bdfab5a450066a0315f6566527b"}, + {file = "orjson-3.10.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:465b4a8a3e459f8d304c19071b4badaa9b267c59207a005a7dd9dfe13d3a423f"}, + {file = "orjson-3.10.4-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:35858d260728c434a3d91b60685ab32418318567e8902039837e1c2af2719e0b"}, + {file = "orjson-3.10.4-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8a5ba090d40c4460312dd69c232b38c2ff67a823185cfe667e841c9dd5c06841"}, + {file = "orjson-3.10.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5dde86755d064664e62e3612a166c28298aa8dfd35a991553faa58855ae739cc"}, + {file = "orjson-3.10.4-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:020a9e9001cfec85c156ef3b185ff758b62ef986cefdb8384c4579facd5ce126"}, + {file = "orjson-3.10.4-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:3bf8e6e3388a2e83a86466c912387e0f0a765494c65caa7e865f99969b76ba0d"}, + {file = "orjson-3.10.4-cp38-none-win32.whl", hash = "sha256:c5a1cca6a4a3129db3da68a25dc0a459a62ae58e284e363b35ab304202d9ba9e"}, + {file = "orjson-3.10.4-cp38-none-win_amd64.whl", hash = "sha256:ecd97d98d7bee3e3d51d0b51c92c457f05db4993329eea7c69764f9820e27eb3"}, + {file = "orjson-3.10.4-cp39-cp39-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:71362daa330a2fc85553a1469185ac448547392a8f83d34e67779f8df3a52743"}, + {file = "orjson-3.10.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d24b59d1fecb0fd080c177306118a143f7322335309640c55ed9580d2044e363"}, + {file = "orjson-3.10.4-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e906670aea5a605b083ebb58d575c35e88cf880fa372f7cedaac3d51e98ff164"}, + {file = "orjson-3.10.4-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7ce32ed4bc4d632268e4978e595fe5ea07e026b751482b4a0feec48f66a90abc"}, + {file = "orjson-3.10.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1dcd34286246e0c5edd0e230d1da2daab2c1b465fcb6bac85b8d44057229d40a"}, + {file = "orjson-3.10.4-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:c45d4b8c403e50beedb1d006a8916d9910ed56bceaf2035dc253618b44d0a161"}, + {file = "orjson-3.10.4-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:aaed3253041b5002a4f5bfdf6f7b5cce657d974472b0699a469d439beba40381"}, + {file = "orjson-3.10.4-cp39-none-win32.whl", hash = "sha256:9a4f41b7dbf7896f8dbf559b9b43dcd99e31e0d49ac1b59d74f52ce51ab10eb9"}, + {file = "orjson-3.10.4-cp39-none-win_amd64.whl", hash = "sha256:6c4eb7d867ed91cb61e6514cb4f457aa01d7b0fd663089df60a69f3d38b69d4c"}, + {file = "orjson-3.10.4.tar.gz", hash = "sha256:c912ed25b787c73fe994a5decd81c3f3b256599b8a87d410d799d5d52013af2a"}, ] [[package]] name = "packaging" -version = "24.0" +version = "24.1" description = "Core utilities for Python packages" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "packaging-24.0-py3-none-any.whl", hash = "sha256:2ddfb553fdf02fb784c234c7ba6ccc288296ceabec964ad2eae3777778130bc5"}, - {file = "packaging-24.0.tar.gz", hash = "sha256:eb82c5e3e56209074766e6885bb04b8c38a0c015d0a30036ebe7ece34c9989e9"}, + {file = "packaging-24.1-py3-none-any.whl", hash = "sha256:5b8f2217dbdbd2f7f384c41c628544e6d52f2d0f53c6d0c3ea61aa5d1d7ff124"}, + {file = "packaging-24.1.tar.gz", hash = "sha256:026ed72c8ed3fcce5bf8950572258698927fd1dbda10a5e981cdf0ac37f4f002"}, ] [[package]] @@ -1588,13 +1588,13 @@ typing-extensions = ">=3.7.4.3" [[package]] name = "typing-extensions" -version = "4.12.1" +version = "4.12.2" description = "Backported and Experimental Type Hints for Python 3.8+" optional = false python-versions = ">=3.8" files = [ - {file = "typing_extensions-4.12.1-py3-none-any.whl", hash = "sha256:6024b58b69089e5a89c347397254e35f1bf02a907728ec7fee9bf0fe837d203a"}, - {file = "typing_extensions-4.12.1.tar.gz", hash = "sha256:915f5e35ff76f56588223f15fdd5938f9a1cf9195c0de25130c627e4d597f6d1"}, + {file = "typing_extensions-4.12.2-py3-none-any.whl", hash = "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d"}, + {file = "typing_extensions-4.12.2.tar.gz", hash = "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8"}, ] [[package]] @@ -1962,4 +1962,4 @@ files = [ [metadata] lock-version = "2.0" python-versions = "^3.10" -content-hash = "ff0867c4b530a26f3e3ace242fa75bc143ac2bc1b60d9d3b00b7814ff6f9a34b" +content-hash = "71302a929991fc2b1c1469cec6658f39910ec0428fe78324103da3c07a7d4fbe" diff --git a/.env b/prod.env similarity index 80% rename from .env rename to prod.env index b21bb0ee3..331ae2e21 100644 --- a/.env +++ b/prod.env @@ -1,9 +1,5 @@ -# Domain -DOMAIN=localhost - -# Environment: local, staging, production -ENVIRONMENT=local -PROJECT_NAME="TAD" +# Environment: local, production, demo +ENVIRONMENT=production # TAD backend BACKEND_CORS_ORIGINS="http://localhost,https://localhost,http://127.0.0.1,https://127.0.0.1" @@ -22,3 +18,4 @@ POSTGRES_PASSWORD=changethis # Database viewer PGADMIN_DEFAULT_PASSWORD=changethis +PGADMIN_DEFAULT_EMAIL=admin@admin.com diff --git a/pyproject.toml b/pyproject.toml index dbac15a55..7f2fff0c6 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -37,7 +37,6 @@ pyyaml = "^6.0.1" pytest = "^8.2.1" coverage = "^7.5.3" httpx = "^0.27.0" -urllib3 = "^2.2.1" playwright = "^1.44.0" pytest-playwright = "^0.5.0" @@ -107,10 +106,12 @@ title = "tad" testpaths = [ "tests" ] -addopts = "--strict-markers" +addopts = "--strict-markers -v -q" filterwarnings = [ "ignore::UserWarning" ] +log_cli = true +log_cli_level = "INFO" [tool.liccheck] level = "PARANOID" diff --git a/script/build b/script/build new file mode 100755 index 000000000..9252a9d78 --- /dev/null +++ b/script/build @@ -0,0 +1,5 @@ +#!/usr/bin/env bash + +set -x + +docker build . -t ghcr.io/minbzk/tad:latest "$@" diff --git a/script/format b/script/format index 00fd706fb..a8cc2b546 100755 --- a/script/format +++ b/script/format @@ -2,4 +2,4 @@ set -x -ruff format $@ +ruff format "$@" diff --git a/script/lint b/script/lint index 971b7902c..dd730f1b4 100755 --- a/script/lint +++ b/script/lint @@ -1,6 +1,5 @@ #!/usr/bin/env bash -set -e set -x -ruff check --fix $@ +ruff check --fix "$@" diff --git a/script/test b/script/test index 1f8a39418..b17492e52 100755 --- a/script/test +++ b/script/test @@ -3,8 +3,8 @@ set -e set -x -coverage run -m pytest $@ -if [ $? -ne 0 ]; then + +if ! coverage run -m pytest "$@" ; then echo "Test failed" exit 1 fi @@ -12,8 +12,8 @@ fi coverage report coverage html coverage lcov -pyright $@ -if [ $? -ne 0 ]; then + +if ! pyright; then echo "Typecheck failed" exit 1 fi diff --git a/tad/api/deps.py b/tad/api/deps.py new file mode 100644 index 000000000..a29bb2068 --- /dev/null +++ b/tad/api/deps.py @@ -0,0 +1,14 @@ +from fastapi import Request +from fastapi.templating import Jinja2Templates +from jinja2 import Environment + +from tad.core.config import get_settings + + +def version_context_processor(request: Request): + return {"version": get_settings().VERSION} + +env = Environment( + autoescape=True, +) +templates = Jinja2Templates(directory="tad/site/templates/",context_processors=[version_context_processor], env=env) diff --git a/tad/api/routes/deps.py b/tad/api/routes/deps.py deleted file mode 100644 index d5e96e330..000000000 --- a/tad/api/routes/deps.py +++ /dev/null @@ -1,15 +0,0 @@ -from fastapi import Request -from fastapi.templating import Jinja2Templates -from jinja2 import Environment - -from tad.core.config import settings - - -def version_context_processor(request: Request): - return {"version": settings.VERSION} - - -env = Environment( - autoescape=True, -) -templates = Jinja2Templates(directory=settings.TEMPLATE_DIR, context_processors=[version_context_processor], env=env) diff --git a/tad/api/routes/pages.py b/tad/api/routes/pages.py index f07972a69..648f6ac7f 100644 --- a/tad/api/routes/pages.py +++ b/tad/api/routes/pages.py @@ -3,7 +3,7 @@ from fastapi import APIRouter, Depends, Request from fastapi.responses import HTMLResponse -from tad.api.routes.deps import templates +from tad.api.deps import templates from tad.services.statuses import StatusesService from tad.services.tasks import TasksService diff --git a/tad/api/routes/root.py b/tad/api/routes/root.py index 7293628f3..1ba579ee4 100644 --- a/tad/api/routes/root.py +++ b/tad/api/routes/root.py @@ -1,7 +1,5 @@ from fastapi import APIRouter -from fastapi.responses import FileResponse, RedirectResponse - -from tad.core.config import settings +from fastapi.responses import RedirectResponse router = APIRouter() @@ -9,8 +7,3 @@ @router.get("/") async def base() -> RedirectResponse: return RedirectResponse("/pages/") - - -@router.get("/favicon.ico", include_in_schema=False) -async def favicon(): - return FileResponse(settings.STATIC_DIR + "/favicon.ico") diff --git a/tad/api/routes/tasks.py b/tad/api/routes/tasks.py index 1bc86e158..120434d29 100644 --- a/tad/api/routes/tasks.py +++ b/tad/api/routes/tasks.py @@ -3,7 +3,7 @@ from fastapi import APIRouter, Depends, Request, status from fastapi.responses import HTMLResponse -from tad.api.routes.deps import templates +from tad.api.deps import templates from tad.schema.task import MovedTask from tad.services.tasks import TasksService diff --git a/tad/core/config.py b/tad/core/config.py index a1bec7ae1..71b802014 100644 --- a/tad/core/config.py +++ b/tad/core/config.py @@ -1,5 +1,6 @@ import logging import secrets +from functools import lru_cache from typing import Any, TypeVar from pydantic import ( @@ -12,38 +13,26 @@ from tad.core.exceptions import SettingsError from tad.core.types import DatabaseSchemaType, EnvironmentType, LoggingLevelType +logger = logging.getLogger(__name__) + # Self type is not available in Python 3.10 so create our own with TypeVar SelfSettings = TypeVar("SelfSettings", bound="Settings") +PROJECT_NAME: str = "TAD" +PROJECT_DESCRIPTION: str = "Transparency of Algorithmic Decision making" +VERSION: str = "0.1.0" # replace in CI/CD pipeline + class Settings(BaseSettings): - # todo(berry): investigate yaml, toml or json file support for SettingsConfigDict - # todo(berry): investigate multiple .env files support for SettingsConfigDict - model_config = SettingsConfigDict( - env_file=(".env", ".env.test", ".env.prod"), env_ignore_empty=True, extra="ignore" - ) SECRET_KEY: str = secrets.token_urlsafe(32) - DOMAIN: str = "localhost" ENVIRONMENT: EnvironmentType = "local" - @computed_field # type: ignore[misc] - @property - def server_host(self) -> str: - if self.ENVIRONMENT == "local": - return f"http://{self.DOMAIN}" - return f"https://{self.DOMAIN}" - - VERSION: str = "0.1.0" - LOGGING_LEVEL: LoggingLevelType = "INFO" LOGGING_CONFIG: dict[str, Any] | None = None - PROJECT_NAME: str = "TAD" - PROJECT_DESCRIPTION: str = "Transparency of Algorithmic Decision making" - - STATIC_DIR: str = "tad/site/static/" - TEMPLATE_DIR: str = "tad/site/templates" + DEBUG: bool = False + AUTO_CREATE_SCHEMA: bool = False # todo(berry): create submodel for database settings APP_DATABASE_SCHEME: DatabaseSchemaType = "sqlite" @@ -55,22 +44,27 @@ def server_host(self) -> str: APP_DATABASE_PASSWORD: str | None = None APP_DATABASE_DB: str = "tad" - APP_DATABASE_FILE: str = "database.sqlite3" + APP_DATABASE_FILE: str = "/database.sqlite3" + + model_config = SettingsConfigDict(extra="ignore") @computed_field # type: ignore[misc] @property - def SQLALCHEMY_DATABASE_URI(self) -> str: - logging.info(f"test: {self.APP_DATABASE_SCHEME}") - - if self.APP_DATABASE_SCHEME == "sqlite": - return str(MultiHostUrl.build(scheme=self.APP_DATABASE_SCHEME, host="", path=self.APP_DATABASE_FILE)) + def SQLALCHEMY_ECHO(self) -> bool: + return self.DEBUG + @computed_field # type: ignore[misc] + @property + def SQLALCHEMY_DATABASE_URI(self) -> str: scheme: str = ( f"{self.APP_DATABASE_SCHEME}+{self.APP_DATABASE_DRIVER}" if isinstance(self.APP_DATABASE_DRIVER, str) else self.APP_DATABASE_SCHEME ) + if self.APP_DATABASE_SCHEME == "sqlite": + return f"{scheme}://{self.APP_DATABASE_FILE}" + return str( MultiHostUrl.build( scheme=scheme, @@ -84,9 +78,26 @@ def SQLALCHEMY_DATABASE_URI(self) -> str: @model_validator(mode="after") def _enforce_database_rules(self: SelfSettings) -> SelfSettings: - if self.ENVIRONMENT != "local" and self.APP_DATABASE_SCHEME == "sqlite": - raise SettingsError("SQLite is not supported in production") + if self.ENVIRONMENT == "production" and self.APP_DATABASE_SCHEME == "sqlite": + raise SettingsError("APP_DATABASE_SCHEME=SQLITE is not supported in production") + return self + + @model_validator(mode="after") + def _enforce_debug_rules(self: SelfSettings) -> SelfSettings: + if self.ENVIRONMENT == "production" and self.DEBUG: + raise SettingsError("DEBUG=True is not supported in production") + return self + + @model_validator(mode="after") + def _enforce_autocreate_rules(self: SelfSettings) -> SelfSettings: + if self.ENVIRONMENT == "production" and self.AUTO_CREATE_SCHEMA: + raise SettingsError("AUTO_CREATE_SCHEMA=True is not supported in production") return self -settings = Settings() # type: ignore +# TODO(berry): make it a function with lrucache + + +@lru_cache(maxsize=8) +def get_settings() -> Settings: + return Settings() diff --git a/tad/core/db.py b/tad/core/db.py index dda4f7c4a..e07be8b13 100644 --- a/tad/core/db.py +++ b/tad/core/db.py @@ -1,18 +1,63 @@ -from sqlalchemy.engine.base import Engine -from sqlmodel import Session, create_engine, select +import logging +from functools import lru_cache -from tad.core.config import settings +from sqlalchemy.engine import Engine +from sqlalchemy.pool import QueuePool, StaticPool +from sqlmodel import Session, SQLModel, create_engine, select -_engine: None | Engine = None +from tad.core.config import get_settings +from tad.models import Status, Task, User +logger = logging.getLogger(__name__) + +@lru_cache(maxsize=8) def get_engine() -> Engine: - global _engine - if _engine is None: - _engine = create_engine(settings.SQLALCHEMY_DATABASE_URI) - return _engine + connect_args = ( + {"check_same_thread": False, "isolation_level": None} if get_settings().APP_DATABASE_SCHEME == "sqlite" else {} + ) + poolclass = StaticPool if get_settings().APP_DATABASE_SCHEME == "sqlite" else QueuePool + + return create_engine( + str(get_settings().SQLALCHEMY_DATABASE_URI), + connect_args=connect_args, + poolclass=poolclass, + echo=get_settings().SQLALCHEMY_ECHO, + ) -async def check_db(): +def check_db(): + logger.info("Checking database connection") with Session(get_engine()) as session: session.exec(select(1)) + + logger.info("Finisch Checking database connection") + + +def init_db(): + logger.info("Initializing database") + + if get_settings().AUTO_CREATE_SCHEMA: + logger.info("Creating database schema") + SQLModel.metadata.create_all(get_engine()) + + with Session(get_engine()) as session: + if get_settings().ENVIRONMENT == "demo": + logger.info("Creating demo data") + + user = session.exec(select(User).where(User.name == "Robbert")).first() + if not user: + user = User(name="Robbert", avatar=None) + session.add(user) + + status = session.exec(select(Status).where(Status.name == "Todo")).first() + if not status: + status = Status(name="Todo", sort_order=1) + session.add(status) + + task = session.exec(select(Task).where(Task.title == "First task")).first() + if not task: + task = Task(title="First task", description="This is the first task", sort_order=1, status_id=status.id) + session.add(task) + session.commit() + logger.info("Finished initializing database") diff --git a/tad/core/log.py b/tad/core/log.py index 4850c4198..92eec496d 100644 --- a/tad/core/log.py +++ b/tad/core/log.py @@ -30,7 +30,7 @@ }, }, "loggers": { - "tad": {"handlers": ["console", "file"], "level": "DEBUG", "propagate": False}, + "": {"handlers": ["console", "file"], "level": "DEBUG", "propagate": False}, }, } diff --git a/tad/core/types.py b/tad/core/types.py index 4848e7f3b..a18f41398 100644 --- a/tad/core/types.py +++ b/tad/core/types.py @@ -1,5 +1,6 @@ from typing import Literal -EnvironmentType = Literal["local", "staging", "production"] +# TODO(berry): make enums and convert to types +EnvironmentType = Literal["local", "production", "demo"] LoggingLevelType = Literal["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"] DatabaseSchemaType = Literal["sqlite", "postgresql", "mysql", "oracle"] diff --git a/tad/main.py b/tad/main.py index 68740234f..f219d1aa7 100644 --- a/tad/main.py +++ b/tad/main.py @@ -5,12 +5,11 @@ from fastapi.exceptions import RequestValidationError from fastapi.responses import HTMLResponse from fastapi.staticfiles import StaticFiles -from fastapi.templating import Jinja2Templates from starlette.exceptions import HTTPException as StarletteHTTPException from tad.api.main import api_router -from tad.core.config import settings -from tad.core.db import check_db +from tad.core.config import PROJECT_DESCRIPTION, PROJECT_NAME, VERSION, get_settings +from tad.core.db import check_db, init_db from tad.core.exception_handlers import ( http_exception_handler as tad_http_exception_handler, ) @@ -22,39 +21,38 @@ from .middleware.route_logging import RequestLoggingMiddleware -configure_logging(settings.LOGGING_LEVEL, settings.LOGGING_CONFIG) - +configure_logging(get_settings().LOGGING_LEVEL, get_settings().LOGGING_CONFIG) logger = logging.getLogger(__name__) -mask = Mask(mask_keywords=["database_uri"]) # todo(berry): move lifespan to own file @asynccontextmanager async def lifespan(app: FastAPI): - logger.info(f"Starting {settings.PROJECT_NAME} version {settings.VERSION}") - logger.info(f"Settings: {mask.secrets(settings.model_dump())}") - # todo(berry): setup database connection - await check_db() + mask = Mask(mask_keywords=["database_uri"]) + check_db() + init_db() + logger.info(f"Starting {PROJECT_NAME} version {VERSION}") + logger.info(f"Settings: {mask.secrets(get_settings().model_dump())}") yield - logger.info(f"Stopping application {settings.PROJECT_NAME} version {settings.VERSION}") + logger.info(f"Stopping application {PROJECT_NAME} version {VERSION}") logging.shutdown() -templates = Jinja2Templates(directory="templates") - +# todo(berry): Create factor for FastAPI app app = FastAPI( lifespan=lifespan, - title=settings.PROJECT_NAME, - summary=settings.PROJECT_DESCRIPTION, - version=settings.VERSION, + title=PROJECT_NAME, + summary=PROJECT_DESCRIPTION, + version=VERSION, openapi_url=None, default_response_class=HTMLResponse, redirect_slashes=False, + debug=get_settings().DEBUG, ) app.add_middleware(RequestLoggingMiddleware) -app.mount("/static", StaticFiles(directory=settings.STATIC_DIR), name="static") +app.mount("/static", StaticFiles(directory="tad/site/static/"), name="static") @app.exception_handler(StarletteHTTPException) @@ -68,5 +66,3 @@ async def validation_exception_handler(request: Request, exc: RequestValidationE app.include_router(api_router) - -# todo (robbert) add init code for example tasks and statuses diff --git a/tad/migrations/versions/006c480a1920_a_message.py b/tad/migrations/versions/006c480a1920_a_message.py deleted file mode 100644 index fa83e759e..000000000 --- a/tad/migrations/versions/006c480a1920_a_message.py +++ /dev/null @@ -1,36 +0,0 @@ -"""a message - -Revision ID: 006c480a1920 -Revises: -Create Date: 2024-05-13 12:36:32.647256 - -""" - -from collections.abc import Sequence - -import sqlalchemy as sa -import sqlmodel.sql.sqltypes -from alembic import op - -# revision identifiers, used by Alembic. -revision: str = "006c480a1920" -down_revision: str | None = None -branch_labels: str | Sequence[str] | None = None -depends_on: str | Sequence[str] | None = None - - -def upgrade() -> None: - # ### commands auto generated by Alembic - please adjust! ### - op.create_table( - "hero", - sa.Column("id", sa.Integer(), nullable=False), - sa.Column("name", sqlmodel.sql.sqltypes.AutoString(), nullable=False), - sa.PrimaryKeyConstraint("id"), - ) - # ### end Alembic commands ### - - -def downgrade() -> None: - # ### commands auto generated by Alembic - please adjust! ### - op.drop_table("hero") - # ### end Alembic commands ### diff --git a/tad/migrations/versions/eb2eed884ae9_a_message.py b/tad/migrations/versions/b62dbd9468e4_create_status_user_and_task_table.py similarity index 70% rename from tad/migrations/versions/eb2eed884ae9_a_message.py rename to tad/migrations/versions/b62dbd9468e4_create_status_user_and_task_table.py index 4a8614219..d5912d550 100644 --- a/tad/migrations/versions/eb2eed884ae9_a_message.py +++ b/tad/migrations/versions/b62dbd9468e4_create_status_user_and_task_table.py @@ -1,8 +1,8 @@ -"""Create the user, status and task tables, drop table hero +"""Create Status, User and Task table -Revision ID: eb2eed884ae9 +Revision ID: b62dbd9468e4 Revises: -Create Date: 2024-05-14 13:36:23.551663 +Create Date: 2024-06-06 09:18:14.989874 """ @@ -13,7 +13,7 @@ from alembic import op # revision identifiers, used by Alembic. -revision: str = "eb2eed884ae9" +revision: str = "b62dbd9468e4" down_revision: str | None = None branch_labels: str | Sequence[str] | None = None depends_on: str | Sequence[str] | None = None @@ -21,14 +21,14 @@ def upgrade() -> None: # ### commands auto generated by Alembic - please adjust! ### - op.drop_table("hero") - op.create_table( + status = op.create_table( "status", sa.Column("id", sa.Integer(), nullable=False), sa.Column("name", sqlmodel.sql.sqltypes.AutoString(), nullable=False), sa.Column("sort_order", sa.Float(), nullable=False), sa.PrimaryKeyConstraint("id"), ) + op.create_table( "user", sa.Column("id", sa.Integer(), nullable=False), @@ -42,8 +42,12 @@ def upgrade() -> None: sa.Column("title", sqlmodel.sql.sqltypes.AutoString(), nullable=False), sa.Column("description", sqlmodel.sql.sqltypes.AutoString(), nullable=False), sa.Column("sort_order", sa.Float(), nullable=False), - sa.Column("status_id", sa.Integer(), nullable=False), + sa.Column("status_id", sa.Integer(), nullable=True), sa.Column("user_id", sa.Integer(), nullable=True), + sa.ForeignKeyConstraint( + ["status_id"], + ["status.id"], + ), sa.ForeignKeyConstraint( ["user_id"], ["user.id"], @@ -52,9 +56,21 @@ def upgrade() -> None: ) # ### end Alembic commands ### + # ### custom commands ### + op.bulk_insert( + status, + [ + {"name": "Todo", "sort_order": 1}, + {"name": "In Progress", "sort_order": 2}, + {"name": "Review", "sort_order": 3}, + {"name": "Done", "sort_order": 4}, + ], + ) + def downgrade() -> None: - # we do not delete any tables on a downgrade - pass # ### commands auto generated by Alembic - please adjust! ### + op.drop_table("task") + op.drop_table("user") + op.drop_table("status") # ### end Alembic commands ### diff --git a/tad/services/storage.py b/tad/services/storage.py index f62de852f..fdf842fb7 100644 --- a/tad/services/storage.py +++ b/tad/services/storage.py @@ -17,18 +17,18 @@ def close(self) -> None: class WriterFactory: @staticmethod - def get_writer(writer_type: str = "file", **kwargs: str) -> Writer: + def get_writer(writer_type: str = "file", **kwargs: Any) -> Writer: match writer_type: case "file": if not all(k in kwargs for k in ("location", "filename")): raise KeyError("The `location` or `filename` variables are not provided as input for get_writer()") - return FileSystemWriteService(location=str(kwargs["location"]), filename=str(kwargs["filename"])) + return FileSystemWriteService(location=Path(kwargs["location"]), filename=str(kwargs["filename"])) case _: raise ValueError(f"Unknown writer type: {writer_type}") class FileSystemWriteService(Writer): - def __init__(self, location: str = "./tests/data", filename: str = "system_card.yaml") -> None: + def __init__(self, location: str | Path = "./tests/data", filename: str = "system_card.yaml") -> None: self.location = location if not filename.endswith(".yaml"): raise ValueError(f"Filename {filename} must end with .yaml instead of .{filename.split('.')[-1]}") diff --git a/tad/services/tasks.py b/tad/services/tasks.py index 363b821dd..13c70ac42 100644 --- a/tad/services/tasks.py +++ b/tad/services/tasks.py @@ -52,8 +52,11 @@ def move_task( self.system_card.title = task.title self.storage_writer.write(self.system_card.model_dump()) + if not isinstance(status.id, int): + raise TypeError("status_id must be an integer") # pragma: no cover + # assign the task to the current user - if status.name == "in_progress": + if status.id > 1: task.user_id = 1 # update the status for the task (this may not be needed if the status has not changed) diff --git a/tad/site/templates/default_layout.jinja b/tad/site/templates/default_layout.jinja index 2dedbb795..cbea94c68 100644 --- a/tad/site/templates/default_layout.jinja +++ b/tad/site/templates/default_layout.jinja @@ -22,6 +22,7 @@ + diff --git a/tests/api/routes/test_pages.py b/tests/api/routes/test_pages.py index c177102ab..8111d9d9d 100644 --- a/tests/api/routes/test_pages.py +++ b/tests/api/routes/test_pages.py @@ -1,17 +1,17 @@ from fastapi.testclient import TestClient +from tests.constants import all_statusses, default_task from tests.database_test_utils import DatabaseTestUtils def test_get_main_page(client: TestClient, db: DatabaseTestUtils) -> None: - db.init( - [ - {"table": "status", "id": 1}, - {"table": "task", "id": 1, "status_id": 1}, - {"table": "task", "id": 2, "status_id": 1}, - ] - ) + # given + db.given([*all_statusses(), default_task()]) + + # when response = client.get("/pages/") + + # then assert response.status_code == 200 assert response.headers["content-type"] == "text/html; charset=utf-8" assert b"" in response.content diff --git a/tests/api/routes/test_root.py b/tests/api/routes/test_root.py index 6ff8b1f8f..735bb6252 100644 --- a/tests/api/routes/test_root.py +++ b/tests/api/routes/test_root.py @@ -4,15 +4,7 @@ def test_get_root(client: TestClient) -> None: response = client.get( "/", + follow_redirects=False, ) # todo (robbert) this is a quick test to see if we (most likely) get the expected page - assert response.status_code == 200 - assert response.headers["content-type"] == "text/html; charset=utf-8" - assert b"