A small Python CLI that caches Slack threads, channel messages, users, and channels to a local SQLite database.
Given a Slack thread URL (or an explicit channel id and root timestamp), it
fetches the thread via conversations.replies and stores every message in a
SQLite cache.
On subsequent runs it only fetches new replies (and detects edits) by passing
oldest to the API based on the highest cached ts.
It can also cache every workspace user and visible channel, so that threads can be rendered with human-readable author names.
uv syncRun it:
uv run slack-cached --helpCredentials are loaded in this order:
- Environment variables:
SLACK_TOKEN(and optionalSLACK_COOKIEfor xoxc/web-client tokens). - A config file at
$XDG_CONFIG_HOME/slack-cached/config(defaults to~/.config/slack-cached/config). It uses a simpleKEY=VALUEformat:SLACK_TOKEN=xoxb-... SLACK_COOKIE=... SLACK_API_BASE_URL=https://slack.com/api
The default cache database lives at
$XDG_CACHE_HOME/slack-cached/threads.db (or ~/.cache/slack-cached/threads.db).
Override with --db /path/to/file.db.
All commands accept -v/--verbose for debug logging on stderr, --db to
override the cache location, and --api-base-url to override the Slack API
base URL (defaults to https://slack.com/api; also settable via
SLACK_API_BASE_URL).
Cache or refresh a thread (no thread output, only a summary on stderr):
slack-cached fetch https://acme.slack.com/archives/C0123ABCDEF/p1700000000123456Or with explicit channel/ts:
slack-cached fetch --channel C0123ABCDEF --ts 1700000000.123456Show a cached thread (human-readable by default; use --json for JSON, or
--jsonl for the whole payload as a single compact JSON line). It auto-fetches
if the thread is missing; pass --no-fetch to disable that:
slack-cached show https://acme.slack.com/archives/C0123ABCDEF/p1700000000123456
slack-cached show --json https://acme.slack.com/archives/C0123ABCDEF/p1700000000123456
slack-cached show --jsonl --channel C0123ABCDEF --ts 1700000000.123456 >> threads.jsonlFetch all top-level messages in a channel via conversations.history:
slack-cached fetch --channel C0123ABCDEFAdd --full-threads to also fetch every reply thread for messages that have
replies:
slack-cached fetch --channel C0123ABCDEF --full-threadsSearch the workspace with the same query syntax as the Slack search box. Every
matched message is cached under its (channel, thread_ts) so it can be revisited
later with show. Search is always a live API call:
slack-cached search "deploy failed"
slack-cached search "from:@alice after:2024-01-01" --json
slack-cached search "incident" --jsonl # one JSON line per run, easy to appendAdd --full-threads to also fetch every reply for each thread a match belongs to:
slack-cached search "incident" --full-threadsTune result paging and ordering with --count, --sort (score or timestamp,
default timestamp), and --sort-dir (asc or desc, default desc).
slack-cached search "RFC" --count 5 --sort score --sort-dir ascPoll multiple channels concurrently for new messages:
slack-cached poll --channels C001,#general,random --interval 5m --last 5m --concurrency 3Uses httpx.AsyncClient with an asyncio.Semaphore for concurrent, non-blocking
HTTP requests. Reads X-RateLimit-Remaining headers to proactively throttle
before hitting 429s. Add --full-threads to expand threads, and --json to get
per-cycle JSON summaries on stdout. Stops gracefully with Ctrl+C.
Cache or refresh every workspace user or visible channel:
slack-cached fetch-users
slack-cached fetch-channelsShow cached users or channels (human-readable by default, --json for pretty
JSON, --jsonl for a single compact JSON line; both auto-fetch when empty
unless --no-fetch is given):
slack-cached show-users
slack-cached show-channels --json
slack-cached show-channels --jsonlWhen a thread's authors are present in the cached users, show renders their
real name and handle (e.g. Alice Smith (alice)) instead of raw user ids.
fetch always reaches out to Slack.
If the thread is already cached, it requests conversations.replies with
oldest=<latest_cached_ts> so the API returns only new replies (and any
recent edits at that boundary).
Messages are upserted by ts, so edits replace the older version in place.
HTTP 429 / ratelimited responses are retried automatically with exponential
backoff (up to 5 attempts), respecting the Retry-After header.
A built-in fake Slack API server for testing and development:
uv run slack-fake-server --help
uv run slack-fake-server --port 8199 --num-threads 50It serves deterministic workspace data (conversations.list,
conversations.replies, conversations.history, users.list) and can
simulate Slack-tier rate limiting with --rate-limits.
Point slack-cached at it with:
slack-cached --api-base-url http://localhost:8199/api fetch ...uv sync
uv run pytest
uv run ruff check
uv run ruff format --check