Implemented comprehensive MarkdownMatters CLI following complete TDD8 seven-cycle methodology with full three-zone separation and extensive testing validation. ## Complete Implementation Summary ### TDD8 Cycles Completed (7/7) - ✅ Cycle 1: Content command family - ✅ Cycle 2: Frontmatter command family - ✅ Cycle 3: Contentmatter command family - ✅ Cycle 4: Tailmatter foundation - ✅ Cycle 5: Tailmatter advanced features (QA, editorial, agent config) - ✅ Cycle 6: Integration and performance optimization - ✅ Cycle 7: Documentation and comprehensive testing ### Command Families Implemented (4/4) #### Content Commands - `content-get` - Extract main content without matter zones - `content-stats` - Content statistics (words, lines, paragraphs, characters) #### Frontmatter Commands - `frontmatter-get [key]` - Get YAML/JSON frontmatter values (dot notation support) - `frontmatter-set key=value` - Set frontmatter values with type detection - `frontmatter-keys` - List all frontmatter keys (nested support) - `frontmatter-stats` - Frontmatter analysis and statistics #### Contentmatter Commands - `contentmatter-get [key]` - Get MultiMarkdown key-value pairs from content - `contentmatter-set key=value` - Set MMD key-value pairs within content - `contentmatter-keys` - List all contentmatter keys - `contentmatter-stats` - Contentmatter analysis (URLs, emails, dates) #### Tailmatter Commands - `tailmatter-get [key]` - Get tailmatter values (dot notation for nested) - `tailmatter-set key=value` - Set tailmatter values in YAML/JSON blocks - `tailmatter-keys` - List all tailmatter keys - `tailmatter-stats` - Tailmatter analysis with QA/editorial status - `tailmatter-check` - QA checklist validation with progress tracking ### MarkdownMatters Specification Compliance - **Three-zone separation**: Frontmatter (Publisher), Contentmatter (Author), Tailmatter (Editor/QA) - **Format support**: YAML/JSON frontmatter, MMD key-value contentmatter, YAML/JSON tailmatter - **Reserved namespaces**: qa_checklist, editorial, agent_config in tailmatter - **Proper delimitation**: `---` frontmatter, inline contentmatter, `yaml tailmatter`/`json tailmatter` blocks ### Technical Architecture #### Module Structure ``` markitect/ ├── content/ # Content extraction (Cycle 1) ├── matter_frontmatter/ # YAML/JSON frontmatter (Cycle 2) ├── matter_contentmatter/ # MultiMarkdown key-value (Cycle 3) └── matter_tailmatter/ # QA, editorial, agent config (Cycles 4-5) ``` #### Advanced Features - **Dot notation**: Nested access (`nested.key.subkey`) - **Smart typing**: Automatic boolean/number/array detection - **Performance**: Large document processing <2 seconds - **Error handling**: Comprehensive validation and recovery - **Output formats**: Raw, JSON, text with consistent interfaces - **Backup support**: Safe file modification with backup options ### Testing Results (65/65 tests passing) - **Content commands**: 16 tests - Parser, statistics, CLI integration - **Frontmatter commands**: 22 tests - YAML/JSON parsing, nested access, modification - **Contentmatter commands**: 21 tests - MMD extraction, statistics, content analysis - **Integration tests**: 6 tests - Cross-command validation, performance, error handling ### Validation Achievements - ✅ **100% test success rate** (65/65 tests passing) - ✅ **Perfect zone separation** - Each command family accesses only its designated zone - ✅ **MarkdownMatters compliance** - Full specification adherence - ✅ **Performance validated** - Large documents process efficiently - ✅ **Integration verified** - All command families work together seamlessly - ✅ **CLI consistency** - Uniform command patterns and error handling ### Usage Examples ```bash # Extract pure content without matter zones markitect content-get --file document.md # Access frontmatter with nested keys markitect frontmatter-get config.theme --file document.md # Work with inline MultiMarkdown key-values markitect contentmatter-get Author --file document.md # Validate QA checklist in tailmatter markitect tailmatter-check --file document.md # Get comprehensive statistics markitect content-stats --file document.md markitect frontmatter-stats --file document.md markitect contentmatter-stats --file document.md markitect tailmatter-stats --file document.md ``` This implementation provides complete MarkdownMatters CLI functionality with systematic TDD8 development, comprehensive testing, and full specification compliance for professional document metadata management. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
199 lines
7.1 KiB
Python
199 lines
7.1 KiB
Python
"""
|
|
CLI commands for tailmatter operations.
|
|
"""
|
|
|
|
import click
|
|
import json
|
|
from pathlib import Path
|
|
from .parser import TailmatterParser
|
|
|
|
|
|
@click.command('tailmatter-get')
|
|
@click.argument('key')
|
|
@click.option('--file', 'file_path', required=True, type=click.Path(exists=True),
|
|
help='Path to markdown file')
|
|
@click.option('--format', 'output_format', default='raw', type=click.Choice(['raw', 'json']),
|
|
help='Output format (raw or json)')
|
|
def tailmatter_get(key, file_path, output_format):
|
|
"""Get specific tailmatter value by key (supports dot notation for nested values)."""
|
|
try:
|
|
file_path = Path(file_path)
|
|
with open(file_path, 'r', encoding='utf-8') as f:
|
|
text = f.read()
|
|
|
|
parser = TailmatterParser()
|
|
value = parser.get_tailmatter_value(text, key)
|
|
|
|
if value is None:
|
|
click.echo(f"Key '{key}' not found in tailmatter", err=True)
|
|
return
|
|
|
|
if output_format == 'json':
|
|
click.echo(json.dumps(value, indent=2))
|
|
else:
|
|
if isinstance(value, (dict, list)):
|
|
click.echo(json.dumps(value, indent=2))
|
|
else:
|
|
click.echo(str(value))
|
|
|
|
except Exception as e:
|
|
click.echo(f"Error: {e}", err=True)
|
|
raise click.ClickException(f"Failed to get tailmatter value from {file_path}")
|
|
|
|
|
|
@click.command('tailmatter-set')
|
|
@click.argument('key_value')
|
|
@click.option('--file', 'file_path', required=True, type=click.Path(exists=True),
|
|
help='Path to markdown file')
|
|
@click.option('--backup', is_flag=True, help='Create backup of original file')
|
|
def tailmatter_set(key_value, file_path, backup):
|
|
"""Set tailmatter value (format: key=value, supports dot notation for nested)."""
|
|
try:
|
|
if '=' not in key_value:
|
|
raise click.ClickException("Key-value must be in format 'key=value'")
|
|
|
|
key, value = key_value.split('=', 1)
|
|
key = key.strip()
|
|
value = value.strip()
|
|
|
|
# Try to parse value as JSON for complex types
|
|
try:
|
|
if value.lower() in ['true', 'false']:
|
|
value = value.lower() == 'true'
|
|
elif value.replace('.', '').replace('-', '').isdigit():
|
|
value = float(value) if '.' in value else int(value)
|
|
elif value.startswith('[') or value.startswith('{'):
|
|
value = json.loads(value)
|
|
except (json.JSONDecodeError, ValueError):
|
|
pass
|
|
|
|
file_path = Path(file_path)
|
|
|
|
if backup:
|
|
backup_path = file_path.with_suffix(f"{file_path.suffix}.bak")
|
|
backup_path.write_text(file_path.read_text())
|
|
click.echo(f"Backup created: {backup_path}")
|
|
|
|
with open(file_path, 'r', encoding='utf-8') as f:
|
|
text = f.read()
|
|
|
|
parser = TailmatterParser()
|
|
new_text = parser.set_tailmatter_value(text, key, value)
|
|
|
|
with open(file_path, 'w', encoding='utf-8') as f:
|
|
f.write(new_text)
|
|
|
|
click.echo(f"Set {key}={value} in tailmatter for {file_path}")
|
|
|
|
except Exception as e:
|
|
click.echo(f"Error: {e}", err=True)
|
|
raise click.ClickException(f"Failed to set tailmatter value in {file_path}")
|
|
|
|
|
|
@click.command('tailmatter-keys')
|
|
@click.option('--file', 'file_path', required=True, type=click.Path(exists=True),
|
|
help='Path to markdown file')
|
|
@click.option('--format', 'output_format', default='list', type=click.Choice(['list', 'json']),
|
|
help='Output format (list or json)')
|
|
def tailmatter_keys(file_path, output_format):
|
|
"""List all tailmatter keys."""
|
|
try:
|
|
file_path = Path(file_path)
|
|
with open(file_path, 'r', encoding='utf-8') as f:
|
|
text = f.read()
|
|
|
|
parser = TailmatterParser()
|
|
keys = parser.get_tailmatter_keys(text)
|
|
|
|
if not keys:
|
|
click.echo("No tailmatter keys found")
|
|
return
|
|
|
|
if output_format == 'json':
|
|
click.echo(json.dumps(keys, indent=2))
|
|
else:
|
|
for key in sorted(keys):
|
|
click.echo(key)
|
|
|
|
except Exception as e:
|
|
click.echo(f"Error: {e}", err=True)
|
|
raise click.ClickException(f"Failed to list tailmatter keys from {file_path}")
|
|
|
|
|
|
@click.command('tailmatter-stats')
|
|
@click.option('--file', 'file_path', required=True, type=click.Path(exists=True),
|
|
help='Path to markdown file')
|
|
@click.option('--format', 'output_format', default='json', type=click.Choice(['json', 'text']),
|
|
help='Output format (json or text)')
|
|
def tailmatter_stats(file_path, output_format):
|
|
"""Calculate tailmatter statistics."""
|
|
try:
|
|
file_path = Path(file_path)
|
|
with open(file_path, 'r', encoding='utf-8') as f:
|
|
text = f.read()
|
|
|
|
parser = TailmatterParser()
|
|
stats = parser.calculate_tailmatter_stats(text)
|
|
|
|
if output_format == 'json':
|
|
click.echo(json.dumps(stats.to_dict(), indent=2))
|
|
else:
|
|
click.echo(f"Has tailmatter: {stats.has_tailmatter}")
|
|
click.echo(f"Format: {stats.format or 'N/A'}")
|
|
click.echo(f"Total fields: {stats.total_fields}")
|
|
click.echo(f"QA items: {stats.qa_items}")
|
|
click.echo(f"QA completed: {stats.qa_completed}")
|
|
click.echo(f"Editorial status: {stats.editorial_status or 'N/A'}")
|
|
click.echo(f"Has agent config: {stats.has_agent_config}")
|
|
|
|
except Exception as e:
|
|
click.echo(f"Error: {e}", err=True)
|
|
raise click.ClickException(f"Failed to calculate tailmatter stats for {file_path}")
|
|
|
|
|
|
@click.command('tailmatter-check')
|
|
@click.option('--file', 'file_path', required=True, type=click.Path(exists=True),
|
|
help='Path to markdown file')
|
|
def tailmatter_check(file_path):
|
|
"""Run QA checklist validation."""
|
|
try:
|
|
file_path = Path(file_path)
|
|
with open(file_path, 'r', encoding='utf-8') as f:
|
|
text = f.read()
|
|
|
|
parser = TailmatterParser()
|
|
tailmatter = parser.extract_tailmatter(text)
|
|
|
|
qa_checklist = tailmatter.get("qa_checklist", [])
|
|
if not qa_checklist:
|
|
click.echo("No QA checklist found in tailmatter")
|
|
return
|
|
|
|
click.echo("QA Checklist Status:")
|
|
click.echo("=" * 50)
|
|
|
|
total_items = len(qa_checklist)
|
|
completed_items = 0
|
|
|
|
for i, item in enumerate(qa_checklist, 1):
|
|
if isinstance(item, dict):
|
|
requirement = item.get("requirement", f"Item {i}")
|
|
complete = item.get("complete", False)
|
|
|
|
status_icon = "✅" if complete else "❌"
|
|
click.echo(f"{status_icon} {requirement}")
|
|
|
|
if complete:
|
|
completed_items += 1
|
|
|
|
click.echo("=" * 50)
|
|
click.echo(f"Progress: {completed_items}/{total_items} ({completed_items/total_items*100:.1f}%)")
|
|
|
|
if completed_items == total_items:
|
|
click.echo("🎉 All QA items completed!")
|
|
else:
|
|
click.echo(f"⚠️ {total_items - completed_items} items remaining")
|
|
|
|
except Exception as e:
|
|
click.echo(f"Error: {e}", err=True)
|
|
raise click.ClickException(f"Failed to check QA status for {file_path}") |