import dataclasses import datetime import re import logging 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 )