Fork of
rcarmo/python-office-mcp-serverwith--namespacefiltering for selective tool loading and reduced token overhead.
MCP server providing tools to extract, convert, and generate Microsoft Office documents (Word, Excel, PowerPoint). Originally developed by @rcarmo in parallel with go-ooxml.
The --namespace flag controls which tool modules load at startup. When used with LLM agents, this reduces the tool schema injected into context, saving tokens.
# Load only Word tools + unified office tools
office-mcp-server --namespace word
# Combine multiple namespaces
office-mcp-server --namespace word --namespace excel
# No flag = all 52 tools (backward compatible)
office-mcp-serverAvailable namespaces: word, excel, pptx, web, azure-pricing
Each document namespace (word, excel, pptx) automatically includes the unified office_* tools (read, inspect, patch, comment, table, template, audit, image) since they are the primary interface for any document type.
| Load | Tools | Estimated Tokens | vs Full |
|---|---|---|---|
| All (no namespace) | 52 | ~19,400 | — |
--namespace word |
22 | ~9,700 | -50% |
--namespace excel |
16 | ~7,700 | -60% |
--namespace pptx |
24 | ~9,500 | -51% |
--namespace web |
5 | ~1,800 | -91% |
--namespace azure-pricing |
7 | ~2,800 | -86% |
--namespace word --namespace excel |
27 | ~11,700 | -40% |
Token estimates based on JSON tool schema size (tool names, descriptions, parameter schemas).
These 8 tools auto-detect document format from file extension:
| Tool | Description |
|---|---|
office_read |
Read content from Word/Excel/PowerPoint as JSON or Markdown |
office_inspect |
Get document structure (sheets, slides, sections, tables, comments) |
office_patch |
Edit cells, shapes, sections, or replace placeholders |
office_comment |
Add or get comments from any document type |
office_table |
Table operations: add rows, create tables, add bullets |
office_template |
Copy templates or analyze template structure |
office_audit |
Audit for placeholders, completion, or tracking status |
office_image |
Insert images into Word, Excel, or PowerPoint documents |
These were a proof-of-concept approach fod managing and updating specific document templates - all the tools marked sow are deprecated and kept only for historical interest.
| Tool | Description |
|---|---|
word_generate_sow |
Fill SOW template with structured data |
word_cleanup_sow |
Remove template artifacts and guidance (tracked) |
word_get_section_guidance |
Extract template instructions from a section |
word_parse_sow_template |
Analyze SOW template structure |
word_create_sow_from_markdown |
Create SOW from Markdown content |
word_extract_sow_structure |
Extract structured data from existing SOW |
word_enable_track_changes |
Enable Word's track changes mode |
word_patch_with_track_changes |
Replace text with revision marks |
| Tool | Description |
|---|---|
pptx_add_slide |
Add new slide with specified layout |
pptx_delete_slide |
Remove a slide |
pptx_duplicate_slide |
Copy a slide |
pptx_reorder_slides |
Change slide order |
pptx_hide_slide |
Hide/unhide a slide |
pptx_set_notes |
Set speaker notes |
pptx_recommend_layout |
Get best layout for content type |
pptx_log_changes |
Add change log slide |
| Tool | Description |
|---|---|
word_from_markdown |
Create Word document from Markdown (supports inline text or markdown_file path for large inputs) |
excel_from_markdown |
Create Excel workbook from Markdown tables (supports inline text or markdown_file for large inputs) |
pptx_from_markdown |
Create PowerPoint from Markdown slides (supports inline text or markdown_file for large inputs) |
| Tool | Description |
|---|---|
restart_server |
Hot-reload the server after code changes |
list_supported_formats |
Show available document formats |
# Read Excel as Markdown
office_read(file_path="data.xlsx", output_format="markdown")
# Read specific range
office_read(file_path="data.xlsx", scope="Sheet1!A1:D10")
# Read a single worksheet
office_read(file_path="data.xlsx", scope="Sheet1")
# Read Word document
office_read(file_path="report.docx", output_format="markdown")# List Excel sheets
office_inspect(file_path="data.xlsx", what="sheets")
# List Word tables
office_inspect(file_path="report.docx", what="tables")
# List PowerPoint slides
office_inspect(file_path="deck.pptx", what="slides")# Patch Excel cell
office_patch(
file_path="data.xlsx",
changes=[{"target": "A1", "value": "New Value"}]
)
# Patch Word placeholder
office_patch(
file_path="report.docx",
changes=[{"target": "<Customer>", "value": "Contoso"}]
)
# Patch PowerPoint shape
office_patch(
file_path="deck.pptx",
changes=[{"target": "slide:1/Title 1", "value": "New Title"}]
)
# PowerPoint soft return in a single text box
office_patch(
file_path="deck.pptx",
changes=[{"target": "slide:1/Title 2", "value": "Contoso{br}Project"}]
)# Add row to Word table
office_table(
file_path="report.docx",
operation="add_row",
table_id="staffing",
data={"Role": "PM", "Count": "1", "Notes": "Lead"}
)
# Create table in Word
office_table(
file_path="report.docx",
operation="create",
data={
"headers": ["Phase", "Owner", "Target Date"],
"rows": [{"Phase": "Discovery", "Owner": "PM", "Target Date": "2026-04-01"}],
"insert_after_section": "Delivery Plan"
}
)
# Add table to PowerPoint
office_table(
file_path="deck.pptx",
operation="create",
table_id="3",
data={
"headers": ["Phase", "Duration"],
"rows": [["Discovery", "2 weeks"]]
}
)# Avoid MCP argument-size limits by passing a markdown_file path
word_from_markdown(
output_path="report.docx",
markdown_file="inputs/large-report.md"
)
excel_from_markdown(
output_path="budget.xlsx",
markdown_file="inputs/budget-tables.md"
)
pptx_from_markdown(
output_path="deck.pptx",
markdown_file="inputs/deck.md"
)
word_create_sow_from_markdown(
output_path="sow.docx",
template_path="templates/Agile.docx",
markdown_file="inputs/sow.md"
)# Audit for placeholders
office_audit(file_path="report.docx", checks=["placeholders"])
# Audit for completion
office_audit(file_path="report.docx", checks=["completion"])The recommended workflow for reviewing documents is this:
1. office_template(operation="copy") → Create working document from template
2. office_template(operation="analyze") → Understand what to preserve vs fill
3. office_inspect(what="tables") → Get EXACT column names for all tables
4. word_generate_sow → Fill placeholders and tables with data
5. office_patch(operation="section") → Add prose to Introduction, Business Context
6. office_table(operation="insert_row") → Add engagement-specific rows to tables
7. office_patch(operation="fix_split") → Replace any remaining split placeholders
8. office_comment(operation="add") → Add review comments for stakeholders
9. word_cleanup_sow → Remove template guidance (tracked)
10. office_audit(checks=["completion"]) → Verify completion score ≥ 80%
Quality Bar: All review tools preserve document structure by editing templates rather than creating new documents from scratch. All changes are tracked for stakeholder review.
Using uv:
uv pip install "git+https://github.com/cjnova/python-office-mcp-server.git"Using pip:
pip install "git+https://github.com/cjnova/python-office-mcp-server.git"This installs the office-mcp-server command. Requires Python ≥3.10 (tested with 3.12).
git clone https://github.com/cjnova/python-office-mcp-server.git
cd python-office-mcp-server
uv pip install .
# or: pip install .
# or for development: pip install -e .git clone https://github.com/cjnova/python-office-mcp-server.git
cd python-office-mcp-server
pip install -r requirements.txt
python office_server.pyVS Code (.vscode/mcp.json):
{
"servers": {
"office": {
"command": "office-mcp-server"
}
}
}With namespace filtering:
{
"servers": {
"office-word": {
"command": "office-mcp-server",
"args": ["--namespace", "word"]
},
"office-excel": {
"command": "office-mcp-server",
"args": ["--namespace", "excel"]
}
}
}Claude Desktop (claude_desktop_config.json):
{
"mcpServers": {
"office": {
"command": "office-mcp-server",
"args": ["--namespace", "word", "--namespace", "excel"]
}
}
}"args": ["--from", "git+https://github.com/cjnova/python-office-mcp-server.git", "office-mcp-server"]
Using uvx (no pre-install needed):
{
"command": "uvx",
"args": ["--from", "git+https://github.com/cjnova/python-office-mcp-server.git", "office-mcp-server", "--namespace", "word"]
}Using uv tool install (pre-installed, no startup latency):
uv tool install "git+https://github.com/cjnova/python-office-mcp-server.git"Some MCP clients can start with subsets of tools disabled by policy/session settings. If a call returns a disabled-tool error, enable the corresponding MCP tools in the client first, then retry. This enablement behavior is controlled by the MCP client/host, not by this server.
The server can be set to be auto-discovered from .vscode/mcp.json. That is left as an exercise to the reader, but to verify: Open Command Palette → MCP: List Servers → confirm officeServer is listed.
Add the server to your Copilot CLI configuration:
# Open config file
code ~/.config/github-copilot/config.json
# Add this to the mcpServers section:
{
"mcpServers": {
"officeServer": {
"command": "python",
"args": ["/path/to/.github/mcp/office_server.py"]
}
}
}cd .github/mcp
pip install -r requirements.txt
python office_server.pyBuild a standalone .exe using PyInstaller.
cd .github/mcp
python -m pip install -r requirements.txt
python -m pip install -r requirements-build.txt
python build_windows_onefile.py --cleanOutput artifact:
dist/office-mcp-server.exe
python build_windows_onefile.py --name office-server-proddist\office-mcp-server.exeUse the generated executable in MCP client configuration by pointing command to the .exe path.
python-docx— Word document handlingopenpyxl— Excel workbook handlingpython-pptx— PowerPoint presentation handlingaioumcp— Async MCP server frameworkpyinstaller— Build-time dependency for one-file Windows executable
The server dynamically loads tool modules from tools/:
office_unified_tools.py— Unified interface (7 tools)word_tools.py— Word conversion toolsword_advanced_tools.py— SOW-specific toolsexcel_tools.py— Excel conversion toolsexcel_advanced_tools.py— Excel advanced operations (internal)pptx_tools.py— PowerPoint conversion toolspptx_advanced_tools.py— Slide management toolsweb_tools.py— Web utilities (fetch, extract, search)azure_pricing_tools.py— Azure pricing queries
Tools are discovered automatically by class name pattern (*Tools).
The --namespace flag is parsed in office_server.py via _parse_namespace_args(). It calls tools.set_namespace_filter() which restricts _discover_tool_module_names() to only return modules mapped in NAMESPACE_MODULES:
NAMESPACE_MODULES = {
"office": ["office_unified_tools"],
"word": ["office_unified_tools", "word_tools", "word_advanced_tools"],
"excel": ["office_unified_tools", "excel_tools", "excel_advanced_tools"],
"pptx": ["office_unified_tools", "pptx_tools", "pptx_advanced_tools"],
"web": ["web_tools"],
"azure-pricing": ["azure_pricing_tools"],
}Tool classes are loaded lazily after namespace filtering, then composed into the server class via multiple inheritance (MRO). The unified office_* tools delegate to format-specific tools via self.tool_* method resolution, which is why they must be loaded alongside any document namespace.