The Things MCP server enables Claude to interact with Cultured Code's Things 3 task management application on macOS by providing access to lists (Inbox, Today, Upcoming), projects, areas, tags, and search capabilities. It leverages the Things.py library and URL scheme to allow users to create tasks, analyze projects, manage priorities, and conduct productivity reviews through natural language conversation. This integration solves the problem of bridging Claude's capabilities with Things 3 data, enabling AI-assisted task management and productivity analysis without leaving the Claude interface.
This Model Context Protocol (MCP) server lets you use Claude Desktop to interact with your task management data in Things 3 from Cultured Code. You can ask Claude to create tasks, analyze projects, help manage priorities, and more.
This server leverages the Things.py library and the Things URL Scheme.
If you find this project helpful, consider supporting its development:
brew install uvThings MCP is published on PyPI and can be run directly with uvx:
uvx things-mcp
Configure your MCP client to use uvx with things-mcp as the argument.
.mcpb file{
"mcpServers": {
"things": {
"command": "uvx",
"args": ["things-mcp"]
}
}
}
claude mcp add-json things '{"command":"uvx","args":["things-mcp"]}'
To make it available globally (across all projects), add -s user:
claude mcp add-json -s user things '{"command":"uvx","args":["things-mcp"]}'
After installation:
get-inbox - Get todos from Inboxget-today - Get todos due todayget-upcoming - Get upcoming todosget-anytime - Get todos from Anytime listget-someday - Get todos from Someday list, including tasks in Someday projectsget-logbook - Get completed todosget-trash - Get trashed todosget-todos - Get todos, optionally filtered by projectget-projects - Get all projectsget-areas - Get all areasget-tags - Get all tagsget-tagged-items - Get items with a specific tagget-tag-usage - Report how many items use each tag, sorted by usage; flag unused tags for cleanupsearch-todos - Simple search by title/notessearch-advanced - Advanced search with multiple filtersget-recent - Get recently created itemsadd-todo - Create a new todoadd-project - Create a new projectadd-area - Create a new Area (via AppleScript; Things URL scheme has no add-area command)update-area - Update an existing Area: rename or set tags (via AppleScript)update-todo - Update an existing todobulk-update-todos - Apply the same update to many todos in a single operationupdate-project - Update an existing projectshow-item - Show a specific item or list in Thingssearch-items - Search for items in ThingsThe list/search read tools (get-inbox, get-today, get-upcoming, get-anytime, get-someday, get-logbook, get-trash, get-todos, get-projects, get-areas, get-tags, get-tagged-items, get-headings, search-todos, search-advanced, get-recent) accept optional pagination:
limit - Maximum number of items to return (default: all; get-logbook defaults to 50)offset - Number of items to skip from the start (default: 0)When neither is set, output is unchanged. When set, a Showing X-Y of Z items header is prepended so you know how much more there is. An offset past the end is reported distinctly from an empty result.
These same read tools also return structured content alongside the human-readable text: MCP clients receive the raw item dicts plus count/total/offset/limit under structured_content, so data can be consumed programmatically without parsing the formatted text.
project_uuid (optional) - Filter todos by projectinclude_items (optional, default: true) - Include checklist itemsinclude_items (optional, default: false) - Include contained itemsstatus - Filter by status (incomplete/completed/canceled)start_date - Filter by start date (YYYY-MM-DD)deadline - Filter by deadline (YYYY-MM-DD)tag - Filter by tagarea - Filter by area UUIDtype - Filter by item type (to-do/project/heading)last - Filter by creation date (e.g., '3d' for last 3 days, '1w' for last week)period - Time period (e.g., '3d', '1w', '2m', '1y')period (optional, default: '7d') - Look-back window by completion date. Accepts d/w/m/y (e.g., '3d', '1w', '2m', '1y'); months ≈ 30 days, years ≈ 365 dayslimit (optional, default: 50) - Maximum number of completed items to returnonly_unused (optional, default: false) - Return only tags that no item references (cleanup candidates)tags - Replace all tags on the todoadd_tags - Append tags without removing existing oneschecklist_items - Replace the entire checklist with this listprepend_checklist_items - Add these items to the top of the checklistappend_checklist_items - Add these items to the bottom of the checklistApplies the same change to every todo in ids in a single operation. Requires the Things auth token to be enabled (Things → Settings → General → Enable Things URLs → Manage); the server reads it automatically.
ids (required) - List of todo UUIDs to updatelist / list_id - Move all into a project/area (by title or UUID)tags / add_tags - Replace or append tags on allwhen - Reschedule all (keyword or YYYY-MM-DD)deadline - Set deadline on all (YYYY-MM-DD)heading / heading_id - Move all under a heading (by title or UUID)completed / canceled - Mark all completed or canceledwhen - Accepts multiple formats:
today, tomorrow, evening, anytime, somedayYYYY-MM-DD (e.g., 2024-01-15)YYYY-MM-DD@HH:MM (e.g., 2024-01-15@14:30)If it's not working:
Make sure Things 3 is installed and has been opened at least once
Check that "Enable Things URLs" is turned on
Claude Desktop can't find uvx
brew install uv)uvx in your config. Find it with which uvx (typically /Users/USERNAME/.local/bin/uvx)The project includes a comprehensive unit test suite for the URL scheme and formatter modules.
# Install test dependencies
uv sync --extra test
# Run all tests
uv run pytest
# Run tests with verbose output
uv run pytest -v
# Run a specific test file
uv run pytest tests/test_url_scheme.py
# Run tests matching a pattern
uv run pytest -k "test_add_todo"
The project includes an integration test plan that can be executed by Claude (via Claude Cowork or Claude Code) to verify all MCP tools work correctly against a live Things database.
See docs/mcp_integration_test_plan.md for the full test plan.
things-mcp/
├── src/things_mcp/ # Main package
│ ├── __init__.py # Package exports
│ ├── __main__.py # Entry point for python -m
│ ├── server.py # MCP server implementation
│ ├── url_scheme.py # Things URL scheme implementation
│ └── formatters.py # Data formatting utilities
├── tests/ # Unit tests
│ ├── conftest.py # Test fixtures and configuration
│ ├── test_url_scheme.py
│ ├── test_formatters.py
│ ├── test_someday_filtering.py
│ └── test_mcp_server_filtering.py
├── docs/ # Documentation
│ └── mcp_integration_test_plan.md # Claude-executable integration test
├── manifest.json # MCPB package manifest
├── build_mcpb.sh # MCPB package build script
├── pyproject.toml # Project dependencies, build config, and pytest config
├── .env.example # Sample environment configuration
└── run.sh # Convenience runner script
By default, the server uses stdio transport for communication with MCP clients. For remote access scenarios, you can run the server with HTTP transport.
Set these environment variables to enable HTTP transport:
| Variable | Default | Description |
|---|---|---|
THINGS_MCP_TRANSPORT | stdio | Transport type: stdio or http |
THINGS_MCP_HOST | 127.0.0.1 | HTTP server bind address |
THINGS_MCP_PORT | 8000 | HTTP server port |
# Using uvx
THINGS_MCP_TRANSPORT=http THINGS_MCP_HOST=0.0.0.0 THINGS_MCP_PORT=8000 uvx things-mcp
# Or from source
THINGS_MCP_TRANSPORT=http THINGS_MCP_HOST=0.0.0.0 THINGS_MCP_PORT=8000 uv run things-mcp
See .env.example for a sample configuration file.
com.mcparmory/google-sheets
domdomegg/google-sheets-mcp
henilcalagiya/google-sheets-mcp
cct15/war-dashboard-data
moooonad/mcp-google-sheets-full
io.github.br0ski777/csv-to-json