| name | setting-up-relayfile |
| description | Use when an agent or human needs to set up relayfile end-to-end so agents can read and write provider files through a local mount. Covers `relayfile setup`, dynamic integration discovery with `relayfile integration available/search`, Nango and Composio backend selection, Atlassian site selection and metadata, cloud login, OAuth/connect flows, mount verification, `RELAYFILE_LOCAL_DIR` handoff, writeback status and retry commands, and key May 2026 cloud-mount gotchas. |
Setting Up Relayfile (Mount + Writeback for Agents)
Overview
Relayfile mounts a provider (Notion, Linear, Slack, GitHub, and other adapter-backed integrations) as ordinary files on disk so an agent can read and write through the filesystem instead of calling APIs. This skill is the canonical setup recipe. Follow it top-to-bottom for first-time setup; jump to Recovering from breakage if a working mount has gone wrong.
When to use this skill
- An agent needs read access to a provider (e.g., "summarize this Notion database").
- An agent needs to write back to a provider (e.g., "post a review on this Notion page", "update this Linear issue").
- A human is setting up a mount before delegating work to an agent.
- A mount stopped reflecting changes and you need to diagnose where.
What you get
After setup, files appear under <local-dir>/<provider>/...:
~/relayfile-mount/notion/
āāā databases/
ā āāā <slug>--<id>/
ā ā āāā metadata.json ā database schema (read-only)
ā ā āāā pages/
ā ā āāā <slug>--<id>.json ā page metadata
ā ā āāā <slug>--<id>/
ā ā āāā content.md ā page body (READ + WRITE)
ā ā āāā blocks/<id>.json ā raw Notion block tree
āāā pages/ ā top-level pages (not in a database)
Read = cat. Write = overwrite, create, or remove files in writable adapter resource directories. The mount daemon picks up the change, queues a writeback, and the cloud delivers to the provider's API.
Current mounts are also self-describing. Start with <local-dir>/LAYOUT.md, then read provider-specific <provider>/LAYOUT.md files and nearby _index.json files instead of hard-coding paths from memory. Entity filenames may use a <sanitized-name>__<id> convention, and some providers expose alias views such as by-title/, by-id/, by-name/, or by-state/.
Prerequisites
- Recent
relayfile CLI on $PATH. Verify: relayfile --help should list setup, integration, writeback, and the integration available / integration search / integration set-metadata subcommands.
- A modern macOS or Linux shell with
jq for JSON inspection. AWS CLI access is optional and only needed for internal cloud log diagnostics.
- Network access to
agentrelay.com/cloud (cloud control plane), api.relayfile.dev (relayfile API), connect.nango.dev (Nango OAuth), and Composio connect endpoints when using --backend composio.
Step 1 ā Run setup (interactive happy path)
relayfile setup \
--provider notion \
--workspace my-agent \
--local-dir ~/relayfile-mount \
--no-open
What this does, in order:
- Cloud login. Opens a localhost callback server, prints a URL to
agentrelay.com/cloud/api/v1/cli/login?.... You complete the login in the browser; the cloud redirects back to 127.0.0.1:<port>/callback with an access token. The CLI stores cloud credentials in ~/.relayfile/cloud-credentials.json and the active Relayfile workspace token in ~/.relayfile/credentials.json.
- Workspace create. POSTs
/api/v1/workspaces with {"name": "my-agent"}. Returns { workspaceId: "rw_<8hex>", relaycastApiKey, relayfileUrl, ... }. The workspace ID is the prefix-style rw_* format ā not a UUID.
- Integration connect. By default, mints a Nango Connect URL like
https://connect.nango.dev/?session_token=nango_connect_session_<hash> and opens it (or prints it, with --no-open). With --backend composio, the cloud resolves the provider to a Composio toolkit, finds or creates the Composio auth config when Composio supports automatic managed auth, and mints a Composio connect URL. You complete the provider auth there. The provider callback inserts a row into workspace_integrations and queues an initial sync. For Jira and Confluence, the CLI then lists the Atlassian sites covered by the OAuth grant and asks which site to bind when more than one is available.
- Initial sync. The cloud nango-sync-worker pulls page metadata + content from the provider and writes it to relayfile. Takes ~30s for a small workspace.
- Mount. Starts a local daemon that polls
api.relayfile.dev/v1/workspaces/<id>/sync/status every 30s and reflects changes into <local-dir>/<provider>/.
Use --no-open if you're an agent: the wizard otherwise tries to open a browser, which usually fails in headless environments and burns the OAuth state.
Step 2 ā Verify the mount is healthy
relayfile status my-agent
Healthy output:
workspace rw_xxxxxxxx (my-agent) mode: poll lag: 4s
local mirror: /Users/you/relayfile-mount
daemon: running (pid 12345)
notion ready 214 files last event 2s ago
pending writebacks: 0 failed: 0 dead-lettered: 0
What each row means:
lag: <N>s ā how stale the mirror is relative to the cloud. >60s means investigate.
daemon: not running ā the mount poller exited. Start it with relayfile mount my-agent ~/relayfile-mount &.
pending writebacks ā local writes queued for upload. Should drain to 0 within ~30s.
failed ā lifetime counter of non-2xx responses from the cloud's PUT endpoint. Informational; don't gate on this.
dead-lettered ā count of writebacks that exhausted retries and got persisted under <local-dir>/.relay/dead-letter/<opId>.json. Gate on this.
If dead-lettered > 0, see Recovering from breakage below.
Step 3 ā Hand off to an agent
Two patterns, depending on where the agent runs:
Pattern A: local agent (Claude Code, scripts, Cursor)
The agent reads files directly:
export RELAYFILE_LOCAL_DIR=~/relayfile-mount
Mental model for the agent: ordinary files. Use Read, Write, Edit, Glob, Grep ā same as any project directory. Writes propagate within ~30s.
Before writing, read the relevant _PERMISSIONS.md or discovery files for the target subtree. If a path is denied, Relayfile preserves the local copy and records the denial in <local-dir>/.relay/permissions-denied.log.
Pattern B: remote agent / SDK access (no disk mirror)
Use @relayfile/sdk against the workspace token:
import { RelayFileClient } from "@relayfile/sdk";
const token = process.env.RELAYFILE_TOKEN;
const client = new RelayFileClient({ token, server: "https://api.relayfile.dev" });
const file = await client.getFile("rw_xxxxxxxx", "/notion/pages/xxx/content.md");
await client.putFile("rw_xxxxxxxx", "/notion/pages/xxx/content.md", {
content: "# New body\n\nā¦",
contentType: "text/markdown",
});
The token issued by relayfile setup carries (as of May 2026): fs:read, fs:write, sync:read, sync:trigger, ops:read. The last two were added so agents can introspect the writeback pipeline (relayfile pull, relayfile ops list, GET /v1/workspaces/<id>/ops/<opId>).
Step 4 ā Verify writeback works (optional but recommended)
Skip-able if the agent only reads. Required if the agent will write.
- Pick a throwaway page in the provider.
- Write a marker:
echo "[writeback test $(date -u +%FT%TZ)]" > ~/relayfile-mount/notion/pages/<throwaway-page>/content.md
- Wait 30s.
- Open the provider's web UI; the marker should appear.
- Run
relayfile writeback status ā dead-lettered should still be 0.
If the marker doesn't appear in step 4, see Recovering from breakage.
Discover writeback contracts before writing
Do not guess writeback shapes and do not use a magic new.json filename. Current relayfile adapters ship discovery documents for writable resources:
A typical discovery surface looks like:
<provider>/
āāā .adapter.md ā adapter overview, operations, ID patterns
āāā <resource>/
āāā .schema.json ā full-record JSON Schema, draft 2020-12
āāā .create.example.json ā minimal create payload
- First check which writeback contract the mounted workspace exposes. Run
find "$RELAYFILE_LOCAL_DIR" \( -name '.adapter.md' -o -name '.schema.json' -o -name 'new.json' \) | head -40. If discovery files are absent and new.json templates are present, the mounted workspace is still on the pre-file-native adapter bundle; do not apply the create-by-filename flow until the cloud/adapter deployment has refreshed that workspace.
- Read the provider
.adapter.md first. In mounted workspaces this may appear under the provider tree or under <local-dir>/discovery/<provider>/.adapter.md; if unsure, run find "$RELAYFILE_LOCAL_DIR" -path '*/.adapter.md'.
- Read the resource
.schema.json before writing JSON. It is JSON Schema draft 2020-12 for the full synced record. Fields with "readOnly": true are server-managed and must not be written. Common schema paths are resource-local, such as /linear/issues/.schema.json; packaged adapters also carry a discovery copy under discovery/<provider>/....
- For creates, start from the sibling
.create.example.json. The create example intentionally omits read-only fields.
- For edits, write only mutable fields to a canonical
<id>.json; omitted fields are left alone.
- For creates, write a valid JSON document to any non-canonical filename in the resource directory, such as
draft-message.json or create issue.json. The adapter creates the provider record at the real <id>.json and rewrites the draft file as a receipt/pointer.
- For deletes, remove the canonical
<id>.json only when the resource's .adapter.md says delete is supported.
The <id> pattern is resource-specific. A Linear issue ID is a UUID; a Slack message ID is a timestamp-like token; GitHub and many CRM IDs are integers. The .adapter.md ID pattern section is the source of truth for whether a filename routes to PATCH/DELETE or CREATE.
Path conventions per provider
| Provider | Read paths | Write paths |
|---|
| Notion | /notion/pages/<slug>--<id>/content.md, /notion/databases/<id>/pages/.../content.md, <slug>.json (metadata) | same paths overwrite the body / properties |
| Slack | /slack/channels/<id>/messages/ plus .adapter.md / .schema.json discovery | create by writing a valid message JSON to /slack/channels/<id>/messages/<non-canonical>.json; edit/delete canonical message files when supported |
| Linear | /linear/issues/<id>.json, comments under issue resources, plus .adapter.md / .schema.json discovery | create by writing a valid issue/comment JSON to a non-canonical filename; edit/delete canonical issue files when supported |
| GitHub | /github/repos/<owner>/<repo>/pulls/<n>/metadata.json, files.json, plus .adapter.md / .schema.json discovery | create a review by writing the review JSON to a non-canonical file under the reviews resource |
new.json is not special in the file-native adapter contract. If a current .adapter.md and .schema.json are present, translate older examples using /messages/new.json or /comments/new.json to "write the create payload to any non-canonical filename in the resource directory." If the live mount only exposes new.json, treat that as an older deployment surface and follow the mounted template or wait for the workspace to refresh onto the new adapter version.
<local-dir>/.relay/ is reserved ā never write there. Anything you put under it gets ignored or treated as daemon state.
Adding more integrations after setup
Do not guess provider names. Ask the CLI for the live catalog first; it pulls static Relayfile integrations plus dynamic Nango providers and Composio toolkits from the cloud, then caches the result locally.
relayfile integration available --refresh
relayfile integration search docker --backend composio --refresh
relayfile integration available --backend nango --search notion
Use available when you want to browse or filter the catalog. Use search when you already have a term. --refresh bypasses the local catalog cache and is worth using when a provider was just added in Nango or Composio. Add --json when an agent needs machine-readable output.
Then connect the provider, optionally selecting the backend:
relayfile integration connect linear --workspace my-agent
relayfile integration connect slack --workspace my-agent
relayfile integration connect dockerhub --backend composio --workspace my-agent --no-open
relayfile integration list --workspace my-agent
For Jira and Confluence, a single Atlassian OAuth grant can cover multiple sites. After a fresh relayfile setup --provider jira|confluence or relayfile integration connect jira|confluence, the CLI calls Cloud's accessible-resources endpoint. If there is one site, it auto-selects it; if there are multiple sites, it prompts for a numbered choice before waiting for initial sync. The selected site's cloudId and baseUrl are saved as integration metadata so Cloud knows which tenant to sync.
If the picker was skipped, the wrong site was chosen, or an operator needs to update provider metadata later, use integration set-metadata. The command replaces the provider metadata namespace, so include every key you want to keep:
relayfile integration set-metadata jira \
cloudId=abc-123 \
baseUrl=https://example.atlassian.net \
--workspace my-agent \
--yes
set-metadata accepts flat KEY=VALUE pairs only. Keys such as site.cloudId or site[cloudId] are rejected locally; nested metadata is not part of the v1 CLI contract. Re-running relayfile integration connect jira or confluence for an already-connected provider should not overwrite existing metadata unless it starts a fresh OAuth connect.
Backend rules:
- Nango is the default backend for the standard Relayfile providers such as Notion, Linear, Slack, and GitHub.
- Composio can be requested explicitly with
--backend composio for supported providers and dynamic Composio toolkits.
- User-facing aliases are allowed where the cloud knows them. For example,
dockerhub resolves to the Composio toolkit slug docker_hub; if discovery shows docker_hub, either spelling is acceptable for connect.
- For Composio, the cloud first lists existing auth configs for the toolkit. If none exists, it attempts to create a managed/default auth config automatically. If Composio cannot create managed auth for that toolkit, the command returns an actionable error; at that point the human must add a custom auth config in Composio Authentication Management and retry the same
relayfile integration connect ... command.
Each provider gets its own subtree under <local-dir>/. Disconnect with relayfile integration disconnect <provider> ā leaves a marker at .relay/disconnected/<provider>.json and removes the provider's tree from the mirror.
Common gotchas
G1 ā Cold-start 500 on workspace create
POST /api/v1/workspaces sometimes 500s on the first call after the cloud Lambda has been idle. Retry once before doing anything diagnostic. Verified May 2026: same call succeeded immediately on retry.
If it 500s twice in a row, check aws logs tail /aws/lambda/clou-production-AgentRelayCloudWebServerUseast1Function-<suffix> --since 5m --follow for the actual stack trace.
G2 ā OAuth callback timing trap
The wizard prints the cloud-login URL, opens a localhost callback server, then waits. If you complete the login after the wizard has timed out (or if you click the callback URL by hand later), the redirect-to-localhost won't load ā that's expected. The login already completed; the wizard just isn't listening anymore. Re-run relayfile setup from scratch.
Same trap on the Nango Connect URL: it has a ~30 minute TTL. If the wizard exited and you click it later, you may need to mint a fresh one:
curl -sS -X POST "https://agentrelay.com/cloud/api/v1/workspaces/<id>/integrations/connect-session" \
-H "Authorization: Bearer $(cat ~/.relayfile/credentials.json | jq -r .token)" \
-H "Content-Type: application/json" \
-d '{"allowedIntegrations":["notion"]}'
G3 ā Dynamic provider discovery and Composio names
When adding a non-standard provider, use discovery before connecting:
relayfile integration search <term> --backend composio --refresh
relayfile integration search <term> --backend nango --refresh
The cloud catalog normalizes provider IDs and backend names. Nango results use Nango provider/config names; Composio results use toolkit slugs. Some human names differ from backend slugs: Docker Hub is exposed by Composio as docker_hub, while the CLI also accepts the common alias dockerhub for connect. Prefer the exact ID printed by integration available/search when scripting.
If relayfile integration connect <provider> --backend composio fails with a message that the toolkit does not support automatic managed auth config creation, the provider was discovered correctly but Composio cannot create a default auth config without extra credentials. A human must add a custom auth config for that toolkit in Composio Authentication Management, then rerun the same command. Cloud will re-check Composio and use the new auth config dynamically.
G4 ā Jira / Confluence sync says cloudId is missing
Jira and Confluence sync require metadata.cloudId when the Atlassian OAuth grant covers multiple sites. A recent CLI handles this through the post-OAuth picker. If an older setup or manual Cloud change left metadata unset, rerun a fresh connect or set the metadata explicitly:
relayfile integration connect jira --workspace my-agent --no-open
relayfile integration set-metadata jira cloudId=<cloud-id> baseUrl=https://<site>.atlassian.net --workspace my-agent --yes
Do not guess the cloudId. Prefer the CLI picker from a fresh OAuth connect; otherwise get the exact site id from Cloud/support tooling before using set-metadata.
G5 ā Workspace ID format
Workspaces created by the productized cloud-mount flow are rw_<8hex>. Older workspaces (and most internal API surfaces) use UUIDs. Most schema columns still type workspace_id as uuid ā see docs/architecture/workspace-id-unification.md in the cloud repo for the broader migration. For this skill: don't substitute a UUID workspace id when the CLI gave you rw_*. They are not interchangeable.
G6 ā Mount mirror dir conventions
<local-dir>/
āāā <provider>/... ā actual files
āāā .relay/
ā āāā state.json ā daemon's live state (workspaceId, lag, counters, remoteRoot)
ā āāā integrations/<provider>.json ā per-integration metadata
ā āāā dead-letter/<opId>.json ā failed writebacks (Phase 1 dead-letter)
ā āāā permissions-denied.log ā read/write denials preserved for diagnosis
ā āāā disconnected/<provider>.json ā marker after `integration disconnect`
ā āāā conflicts/<resolved-conflicts>
āāā .relayfile-mount-state.json ā sync revisions per file
The dead-letter dir is the agent's primary debugging surface for stuck writebacks. Each file is a JSON record with { opId, path, attempts, lastStatus, lastBody, ts }.
Recovering from breakage
Symptom: file edits don't appear in the provider
Run:
relayfile writeback status my-agent --json | jq
Three cases:
Case 1 ā dead-lettered is non-empty.
The mount tried, retried, gave up. Inspect the dead-letter records:
ls ~/relayfile-mount/.relay/dead-letter/
cat ~/relayfile-mount/.relay/dead-letter/op_*.json | jq '{opId, path, lastStatus, lastBody, attempts}'
lastStatus tells you what the cloud rejected with. lastBody is truncated to 1KB. Once you've fixed the underlying issue (e.g. the file had bad JSON properties, or the page was archived in the provider), retry:
relayfile writeback retry --opId <opId> my-agent
The dead-letter file gets removed if the retry succeeds.
Case 2 ā pending is non-zero and not draining.
The daemon enqueued a writeback but isn't getting through to the cloud. Check the daemon log:
relayfile logs my-agent | tail -40
Look for WARN writeback request failed lines (Phase 1 logging contract). Most common cause: token expired. Fix:
relayfile login --server https://agentrelay.com/cloud
then re-run mount.
Case 3 ā counts are zero but provider isn't updating.
The write didn't make it past the local file system into the daemon's queue. Verify:
- File mtime updated (
stat <file>)
- Daemon is actually running (
pgrep -fl "relayfile mount")
- Path is under a writable subtree (e.g.
content.md paths are write-enabled; metadata.json paths sometimes aren't, depending on the adapter)
If the daemon exited, restart:
relayfile mount my-agent ~/relayfile-mount &
Symptom: relayfile setup hangs at "Connect notion: "
The wizard is polling the cloud's /integrations/<provider>/status endpoint waiting for OAuth to complete. Either:
- Open the URL and complete the OAuth (the wizard will detect and proceed within ~5s of the Nango webhook firing).
- Cancel with Ctrl-C and run
relayfile integration connect notion --workspace my-agent --no-open separately so the wizard isn't blocked on the OAuth step.
Symptom: relayfile writeback status exits non-zero but I don't see why
The CLI exit code is non-zero only when there are dead-lettered ops (not just because the lifetime failedWritebacks counter is non-zero ā that's by design as of May 2026). Run with --json and look at deadLettered.length. If it's 0, the non-zero exit is a regression ā file a bug.
Cleaning up
When you're done with a mount and want to tear down:
relayfile stop my-agent
relayfile integration disconnect notion --workspace my-agent --yes
relayfile workspace delete my-agent --yes
rm -rf ~/relayfile-mount
The cloud-side workspace persists indefinitely ā there's no public DELETE endpoint as of May 2026. It's an inert orphan.
What this skill does NOT cover
- Self-hosted relayfile (running your own
relayfile-server Go binary against a private Nango). For most agent use-cases the managed cloud at agentrelay.com is the right choice; self-hosted is for environments where data residency rules out the cloud.
- Multi-workspace agents. A single agent talking to multiple workspaces simultaneously needs careful token handling that's out of scope here.
- GitHub-via-relayfile for source code. The GitHub adapter exists but the productized cloud-mount workflow is heavier-weight than
git clone; only use it if the agent specifically benefits from filesystem-shaped access to PR metadata, reviews, etc.
Quick reference
| Command | Purpose |
|---|
relayfile setup --provider <p> --workspace <name> --local-dir <path> | First-time setup |
relayfile status <workspace> | Health overview |
relayfile mount <workspace> <local-dir> | Restart the daemon |
relayfile stop <workspace> | Stop the daemon |
relayfile integration available [--search <q>] [--backend <nango|composio>] [--json] [--refresh] | Browse the live provider catalog |
relayfile integration search <q> [--backend <nango|composio>] [--json] [--refresh] | Search dynamic Nango providers and Composio toolkits |
relayfile integration list --workspace <name> --json | List connected providers |
relayfile integration connect <provider> [--backend <nango|composio>] --workspace <name> | Add another provider |
relayfile integration set-metadata <provider> KEY=VALUE... --workspace <name> --yes | Replace flat provider metadata, such as Jira/Confluence cloudId and baseUrl |
relayfile integration disconnect <provider> --workspace <name> --yes | Remove a provider |
relayfile tree <workspace> <path> | Live cloud-side directory listing |
relayfile read <workspace> <path> | Live cloud-side file read |
relayfile writeback status <workspace> [--json] | Pending / failed / dead-lettered counts |
relayfile writeback retry --opId <op> <workspace> | Re-enqueue a dead-lettered op |
relayfile pull --workspace <name> | Force a refresh from provider |
relayfile ops list --workspace <name> --json | Cloud-side operation log |
relayfile workspace delete <name> --yes | Remove from local registry |