""" Database management functionality for MarkiTect. This module provides SQLite database initialization and markdown file storage with front matter support. """ import sqlite3 import json import os from datetime import datetime from pathlib import Path from typing import Optional, Dict, Any from .frontmatter import FrontMatterParser class DatabaseManager: """Manager for SQLite database operations.""" def __init__(self, db_path: str): """ Initialize database manager. Args: db_path: Path to SQLite database file """ self.db_path = db_path self.front_matter_parser = FrontMatterParser() def initialize_database(self) -> None: """ Initialize SQLite database with required tables. Creates the markdown_files table with the following schema: - id: INTEGER PRIMARY KEY - filename: TEXT NOT NULL - front_matter: TEXT (JSON) - content: TEXT - created_at: TIMESTAMP DEFAULT CURRENT_TIMESTAMP """ # Ensure directory exists db_dir = os.path.dirname(self.db_path) if db_dir and not os.path.exists(db_dir): os.makedirs(db_dir) conn = sqlite3.connect(self.db_path) cursor = conn.cursor() # Create markdown_files table cursor.execute(''' CREATE TABLE IF NOT EXISTS markdown_files ( id INTEGER PRIMARY KEY AUTOINCREMENT, filename TEXT NOT NULL, front_matter TEXT, content TEXT, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ) ''') conn.commit() conn.close() def store_markdown_file(self, filename: str, content: str) -> Optional[int]: """ Store a markdown file in the database. Args: filename: Name of the markdown file content: Raw markdown content with optional front matter Returns: ID of the inserted record, or None if insertion failed """ # Parse front matter and content front_matter, markdown_content = self.front_matter_parser.parse(content) # Convert front matter to JSON string front_matter_json = json.dumps(front_matter) if front_matter else '{}' conn = sqlite3.connect(self.db_path) cursor = conn.cursor() try: cursor.execute(''' INSERT INTO markdown_files (filename, front_matter, content, created_at) VALUES (?, ?, ?, ?) ''', (filename, front_matter_json, markdown_content, datetime.now())) record_id = cursor.lastrowid conn.commit() return record_id except sqlite3.Error: conn.rollback() return None finally: conn.close() def get_markdown_file(self, filename: str) -> Optional[Dict[str, Any]]: """ Retrieve a markdown file from the database. Args: filename: Name of the markdown file to retrieve Returns: Dictionary containing file data, or None if not found """ conn = sqlite3.connect(self.db_path) cursor = conn.cursor() cursor.execute(''' SELECT id, filename, front_matter, content, created_at FROM markdown_files WHERE filename = ? ''', (filename,)) row = cursor.fetchone() conn.close() if row: return { 'id': row[0], 'filename': row[1], 'front_matter': json.loads(row[2]) if row[2] else {}, 'content': row[3], 'created_at': row[4] } return None def list_markdown_files(self) -> list: """ List all markdown files in the database. Returns: List of dictionaries containing file metadata """ conn = sqlite3.connect(self.db_path) cursor = conn.cursor() cursor.execute(''' SELECT id, filename, front_matter, created_at FROM markdown_files ORDER BY created_at DESC ''') rows = cursor.fetchall() conn.close() files = [] for row in rows: files.append({ 'id': row[0], 'filename': row[1], 'front_matter': json.loads(row[2]) if row[2] else {}, 'created_at': row[3] }) return files