A local-first notebook system for Python, inspired by Clerk for Clojure. Plaque turns regular Python files into interactive notebooks with real-time updates and smart dependency tracking.
👉 Check out examples/getting-started.py and its rendered version at examples/getting-started.html to see Plaque in action or watch the 🎥Demo Video
- Local-first: Uses plain Python files as the source - and your own editor - no special file formats
- Live Updates: Browser preview updates in real-time as you edit
- Rich Output: Supports Markdown, LaTeX equations, plots, DataFrames, and more
- Flexible Format: Supports both
# %%markers and multiline comments for cells - Python-native: Use standard Python syntax for both code and documentation
- IPython Support: Magic commands (
%timeit,%%time, etc.) and top-level async/await - Download Button: Rendered notebooks include a download link to get the original Python source
Many systems support reactive notebooks, like clerk, marimo, observable, pluto, etc. Plaque is meant to be a simple thing that provides 80% of the utility with a very simple package. The core idea is that your files should only run as they would if you ran them from scratch from top to bottom, but we don't actually have to rerun every cell every time. Instead, we only ever re-execute any cell you modify and any cells later in the document.
In this way, you can have most of the benefits for reactivity and live updating, but still get caching and some gaurentee that you don't have to re-evaluate expensive computations.
Plaque supports two different styles for creating notebooks:
Using # %% markers, similar to VS Code notebooks:
# Code cell
x = 42
print(f"The answer is {x}")
# %% [markdown]
# # This is a markdown cell
#
# With support for:
# - Lists
# - **Bold text**
# - And more!
# %%
# Another code cell
import matplotlib.pyplot as plt
plt.plot([1, 2, 3], [1, 4, 9])
plt.show()Using Python's multiline strings (""" or ''') for documentation:
"""
# Getting Started
This notebook demonstrates using multiline comments as markdown cells.
All standard markdown features are supported:
1. **Bold text**
2. *Italic text*
3. Code blocks
4. LaTeX equations: $E = mc^2$
"""
# Code is automatically treated as a code cell
x = 42
print(f"The answer is {x}")
"""
## Data Visualization
Now let's make a plot:
"""
import matplotlib.pyplot as plt
plt.plot([1, 2, 3], [1, 4, 9])
plt.show()
"""
The plot shows a quadratic relationship between x and y.
"""Using f-string style comments for programmatic templated output:
f"""
# Dynamic Report for {dataset_name}
Results generated on {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}
## Summary Statistics
- Total samples: {len(data)}
- Mean value: {np.mean(data):.2f}
- Standard deviation: {np.std(data):.2f}
"""
dataset_name = "Sales Data Q4"
data = [1, 2, 3, 4, 5]
f"""
## Analysis Results
The dataset '{dataset_name}' contains {len(data)} data points.
Maximum value observed: {max(data)}
"""F-style comments allow you to create dynamic markdown cells that incorporate variables and expressions, making them perfect for automated reports and templated notebooks.
Both styles support:
- Markdown formatting with bold, italic, lists, etc.
- LaTeX equations (both inline and display)
- Code syntax highlighting
- Rich output (plots, DataFrames, etc.)
Plaque uses IPython as its execution engine, which means you can use:
Magic Commands:
# Time a single line
%timeit sum(range(1000))
# Time an entire cell
%%time
result = expensive_computation()
# List variables in namespace
%who
# Get help on objects
my_function?Top-level Async/Await:
import asyncio
async def fetch_data(url):
await asyncio.sleep(0.1)
return {"data": "result"}
# Use await directly at the top level - no asyncio.run() needed!
result = await fetch_data("https://api.example.com")
resultShell Commands:
# Run shell commands with !
!ls -la
!pip list | grep numpyWhen using multiline comments as cells:
- Top-level comments become markdown cells
- Function/method docstrings remain as code
- You can mix code and documentation freely
- Both
"""and'''are supported
You can install Plaque using either pip or uv:
# Using pip
pip install plaque
# Using uv (recommended)
uv pip install plaque# Clone the repository
git clone https://github.com/alexalemi/plaque.git
cd plaque
# Install in development mode
uv pip install -e .
# or
pip install -e .For development work with testing and additional tools:
# Clone the repository
git clone https://github.com/alexalemi/plaque.git
cd plaque
# Create and activate a virtual environment (optional)
python -m venv venv
source venv/bin/activate # On Windows: venv\Scripts\activate
# Install with development dependencies
uv pip install -e ".[dev]"
# or
pip install -e ".[dev]"To render a notebook:
# Generate static HTML
plaque render my_notebook.py
# Generate static HTML with custom output path
plaque render my_notebook.py output.html
# Start a live re-render with caching.
plaque watch my_notebook.py
# Start live server with auto-reload
plaque serve my_notebook.py
# Specify a custom port (default is 5000)
plaque serve my_notebook.py --port 8000
# Open browser automatically
plaque serve my_notebook.py --open