Skip to main content

Parse Markdown and turn it into nicely-formatted text for terminal display

Project description

md2term

GitHub repo PyPI Changelog Tests License

A streaming markdown-to-terminal converter that renders markdown with rich formatting and syntax highlighting. Streaming is supported so you can pipe your favorite LLM CLI tool to it, like llm or Ollama, like this:

demo

[!NOTE] This software was created almost entirely by AI with Cursor and Claude 4 Sonnet.

Installation

Install md2term using uv:

uv tool install md2term

Or with pip or pipx:

pip install md2term
# or
pipx install md2term

Verify the installation:

md2term --version

Usage

Command Line

# Convert a markdown file
md2term README.md

# Read from stdin
cat README.md | md2term

# Pipe from other commands
curl -s https://raw.githubusercontent.com/user/repo/main/README.md | md2term

# Or commands with slow output
llm 'tell me long a story about cheesecakes using markdown formatting' | md2term

# Override terminal width
md2term --width 100 README.md

# Show version
md2term --version

# Show help
md2term --help

Python Library

You can also use md2term as a Python library for integrating markdown rendering into your applications:

from rich.console import Console
from md2term import convert, StreamingRenderer

# Simple conversion
markdown_text = "# Hello\n\nThis is **bold** text."
convert(markdown_text)

# Streaming usage (great for LLM applications)
console = Console(force_terminal=True)
renderer = StreamingRenderer(console)

try:
    # Add content incrementally
    renderer.add_text("# Streaming Example\n\n")
    renderer.add_text("This content appears **immediately** as it's added.\n")
    renderer.add_text("\n```python\nprint('Hello, World!')\n```\n")
finally:
    # Always finalize to ensure complete rendering
    renderer.finalize()

The streaming functionality is particularly useful for:

  • LLM/AI applications that generate content in real-time
  • Processing large files with immediate visual feedback
  • Building interactive CLI tools with progressive output

See example_streaming.py for more detailed examples and patterns.

Examples

For a comprehensive example of markdown features, see example.md in this repository.

Design Decisions

Streaming and Timing Strategy

The program uses a sophisticated streaming approach designed to provide responsive real-time rendering while minimizing visual flickering and corruption:

Smart Update Frequency Control

The streaming renderer uses balanced update conditions to determine when to re-render content:

  1. Content-based triggers: Updates when 80+ characters are added AND at least 50ms have passed
  2. Paragraph breaks: Immediate updates on double newlines (\n\n) for clear content boundaries
  3. Time-based updates: Every 100ms if at least 20 characters have been added
  4. Completion detection: Updates when content appears complete (headings, list items, blockquotes) AND 80ms have passed

This approach balances responsiveness with performance, avoiding excessive re-rendering while ensuring users see content as it streams in.

Backtracking and Re-rendering

  • Minimal re-rendering: Only re-renders when buffer content actually changes
  • Accurate line counting: Uses a temporary console to count output lines before clearing previous content
  • ANSI escape sequences: Clears previous output using \033[1A\033[2K (move up, clear line) for each rendered line
  • Fallback handling: Gracefully falls back to plain text if markdown parsing fails during streaming

Input Processing Strategies

The program adapts its reading strategy based on input type:

  • File input: Reads entire content at once for optimal performance
  • Stdin streaming: Uses character-by-character reading with select() optimization when available
  • Chunk optimization: Reads 64-character chunks when data is readily available, falls back to single characters otherwise
  • Cross-platform compatibility: Gracefully handles systems where select() or fileno() are not available

Completion Detection

The renderer intelligently detects when markdown elements appear complete:

  • Double newlines: Clear paragraph boundaries
  • Structural elements: Headings (#), list items (- ), blockquotes (> ), horizontal rules (---)
  • Empty lines: Natural content breaks

This allows for responsive updates without waiting for arbitrary timeouts.

Code Block Handling

The program uses a smart approach to handle multi-line code blocks:

  1. Streaming Processing: For stdin input, the program processes content in chunks, buffering until it encounters blank lines (when not in a code block)
  2. Code Fence Detection: Detects triple backticks (\```) to track when we're inside code blocks
  3. No Backtracking: Instead of clearing previous lines, the program assumes that triple backticks always indicate the start/end of code blocks

This approach is efficient and works well for typical markdown usage patterns.

Color Scheme

  • H1: Bright cyan with rules above and below, centered
  • H2: Bright blue with rule below
  • H3: Bright magenta
  • H4: Bright yellow
  • H5: Bright green
  • H6: Bright white
  • Code spans: Red text on dark gray background
  • Links: Blue underlined text with URL in parentheses
  • Lists: Yellow bullets (•) for unordered, cyan numbers for ordered
  • Blockquotes: Blue italic text in a panel

Terminal Width Handling

The program automatically detects terminal width and wraps text accordingly. You can override this with the --width option for testing or specific formatting needs.

Development

To install for development:

# Clone the repository
git clone https://github.com/statico/md2term
cd md2term

# Install in development mode
uv tool install --editable .

# Or install dependencies for local development
uv sync
uv run md2term README.md

Running Tests

The project uses pytest with snapshot testing via syrupy to ensure consistent output formatting:

# Install test dependencies
uv sync --group test

# Run all tests
uv run pytest

# Run tests with coverage
uv run pytest --cov=md2term

# Run specific test file
uv run pytest tests/test_md2term.py

# Run specific test
uv run pytest tests/test_md2term.py::TestMarkdownFeatures::test_headings_all_levels

Snapshot Testing

The tests use snapshot testing to verify that markdown rendering produces consistent terminal output. Snapshots capture the exact ANSI escape sequences and formatting that would appear in the terminal.

# Update snapshots when output changes (after verifying changes are correct)
uv run pytest --snapshot-update

# Review snapshot differences
uv run pytest --snapshot-details

Important: When modifying rendering logic, always:

  1. Run tests to see what changed
  2. Manually verify the output looks correct with uv run md2term example.md
  3. Update snapshots only if the changes are intentional and correct

The snapshot file is located at tests/__snapshots__/test_md2term.ambr and contains the expected terminal output for various markdown inputs.

License

This project is licensed under the Apache License 2.0. See the LICENSE file for details.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

md2term-1.0.2.tar.gz (72.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

md2term-1.0.2-py3-none-any.whl (16.0 kB view details)

Uploaded Python 3

File details

Details for the file md2term-1.0.2.tar.gz.

File metadata

  • Download URL: md2term-1.0.2.tar.gz
  • Upload date:
  • Size: 72.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for md2term-1.0.2.tar.gz
Algorithm Hash digest
SHA256 159dbe2843fab9f6714d7e0f4b64dfd104b2c2c8722004f7b3edc2e02491ed5b
MD5 5fcfea1725d5fba8624f268403ed0a17
BLAKE2b-256 18c671841131fb745cb11c14bb44650cea68ebb7e0f6f22f359eee43d98c8dc0

See more details on using hashes here.

Provenance

The following attestation bundles were made for md2term-1.0.2.tar.gz:

Publisher: publish.yml on statico/md2term

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file md2term-1.0.2-py3-none-any.whl.

File metadata

  • Download URL: md2term-1.0.2-py3-none-any.whl
  • Upload date:
  • Size: 16.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.12.9

File hashes

Hashes for md2term-1.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 e3022798f8fb9f7836287cbbd3be01f00004b9cbf8f9d823147ef0ef1839ffaf
MD5 1007fefd474be84f7ab9843c93d9c035
BLAKE2b-256 c93b7993b1f58457ab61c924f42c54217c034faae66d98f75ab7bdf05522b62c

See more details on using hashes here.

Provenance

The following attestation bundles were made for md2term-1.0.2-py3-none-any.whl:

Publisher: publish.yml on statico/md2term

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page