Skip to main content

Drive a running Microsoft Word instance from Python — xlwings, but for Word, and LLM-friendly.

Project description

wordlive

Drive a running Microsoft Word instance from Python — xlwings, but for Word. Built for both human scripting and LLM agents. Windows-only.

Install

pip install wordlive

(Requires Python 3.13+ and pywin32 on Windows.)

Python

import wordlive as wl

with wl.attach() as word:
    doc = word.documents.active

    # Reads
    outline = doc.outline()
    bookmarks = doc.bookmarks.list()

    # Polite writes — preserves the user's cursor and view, atomic Ctrl-Z.
    with doc.edit("Update address block"):
        doc.bookmarks["Address"].set_text("123 Main St")
        doc.content_controls["Signatory"].set_text("Jane Doe")
        doc.heading("Introduction").insert_paragraph_after("New context paragraph.")

CLI

JSON in, JSON out — designed to drop straight into an LLM tool-use loop:

wordlive status
wordlive outline                  # heading structure (heading:N)
wordlive outline --all            # every paragraph (para:N) — alias of `paragraphs`
wordlive paragraphs               # same: para:N, level, offsets, text
wordlive read bookmark Address
wordlive write bookmark Address --text "123 Main St"

# Insert a new paragraph relative to ANY anchor (heading, paragraph, bookmark, …):
wordlive insert --anchor-id heading:1 --text "..."          # after (default)
wordlive insert --anchor-id para:3 --text "..." --before

# Address anchors by ID (the IDs `outline`/`paragraphs` emit — `heading:N`, `para:N`, `bookmark:NAME`, `cc:NAME`):
wordlive replace --anchor-id heading:3 --text "Updated section text"
wordlive go-to --anchor-id bookmark:Address

# Explicit cursor surface (the non-preferred mode — deliberately moves the cursor):
wordlive cursor read                              # where is the cursor? which para:N?
wordlive cursor write --text "inserted here"      # type at the cursor

# Styles + paragraph formatting (atomic-undo):
wordlive style list
wordlive style apply --anchor-id heading:3 --name "Heading 2"
wordlive format-paragraph --anchor-id heading:3 --alignment center --space-before 6

# Tables (cells are anchors: table:N:R:C):
wordlive table list
wordlive table read 1
wordlive replace --anchor-id table:1:2:2 --text "$450"
wordlive table add-row --table 1 --values '["Lodging", "$600"]'

# Collaboration: comments + track changes (the polite, non-destructive surface):
wordlive comment add --anchor-id heading:3 --text "Please expand this." --author Bot
wordlive comment list
wordlive comment resolve --index 1
wordlive track on            # record edits as revisions; `track off` to stop

# Lists & numbering (any anchor's paragraphs):
wordlive list apply --anchor-id heading:6 --type numbered
wordlive list restart --anchor-id heading:6

# Sections, headers & footers (header:S:WHICH / footer:S:WHICH):
wordlive section list
wordlive header write --section 1 --text "ACME Corporation"
wordlive footer read --section 1

# Images — from a file or base64 (--wrap is required: inline | auto | square | …):
wordlive insert-image --anchor-id heading:3 --path diagram.png --wrap auto
base64 logo.png | wordlive insert-image --anchor-id bookmark:Logo --base64 - --wrap inline --width 96

# Batch multiple ops in a single Ctrl-Z:
wordlive exec --script ops.json

Where ops.json looks like:

{
  "label": "Update report",
  "ops": [
    {"op": "write_bookmark", "name": "Address", "text": "123 Main St"},
    {"op": "write_cc", "name": "Signatory", "text": "Jane Doe"},
    {"op": "insert_paragraph", "anchor_id": "heading:3", "text": "New risk paragraph."},
    {"op": "replace", "anchor_id": "heading:3", "text": "Updated section text"},
    {"op": "apply_style", "anchor_id": "heading:3", "name": "Heading 2"},
    {"op": "format_paragraph", "anchor_id": "heading:3", "alignment": "center", "space_before": 6}
  ]
}

Exit codes: 0 ok, 2 anchor-not-found, 3 Word-busy, 4 Word-not-running, 1 other.

Agent skill

wordlive ships an LLM-facing skill (SKILL.md) — a concise CLI reference for agents. Drop it into a project or your home directory so coding tools discover it:

wordlive install-skill            # ./.agents/skills/wordlive/SKILL.md
wordlive install-skill --system   # ~/.agents/skills/wordlive/SKILL.md

Design

  • Politeness first — operations preserve the user's Selection, view, and scroll. The user keeps editing alongside you.
  • Semantic anchors over Selection — operations target bookmarks, content controls, or headings — never the live cursor unless you ask.
  • Atomic undo — every doc.edit() opens a Word UndoRecord, so a single Ctrl-Z reverts the whole block.
  • Escape hatch — every wrapper exposes .com for the raw COM object; you're never blocked by missing coverage.

See spec.md for the full design.

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

wordlive-0.8.1.tar.gz (165.1 kB view details)

Uploaded Source

Built Distribution

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

wordlive-0.8.1-py3-none-any.whl (60.4 kB view details)

Uploaded Python 3

File details

Details for the file wordlive-0.8.1.tar.gz.

File metadata

  • Download URL: wordlive-0.8.1.tar.gz
  • Upload date:
  • Size: 165.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for wordlive-0.8.1.tar.gz
Algorithm Hash digest
SHA256 d281537c83774d09607181d2914fd58f969a97d3ee27156e9a7b4dcbf523e524
MD5 db7a2e1b013c9ae26ec4ec5c821d055f
BLAKE2b-256 31fdf3a67de76d582492ecde996eb7a6001ae2c57cba5c30ee95cf58c5aa5bea

See more details on using hashes here.

Provenance

The following attestation bundles were made for wordlive-0.8.1.tar.gz:

Publisher: release.yml on thomas-villani/wordlive

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

File details

Details for the file wordlive-0.8.1-py3-none-any.whl.

File metadata

  • Download URL: wordlive-0.8.1-py3-none-any.whl
  • Upload date:
  • Size: 60.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for wordlive-0.8.1-py3-none-any.whl
Algorithm Hash digest
SHA256 ee86376116ec065e4096567372a4544994948a53f29c506ed317eb67515469df
MD5 3bb287e8732755633a197c8c70b1e510
BLAKE2b-256 01ed6793561ecab275e7a0fda414db7d1478b8f885645f0ea98c267e711d394b

See more details on using hashes here.

Provenance

The following attestation bundles were made for wordlive-0.8.1-py3-none-any.whl:

Publisher: release.yml on thomas-villani/wordlive

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