CUPT stands for "ClickUP Terminal," a command-line interface for accessing your tasks in ClickUp from the terminal or via your favorite AI-based tool.
- Task listing with deep paging, date filters, and subtask nesting (
↳). - Tag and team filters —
cupt list --tag ai_ready --team MattTechscopes to exactly what you're working on. - Hierarchical context —
cupt context <id>shows a task's parent and siblings. - Status-aware completion —
cupt doneresolves the correct "closed" status per task's list automatically;--dry-runlets you preview before writing. - Stateful interactive UX —
cupt listshows a stable#column (1, 2, 3...) you can type instead of long ClickUp IDs.cupt start <id>sets a session active task so subsequent commands (note,done,show,time start) don't need an ID. Both features auto-hide when stdout is a pipe, so scripts and agents keep their stateless behavior. - Quick capture —
cupt add "task name"creates a new task in the active task's list (or your default list).--parent this/--blocks thislink it to whatever you're working on. - Time tracking — start/stop timers and add manual entries.
- Notes — quick comments and a list view per task.
- Attachments — list, download, and upload files on tasks.
- Flexible auth — OAuth or Personal API Token.
- Offline support —
cupt listtransparently caches what it just showed;cupt show <id> --offlineworks without a network.cupt prefetchpopulates the cache eagerly. - JSON output everywhere — every read command supports
--jsonfor piping intojqor feeding an agent. - Agent skill bundled —
skill/cupt-clickup/is a portable SKILL.md that teaches Claude Code, OpenCode, Codex, and other agents how to drive cupt efficiently. See For AI agents below.
pipx install cupt
pipx upgrade cupt # later, when a new version is publishedpipx installs cupt in an isolated environment that's available globally — same convenience as a system package, no chance of conflicting with other Python projects. If you don't have pipx: brew install pipx && pipx ensurepath, or see https://pipx.pypa.io/stable/installation/.
Plain pip install cupt works too; pipx is just the friendlier default for CLI tools.
git clone https://github.com/newz2000/cupt.git cd cupt python -m venv venv && source venv/bin/activate pip install -e .
To make a local checkout available system-wide: pipx install --force .
This walkthrough takes about five minutes and ends with you running real queries against your ClickUp workspace.
cupt auth
You'll be asked to pick OAuth or a Personal API Token. For most users a Personal API Token is faster — grab one from https://app.clickup.com/settings/apps (it starts with pk_) and paste it when prompted. OAuth is the right choice if you're sharing this install with a team.
After auth, cupt automatically picks your first workspace as the default and you're ready to go:
cupt status # confirm: shows your username and workspacecupt list # tasks assigned to you, sorted by due date cupt list --today # just today cupt list --overdue # overdue, oldest first cupt list --week # the next seven days cupt list --all # everyone's tasks in the workspace, not just yours cupt list -n <N> --verbose # cap to N rows, include assignee/estimate/tracked columns
The default view shows ID, status, due date, and name. Every row's ID is what you'll feed into the other commands.
cupt show <task-id> # description, status, assignees, tags, list, folder cupt show <task-id> --notes # also include all comments cupt context <task-id> # parent + siblings/subtasks
In an interactive terminal cupt list shows a # column with stable short IDs (1, 2, 3...) you can use anywhere a task ID is expected:
# ID Status Due Name
-----------------------------------------------------------------
1 868abc open 2026年06月09日 17:00 Fix login bug
2 868def open 2026年06月10日 17:00 Review onboarding doc
3 868ghi open — Reply to client
These numbers stay stable across sessions until the underlying task is closed (or moves out of your pending list). cupt show 2, cupt note 2 "...", cupt done 2 all work.
When you pick a task to focus on, mark it active:
cupt start 1 # or `cupt start 868abc` # every subsequent command falls back to this task when no ID is given: cupt note "made progress on the auth flow" cupt time start # optional — start a timer for it cupt show # re-read the description without retyping cupt done --note "shipping it" # closes the task and clears active
cupt active shows the current active task; cupt stop clears it without closing the task (use this when you get interrupted). Every cupt list ends with a one-line footer reminding you what's active, so you can spot "wrong terminal" mistakes before they happen.
Mid-task you'll often realize a new task needs to exist. cupt add captures it without leaving the terminal:
cupt add "Migrate the legacy auth tokens" # lands in the active task's list cupt add "Subtask of what I'm doing" --parent this # subtask of active cupt add "Have to do this first" --blocks this # active task now depends on this one cupt add "Reply to legal" --due tomorrow --tag urgent # with metadata
Pass an explicit ID instead of this (--parent 868xyz or --parent 2) to link to a different task. Without --parent or --blocks, the new task is a peer — no relationship — so capture stays fast when you just want to write something down.
Hidden in scripts. Short IDs, the active-task fallback, and the list footer all disappear when stdout is a pipe (or when CUPT_INTERACTIVE=0 / --no-interactive is set). cupt list | grep ..., cupt show abc --json, and an agent calling cupt add "..." --list <id> --json all behave the way they did in 0.7.x — stateless and predictable.
ClickUp tags (ai_ready, urgent, waiting, etc.) and teams (what ClickUp calls user-groups in its UI, e.g. MattTech, AI Agent) are how you carve a busy workspace into something workable.
cupt list --tag <tag> # must have this tag cupt list --tag <tag-a> --tag <tag-b> # must have BOTH (AND) cupt list --no-tag <tag> # must NOT have this tag cupt teams # discover team names and IDs cupt list --team <team-name> # only tasks assigned to that team cupt list --team <team-a> --team <team-b> # either team (OR) cupt list --team <team> --tag <tag> --mine # stack filters freely
--mine is on by default. Add --all (or omit --mine) to see the whole workspace.
A note on --team performance. ClickUp's API has no server-side filter for teams, so cupt has to walk extra pages to find matches. On big workspaces this can take tens of seconds for --all --team; local v1.0 verification against one real workspace measured about 31–37 seconds when the command hit the 10-page cap. After the table you'll see a footer like (team filter: searched N pages in T.Ts) so the cost is honest. For the fastest, most reliable results on large workspaces, pair the team filter with a discriminating tag — the tag narrows server-side before the team filter runs:
cupt list --team MattTech --tag ai_ready # faster: server narrows firstIf you see hit page cap — pair with --tag for full coverage in the footer, that's a hint that matches may exist on pages we didn't walk.
Different ClickUp lists carry different status names (one uses Done, another Complete, another Resolved). cupt done figures out the right one per task's list automatically. If you want to double-check before writing:
cupt statuses <task-id> # show all statuses for the task's list, # marking the one that done would apply cupt done <task-id> --dry-run # preview the resolved status, no write cupt done <task-id> # do it cupt done <task-id> --note "Shipped behind the AI_v2 flag"
cupt time start <task-id> # start a timer cupt time stop # stop the running timer cupt time add <task-id> 1h30m # log time after the fact cupt note <task-id> "Talked to the client, they want the v2 layout" cupt notes <task-id> # list all comments cupt attach list <task-id> cupt attach add <task-id> <file> cupt attach get <task-id> <selector>
Every cupt list invocation silently caches the tasks it just showed (plus their details and comments) for offline reads. When you know you're about to lose network, run:
cupt prefetch # eagerly cache details for the current task set cupt list --offline # later, on the plane cupt show <task-id> --offline
When you want to work down a queue without re-running cupt list, use sequential focus mode:
cupt work --tag ai_ready
cupt work --today
cupt work --team MattTech --json # inspect the queue without promptingInteractive mode presents one task at a time and accepts [w]ork, [s]kip, [d]one, or [q]uit. Selecting work sets the active task and starts a timer when possible; selecting done resolves the task's completion status per list.
Every read command supports --json:
cupt list --tag ai_ready --json | jq '.[] | .name' cupt statuses <task-id> --json # agent-friendly: target + all statuses
Shell completion snippets for bash, zsh, and fish live in docs/shell-completion.md. Agent JSON and exit-code contracts live in docs/agent-contract.md.
You now know enough to be productive. The command reference below is a quicker reminder once these basics are in muscle memory.
| Command | What it does |
|---|---|
cupt auth / cupt logout / cupt status |
Manage credentials and check current account/workspace |
cupt config --workspace-id <id> |
Override the default workspace |
cupt teams |
List ClickUp teams (user-groups) in the workspace |
cupt list [--overdue|--today|--week] [--tag X] [--no-tag X] [--team X] [--mine|--all] [--json] [--offline] |
List tasks with stackable filters. Interactive sessions get a # short-ID column. |
cupt show [<id>] [--notes] [--json] [--offline] |
Full task details. Falls back to the active task when no ID is given. |
cupt context [<id>] |
Parent + sibling/subtask view. Falls back to active. |
cupt statuses <id> [--list] [--json] |
Show available statuses for a task's list (or pass --list <list-id>) |
cupt done [<id>] [--note "..."] [--dry-run] |
Mark complete; clears the active pointer on success. --dry-run previews the resolved status. |
cupt add "<name>" [--list X] [--parent <id|this>] [--blocks <id|this>] [-d "..."] [--due ...] [--tag X] [--json] |
Create a new task. Defaults: active task's list, you as assignee, no link. |
cupt start <id> / cupt stop / cupt active |
Set / clear / show the session's active task (interactive only). |
cupt work [--tag X] [--team X] [--today|--overdue|--week] [--json] |
Sequential focus mode: work, skip, complete, or quit one task at a time. |
cupt summary [--all] [--json] |
Daily due/overdue/completed/time summary. |
cupt tag add|remove <id> <name> |
Tag management |
cupt time start [<id>] / cupt time stop / cupt time add [<id>] <dur> / cupt time status |
Time tracking. start and add fall back to the active task. |
cupt note [<id>] "<text>" / cupt notes [<id>] |
Add or list comments; both fall back to the active task. |
cupt attach list|add|get <id> [args] |
Attachment management |
cupt prefetch |
Cache details for the current task set for offline use |
Global: --interactive / --no-interactive, --lang, CUPT_INTERACTIVE=1|0, CUPT_LANG |
Force interactive (short IDs + active task) or stateless mode. Default: enabled when stdout is a TTY. |
cupt is usable as a dependency in your own Python code. Importing it does no I/O — no config directory is created, no network calls happen until you make one explicitly.
cupt includes beta CLI localization catalogs for ClickUp's supported app
languages:
cupt --lang es_LA --help CUPT_LANG=pt_BR cupt list --help
Available catalogs are es_LA, es_ES, fr, de, it, and pt_BR.
The es, es-419, and es-LA aliases resolve to Latin American Spanish;
pt-BR resolves to pt_BR.
The Spanish catalogs are AI-generated beta translations intended for feedback.
The French, German, Italian, and Brazilian Portuguese catalogs are present so
reviewers can work against the same .pot source and currently fall back to
English for most strings. Command names, flags, JSON keys, IDs, task content,
and low-level API error payloads stay literal so scripts remain stable and
troubleshooting stays searchable. Click's built-in help headings such as
Usage, Options, and Commands may still appear in English.
from cupt import ClickUpClient, TaskService, APIError client = ClickUpClient("pk_xxxxxxxxxxxxxxxx") # personal API token service = TaskService(client) try: tasks = service.list_tasks( workspace_id="123456", tags=["urgent"], # server-side filter include_closed=False, ) urgent_billing = service.filter_by_tags( tasks, required=["urgent", "billing"] ) for t in urgent_billing: print(t["id"], t["name"]) except APIError as e: print(f"ClickUp request failed: {e}")
For agents that need to mark tasks complete across lists with diverging status names, use TaskService.resolve_completion_status(task_id) to discover the target status before writing. See AGENTS.md for the canonical pattern.
Public API surface (anything importable from cupt top-level):
| Symbol | Purpose |
|---|---|
ClickUpClient |
Thin HTTP wrapper around the ClickUp v2 REST API. |
TaskService |
List/filter/complete tasks; resolve parent names; resolve closed-status per list. |
TimeService |
Start/stop timers, add time entries, fetch totals. |
NoteService |
Add and list task comments. |
CuptError |
Base exception. All cupt errors subclass this. |
APIError |
HTTP failure, timeout, or invalid JSON from ClickUp. |
AuthError |
Missing or invalid credentials. |
ConfigError |
Configuration is missing or malformed. |
Other modules (cupt.config, cupt.context, command modules) are internal to the CLI and may change between releases.
cupt ships a portable agent skill (skill/cupt-clickup/) that teaches AI
agents — Claude Code, OpenCode, Codex, and any other tool that supports the
SKILL.md convention — how to use cupt efficiently. The skill is plain
markdown with YAML frontmatter, so the same files work across agents; only
the install path differs.
Pick the line(s) that match the agent(s) you use:
# Claude Code (user-scope — available in every project) mkdir -p ~/.claude/skills && cp -r skill/cupt-clickup ~/.claude/skills/ # Claude Code (project-scope — checked into the repo you're working in) mkdir -p .claude/skills && cp -r skill/cupt-clickup .claude/skills/ # OpenCode (also reads from ~/.claude/skills, so the Claude install above # works too — these are if you prefer OpenCode's native paths) mkdir -p ~/.config/opencode/skills && cp -r skill/cupt-clickup ~/.config/opencode/skills/ # Codex (uses its own .agents path, does NOT read .claude/skills) mkdir -p ~/.agents/skills && cp -r skill/cupt-clickup ~/.agents/skills/
The skill includes a pre-flight check (cupt --version, cupt status)
so agents detect a missing install or unauthenticated account and prompt
the user instead of failing silently. It does not attempt to install cupt
itself — that's a user action.
See skill/cupt-clickup/examples.md for the multi-step agent workflows
the skill teaches (tagged work queues, team-scoped processing, safe
multi-list completion, human handoff, empty-queue handling).
ClickUp's REST API still uses "team" for what its UI now calls "Workspace," and uses "group" for what its UI calls "Team." cupt follows the current UI:
- Workspace = the top-level container (
cupt config --workspace-id). - Team = a user-group within a workspace (
cupt list --team,cupt teams).
If you're reading ClickUp's API docs and see team_id, that's the workspace ID. The CLI shields you from this.
cupt is built with a focus on stability and testability.
- Coverage: CI enforces at least 80%.
- Tests: unit tests use
pytestand mocks.
pytest --cov=cupt tests/
Post-1.0 experiments are tracked in PLAN.md, including saved views, policy ingestion, and optional AI plugin ideas.
cupt/— package rootcupt/services/— core business logic (TaskService, TimeService, NoteService)cupt/api.py— ClickUp API client wrappercupt/main.py— CLI entry pointtests/— unit tests
If you or your favorite AI tool want to make improvements, please submit a pull request. Read AGENTS.md before starting — it captures the project's conventions, known gotchas, and quality bar. Install the pre-commit hook (pip install pre-commit && pre-commit install) before committing.