diff --git a/.cookiecutter.json b/.cookiecutter.json index e80c2a10..202857ac 100644 --- a/.cookiecutter.json +++ b/.cookiecutter.json @@ -2,11 +2,12 @@ "_copy_without_render": [ "*.github" ], - "_template": "https://github.com/MartinBernstorff/swift-python-cookiecutter", + "_template": "https://github.com/MartinBernstorff/nimble-python-cookiecutter", "author": "Martin Bernstorff", "copyright_year": "2023", - "email": "martinbernstorfff@gmail.com", + "email": "martinbernstorff@gmail.com", "friendly_name": "Personal Mnemonic Medium", + "github_repo": "personal-mnemonic-medium", "github_user": "MartinBernstorff", "license": "MIT", "package_name": "personal_mnemonic_medium", diff --git a/.cruft.json b/.cruft.json index 3f72bab4..d6b1af7a 100644 --- a/.cruft.json +++ b/.cruft.json @@ -1,6 +1,6 @@ { - "template": "https://github.com/MartinBernstorff/swift-python-cookiecutter", - "commit": "525b1f682621953d7ef9deb6ad6cf7359ebd43c8", + "template": "https://github.com/MartinBernstorff/nimble-python-cookiecutter", + "commit": "80b90f37d8ce87ffb1ab97cf2b518fc0fd431dfb", "checkout": null, "context": { "cookiecutter": { @@ -8,15 +8,16 @@ "package_name": "personal_mnemonic_medium", "friendly_name": "Personal Mnemonic Medium", "author": "Martin Bernstorff", - "email": "martinbernstorfff@gmail.com", + "email": "martinbernstorff@gmail.com", "github_user": "MartinBernstorff", + "github_repo": "personal-mnemonic-medium", "version": "0.0.0", "copyright_year": "2023", "license": "MIT", "_copy_without_render": [ "*.github" ], - "_template": "https://github.com/MartinBernstorff/swift-python-cookiecutter" + "_template": "https://github.com/MartinBernstorff/nimble-python-cookiecutter" } }, "directory": null diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json index 841db021..ef8fca22 100644 --- a/.devcontainer/devcontainer.json +++ b/.devcontainer/devcontainer.json @@ -1,37 +1,39 @@ // For format details, see https://aka.ms/devcontainer.json. For config options, see the // README at: https://github.com/devcontainers/templates/tree/main/src/docker-existing-dockerfile { - "name": "Existing Dockerfile", - "build": { - // Sets the run context to one level up instead of the .devcontainer folder. - "context": "..", - // Update the 'dockerFile' property if you aren't using the standard 'Dockerfile' filename. - "dockerfile": "../Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": [ - "ms-python.python", - "charliermarsh.ruff", - "ms-python.black-formatter", - "ms-azuretools.vscode-docker", - "ms-vscode.makefile-tools", - "github.vscode-github-actions" - ] - } - }, - "features": { - "ghcr.io/devcontainers/features/github-cli:1": {} - }, - "postStartCommand": "pip install -e ." - // Features to add to the dev container. More info: https://containers.dev/features. - // "features": {}, - // Use 'forwardPorts' to make a list of ports inside the container available locally. - // "forwardPorts": [], - // Uncomment the next line to run commands after the container is created. - // "postCreateCommand": "cat /etc/os-release", - // Configure tool-specific properties. - // "customizations": {}, - // Uncomment to connect as an existing user other than the container default. More info: https://aka.ms/dev-containers-non-root. - // "remoteUser": "devcontainer" + "name": "Existing Dockerfile", + "build": { + // Sets the run context to one level up instead of the .devcontainer folder. + "context": "..", + // Update the 'dockerFile' property if you aren't using the standard 'Dockerfile' filename. + "dockerfile": "../Dockerfile", + "cacheFrom": "ghcr.io/martinbernstorff/personal-mnemonic-medium-devcontainer:latest" + }, + // "features": {}, + "customizations": { + "vscode": { + "extensions": [ + "GitHub.copilot", + "charliermarsh.ruff", + "ms-python.python", + "ms-python.vscode-pylance", + "GitHub.vscode-pull-request-github", + "ms-vscode.makefile-tools", + "github.vscode-github-actions", + ] + } + }, + "features": { + "ghcr.io/devcontainers/features/github-cli:1": {} + }, + "postStartCommand": "make install" + // Features to add to the dev container. More info: https://containers.dev/features. + // "features": {}, + // Use 'forwardPorts' to make a list of ports inside the container available locally. + // "forwardPorts": [], + // Uncomment the next line to run commands after the container is created. + // Configure tool-specific properties. + // "customizations": {}, + // Uncomment to connect as an existing user other than the container default. More info: https://aka.ms/dev-containers-non-root. + // "remoteUser": "devcontainer" } \ No newline at end of file diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md deleted file mode 100644 index 56f513c2..00000000 --- a/.github/pull_request_template.md +++ /dev/null @@ -1,8 +0,0 @@ -- [ ] I have considered whether this PR needs review, and requested a review if necessary. - -Fixes issue # - -# Notes for reviewers -Reviewers can skip X, but should pay attention to Y. - - \ No newline at end of file diff --git a/.github/workflows/dependabot_automerge.yml b/.github/workflows/dependabot_automerge.yml deleted file mode 100644 index 22d2ecd7..00000000 --- a/.github/workflows/dependabot_automerge.yml +++ /dev/null @@ -1,30 +0,0 @@ -# GitHub action to automerge dependabot PRs. Only merges if tests passes the -# branch protections in the repository settings. -# You can set branch protections in the repository under Settings > Branches > Add rule -name: automerge-bot-prs - -on: pull_request - -permissions: - contents: write - pull-requests: write - -jobs: - dependabot-automerge: - runs-on: ubuntu-latest - # if actor is dependabot or pre-commit-ci[bot] then run - if: ${{ github.actor == 'dependabot[bot]' }} - - steps: - # Checkout action is required for token to persist - - name: Enable auto-merge for Dependabot PRs - run: gh pr merge --auto --merge "$PR_URL" # Use Github CLI to merge automatically the PR - env: - PR_URL: ${{github.event.pull_request.html_url}} - GH_TOKEN: ${{ secrets.GITHUB_TOKEN }} - - - name: Auto approve dependabot PRs - if: ${{ github.actor == 'dependabot[bot]' }} - uses: hmarr/auto-approve-action@v3.1.0 - with: - github-token: ${{ secrets.GITHUB_TOKEN }} diff --git a/.github/workflows/validate.yml b/.github/workflows/tests.yml similarity index 54% rename from .github/workflows/validate.yml rename to .github/workflows/tests.yml index 8064d438..ac592502 100644 --- a/.github/workflows/validate.yml +++ b/.github/workflows/tests.yml @@ -1,15 +1,17 @@ -# GitHub action to check if pre-commit has been run. Runs from .pre-commit-config.yaml, where the pre-commit actions are. - -name: validate +# This workflow will install Python dependencies, run pytests and run notebooks +# then it will in python 3.9 (ubuntu-latest) create a badge with the coverage +# and add it to the PR. This badge will be updated if the PR is updated. +name: Tests on: - pull_request: - branches: [main] push: branches: [main] + pull_request: + branches: [main] jobs: - build: + build-and-test: + permissions: write-all concurrency: group: "${{ github.workflow }} @ ${{ github.ref }}" cancel-in-progress: true @@ -22,14 +24,15 @@ jobs: uses: docker/login-action@v2 with: registry: ghcr.io - username: ${{ github.repository_owner }} + username: MartinBernstorff password: ${{ secrets.GITHUB_TOKEN }} - name: Pre-build dev container image uses: devcontainers/ci@v0.3 with: - imageName: ghcr.io/martinbernstorff/personal-mnemonic-medium - cacheFrom: ghcr.io/martinbernstorff/personal-mnemonic-medium - push: always + imageName: ghcr.io/martinbernstorff/personal-mnemonic-medium-devcontainer + # cacheFrom: ghcr.io/martinbernstorff/personal-mnemonic-medium-devcontainer:latest + push: filter + refFilterForPush: refs/heads/main runCmd: - make validate \ No newline at end of file + make validate diff --git a/.vscode/launch.json b/.vscode/launch.json deleted file mode 100644 index b7594878..00000000 --- a/.vscode/launch.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - // Use IntelliSense to learn about possible attributes. - // Hover to view descriptions of existing attributes. - // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 - "version": "0.2.0", - "configurations": [ - { - "name": "Run main on Life Lessons", - "type": "python", - "request": "launch", - "program": "${file}", - "console": "integratedTerminal", - "args": [ - "/input/", - "Life.apkg" - ], - } - ] -} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json index 190db2e1..2483f957 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -1,12 +1,19 @@ { + "python.analysis.typeCheckingMode": "strict", "python.testing.pytestArgs": [ - "tests" + "personal_mnemonic_medium" ], "python.testing.unittestEnabled": false, "python.testing.pytestEnabled": true, - "python.analysis.typeCheckingMode": "strict", - "[python]": { - "editor.defaultFormatter": "ms-python.black-formatter" + "explorer.excludeGitIgnore": false, + "files.exclude": { + "**/.git": true, + "**/.svn": true, + "**/.hg": true, + "**/CVS": true, + "**/.DS_Store": true, + "**/Thumbs.db": true, + "**/BUILD": true }, - "python.formatting.provider": "none" + "python.analysis.diagnosticMode": "workspace" } \ No newline at end of file diff --git a/Dockerfile b/Dockerfile index 2c144c25..a01c1081 100644 --- a/Dockerfile +++ b/Dockerfile @@ -3,16 +3,16 @@ FROM python:3.11-bookworm # Set the working directory to /app WORKDIR /app +RUN pip install pyright +RUN pyright . # Install deps COPY pyproject.toml ./ -RUN pip install .[dev] -RUN pip install .[tests] +RUN pip install --upgrade .[dev] +RUN pip install --upgrade .[tests] # Ensure pyright builds correctly. # If run in make validate, it is run in parallel, which breaks its installation. -RUN pyright . - # Install the entire app COPY . /app RUN pip install -e . diff --git a/application/main.py b/application/main.py index 5a110106..22d55ec4 100644 --- a/application/main.py +++ b/application/main.py @@ -1,27 +1,31 @@ -from collections import defaultdict from pathlib import Path from time import sleep -from typing import Annotated, Any, Dict +from typing import Annotated, Any import sentry_sdk import typer +from functionalpy import Seq from personal_mnemonic_medium.card_pipeline import CardPipeline from personal_mnemonic_medium.exporters.anki.package_generator import ( AnkiPackageGenerator, ) from personal_mnemonic_medium.exporters.anki.sync import sync_deck -from personal_mnemonic_medium.note_factories.markdown import MarkdownNoteFactory +from personal_mnemonic_medium.note_factories.markdown import ( + MarkdownNoteFactory, +) from personal_mnemonic_medium.prompt_extractors.cloze_extractor import ( ClozePromptExtractor, ) -from personal_mnemonic_medium.prompt_extractors.qa_extractor import QAPromptExtractor +from personal_mnemonic_medium.prompt_extractors.qa_extractor import ( + QAPromptExtractor, +) from wasabi import Printer msg = Printer(timestamp=True) # helper for creating anki connect requests -def request(action: Any, **params: Any) -> Dict[str, Any]: +def request(action: Any, **params: Any) -> dict[str, Any]: return {"action": action, "params": params, "version": 6} @@ -30,12 +34,16 @@ def main( host_output_dir: Path, watch: Annotated[ bool, - typer.Option(help="Keep running, updating Anki deck every 15 seconds"), + typer.Option( + help="Keep running, updating Anki deck every 15 seconds" + ), ], ): """Run the thing.""" if not input_dir.exists(): - raise FileNotFoundError(f"Input directory {input_dir} does not exist") + raise FileNotFoundError( + f"Input directory {input_dir} does not exist" + ) if not host_output_dir.exists(): msg.info(f"Creating output directory {host_output_dir}") @@ -59,17 +67,17 @@ def main( ClozePromptExtractor(), ], card_exporter=AnkiPackageGenerator(), # Step 3, get the cards from the prompts - ).run( - input_path=input_dir, - ) - - decks = defaultdict(list) + ).run(input_path=input_dir) - for card in cards: - decks[card.deckname] += [card] + grouped_cards = ( + Seq(cards).group_by(lambda card: card.deckname).to_iter() + ) - for deck in decks: - deck_bundle = AnkiPackageGenerator().cards_to_deck_bundle(cards=decks[deck]) + for group in grouped_cards: + cards = group.group_contents.to_list() + deck_bundle = AnkiPackageGenerator().cards_to_deck_bundle( + cards=cards + ) sync_deck( deck_bundle=deck_bundle, sync_dir_path=host_output_dir, @@ -79,10 +87,16 @@ def main( if watch: sleep_seconds = 60 - msg.good(f"Sync complete, sleeping for {sleep_seconds} seconds") + msg.good( + f"Sync complete, sleeping for {sleep_seconds} seconds" + ) sleep(sleep_seconds) - main(input_dir=input_dir, watch=watch, host_output_dir=host_output_dir) + main( + input_dir=input_dir, + watch=watch, + host_output_dir=host_output_dir, + ) + - if __name__ == "__main__": typer.run(main) diff --git a/makefile b/makefile index a547efa1..f256dbf9 100644 --- a/makefile +++ b/makefile @@ -1,28 +1,41 @@ -lint: - @echo Running black - black . +SRC_PATH = src/personal_mnemonic_medium - @echo Running ruff - ruff check . --fix +install-dev: + pip install --upgrade .[dev] -test: - @echo ––– Testing ––– - pytest -n auto -rfE --failed-first --disable-warnings -q +install: + make install-dev + pip install -e . -type-check: - @echo ––– Running static type checks ––– - pyright . +test: ## Run tests + pytest tests -install: - pip install --upgrade -e .[dev,tests] +lint: ## Format code + ruff format . + ruff . --fix \ + --extend-select F401 \ + --extend-select F841 -validate: - @echo ––– Ensuring dependencies are up to date. This will take a few moments --- - @make install > /dev/null - @make lint && make type-check && make test +type-check: ## Type-check code + pyright $(SRC_PATH) -pr: - gh pr create -w - make validate +validate: ## Run all checks + make lint + make type-check + make test + +sync-pr: + git push --set-upstream origin HEAD git push - gh pr merge --auto --merge \ No newline at end of file + +create-pr: + gh pr create -w || true + +merge-pr: + gh pr merge --auto --merge --delete-branch + +pr: ## Run relevant tests before PR + make sync-pr + make create-pr + make validate + make merge-pr \ No newline at end of file diff --git a/pyproject.toml b/pyproject.toml index 66fb2f30..eb153203 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,24 +1,17 @@ -# v1 -# v2 [build-system] requires = ["setuptools>=61.0.0", "wheel", "setuptools_scm"] build-backend = "setuptools.build_meta" [project] name = "personal-mnemonic-medium" -version = "0.2.0" -authors = [ - { name = "Martin Bernstorff", email = "martinbernstorfff@gmail.com" }, -] +version = "0.0.0" +authors = [{ name = "Martin Bernstorff", email = "martinbernstorff@gmail.com" }] description = "Personal Mnemonic Medium" -classifiers = [ - "Operating System :: POSIX :: Linux", - "Operating System :: MacOS :: MacOS X", - "Operating System :: Microsoft :: Windows", - "Programming Language :: Python :: 3.10", -] -requires-python = ">=3.10" +classifiers = ["Programming Language :: Python :: 3.11"] +requires-python = ">=3.11" + dependencies = [ + "functionalpy==0.6.0", "misaka==2.1.1", "genanki==0.13.0", "typer==0.9.0", @@ -27,12 +20,19 @@ dependencies = [ "sentry-sdk==1.32.0", ] +[project.license] +file = "LICENSE" + +[project.readme] +file = "README.md" +content-type = "text/markdown" + [project.optional-dependencies] dev = [ "cruft==2.15.0", "pyright==1.1.328", "pre-commit==2.20.0", - "ruff==0.0.254", + "ruff==0.1.3", "black==22.8.0", ] tests = [ @@ -48,19 +48,20 @@ repository = "https://github.com/MartinBernstorff/personal-mnemonic-medium" documentation = "https://MartinBernstorff.github.io/personal-mnemonic-medium/" [tool.pyright] -exclude = [".*venv*", ".tox", "*.apkg"] +exclude = [".*venv*"] pythonPlatform = "Darwin" -typeCheckingMode = "basic" - +reportMissingTypeStubs = false [tool.ruff] # Enable pycodestyle (`E`) and Pyflakes (`F`) codes by default. +line-length = 70 select = [ "A", "ANN", "ARG", "B", "C4", + "C90", "COM", "D417", "E", @@ -75,7 +76,6 @@ select = [ "PLW", "PT", "UP", - "Q", "PTH", "RSE", "RET", @@ -83,7 +83,18 @@ select = [ "SIM", "W", ] -ignore = ["ANN101", "ANN401", "E402", "E501", "F401", "F841", "UP006", "RET504"] +ignore = [ + "ANN101", + "ANN401", + "E402", + "E501", + "F841", + "RET504", + "COM812", + "COM819", + "W191", +] +ignore-init-module-imports = true # Allow autofix for all enabled rules (when `--fix`) is provided. unfixable = ["ERA"] # Exclude a variety of commonly ignored directories. @@ -112,7 +123,10 @@ exclude = [ ] # Allow unused variables when underscore-prefixed. dummy-variable-rgx = "^(_+|(_+[a-zA-Z0-9_]*[a-zA-Z0-9]+?))$" -target-version = "py39" +target-version = "py311" + +[tool.ruff.format] +skip-magic-trailing-comma = true [tool.ruff.flake8-annotations] mypy-init-return = true @@ -120,47 +134,11 @@ suppress-none-returning = true [tool.ruff.isort] known-third-party = ["wandb"] +split-on-trailing-comma = false [tool.ruff.mccabe] # Unlike Flake8, default to a complexity level of 10. max-complexity = 10 -[tool.semantic_release] -branch = "main" -version_variable = ["pyproject.toml:version"] -upload_to_pypi = false -upload_to_release = false -build_command = "python -m pip install build; python -m build" - [tool.setuptools] include-package-data = true - - -[tool.tox] -legacy_tox_ini = """ -[tox] -envlist = py{39} - -[testenv] -description: run unit tests -extras = tests -use_develop = true -commands = - pytest -n auto {posargs:test} - -[testenv:type] -description: run type checks -extras = tests, dev -basepython = py39 # Setting these explicitly avoid recreating env if your shell is set to a different version -use_develop = true -commands = - pyright . - -[testenv:docs] -description: build docs -extras = docs -basepython = py39 # Setting these explicitly avoid recreating env if your shell is set to a different version -use_develop = true -commands = - sphinx-build -b html docs docs/_build/html -""" diff --git a/readme.md b/readme.md index 85108c0a..6116b5ec 100644 --- a/readme.md +++ b/readme.md @@ -1,5 +1,18 @@ # Personal Mnemonic Medium -[![Open in Dev Containers](https://img.shields.io/static/v1?label=Dev%20Containers&message=Open&color=blue&logo=visualstudiocode)](https://vscode.dev/redirect?url=vscode://ms-vscode-remote.remote-containers/cloneInVolume?url=https://github.com/MartinBernstorff/personal-mnemonic-medium/) + +[![PyPI](https://img.shields.io/pypi/v/personal-mnemonic-medium.svg)][pypi status] +[![Python Version](https://img.shields.io/pypi/pyversions/personal-mnemonic-medium)][pypi status] +[![documentation](https://github.com/MartinBernstorff/personal-mnemonic-medium/actions/workflows/documentation.yml/badge.svg)][documentation] +[![Tests](https://github.com/MartinBernstorff/personal-mnemonic-medium/actions/workflows/tests.yml/badge.svg)][tests] +[![Black](https://img.shields.io/badge/code%20style-black-000000.svg)][black] + +[pypi status]: https://pypi.org/project/personal-mnemonic-medium/ +[documentation]: https://MartinBernstorff.github.io/personal-mnemonic-medium/ +[tests]: https://github.com/MartinBernstorff/personal-mnemonic-medium/actions?workflow=Tests +[black]: https://github.com/psf/black + + + Extracting spaced repetition prompts (flashcards) from documents. @@ -11,7 +24,10 @@ A [Zettelkasten](https://medium.com/@martinbernstorf/why-you-need-an-idea-manage This thinking is largely inspired by Andy Matuschak's [Personal Mnemonic Medium](https://notes.andymatuschak.org/The_mnemonic_medium_can_be_extended_to_one%E2%80%99s_personal_notes), and the code is based on the unmaintained [Ankdown](https://github.com/benwr/ankdown). -FYI-style open source, maintenance is not guaranteed. + + +## Installation + ## Pipeline The left path describes the abstract pipeline, the right path the current instantiation in this repo. @@ -33,10 +49,27 @@ graph TD Prompts -- AnkiPackageGenerator --> Cards ``` -## Contributing -To get a full +### Setting up a dev environment +1. Install [Orbstack](https://orbstack.dev/) or Docker Desktop. Make sure to complete the full install process before continuing. +2. If not installed, install VSCode +3. Press this [link](https://vscode.dev/redirect?url=vscode://ms-vscode-remote.remote-containers/cloneInVolume?url=https://github.com/Aarhus-Psychiatry-Research/psycop-common) +4. Complete the setup process + +## Usage + +TODO: Add minimal usage example + +To see more examples, see the [documentation]. + +# 💬 Where to ask questions + +| Type | | +| ------------------------------ | ---------------------- | +| 🚨 **Bug Reports** | [GitHub Issue Tracker] | +| 🎁 **Feature Requests & Ideas** | [GitHub Issue Tracker] | +| 👩💻 **Usage Questions** | [GitHub Discussions] | +| 🗯 **General Discussion** | [GitHub Discussions] | -## Running through docker -To build and run the container, see `docker_cmd.sh`. +[github issue tracker]: https://github.com/MartinBernstorff/personal-mnemonic-medium/issues +[github discussions]: https://github.com/MartinBernstorff/personal-mnemonic-medium/discussions - diff --git a/src/personal_mnemonic_medium/card_pipeline.py b/src/personal_mnemonic_medium/card_pipeline.py index af01007b..c11becad 100644 --- a/src/personal_mnemonic_medium/card_pipeline.py +++ b/src/personal_mnemonic_medium/card_pipeline.py @@ -1,12 +1,17 @@ from collections.abc import Sequence from pathlib import Path -from typing import List -from personal_mnemonic_medium.exporters.anki.card_types.base import AnkiCard +from personal_mnemonic_medium.exporters.anki.card_types.base import ( + AnkiCard, +) from personal_mnemonic_medium.exporters.base import CardExporter -from personal_mnemonic_medium.note_factories.base import DocumentFactory +from personal_mnemonic_medium.note_factories.base import ( + DocumentFactory, +) from personal_mnemonic_medium.note_factories.note import Document -from personal_mnemonic_medium.prompt_extractors.base import PromptExtractor +from personal_mnemonic_medium.prompt_extractors.base import ( + PromptExtractor, +) from personal_mnemonic_medium.prompt_extractors.prompt import Prompt @@ -21,17 +26,18 @@ def __init__( self.prompt_extractors = prompt_extractors self.card_exporter = card_exporter - def run( - self, - input_path: Path, - ) -> List[AnkiCard]: - notes: List[Document] = [] + def run(self, input_path: Path) -> list[AnkiCard]: + notes: list[Document] = [] if input_path.is_dir(): - notes += list(self.document_factory.get_notes_from_dir(dir_path=input_path)) + notes += list( + self.document_factory.get_notes_from_dir( + dir_path=input_path + ) + ) if not input_path.is_dir(): note_from_file = self.document_factory.get_note_from_file( - file_path=input_path, + file_path=input_path ) notes.append(note_from_file) @@ -42,6 +48,6 @@ def run( collected_prompts += extractor.extract_prompts(note) cards: list[AnkiCard] = self.card_exporter.prompts_to_cards( - prompts=collected_prompts, + prompts=collected_prompts ) return cards diff --git a/src/personal_mnemonic_medium/exporters/anki/card_types/base.py b/src/personal_mnemonic_medium/exporters/anki/card_types/base.py index 109eb1d2..5f78591e 100644 --- a/src/personal_mnemonic_medium/exporters/anki/card_types/base.py +++ b/src/personal_mnemonic_medium/exporters/anki/card_types/base.py @@ -2,11 +2,11 @@ import os import re from abc import ABC, abstractmethod +from collections.abc import Callable, Iterator from pathlib import Path -from typing import Any, Callable, List, Optional, Tuple +from typing import Any import genanki -from personal_mnemonic_medium.exporters.anki.globals import CONFIG from personal_mnemonic_medium.exporters.markdown_to_html.html_compiler import ( compile_field, ) @@ -22,9 +22,11 @@ class AnkiCard(ABC): def __init__( self, - fields: List[str], + fields: list[str], source_prompt: Prompt, - url_generator: Callable[[Path, Optional[int]], str] = get_obsidian_url, + url_generator: Callable[ + [Path, int | None], str + ] = get_obsidian_url, html_compiler: Callable[[str], str] = compile_field, ): self.markdown_fields = fields @@ -38,11 +40,11 @@ def source_markdown(self) -> str: return self.source_doc.content @property - def html_fields(self) -> List[str]: + def html_fields(self) -> list[str]: return list(map(self.html_compiler, self.markdown_fields)) @property - def tags(self) -> List[str]: + def tags(self) -> list[str]: return self.source_doc.tags @property @@ -82,7 +84,7 @@ def deckname(self) -> str: + self.subdeck ) raise ValueError( - "Subdeck length is 0", + "Subdeck length is 0" ) # This is purposefully non-valid code except: # noqa return "0. Don't click me::1. Active::Personal Mnemonic Medium" @@ -98,21 +100,22 @@ def add_field(self, field: Any): def get_source_button(self) -> str: """Get the button to open the source document.""" url = self.url_generator( - self.source_doc.source_path, - self.source_prompt.line_nr, + self.source_doc.source_path, self.source_prompt.line_nr ) html = f'
", "").replace("
", "").strip(): n + n["fields"]["UUID"]["value"] + .replace("", "") + .replace("
", "") + .strip(): n for n in anki_notes_info } @@ -172,10 +210,12 @@ def sync_model(model: Model): model_names_to_ids = {} try: model_names_to_ids = invoke("modelNamesAndIds") - if model.name not in model_names_to_ids: + if model.name not in model_names_to_ids: # type: ignore return except Exception as e: - msg.good("\tUnable to fetch existing model names and ids from anki") + msg.good( + "\tUnable to fetch existing model names and ids from anki" + ) msg.good(f"\t\t{e}") if anki_connect_is_live(): @@ -183,30 +223,29 @@ def sync_model(model: Model): invoke( "updateModelTemplates", model={ - "name": model.name, + "name": model.name, # type: ignore "templates": { t["name"]: { "qfmt": t["qfmt"], "afmt": t["afmt"], } - for t in model.templates + for t in model.templates # type: ignore }, }, ) - msg.good(f"\tUpdated model {model.name} template") + msg.good(f"\tUpdated model {model.name} template") # type: ignore except Exception as e: - msg.good(f"\tUnable to update model {model.name} template") + msg.good( + f"\tUnable to update model {model.name} template" # type: ignore + ) msg.good(f"\t\t{e}") try: invoke( "updateModelStyling", - model={ - "name": model.name, - "css": model.css, - }, + model={"name": model.name, "css": model.css}, # type: ignore ) - msg.good(f"\tUpdated model {model.name} css") + msg.good(f"\tUpdated model {model.name} css") # type: ignore except Exception as e: - msg.good(f"\tUnable to update model {model.name} css") + msg.good(f"\tUnable to update model {model.name} css") # type: ignore msg.good(f"\t\t{e}") diff --git a/src/personal_mnemonic_medium/exporters/base.py b/src/personal_mnemonic_medium/exporters/base.py index f489121c..3e13b75f 100644 --- a/src/personal_mnemonic_medium/exporters/base.py +++ b/src/personal_mnemonic_medium/exporters/base.py @@ -1,11 +1,15 @@ from abc import ABC, abstractmethod from collections.abc import Sequence -from personal_mnemonic_medium.exporters.anki.card_types.base import AnkiCard +from personal_mnemonic_medium.exporters.anki.card_types.base import ( + AnkiCard, +) from personal_mnemonic_medium.prompt_extractors.prompt import Prompt class CardExporter(ABC): @abstractmethod - def prompts_to_cards(self, prompts: Sequence[Prompt]) -> list[AnkiCard]: + def prompts_to_cards( + self, prompts: Sequence[Prompt] + ) -> list[AnkiCard]: pass diff --git a/src/personal_mnemonic_medium/exporters/markdown_to_html/html_compiler.py b/src/personal_mnemonic_medium/exporters/markdown_to_html/html_compiler.py index d2732b52..9ffe3013 100644 --- a/src/personal_mnemonic_medium/exporters/markdown_to_html/html_compiler.py +++ b/src/personal_mnemonic_medium/exporters/markdown_to_html/html_compiler.py @@ -12,7 +12,10 @@ def field_to_html(field: Any) -> str: If math is separated with dollar sign it is converted to brackets. """ if CONFIG["dollar"]: - for sep, (op, cl) in [("$$", (r"\\[", r"\\]")), ("$", (r"\\(", r"\\)"))]: + for sep, (op, cl) in [ + ("$$", (r"\\[", r"\\]")), + ("$", (r"\\(", r"\\)")), + ]: escaped_sep = sep.replace(r"$", r"\$") # ignore escaped dollar signs when splitting the field field = re.split(rf"(? str: token_instances = re.findall(pattern, field) for instance in token_instances: - field = field.replace(instance, replacement + instance[1:-1] + replacement) # type: ignore + field = field.replace( + instance, + replacement + instance[1:-1] + replacement, # type: ignore + ) # type: ignore # Make sure every \n converts into a newline field = field.replace("\n", " \n") @@ -45,7 +51,11 @@ def field_to_html(field: Any) -> str: def compile_field(fieldtext: str) -> str: """Turn source markdown into an HTML field suitable for Anki.""" - fieldtext_sans_wiki = fieldtext.replace("[[", "").replace("]]", "") - fieldtext_sans_comments = re.sub(r"", "", fieldtext_sans_wiki) + fieldtext_sans_wiki = fieldtext.replace("[[", "").replace( + "]]", "" + ) + fieldtext_sans_comments = re.sub( + r"", "", fieldtext_sans_wiki + ) return field_to_html(fieldtext_sans_comments) diff --git a/src/personal_mnemonic_medium/exporters/url_generators/obsidian_url.py b/src/personal_mnemonic_medium/exporters/url_generators/obsidian_url.py index 43f2f92c..75d4d036 100644 --- a/src/personal_mnemonic_medium/exporters/url_generators/obsidian_url.py +++ b/src/personal_mnemonic_medium/exporters/url_generators/obsidian_url.py @@ -1,9 +1,10 @@ import urllib from pathlib import Path -from typing import Optional -def get_obsidian_url(source_path: Path, line_nr: Optional[int] = None) -> str: +def get_obsidian_url( + source_path: Path, line_nr: int | None = None +) -> str: """Get the obsidian URI for the source document.""" vault: str = urllib.parse.quote(source_path.parent.name) # type: ignore file: str = urllib.parse.quote(source_path.name) # type: ignore diff --git a/src/personal_mnemonic_medium/note_factories/base.py b/src/personal_mnemonic_medium/note_factories/base.py index cd64ce8e..75f91c78 100644 --- a/src/personal_mnemonic_medium/note_factories/base.py +++ b/src/personal_mnemonic_medium/note_factories/base.py @@ -7,7 +7,9 @@ class DocumentFactory(ABC): @abstractmethod - def get_notes_from_dir(self, dir_path: Path) -> Sequence[Document]: + def get_notes_from_dir( + self, dir_path: Path + ) -> Sequence[Document]: pass @abstractmethod diff --git a/src/personal_mnemonic_medium/note_factories/markdown.py b/src/personal_mnemonic_medium/note_factories/markdown.py index 8b548618..f6ba2f8d 100644 --- a/src/personal_mnemonic_medium/note_factories/markdown.py +++ b/src/personal_mnemonic_medium/note_factories/markdown.py @@ -2,11 +2,12 @@ import re from collections.abc import Sequence from pathlib import Path -from typing import Optional from tqdm import tqdm -from personal_mnemonic_medium.note_factories.base import DocumentFactory +from personal_mnemonic_medium.note_factories.base import ( + DocumentFactory, +) from personal_mnemonic_medium.note_factories.note import Document @@ -31,7 +32,7 @@ def get_and_append_new_uuid(self, file_path: Path) -> str: def get_note_id(self, file_string: str) -> str: return re.findall(r"" - expected_id = ( - r"" - ) + expected_id = r"" extracted_id = factory.get_note_id(note_str) @@ -133,21 +136,29 @@ def test_get_bear_id(): def test_alias_wiki_link_substitution(): alias = "Here I am [[alias|wiki link]], and another [[alias2|wiki link2]]" - output = Document._replace_alias_wiki_links(alias) - assert output == "Here I am [[wiki link]], and another [[wiki link2]]" + output = Document.replace_alias_wiki_links(alias) + assert ( + output + == "Here I am [[wiki link]], and another [[wiki link2]]" + ) no_alias = "Here I am [[wiki link]] and another [[wiki link2]]" - output = Document._replace_alias_wiki_links(no_alias) - assert output == "Here I am [[wiki link]] and another [[wiki link2]]" + output = Document.replace_alias_wiki_links(no_alias) + assert ( + output == "Here I am [[wiki link]] and another [[wiki link2]]" + ) test_3 = "How was ice climbing [[Franz Josef]] with [[Vibeke Christiansen|Vibeke]]?" - output = Document._replace_alias_wiki_links(test_3) - assert output == "How was ice climbing [[Franz Josef]] with [[Vibeke]]?" + output = Document.replace_alias_wiki_links(test_3) + assert ( + output + == "How was ice climbing [[Franz Josef]] with [[Vibeke]]?" + ) alias = "[[Isolation (database design)|Isolation]]" - output = Document._replace_alias_wiki_links(alias) + output = Document.replace_alias_wiki_links(alias) assert output == "[[Isolation]]" alias = "[[test-test|test-]]" - output = Document._replace_alias_wiki_links(alias) + output = Document.replace_alias_wiki_links(alias) assert output == "[[test-]]" diff --git a/tests/exporters/anki/test_package_generator.py b/tests/exporters/anki/test_package_generator.py index 73c40555..ba41cfdf 100644 --- a/tests/exporters/anki/test_package_generator.py +++ b/tests/exporters/anki/test_package_generator.py @@ -1,13 +1,18 @@ from pathlib import Path -import genanki -from personal_mnemonic_medium.exporters.anki.card_types.base import AnkiCard -from personal_mnemonic_medium.exporters.anki.card_types.qa import AnkiQA +from personal_mnemonic_medium.exporters.anki.card_types.base import ( + AnkiCard, +) +from personal_mnemonic_medium.exporters.anki.card_types.qa import ( + AnkiQA, +) from personal_mnemonic_medium.exporters.anki.package_generator import ( AnkiPackageGenerator, ) from personal_mnemonic_medium.note_factories.note import Document -from personal_mnemonic_medium.prompt_extractors.qa_extractor import QAPrompt +from personal_mnemonic_medium.prompt_extractors.qa_extractor import ( + QAPrompt, +) def test_cards_to_decks(): @@ -31,10 +36,7 @@ def test_cards_to_decks(): for _ in range(4) ] - deck, media = AnkiPackageGenerator().cards_to_deck(cards=genanki_notes) - - assert type(deck) == genanki.Deck - assert type(media) == set + AnkiPackageGenerator().cards_to_deck(cards=genanki_notes) def test_package_generators(): @@ -58,6 +60,4 @@ def test_package_generators(): for _ in range(4) ] - AnkiPackageGenerator().cards_to_deck_bundle( - cards=genanki_notes, - ) + AnkiPackageGenerator().cards_to_deck_bundle(cards=genanki_notes) diff --git a/tests/note_factories/test_markdown_extractor.py b/tests/note_factories/test_markdown_extractor.py index ddd8636c..df8a0459 100644 --- a/tests/note_factories/test_markdown_extractor.py +++ b/tests/note_factories/test_markdown_extractor.py @@ -2,15 +2,27 @@ PROJECT_ROOT = Path(__file__).parent.parent.parent -from personal_mnemonic_medium.note_factories.markdown import MarkdownNoteFactory +from personal_mnemonic_medium.note_factories.markdown import ( + MarkdownNoteFactory, +) def test_get_notes_from_dir(): notes = MarkdownNoteFactory().get_notes_from_dir( - PROJECT_ROOT / "tests" / "test_md_files", + PROJECT_ROOT / "tests" / "test_md_files" ) assert len(notes) == 4 - assert len([note for note in notes if note.title == "test_card_guid"]) == 1 - assert len([note for note in notes if "7696CDCD" in note.content]) == 1 - assert len([note for note in notes if "7696CDCD" in note.uuid]) == 1 + assert ( + len( + [note for note in notes if note.title == "test_card_guid"] + ) + == 1 + ) + assert ( + len([note for note in notes if "7696CDCD" in note.content]) + == 1 + ) + assert ( + len([note for note in notes if "7696CDCD" in note.uuid]) == 1 + ) diff --git a/tests/prompt_extractors/test_cloze_extractor.py b/tests/prompt_extractors/test_cloze_extractor.py index 1d5f5678..8267a4bb 100644 --- a/tests/prompt_extractors/test_cloze_extractor.py +++ b/tests/prompt_extractors/test_cloze_extractor.py @@ -32,6 +32,8 @@ def test_cloze_no_hits(): source_path=Path(__file__), ) - prompts = ClozePromptExtractor().extract_prompts(note_without_cloze) + prompts = ClozePromptExtractor().extract_prompts( + note_without_cloze + ) assert len(prompts) == 0 diff --git a/tests/prompt_extractors/test_qa_prompt_extractor.py b/tests/prompt_extractors/test_qa_prompt_extractor.py index 233ecd49..d3631c9b 100644 --- a/tests/prompt_extractors/test_qa_prompt_extractor.py +++ b/tests/prompt_extractors/test_qa_prompt_extractor.py @@ -2,7 +2,9 @@ import pytest from personal_mnemonic_medium.note_factories.note import Document -from personal_mnemonic_medium.prompt_extractors.qa_extractor import QAPromptExtractor +from personal_mnemonic_medium.prompt_extractors.qa_extractor import ( + QAPromptExtractor, +) @pytest.fixture() @@ -35,18 +37,26 @@ def test_has_qa_matches(qa_extractor: QAPromptExtractor): "QA. Testing something else, even with QA in it!", "\\Q. Testing newlines as well!", ] - matches = [string for string in example_strings if qa_extractor._has_qa(string)] + matches = [ + string + for string in example_strings + if qa_extractor._has_qa(string) # type: ignore + ] assert len(matches) == 3 def test_has_qa_does_not_match(qa_extractor: QAPromptExtractor): - example_strings = ["\nQ.E.D.", "> A question like this, or", "::Q. A comment!::"] + example_strings = [ + "\nQ.E.D.", + "> A question like this, or", + "::Q. A comment!::", + ] matches = 0 for string in example_strings: - if qa_extractor._has_qa(string): + if qa_extractor._has_qa(string): # type: ignore matches += 1 assert matches == 0