name: analysis-workspace description: >- How to use Resources-backed workspace files for large-scale multi-source analyses: scratch/ temporary staging, chunked batch processing with per-item memos, run-code aggregation, saveToFile for big API pulls, and synthesizing across files that exceed one context window.
Analysis Workspace
Analysis files live in the same Resources workspace users inspect and manage in the app. Use normal path conventions:
scratch/...for temporary staging files, per-item memos, raw API pulls, and other agent working data. These are hidden from the Resources view by default.- Descriptive folders such as
analysis/q2-churn/only for files the user should keep, inspect, or manage after the analysis.
The run-code helpers (workspaceRead, workspaceWrite, workspaceAppend,
workspaceList) and saveToFile write through this Resources-backed file
store. Use them to stage intermediate results that would overflow the context
window, then read them back selectively for synthesis.
When to Use
- Batch fan-out with 30+ items (accounts, calls, deals, tickets, messages, documents, events): write a per-item memo file after each item, then synthesize across all memos in a final pass.
- Large API payloads: use
saveToFileonprovider-api-requestorweb-requestto write a 20 MB dataset toscratch/...instead of returning it in context. - Provider-wide search/count/classification: build a durable corpus first,
then search or aggregate it with
run-code. This is required when the user expects broad recall or when a negative answer such as "no mentions" would be misleading if based on a sample. - Multi-step analyses that span multiple conversations or agent turns. Keep
durable files outside
scratch/only when the user wants to keep them. - run-code aggregation: call
workspaceRead/workspaceWriteinside arun-codeblock to load and process data that's too large to print as output.
Workspace File Helpers
Inside run-code, use the workspace helper functions:
| Helper | Use |
|---|---|
workspaceWrite(path, content, contentType?) |
Create or overwrite a file |
workspaceAppend(path, content) |
Append text to a file |
workspaceRead(path, opts?) |
Read content, with optional paging |
workspaceReadMeta(path, opts?) |
Read content plus metadata/truncation info |
workspaceList(prefix?) |
List files under a prefix |
readsupports{ offset, maxChars }for paging large files.listsupports a prefix filter, e.g.workspaceList("scratch/q2/").- Direct writes cap at 2 MB each.
saveToFileallows up to 20 MB per pull. - Temporary files belong under
scratch/; durable user-facing files belong in normal Resources folders.
Chunked Batch Analysis (30+ items)
For large fan-outs (account deep dives, Gong call reviews, deal cohorts):
- Define cohort: fetch the item list (e.g.
hubspot-recordsfor accounts). - Chunk: process 5–10 items per pass to avoid context overflow.
- Per-item memo: for each item, fetch evidence and write a memo file under
scratch/unless the user asked to keep the memos:await workspaceWrite( "scratch/analysis/q2-churn/acme-corp.md", "## Acme Corp\n\n**ARR**: $120k\n**Risk signals**: ...", "text/markdown" ); - Synthesize: after all items are processed, list files and read each memo:
Then read each file and synthesize findings into the final answer or a saved analysis.const files = await workspaceList("scratch/analysis/q2-churn/"); - Promote (optional): write a durable summary outside
scratch/if the user wants to inspect or keep it in Resources.
For very large cohorts (100+ items), use agent-teams sub-agents to process chunks in parallel — each sub-agent writes its memos independently, the orchestrator synthesizes at the end.
Corpus-First Provider Search
Use this workflow for arbitrary provider questions where a canned action is too narrow, where records must be joined across systems, or where absence matters:
- Discover the provider surface with
provider-api-catalogandprovider-api-docswhen endpoint/filter/pagination details are uncertain. - Pull the relevant records with
provider-api-request. UsefetchAllPagesfor cursor pagination,stageAsfor queryable staged datasets, andsaveToFilefor large raw responses. - Use
run-codeto callproviderFetchorappActionin loops, write intermediate files, normalize records, join identity fields, and aggregate. - Validate coverage before synthesis. Track pages fetched, records inspected, truncation flags, aborted calls, and records skipped for missing joins.
- Finalize with the answer plus coverage and caveats. If coverage is partial, say so directly; never state "none found", "all records", or an exhaustive conclusion from sampled, truncated, or aborted data.
saveToFile on Provider API Requests
Attach saveToFile to provider-api-request or web-request to write the
full response body to a Resources-backed workspace file instead of returning it
in context. Allows up to 20 MB per call. Use scratch/... for temporary raw
payloads.
provider-api-request
provider=<provider-id>
method=POST
path=/records/search
body={ filterGroups: [...], limit: 200 }
saveToFile="scratch/analysis/provider-records-2026-q2.json"
Returns: { savedToFile: true, savedTo, status, bytes, contentType, preview }.
Then use run-code to process the saved file:
const raw = await workspaceRead("scratch/analysis/provider-records-2026-q2.json");
const records = JSON.parse(raw);
// … aggregate, filter, join …
fetchAllPages
Use fetchAllPages on provider-api-request to automatically paginate
cursor-based APIs. Combine with saveToFile to write the full dataset:
provider-api-request
provider=hubspot
path=/crm/v3/objects/deals
query={ limit: 100 }
fetchAllPages={
cursorPath: "paging.next.after",
cursorParam: "after",
itemsPath: "results",
maxPages: 20
}
saveToFile="scratch/analysis/all-deals.json"
Common cursor paths:
- HubSpot:
paging.next.after/after - Gong:
records.cursor/cursor - Pylon:
nextCursor/cursor - Slack:
response_metadata.next_cursor/cursor - PostHog:
next(full URL — extract token manually if needed)
run-code + Workspace Integration
Inside a run-code block you have access to workspace helpers:
// Read a previously saved API response
const raw = await workspaceRead("scratch/analysis/deals.json");
const deals = JSON.parse(raw);
// Process data
const byStage = {};
for (const deal of deals) {
const stage = deal.properties.dealstage ?? "unknown";
byStage[stage] = (byStage[stage] ?? 0) + 1;
}
// Write the aggregated result back
await workspaceWrite(
"scratch/analysis/deals-by-stage.json",
JSON.stringify(byStage, null, 2),
"application/json"
);
console.log(JSON.stringify(byStage, null, 2));
Workspace helpers: workspaceRead(path, opts?), workspaceWrite(path, content, contentType?),
workspaceAppend(path, content), workspaceList(prefix?).
Provider API Discovery
When you need an API endpoint that isn't covered by a canned action:
provider-api-catalog— list available providers and their base URLs, auth, and example paths.provider-api-docs provider=<id> url=<docs-url>— fetch any public docs or OpenAPI spec URL. Works for anyhttps://URL.provider-api-request— call the endpoint directly.
Use web-request to fetch public REST docs pages before registering a custom
provider.
Registering Custom Providers
For APIs not in the built-in catalog, register them with
provider-api-register (dispatch action):
provider-api-register
id="my-internal-api"
label="My Internal API"
baseUrl="https://api.mycompany.com"
auth={ type: "bearer", credentialKey: "MY_API_TOKEN" }
docsUrls=["https://docs.mycompany.com/api"]
Then call it with provider-api-request provider=my-internal-api ... and the
agent's credential system handles auth automatically.
Learnings Flywheel
After any significant batch analysis, record discoveries to LEARNINGS.md
via the resources tool (action: "write"). Capture confirmed schema paths,
cursor fields, identity join keys, and pagination patterns.