cd /news/developer-tools/default-workspaces-and-where-new-age… Β· home β€Ί topics β€Ί developer-tools β€Ί article
[ARTICLE Β· art-28431] src=dev.to β†— pub= topic=developer-tools verified=true sentiment=Β· neutral

Default Workspaces and Where New Agents Land

Nylas has detailed how its Agent Accounts land in workspaces, with a default workspace acting as a safety net for unplaced accounts. The workspace carries policy limits and rules that govern agents, and placement can be updated via a single grant PATCH request. The company recommends attaching a conservative baseline policy to the default workspace to ensure unassigned accounts have sane guardrails.

read5 min views1 publishedJun 15, 2026

Every Agent Account you create lands in a workspace β€” the only question is whether you picked it or the platform did:

curl --request POST \
  --url "https://api.us.nylas.com/v3/connect/custom" \
  --header "Authorization: Bearer <NYLAS_API_KEY>" \
  --header "Content-Type: application/json" \
  --data '{
    "provider": "nylas",
    "workspace_id": "<WORKSPACE_ID>",
    "settings": {
      "email": "test@your-application.nylas.email"
    }
  }'

That top-level workspace_id

is optional, and what happens when you omit it is one of the more under-read parts of the Nylas Agent Accounts docs (the feature's in beta, so read with that in mind). Workspaces aren't cosmetic grouping β€” they're the carrier for every policy limit and mail rule that governs your agents. Getting placement wrong means an agent running with the wrong send quota, the wrong spam settings, or no inbound filtering at all.

When a new account is created, placement resolves in this order:

workspace_id

on the requestauto_group

enabled and its domain

matches the new account's email domain, the account joins it automatically.That third bucket is the interesting one.

Every application gets exactly one default workspace, created and managed by the platform. You can't delete it, and you can only update two of its fields: policy_id

and rule_ids

. Everything else is managed for you.

It's easy to read "default" as "throwaway," but it's better understood as your safety net. Any account you forget to place explicitly β€” a quick test from the CLI, a provisioning script missing the workspace parameter β€” inherits whatever the default workspace carries. Which leads to a practical recommendation: attach a conservative baseline policy and your spam-blocking rules to the default workspace on day one. Then unassigned accounts get sane guardrails instead of running wide open.

Because that's the alternative: a workspace with no policy_id

runs its accounts at your billing plan's maximum limits. No custom send quota, no retention tightening, no spam-sensitivity tuning. Fine for a sandbox; probably not what you want for whatever an intern provisions on a Friday.

A workspace carries one policy_id

plus an array of rule_ids

, and every account inside inherits both. The policy bundles limits (attachment size and count, storage, daily send quotas, inbox and spam retention) and spam detection (DNSBL checks, header anomaly detection, and a spam_sensitivity

dial that runs from 0.1 to 5.0). The rules array carries both inbound and outbound rules together β€” the engine filters by trigger

at evaluation time, and rules run in priority order from 0 to 1000, lowest first, with 10 as the default.

The payoff is that none of this configuration lives on individual grants. Update the workspace's policy and every account in it picks up the change. The docs suggest one workspace per agent archetype β€” your sales-outreach agents and your support-triage agents want different send limits and spam tolerances, so give each group its own workspace rather than one catch-all.

Placement isn't permanent. An account that landed in the default workspace can be moved with a single grant update:

curl --request PATCH \
  --url "https://api.us.nylas.com/v3/grants/<NYLAS_GRANT_ID>" \
  --header "Authorization: Bearer <NYLAS_API_KEY>" \
  --header "Content-Type: application/json" \
  --data '{
    "workspace_id": "<WORKSPACE_ID>"
  }'

From that point the account picks up the new workspace's policy and rules. This makes a "graduate the agent" workflow trivial: prototype in the default workspace, then PATCH

it into the production workspace β€” with its stricter outbound rules and tuned spam sensitivity β€” when it's ready.

Inheritance is invisible until something gets blocked or routed, so the platform keeps receipts. Every rule evaluation β€” inbound message, SMTP envelope, or outbound send β€” writes an audit record you can read per grant:

curl --request GET \
  --url "https://api.us.nylas.com/v3/grants/<NYLAS_GRANT_ID>/rule-evaluations?limit=50" \
  --header "Authorization: Bearer <NYLAS_API_KEY>"

Each record names the evaluation stage (smtp_rcpt

for messages rejected before acceptance, inbox_processing

for post-acceptance evaluation, outbound_send

for send-time checks), the matched rule IDs, and the actions applied. If an account is sitting in the wrong workspace, this endpoint is usually where you find out β€” the rules you expected to match never show up in matched_rule_ids

.

One subtlety worth internalizing: rule evaluation fails closed. If a block

rule can't be evaluated because of a transient infrastructure error, the message is blocked rather than let through β€” surfaced as a retryable 503

on sends or an SMTP 451

tempfail inbound, with blocked_by_evaluation_error: true

on the audit record so you can tell an outage from a genuine match.

If you're setting this up fresh (the quickstart gets you a working account in under 5 minutes), here's the order that avoids surprises:

   curl --request POST \
     --url "https://api.us.nylas.com/v3/policies" \
     --header "Authorization: Bearer <NYLAS_API_KEY>" \
     --header "Content-Type: application/json" \
     --data '{
       "name": "Baseline Agent Policy",
       "limits": {
         "limit_attachment_size_limit": 26214400,
         "limit_attachment_count_limit": 20,
         "limit_inbox_retention_period": 365,
         "limit_spam_retention_period": 30
       },
       "spam_detection": {
         "use_list_dnsbl": true,
         "use_header_anomaly_detection": true,
         "spam_sensitivity": 1.0
       }
     }'

Every limit is optional β€” omit one and it defaults to your plan's maximum, while a value above the plan maximum is rejected with an error. Two tuning notes from the docs: start spam_sensitivity

at 1.0 and adjust from observed behavior, and keep limit_spam_retention_period

shorter than limit_inbox_retention_period

so spam clears out ahead of the inbox.

PATCH

with policy_id

) so strays are covered.auto_group

with a domain

if your agent types map cleanly to domains.workspace_id

explicitly in your provisioning code anyway β€” auto-grouping is a backstop, not a substitute for being intentional.The whole model rewards thinking about workspaces before you scale account creation, because retrofitting placement across hundreds of grants means a hundred PATCH

calls.

Next step: list your current grants, check which workspace each actually sits in, and ask whether the default workspace has a policy attached. If the answer is "no policy," your unassigned agents are running at plan maximums right now β€” is that what you intended?

── more in #developer-tools 4 stories Β· sorted by recency
── more on @nylas 3 stories trending now
sponsored brought to you by zahid.host 4,200+ EU-deployed projects
reading about agents? ship yours in a single git push.

Run your AI side-project on zahid.host

EU-based hosting, git-push deploys, automatic HTTPS, no cold starts. Free tier with a custom domain β€” perfect for shipping the agent you just read about.

$git push zahid main
β†’ Live at https://your-agent.zahid.host βœ“
Get free account β†’ Pricing
from €0/mo Β· no card required
LIVE [news/default-workspaces-a…] indexed:0 read:5min 2026-06-15 Β· β€”