1
0
favro-sync/favro_sync/favro_markdown.py
Jon Michael Aanes 016a01cece
All checks were successful
Run Python tests (through Pytest) / Test (push) Successful in 27s
Verify Python project can be installed, loaded and have version checked / Test (push) Successful in 25s
Ruff
2025-01-08 15:26:07 +01:00

184 lines
6.2 KiB
Python

import dataclasses
import datetime
import logging
import re
import frontmatter
import marko
import marko.md_renderer
logger = logging.getLogger(__name__)
################################################################################
# FrontMatter keys
FM_KEY_TODO_LIST_COMPLETED = 'todo-list-completed'
FM_KEY_TAGS = 'tags'
FM_KEY_ASSIGNMENTS = 'assignments'
FM_KEY_DEPENDENCIES = 'dependencies'
FM_KEY_URL = 'url'
FM_KEY_ALIASES = 'aliases'
FM_KEY_ARCHIVED = 'archived'
FM_KEY_DUE_DATE = 'due'
FM_KEY_START_DATE = 'start-date'
################################################################################
@dataclasses.dataclass(frozen=True)
class CardContents:
identifier: str | None
name: str | None
description: str | None
tags: list[str]
assignments: list[str]
card_dependencies: list[str]
url: str
todo_list_completed: bool | None
is_archived: bool
start_date: datetime.date | None
due_date: datetime.date | None
custom_fields: dict[str, str]
def format_obsidian_link(text: str) -> str:
return f'[[{text}]]'
def parse_obsidian_link(text: str) -> str:
if m := re.match(r'^\[\[(.*)\]\]$', text):
return m.group(1)
return text
class CardFileFormatter:
"""Component for formatting and parsing card files."""
def __init__(self, obsidian_mode=True):
"""Initialize card formatter.
Arguments:
- `obsidian_mode`: Configure formatter to enable maximum compatibility
with [Obsidian](https://obsidian.md/). Enables internal links, and
exposes certain aliases.
"""
self.obsidian_mode = obsidian_mode
self.markdown = marko.Markdown()
self.renderer = marko.md_renderer.MarkdownRenderer()
def format_card_contents(self, card: CardContents) -> str:
"""Formats card contents. Mostly the inverse of [`parse_card_contents`]."""
logger.info('Formatting card: %s', card.identifier)
# Choose frontmatter data
frontmatter_data = {}
if self.obsidian_mode:
aliases = []
if card.name:
aliases.append(card.name)
if card.identifier and card.name:
aliases.append(f'{card.identifier}: {card.name}')
if aliases:
frontmatter_data[FM_KEY_ALIASES] = aliases
del aliases
if len(card.tags) > 0:
frontmatter_data[FM_KEY_TAGS] = card.tags
if card.url is not None and self.obsidian_mode:
frontmatter_data[FM_KEY_URL] = card.url
if len(card.assignments) > 0:
frontmatter_data[FM_KEY_ASSIGNMENTS] = card.assignments
if self.obsidian_mode:
frontmatter_data[FM_KEY_ASSIGNMENTS] = [
format_obsidian_link(name)
for name in frontmatter_data[FM_KEY_ASSIGNMENTS]
]
if len(card.card_dependencies) > 0:
frontmatter_data[FM_KEY_DEPENDENCIES] = card.card_dependencies
if self.obsidian_mode:
frontmatter_data[FM_KEY_DEPENDENCIES] = [
format_obsidian_link(name)
for name in frontmatter_data[FM_KEY_DEPENDENCIES]
]
if card.todo_list_completed is not None:
frontmatter_data[FM_KEY_TODO_LIST_COMPLETED] = card.todo_list_completed
if card.is_archived is not None:
frontmatter_data[FM_KEY_ARCHIVED] = card.is_archived
if card.due_date is not None:
frontmatter_data[FM_KEY_DUE_DATE] = card.due_date
if card.start_date is not None:
frontmatter_data[FM_KEY_DUE_DATE] = card.start_date
if len(card.custom_fields) > 0:
frontmatter_data |= card.custom_fields
# Card name
ls = []
if card.name:
ls.append('# ')
ls.append(card.name)
ls.append('\n\n')
# Card contents
if description := card.description:
if self.obsidian_mode:
description = re.sub(
r'\-\s*\[\s*\]',
'- [ ]',
description,
flags=re.MULTILINE,
)
ls.append(description)
del description
fm = frontmatter.Post(''.join(ls), **frontmatter_data)
return frontmatter.dumps(fm)
def parse_card_contents(self, contents: str) -> CardContents:
"""Parses card contents. Mostly the inverse of [`parse_card_contents`].
1. Strips frontmatter and parses certain fields from the header.
2. Parses header
3. Finds content.
"""
logger.info('Parsing card contents (len %d)', len(contents))
fm = frontmatter.loads(contents)
del contents
document = self.markdown.parse(fm.content.strip())
name = None
for elem in document.children:
if isinstance(elem, marko.block.Heading):
name = self.renderer.render_children(elem)
document.children.remove(elem)
break
tags: list[str] = fm.metadata.get(FM_KEY_TAGS, [])
assignments: list[str] = fm.metadata.get(FM_KEY_ASSIGNMENTS, [])
assignments = [parse_obsidian_link(text) for text in assignments]
card_dependencies: list[str] = fm.metadata.get(FM_KEY_DEPENDENCIES, [])
card_dependencies = [parse_obsidian_link(text) for text in card_dependencies]
url: list[str] = fm.metadata.get(FM_KEY_URL)
todo_list_completed: bool | None = fm.metadata.get(FM_KEY_TODO_LIST_COMPLETED)
is_archived: bool = fm.metadata.get(FM_KEY_ARCHIVED)
start_date: datetime.date = fm.metadata.get(FM_KEY_START_DATE)
due_date: datetime.date = fm.metadata.get(FM_KEY_DUE_DATE)
description = self.renderer.render_children(document).strip()
return CardContents(
None,
name,
description,
tags=tags,
assignments=assignments,
card_dependencies=card_dependencies,
url=url,
todo_list_completed=todo_list_completed,
is_archived=is_archived,
start_date=start_date,
due_date=due_date,
custom_fields={}, # TODO
)