Vector-native PDF diagram toolkit for ReportLab Platypus
This project has been archived.
The maintainers of this project have marked this project as archived. No new releases are expected.
Project description
📊 paperforge_diagrams
Vector-native PDF diagram toolkit for ReportLab Platypus. Draw complex diagrams natively inside your ReportLab documents with a clean, pythonic API.
No raster images, no external command-line tools, and no internet connection required. All diagram types compile directly to reportlab.graphics.shapes.Drawing objects that wrap seamlessly as Platypus Flowables.
🚀 Features
- Flowchart: ANSI/ISO flowchart symbols (terminals, processes, decisions, io, connectors).
- SequenceDiagram: UML sequence diagrams (actors, lifelines, activation bars, message arrows, dividers).
- ClassDiagram: UML class diagrams (class boxes, headers, attributes, methods, relations).
- ERDiagram: Entity-Relationship diagrams (entities, relations, primary key and multi-value attributes).
- StateMachine: DFA/NFA and process state diagrams (states, transitions, initial/accepting indicators).
- NetworkDiagram: Network topology (hosts, servers, databases, links).
- ArchitectureDiagram: System architecture topologies with clients, services, databases, queues, and auto-routed connections.
- AWSDiagram: Cloud infrastructure architecture with AWS icons (EC2, S3, RDS, Lambda, SQS).
- GitDiagram: Git branch timeline diagrams with commits, merges, and distinct branch lanes.
- SchemaDiagram: Database table schema definition diagrams with field types, primary keys, and foreign-key links.
- C4ContainerDiagram: C4 Model container diagrams with System Context, Container, Component layers, and descriptive relationships.
- TimingDiagram: Digital waveform timing diagrams.
- LayeredStack: OSI model, TCP/IP stack, memory hierarchy.
- Standalone Export: Save drawings directly to PDF, SVG, and PNG/JPG formats.
📦 Installation
Install the package locally:
pip install ./paperforge_diagrams
🎨 Supported Diagram Types & Examples
1. Flowchart
Draw processes, decisions, and connectors with auto-layout or manual coordinates.
import paperforge_diagrams as pd
# Create a horizontal flowchart with a custom scale factor
fc = pd.Flowchart(
width=400,
height=150,
direction="LR", # 'TB' (top-to-bottom, default) or 'LR' (left-to-right)
scale_factor=1.0, # Optional manual override for text/box sizing (default: auto-computed)
)
fc.terminal("start", "START")
fc.process("calc", "Compute Sum")
fc.decision("check", "Sum > 100?")
fc.terminal("end", "END")
fc.edge("start", "calc")
fc.edge("calc", "check")
fc.edge("check", "end", branch="yes")
fc.edge("check", "calc", branch="no", orthogonal=True)
2. Sequence Diagram
Draw actor interactions over lifelines.
import paperforge_diagrams as pd
seq = pd.SequenceDiagram(width=400, height=220)
seq.actor("c", "Client")
seq.actor("s", "Server")
seq.activate("c")
seq.message("c", "s", "HTTP GET /index.html")
seq.activate("s")
seq.divider("Internal Processing")
# Arrow styles: 'solid' (default), 'dashed', 'solid_open', 'dashed_open'
seq.message("s", "c", "200 OK (HTML Document)", arrow="dashed")
seq.deactivate("s")
seq.deactivate("c")
3. Layered Stack
Draw OSI stacks, memory hierarchies, or layered architectures.
import paperforge_diagrams as pd
stack = pd.LayeredStack(width=300, height=180)
stack.layer("Application", sublabel="HTTP, DNS")
stack.layer("Transport", sublabel="TCP, UDP")
stack.divider() # Draw a thicker divider line after Transport layer
stack.layer("Network", sublabel="IP, ICMP")
stack.layer("Link", sublabel="Ethernet")
4. Network Diagram
Draw network topologies with hosts, clouds, switches, routers, and firewalls.
import paperforge_diagrams as pd
net = pd.NetworkDiagram(width=500, height=220)
net.node("inet", "Internet", x=50, y=110, kind="cloud")
net.node("fw", "Firewall", x=160, y=110, kind="firewall")
net.node("sw", "Core Switch", x=270, y=110, kind="switch")
net.node("srv", "Web Server", x=380, y=150, kind="server")
net.node("db", "Database", x=380, y=70, kind="database")
net.link("inet", "fw")
net.link("fw", "sw")
net.link("sw", "srv")
net.link("sw", "db")
# Also supports building standard topologies programmatically:
# net.star_topology(center_id="sw", center_label="Switch", spoke_ids=["h1", "h2", "h3"])
# net.bus_topology(node_ids=["n1", "n2", "n3"])
# net.ring_topology(node_ids=["r1", "r2", "r3"])
# net.mesh_topology(node_ids=["m1", "m2", "m3"])
# net.tree_topology(parent_child_map={"root": ["child1", "child2"]})
5. UML Class Diagram
Draw UML class diagrams with visibility, attributes, and methods.
import paperforge_diagrams as pd
cd = pd.ClassDiagram(width=400, height=250, class_w=120)
cd.uml_class("Shape", "Shape", stereotype="abstract", methods=["+ area(): double"])
cd.uml_class("Circle", "Circle", attributes=["- radius: double"], methods=["+ area(): double"])
# Relationship kinds: 'inheritance', 'realization', 'composition', 'aggregation', 'association', 'dependency'
cd.relate("Circle", "Shape", kind="inheritance")
6. Entity-Relationship Diagram (ER)
Draw entity-relationship diagrams (Chen notation) with cardinalities.
import paperforge_diagrams as pd
er = pd.ERDiagram(width=450, height=200)
er.entity("Customer")
er.relationship("Buys")
er.entity("Product")
er.entity_attributes("Customer", [("ID", {"pk": True}), "Name"])
er.entity_attributes("Product", [("SKU", {"pk": True}), "Price"])
er.connect("Customer", "Buys", card_from="1", card_to="N")
er.connect("Product", "Buys", card_from="1", card_to="N")
7. State Machine (DFA / Process Transitions)
Draw finite state automata or lifecycle models.
import paperforge_diagrams as pd
sm = pd.StateMachine(width=400, height=180)
sm.state("s0", "Init", x=70, y=90, initial=True)
sm.state("s1", "Active", x=200, y=90)
sm.state("s2", "Success", x=330, y=90, accepting=True)
sm.transition("s0", "s1", label="start")
sm.transition("s1", "s1", label="process")
sm.transition("s1", "s2", label="complete")
8. Timing Diagram
Draw digital timing signal waveforms.
import paperforge_diagrams as pd
td = pd.TimingDiagram(width=400, height=150)
td.clock("CLK", period=20.0, cycles=6)
td.signal("RESET", transitions=[(0, 1), (15, 0)])
td.signal("DATA", transitions=[(0, 0), (35, 1), (75, 0)])
9. Database Schema Diagram
Draw database table structures with primary/foreign keys and relationships.
import paperforge_diagrams as pd
schema = pd.SchemaDiagram(width=450, height=200)
schema.table("users", [
("id", "INTEGER", {"pk": True}),
("email", "VARCHAR", {}),
("created_at", "TIMESTAMP", {})
])
schema.table("orders", [
("id", "INTEGER", {"pk": True}),
("user_id", "INTEGER", {"fk": True}),
("total", "DECIMAL", {})
])
schema.relation("orders", "user_id", "users", "id")
10. Service Architecture Diagram
Draw multi-tier system topologies (clients, microservices, databases, queues) with automatic or manual layouts.
import paperforge_diagrams as pd
# Supports orientation: 'horizontal' (default) or 'vertical'
arch = pd.ArchitectureDiagram(width=450, height=220, orientation="horizontal")
arch.client("web", "Web Client")
arch.service("api", "Gateway API")
arch.database("db", "Main Database")
arch.queue("q", "Message Broker")
arch.connect("web", "api", "HTTPS")
arch.connect("api", "db", "TCP/3306")
arch.connect("api", "q", "AMQP")
11. C4 Container Diagram
Draw C4 Model Container views to document software systems, containers, technologies, and relationships.
import paperforge_diagrams as pd
c4 = pd.C4ContainerDiagram(width=400, height=200)
c4.system("user", "Customer")
c4.container("web", "SPA (React)", "Provides shopping UI")
c4.container("api", "API Application (Go)", "Handles business logic")
c4.container("db", "Database (Postgres)", "Stores order data")
c4.relate("user", "web", "Uses")
c4.relate("web", "api", "Makes API calls to")
c4.relate("api", "db", "Reads/Writes to")
12. Git Branch Flow Diagram
Draw commits, branch lanes, merges, and timelines horizontally.
import paperforge_diagrams as pd
git = pd.GitDiagram(width=400, height=150)
git.commit("main", "C1: Initial Commit")
git.branch("main", "feature")
git.commit("feature", "C2: Implement login")
git.commit("main", "C3: Hotfix bug")
git.merge("feature", "main", "C4: Merge feature/login")
13. AWS Diagram
Draw AWS cloud infrastructure diagrams using vector-native AWS icons.
import paperforge_diagrams as pd
# Supports orientation: 'horizontal' (default) or 'vertical'
aws = pd.AWSDiagram(width=450, height=220, orientation="horizontal")
# Supported node methods: ec2(), rds(), s3(), lambda_fn(), sqs()
aws.ec2("web", "Web Server")
aws.rds("db", "RDS PostgreSQL")
aws.s3("bucket", "S3 Storage")
aws.sqs("queue", "Job Queue")
aws.connect("web", "db", "SQL Connection")
aws.connect("web", "bucket", "Uploads")
aws.connect("web", "queue", "Events")
🎨 Diagram Theming & Integration
paperforge_diagrams comes with built-in dark and light themes, and can dynamically inherit themes from paperforge_notes for consistent visual layouts.
1. Using Preset Diagram Themes
Apply a theme directly to your builder instance:
import paperforge_diagrams as pd
# Use preset theme (pd.DARK or pd.LIGHT)
diagram = pd.Flowchart(width=300, height=200, theme=pd.LIGHT)
2. Matching Notes Theme (Integration)
Derive matching diagram settings from the active document notes theme:
import paperforge_notes as pn
import paperforge_diagrams as pd
# 1. Set the notes theme
pn.set_theme(pn.OCEAN_DARK)
# 2. Derive the matching diagram theme
diag_theme = pd.DiagramTheme.from_notes_theme(pn.get_theme())
# 3. Apply it to your diagrams
fc = pd.Flowchart(width=400, height=200, theme=diag_theme)
3. Customizing & Overriding Themes
Modify a theme on the fly using model_copy(update={...}):
# Create a print-optimized black-and-white diagram theme
bw_theme = pd.DiagramTheme.from_notes_theme(pn.get_theme()).model_copy(
update={
"stack_colors": ("#ffffff", "#ffffff", "#ffffff"),
"stack_stroke": "#000000",
"stack_text": "#000000",
"stack_sublabel_text": "#000000",
"bg": "#ffffff",
"text": "#000000",
"node_fill": "#ffffff",
"node_stroke": "#000000",
"node_text": "#000000",
"font_name": "Times-Roman",
"font_name_bold": "Times-Bold",
"font_name_italic": "Times-Italic",
}
)
# Apply to stack diagram
stack = pd.LayeredStack(width=300, height=180, theme=bw_theme)
💾 Standalone Export
Export your diagrams directly to vector or raster formats without a ReportLab story:
diagram = pd.Flowchart(width=300, height=200)
diagram.terminal("s", "START").terminal("e", "END").edge("s", "e")
# Save to PDF (vector)
diagram.save("output.pdf")
# Save to SVG (vector)
diagram.save("output.svg")
# Save to PNG (raster)
diagram.save("output.png")
📝 Integration with paperforge_notes
To embed a vector diagram into a PaperForge notes document, call .as_flowable() and pass it to pn.add(...):
import paperforge_notes as pn
import paperforge_diagrams as pd
fc = pd.Flowchart(width=pn.CW, height=200)
fc.terminal("s", "START").terminal("e", "END").edge("s", "e")
# Add the diagram flowables directly
pn.add(fc.as_flowable())
Static Type-Checking (Pyright / Pylance)
If you see static type warnings in your IDE:
Argument of type "list[Unknown]" cannot be assigned to parameter "x" of type "Flowable" in function "add"
This happens because .as_flowable() returns a list[Flowable] (which packages the drawing flowable along with its optional caption Paragraph flowable) rather than a single raw Flowable.
paperforge_notes handles this union type cleanly. Calling pn.add(diagram.as_flowable()) is type-safe and fully compliant.
📋 Requirements & License
- Python >= 3.11
- reportlab >= 4.5.1
- pydantic >= 2.13.4
- Licensed under the MIT License.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file paperforge_diagrams-0.1.0.tar.gz.
File metadata
- Download URL: paperforge_diagrams-0.1.0.tar.gz
- Upload date:
- Size: 20.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.6
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
fedd084d616a6de28b5a193984917553b7fbb7ff75fff27e241d4487798631c1
|
|
| MD5 |
6909c610d80c4a2c1676b33c0799e6fb
|
|
| BLAKE2b-256 |
0a783af9dbc1060b105636f436f35c15e5292edcb66210363ff56ddc18850a67
|
File details
Details for the file paperforge_diagrams-0.1.0-py3-none-any.whl.
File metadata
- Download URL: paperforge_diagrams-0.1.0-py3-none-any.whl
- Upload date:
- Size: 6.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.6
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c5fa4ad50f31ccc21c828699cdbc2e27e9c90c4a8ba1b9f36ef0e31840812d26
|
|
| MD5 |
09e97d6cb38074a287f108ba353edf32
|
|
| BLAKE2b-256 |
5c29cfd60221e26fd67c4d758db7223f65aeb001270df0151e6f1b72d7729660
|