Install
openclaw skills install @byungkyu/api-gatewayConnect to external services through Maton-managed API routes. Use this skill only after the user names the target app, account, and task. Start with read/list calls when possible and follow the app-specific reference before any change.
openclaw skills install @byungkyu/api-gatewayManaged API routing for third-party services, provided by Maton. Use this only for a user-requested app, account, and task.
CLI:
maton slack channel list --types public_channel --limit 10
maton api '/slack/api/conversations.list?types=public_channel&limit=10'
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/slack/api/conversations.list?types=public_channel&limit=10')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Use https://api.maton.ai/ with the app-prefixed routes documented in the examples below or in the matching reference file.
Usage protocol:
Read-only route examples:
https://api.maton.ai/slack/api/conversations.list?types=public_channel&limit=10
https://api.maton.ai/google-mail/gmail/v1/users/me/messages
The first path segment is the app identifier listed in Supported Services. For Gmail, use /google-mail/gmail/v1/users/me/messages.
NPM:
npm install -g @maton-ai/cli
Homebrew:
brew install maton-ai/cli/maton
IMPORTANT — Credential Safety:
MATON_API_KEY as a secret. Never log it, echo it, paste it into prompts, or expose it in shared files, command output, or tool results.maton connection delete {id}).CLI:
maton login # Opens browser for API key
maton login --interactive # Skip browser, paste API key directly
maton whoami # Show current auth state
Manual:
MATON_API_KEY:export MATON_API_KEY="YOUR_API_KEY"
CLI:
maton connection list slack --status ACTIVE
maton api -X GET /connections -f app=slack -f status=ACTIVE
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/connections?app=slack&status=ACTIVE')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Query Parameters (optional):
app - Filter by service name (e.g., slack, hubspot, salesforce)status - Filter by connection status (ACTIVE, PENDING, FAILED)Response:
{
"connections": [
{
"connection_id": "{connection_id}",
"status": "ACTIVE",
"creation_time": "2025-12-08T07:20:53.488460Z",
"last_updated_time": "2026-01-31T20:03:32.593153Z",
"url": "https://connect.maton.ai/?session_token=5e9...",
"app": "slack",
"method": "OAUTH2",
"metadata": {}
}
]
}
CLI:
maton connection create slack
maton api /connections -f app=slack
Python:
python <<'EOF'
import urllib.request, os, json
data = json.dumps({'app': 'slack'}).encode()
req = urllib.request.Request('https://api.maton.ai/connections', data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Request Body:
app (required) - Service name (e.g., slack, notion)method (optional) - Connection method (API_KEY, BASIC, OAUTH1, OAUTH2, MCP)CLI:
maton connection get {connection_id}
maton api /connections/{connection_id}
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/connections/{connection_id}')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Response:
{
"connection": {
"connection_id": "{connection_id}",
"status": "ACTIVE",
"creation_time": "2025-12-08T07:20:53.488460Z",
"last_updated_time": "2026-01-31T20:03:32.593153Z",
"url": "https://connect.maton.ai/?session_token=5e9...",
"app": "slack",
"metadata": {}
}
}
Open the returned URL in a browser to complete service authorization.
CLI:
maton connection delete {connection_id} --yes
maton api -X DELETE /connections/{connection_id}
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/connections/{connection_id}', method='DELETE')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
If you have multiple connections for the same app, specify which connection to use:
CLI:
maton slack channel list --types public_channel --limit 10 --connection {connection_id}
maton api '/slack/api/conversations.list?types=public_channel&limit=10' --connection {connection_id}
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/slack/api/conversations.list?types=public_channel&limit=10')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Maton-Connection', '{connection_id}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
If you have multiple connections, always specify the connection to ensure requests go to the intended account.
CLI:
maton trigger list --source github --status ENABLED -L 50
maton api -X GET /triggers -f source=github -f status=ENABLED -f limit=50
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/triggers?source=github&status=ENABLED&limit=50')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Query Parameters (optional): source, status, limit, next_token.
Response:
{
"triggers": [
{
"trigger_id": "{trigger_id}",
"source": "github",
"event_type": "pull_request.opened",
"name": "PR opened",
"description": null,
"parameters": {"repo": "maton-ai/cli"},
"connection_id": "{connection_id}",
"destinations": [
{
"destination_id": "{destination_id}",
"url": "https://httpbin.org/post",
"name": null,
"status": "ENABLED",
"reason": null
}
],
"status": "ENABLED",
"reason": null,
"created_at": "2026-05-25T23:24:38.079501Z",
"updated_at": "2026-05-25T23:24:38.079501Z"
}
],
"next_token": "gAAAAABqN6tD5X7..."
}
CLI:
maton trigger create --source github --event-type pull_request.opened \
--connection-id conn_123 \
--parameter repo=maton-ai/cli \
--destination '{"url":"https://httpbin.org/post","method":"POST","name":"prod"}'
maton api /triggers \
-f source=github -f event_type=pull_request.opened \
-f name='PR opened' -f connection_id=conn_123 \
-F 'parameters[repo]=maton-ai/cli' \
-F 'destinations[][url]=https://httpbin.org/post' \
-F 'destinations[][method]=POST' \
-F 'destinations[][name]=prod'
Python:
python <<'EOF'
import urllib.request, os, json
data = json.dumps({
"source": "github",
"event_type": "pull_request.opened",
"name": "PR opened",
"connection_id": "conn_123",
"parameters": {"repo": "maton-ai/cli"},
"destinations": [{"url": "https://httpbin.org/post", "method": "POST", "name": "prod"}]
}).encode()
req = urllib.request.Request('https://api.maton.ai/triggers', data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Request Body:
source (required)event_type (required)connection_id (optional)name, description (optional)parameters (optional)destinations (optional)Each source's event types and their parameters are documented at references/{source}/triggers.md (e.g. google-mail). Besides the app sources in the Supported Services table, the special time source fires on a cron schedule (schedule.elapsed) and needs no connection.
CLI:
maton trigger get {trigger_id}
maton api /triggers/{trigger_id}
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Response:
{
"trigger": {
"trigger_id": "{trigger_id}",
"source": "stripe",
"event_type": "charge.succeeded",
"name": "Charges",
"description": null,
"parameters": {"event_type": "charge.succeeded"},
"connection_id": "{connection_id}",
"destinations": [
{
"destination_id": "{destination_id}",
"url": "https://httpbin.org/post",
"name": null,
"status": "ENABLED",
"reason": null
}
],
"status": "ENABLED",
"reason": null,
"created_at": "2026-05-25T23:27:50.166333Z",
"updated_at": "2026-05-25T23:27:50.166333Z"
}
}
Edits trigger metadata only. Destinations are managed through their own endpoints.
CLI:
maton trigger update {trigger_id} --parameter repo=maton-ai/cli
maton api -X PATCH /triggers/{trigger_id} -F 'parameters[repo]=maton-ai/cli'
Python:
python <<'EOF'
import urllib.request, os, json
data = json.dumps({"parameters": {"repo": "maton-ai/cli"}}).encode()
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}', data=data, method='PATCH')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Request Body: name, description, status, parameters (replaces all).
CLI:
maton trigger delete {trigger_id} --yes
maton api -X DELETE /triggers/{trigger_id}
Python:
python <<'EOF'
import urllib.request, os
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}', method='DELETE')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
urllib.request.urlopen(req)
EOF
CLI:
maton trigger destination list --trigger {trigger_id}
maton api -X GET /triggers/{trigger_id}/destinations
Response:
{
"destinations": [
{
"destination_id": "{destination_id}",
"url": "https://httpbin.org/post",
"name": null,
"status": "ENABLED",
"reason": null
}
]
}
CLI:
maton trigger destination create --trigger {trigger_id} \
--url https://httpbin.org/post --method POST --name prod \
--header X-Token=secret \
--body-template '{"data": {{ payload.data }}}'
maton api /triggers/{trigger_id}/destinations \
-f url=https://httpbin.org/post -f method=POST -f name=prod \
-F 'headers[X-Token]=secret' \
-f 'body_template={"data": {{ payload.data }}}'
Python:
python <<'EOF'
import urllib.request, os, json
data = json.dumps({
"url": "https://httpbin.org/post",
"method": "POST",
"name": "prod",
"headers": {"X-Token": "secret"},
"body_template": '{"data": {{ payload.data }}}'
}).encode()
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}/destinations', data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Request Body:
url (required)method (optional, default: POST)name (optional)headers (optional)body_template (optional) — JSON template for the outgoing request body, with {{ ... }} placeholders interpolated at delivery time. See references/{source}/triggers.md for each source's payload shape and available fields.Template placeholders:
{{ payload }} — the full event payload, inlined as JSON{{ payload.x.y.z }} — drill into a nested field inside the payload{{ trigger_id }}, {{ trigger_name }}, {{ event_id }}, {{ source }}, {{ event_type }} — scalar metadata{{ received_at }} — when the event was receivedCLI:
maton trigger destination get {destination_id} --trigger {trigger_id}
maton api -X GET /triggers/{trigger_id}/destinations/{destination_id}
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}/destinations/{destination_id}')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Response:
{
"destination": {
"destination_id": "{destination_id}",
"url": "https://httpbin.org/post",
"method": "POST",
"headers": {},
"signing_secret": "••••••••",
"name": null,
"body_template": null,
"status": "ENABLED",
"reason": null,
"created_at": "2026-05-25T23:27:50.166333Z",
"updated_at": "2026-05-25T23:27:50.166333Z"
}
}
signing_secret is masked; retrieve the plaintext value only at create time or via Rotate Destination Secret.
CLI:
maton trigger destination update {destination_id} --trigger {trigger_id} --url https://new.dev/hook
maton api -X PATCH /triggers/{trigger_id}/destinations/{destination_id} -f url=https://new.dev/hook
Python:
python <<'EOF'
import urllib.request, os, json
data = json.dumps({"url": "https://new.dev/hook"}).encode()
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}/destinations/{destination_id}', data=data, method='PATCH')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Request Body: url, method, name, headers (replaces all), body_template, status.
CLI:
maton trigger destination delete {destination_id} --trigger {trigger_id} --yes
maton api -X DELETE /triggers/{trigger_id}/destinations/{destination_id}
Python:
python <<'EOF'
import urllib.request, os
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}/destinations/{destination_id}', method='DELETE')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
urllib.request.urlopen(req)
EOF
CLI:
maton trigger destination rotate-secret {destination_id} --trigger {trigger_id}
maton api -X POST /triggers/{trigger_id}/destinations/{destination_id}/secret:rotate
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request(
'https://api.maton.ai/triggers/{trigger_id}/destinations/{destination_id}/secret:rotate',
data=b'', method='POST',
)
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Response:
{
"signing_secret": "whsec_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
}
The new signing secret is returned in plaintext only once.
Events are stored per-trigger whether or not the trigger has destinations.
CLI:
maton trigger event list --trigger {trigger_id} -L 1
maton api -X GET /triggers/{trigger_id}/events -f limit=1
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}/events?limit=1')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Query Parameters (optional): limit, next_token.
Response:
{
"events": [
{
"event_id": "{event_id}",
"received_at": "2026-06-20T16:00:09.938161Z",
"payload": {
"scheduled_for": "2026-06-20T16:00:00Z",
"cron_expression": "0 9 * * *",
"timezone": "America/Los_Angeles"
},
"delivery_counts": {"total": 0, "succeeded": 0, "failed": 0}
}
],
"next_token": "gAAAAABqN6Xf...="
}
CLI:
maton trigger event replay {event_id} --trigger {trigger_id}
maton api -X POST /triggers/{trigger_id}/events/{event_id}:replay
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request(
'https://api.maton.ai/triggers/{trigger_id}/events/{event_id}:replay',
data=b'', method='POST',
)
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton trigger event get {event_id} --trigger {trigger_id}
maton api /triggers/{trigger_id}/events/{event_id}
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/triggers/{trigger_id}/events/{event_id}')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Response:
{
"event": {
"event_id": "{event_id}",
"received_at": "2026-06-20T16:00:09.938161Z",
"payload": {
"scheduled_for": "2026-06-20T16:00:00Z",
"cron_expression": "0 9 * * *",
"timezone": "America/Los_Angeles"
},
"deliveries": [
{
"delivery_id": "{delivery_id}",
"destination_id": "{destination_id}",
"status": "SUCCEEDED",
"reason": null,
"attempts": 1,
"last_response_status": 200,
"last_response_body": "{}",
"last_response_duration": 105,
"last_error_message": null,
"destination_url": null,
"destination_method": null,
"last_attempt_at": "2026-06-20T16:00:33.860432Z",
"created_at": "2026-06-20T16:00:09.938161Z",
"finished_at": "2026-06-20T16:00:33.860432Z"
}
]
}
}
CLI:
maton trigger event watch -t {trigger_id} --exec ./handle.sh
#!/usr/bin/env bash
EVENT_JSON="$(cat)" python <<'EOF'
import json, os
event_id = os.environ["MATON_EVENT_ID"]
event = json.loads(os.environ["EVENT_JSON"])
print(f"[{event_id}] {event['payload']['threadId']}")
EOF
After each event, the last processed event ID is checkpointed to a per-trigger state file, so restarting the watch resumes after the last handled event and an interrupted batch never re-runs events it already processed.
MATON_API_KEY or OAuth tokens. Verify presence without revealing values.--connection flag (CLI) or Maton-Connection header to ensure requests go to the intended account, especially when the user has multiple connections for the same service.| Service | App Name | Service API Host | Trigger Source |
|---|---|---|---|
| ActiveCampaign | active-campaign | {account}.api-us1.com | |
| Acuity Scheduling | acuity-scheduling | acuityscheduling.com | |
| Airtable | airtable | api.airtable.com | |
| Apify | apify | api.apify.com | |
| Apollo | apollo | api.apollo.io | |
| Asana | asana | app.asana.com | |
| Attio | attio | api.attio.com | |
| Basecamp | basecamp | 3.basecampapi.com | |
| Baserow | baserow | api.baserow.io | |
| beehiiv | beehiiv | api.beehiiv.com | |
| Box | box | api.box.com | |
| Brevo | brevo | api.brevo.com | |
| Brave Search | brave-search | api.search.brave.com | |
| Buffer | buffer | api.buffer.com | |
| Calendly | calendly | api.calendly.com | ✓ |
| Cal.com | cal-com | api.cal.com | |
| CallRail | callrail | api.callrail.com | |
| Chargebee | chargebee | {subdomain}.chargebee.com | |
| ClickFunnels | clickfunnels | {subdomain}.myclickfunnels.com | |
| ClickSend | clicksend | rest.clicksend.com | |
| ClickUp | clickup | api.clickup.com | |
| Clio | clio | app.clio.com | |
| Clockify | clockify | api.clockify.me | |
| Coda | coda | coda.io | |
| Confluence | confluence | api.atlassian.com | |
| CompanyCam | companycam | api.companycam.com | |
| Cognito Forms | cognito-forms | www.cognitoforms.com | |
| Constant Contact | constant-contact | api.cc.email | |
| Dropbox | dropbox | api.dropboxapi.com | |
| Dropbox Business | dropbox-business | api.dropboxapi.com | |
| ElevenLabs | elevenlabs | api.elevenlabs.io | |
| Eventbrite | eventbrite | www.eventbriteapi.com | |
| Exa | exa | api.exa.ai | |
| Facebook Page | facebook-page | graph.facebook.com | |
| fal.ai | fal-ai | queue.fal.run | |
| Fathom | fathom | api.fathom.ai | |
| Firecrawl | firecrawl | api.firecrawl.dev | |
| Firebase | firebase | firebase.googleapis.com | |
| Fireflies | fireflies | api.fireflies.ai | |
| Front | front | api2.frontapp.com | |
| GetResponse | getresponse | api.getresponse.com | |
| Grafana | grafana | User's Grafana instance | |
| GitHub | github | api.github.com | ✓ |
| Gumroad | gumroad | api.gumroad.com | |
| Granola MCP | granola | mcp.granola.ai | |
| Google Ads | google-ads | googleads.googleapis.com | |
| Google BigQuery | google-bigquery | bigquery.googleapis.com | |
| Google Analytics Admin | google-analytics-admin | analyticsadmin.googleapis.com | |
| Google Analytics Data | google-analytics-data | analyticsdata.googleapis.com | |
| Google Apps Script | google-apps-script | script.googleapis.com | |
| Google Calendar | google-calendar | www.googleapis.com | |
| Google Classroom | google-classroom | classroom.googleapis.com | |
| Google Contacts | google-contacts | people.googleapis.com | |
| Google Docs | google-docs | docs.googleapis.com | |
| Google Drive | google-drive | www.googleapis.com | |
| Google Forms | google-forms | forms.googleapis.com | |
| Gmail | google-mail | gmail.googleapis.com | ✓ |
| Google Merchant | google-merchant | merchantapi.googleapis.com | |
| Google Meet | google-meet | meet.googleapis.com | |
| Google Play | google-play | androidpublisher.googleapis.com | |
| Google Search Console | google-search-console | www.googleapis.com | |
| Google Sheets | google-sheets | sheets.googleapis.com | |
| Google Slides | google-slides | slides.googleapis.com | |
| Google Tag Manager | google-tag-manager | tagmanager.googleapis.com | |
| Google Tasks | google-tasks | tasks.googleapis.com | |
| Google Workspace Admin | google-workspace-admin | admin.googleapis.com | |
| GoHighLevel (PIT) | highlevel-pit | services.leadconnectorhq.com | |
| HubSpot | hubspot | api.hubapi.com | ✓ |
| Instantly | instantly | api.instantly.ai | |
| Jira | jira | api.atlassian.com | |
| Jobber | jobber | api.getjobber.com | |
| JotForm | jotform | api.jotform.com | |
| Kaggle | kaggle | api.kaggle.com | |
| Keap | keap | api.infusionsoft.com | |
| Kibana | kibana | User's Kibana instance | |
| Kit | kit | api.kit.com | |
| Klaviyo | klaviyo | a.klaviyo.com | |
| Lemlist | lemlist | api.lemlist.com | |
| Linear | linear | api.linear.app | ✓ |
linkedin | api.linkedin.com | ||
| LinkedIn Community Management | linkedin-community-management | api.linkedin.com | |
| Mailchimp | mailchimp | {dc}.api.mailchimp.com | |
| MailerLite | mailerlite | connect.mailerlite.com | |
| Mailgun | mailgun | api.mailgun.net | |
| Make | make | {zone}.make.com | |
| ManyChat | manychat | api.manychat.com | |
| Manus | manus | api.manus.ai | |
| Memelord | memelord | www.memelord.com | |
| Microsoft Excel | microsoft-excel | graph.microsoft.com | |
| Microsoft Teams | microsoft-teams | graph.microsoft.com | |
| Microsoft To Do | microsoft-to-do | graph.microsoft.com | |
| Monday.com | monday | api.monday.com | |
| Motion | motion | api.usemotion.com | |
| Netlify | netlify | api.netlify.com | |
| Notion | notion | api.notion.com | ✓ |
| Notion MCP | notion | mcp.notion.com | ✓ |
| OneNote | one-note | graph.microsoft.com | |
| OneDrive | one-drive | graph.microsoft.com | |
| Outlook | outlook | graph.microsoft.com | |
| PDF.co | pdf-co | api.pdf.co | |
| Pipedrive | pipedrive | api.pipedrive.com | |
| Podio | podio | api.podio.com | |
| PostHog | posthog | {subdomain}.posthog.com | |
| QuickBooks | quickbooks | quickbooks.api.intuit.com | |
| Quo | quo | api.openphone.com | |
| Reducto | reducto | platform.reducto.ai | |
| Resend | resend | api.resend.com | |
| Salesforce | salesforce | {instance}.salesforce.com | |
| SendGrid | sendgrid | api.sendgrid.com | |
| Sentry | sentry | {subdomain}.sentry.io | |
| SharePoint | sharepoint | graph.microsoft.com | |
| SignNow | signnow | api.signnow.com | |
| Slack | slack | slack.com | ✓ |
| Snapchat | snapchat | adsapi.snapchat.com | |
| Square | squareup | connect.squareup.com | |
| Squarespace | squarespace | api.squarespace.com | |
| Stripe | stripe | api.stripe.com | ✓ |
| Sunsama MCP | sunsama | MCP server | |
| Supabase | supabase | {project_ref}.supabase.co | |
| Systeme.io | systeme | api.systeme.io | |
| Tally | tally | api.tally.so | |
| Tavily | tavily | api.tavily.com | |
| Telegram | telegram | api.telegram.org | |
| TickTick | ticktick | api.ticktick.com | |
| Todoist | todoist | api.todoist.com | |
| Toggl Track | toggl-track | api.track.toggl.com | |
| Trello | trello | api.trello.com | |
| Twilio | twilio | api.twilio.com | |
| Twenty CRM | twenty | api.twenty.com | |
| Typeform | typeform | api.typeform.com | |
| Unbounce | unbounce | api.unbounce.com | |
| Vercel | vercel | api.vercel.com | |
| Vimeo | vimeo | api.vimeo.com | |
| WATI | wati | {tenant}.wati.io | |
| WhatsApp Business | whatsapp-business | graph.facebook.com | |
| WooCommerce | woocommerce | {store-url}/wp-json/wc/v3 | |
| WordPress.com | wordpress | public-api.wordpress.com | |
| Wrike | wrike | www.wrike.com | |
| Xero | xero | api.xero.com | |
| YouTube | youtube | www.googleapis.com | |
| YouTube Analytics | youtube-analytics | youtubeanalytics.googleapis.com | |
| YouTube Reporting | youtube-reporting | youtubereporting.googleapis.com | |
| Zoom | zoom | api.zoom.us | |
| Zoom Admin | zoom-admin | api.zoom.us | |
| Zoho Bigin | zoho-bigin | www.zohoapis.com | |
| Zoho Bookings | zoho-bookings | www.zohoapis.com | |
| Zoho Books | zoho-books | www.zohoapis.com | |
| Zoho Calendar | zoho-calendar | calendar.zoho.com | |
| Zoho CRM | zoho-crm | www.zohoapis.com | |
| Zoho Inventory | zoho-inventory | www.zohoapis.com | |
| Zoho Mail | zoho-mail | mail.zoho.com | |
| Zoho People | zoho-people | people.zoho.com | |
| Zoho Projects | zoho-projects | projectsapi.zoho.com | |
| Zoho Recruit | zoho-recruit | recruit.zoho.com |
See references/ for detailed routing guides per provider:
CLI:
maton google-mail message send --to alice@example.com --subject Hi --body 'Hello!'
maton api /google-mail/gmail/v1/users/me/messages/send -f raw="$RAW_BASE64URL"
Python:
# Native Gmail API: POST https://gmail.googleapis.com/gmail/v1/users/me/messages/send
python <<'EOF'
import urllib.request, os, json, base64
from email.message import EmailMessage
msg = EmailMessage()
msg['To'], msg['Subject'] = 'alice@example.com', 'Hi'
msg.set_content('Hello!')
raw = base64.urlsafe_b64encode(msg.as_bytes()).decode()
data = json.dumps({'raw': raw}).encode()
req = urllib.request.Request('https://api.maton.ai/google-mail/gmail/v1/users/me/messages/send', data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton slack channel list --types public_channel --limit 10
maton api '/slack/api/conversations.list?types=public_channel&limit=10'
Python:
# Native Slack API: GET https://slack.com/api/conversations.list
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/slack/api/conversations.list?types=public_channel&limit=10')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton hubspot contact search --filter createdate:GT:2026-01-01 --properties email,firstname
maton api /hubspot/crm/v3/objects/contacts/search \
-F 'filterGroups[][filters][][propertyName]=createdate' \
-F 'filterGroups[][filters][][operator]=GT' \
-F 'filterGroups[][filters][][value]=2026-01-01' \
-F 'properties[]=email' -F 'properties[]=firstname' -F limit=10
Python:
# Native HubSpot API: POST https://api.hubapi.com/crm/v3/objects/contacts/search
python <<'EOF'
import urllib.request, os, json
data = json.dumps({
"filterGroups": [{"filters": [{"propertyName": "createdate", "operator": "GT", "value": "2026-01-01"}]}],
"properties": ["email", "firstname"],
"limit": 10
}).encode()
req = urllib.request.Request('https://api.maton.ai/hubspot/crm/v3/objects/contacts/search', data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton google-sheets values append {spreadsheet_id} --range A1 --values 'Alice,100,true'
echo '{"values":[["Alice","100","true"]]}' | maton api -X POST \
'/google-sheets/v4/spreadsheets/{spreadsheet_id}/values/A1:append?valueInputOption=USER_ENTERED' --input -
Python:
# Native Sheets API: POST https://sheets.googleapis.com/v4/spreadsheets/{id}/values/{range}:append
python <<'EOF'
import urllib.request, os, json
data = json.dumps({"values": [["Alice", "100", "true"]]}).encode()
req = urllib.request.Request(
'https://api.maton.ai/google-sheets/v4/spreadsheets/{spreadsheet_id}/values/A1:append?valueInputOption=USER_ENTERED',
data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton salesforce query 'SELECT Id,Name FROM Contact LIMIT 10'
Python:
# Native Salesforce API: GET https://{instance}.salesforce.com/services/data/v64.0/query?q=...
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/salesforce/services/data/v64.0/query?q=SELECT+Id,Name+FROM+Contact+LIMIT+10')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton api '/airtable/v0/meta/bases/{base_id}/tables'
Python:
# Native Airtable API: GET https://api.airtable.com/v0/meta/bases/{id}/tables
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/airtable/v0/meta/bases/{base_id}/tables')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton notion data-source query {data_source_id}
Python:
# Native Notion API: POST https://api.notion.com/v1/data_sources/{id}/query
python <<'EOF'
import urllib.request, os, json
data = json.dumps({}).encode()
req = urllib.request.Request('https://api.maton.ai/notion/v1/data_sources/{data_source_id}/query', data=data, method='POST')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Content-Type', 'application/json')
req.add_header('Notion-Version', '2025-09-03')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
CLI:
maton stripe customer list -L 10
Python:
# Native Stripe API: GET https://api.stripe.com/v1/customers
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/stripe/v1/customers?limit=10')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
maton trigger create --source google-mail --event-type email.received \
--connection-id conn_123 \
--parameter labels=INBOX \
--destination '{"url":"https://httpbin.org/post","method":"POST","name":"prod"}'
maton trigger event watch -t {trigger_id} --exec ./handle.sh
#!/usr/bin/env bash
EVENT_JSON="$(cat)" python <<'EOF'
import json, os, urllib.request
event = json.loads(os.environ["EVENT_JSON"])
data = json.dumps({"channel": "C0123456789", "text": f"New email: {event['snippet']}"}).encode()
req = urllib.request.Request("https://api.maton.ai/slack/api/chat.postMessage", data=data, method="POST")
req.add_header("Authorization", f"Bearer {os.environ['MATON_API_KEY']}")
req.add_header("Content-Type", "application/json")
urllib.request.urlopen(req)
EOF
maton trigger create --source google-mail --event-type email.received \
--connection-id conn_123 \
--parameter labels=INBOX \
--destination '{"url":"https://api.maton.ai/slack/api/chat.postMessage","method":"POST","name":"slack","headers":{"Authorization":"Bearer '"$MATON_API_KEY"'","Content-Type":"application/json"},"body_template":"{\"channel\": \"C0123456789\", \"text\": \"New email: {{ payload.snippet }}\"}"}'
# List public slack channels
maton slack channel list --types public_channel --limit 10
# List unread messages with headers
maton google-mail message list --hydrate
# Filter with jq — e.g., only active customers
# Note: --jq requires --json
maton stripe customer list -L 10 --json --jq '.data | map(select(.delinquent == false))'
const response = await fetch('https://api.maton.ai/slack/api/conversations.list?types=public_channel&limit=10', {
headers: {
'Authorization': `Bearer ${process.env.MATON_API_KEY}`
}
});
const data = await response.json();
import os
import requests
response = requests.get(
'https://api.maton.ai/slack/api/conversations.list?types=public_channel&limit=10',
headers={'Authorization': f'Bearer {os.environ["MATON_API_KEY"]}'}
)
data = response.json()
| Status | Meaning |
|---|---|
| 400 | Missing connection for the requested app |
| 401 | Invalid or missing Maton API key |
| 429 | Rate limited (10 requests/second per account) |
| 500 | Internal Server Error |
| 4xx/5xx | Passthrough error from the target API |
Errors from the target API are passed through with their original status codes and response bodies.
CLI:
maton whoami
maton connection list
Manual:
MATON_API_KEY environment variable is set (verify presence only — never print the actual value):[ -n "$MATON_API_KEY" ] && echo "MATON_API_KEY is set" || echo "MATON_API_KEY is not set"
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/connections')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
/google-mail/. For example:https://api.maton.ai/google-mail/gmail/v1/users/me/messageshttps://api.maton.ai/gmail/v1/users/me/messagesCLI:
maton connection list google-mail --status ACTIVE
Python:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://api.maton.ai/connections?app=google-mail&status=ACTIVE')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
A 500 error may indicate expired service authorization. Try creating a new connection via the Connection Management section above and completing service authorization. If the new connection is "ACTIVE", delete the old connection to ensure Maton uses the new one.
fields[], sort[], records[]), use the -g flag to disable glob parsingjq, environment variables may not expand correctly in some shells, which can cause "Invalid API key" errorswww.linkedin.com upload URLs while API calls use api.linkedin.com). These upload URLs are pre-signed and do NOT require an Authorization header. Upload the binary directly to the returned URL. You MUST use Python urllib for these uploads because the URLs contain encoded characters (e.g., %253D) that get corrupted when passed through shell variables or curl. Always parse the JSON response with json.load() and use the URL directly in Python. Safety: Only follow upload URLs returned by the expected API host (e.g., *.linkedin.com for LinkedIn). Never follow upload URLs that point to unexpected domains — confirm the host matches the service before uploading any data.Use native API docs: Refer to each service's official API documentation for endpoint paths and parameters.
Headers are forwarded: Custom headers (except Host and Authorization) are forwarded to the target API.
Query params work: URL query parameters are passed through to the target API.
HTTP methods: Use the method required by the referenced endpoint. Confirm the exact target and expected outcome before methods that change data.
QuickBooks special case: Use :realmId in the path and it will be replaced with the connected realm ID.