HubSpot
Scanned@byungkyu
npx machina-cli add skill @byungkyu/hubspot-api --openclawHubSpot
Access the HubSpot CRM API with managed OAuth authentication. Create and manage contacts, companies, deals, and their associations.
Quick Start
# List contacts
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://gateway.maton.ai/hubspot/crm/v3/objects/contacts?limit=10&properties=email,firstname,lastname')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
Base URL
https://gateway.maton.ai/hubspot/{native-api-path}
Replace {native-api-path} with the actual HubSpot API endpoint path. The gateway proxies requests to api.hubapi.com and automatically injects your OAuth token.
Authentication
All requests require the Maton API key in the Authorization header:
Authorization: Bearer $MATON_API_KEY
Environment Variable: Set your API key as MATON_API_KEY:
export MATON_API_KEY="YOUR_API_KEY"
Getting Your API Key
- Sign in or create an account at maton.ai
- Go to maton.ai/settings
- Copy your API key
Connection Management
Manage your HubSpot OAuth connections at https://ctrl.maton.ai.
List Connections
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://ctrl.maton.ai/connections?app=hubspot&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
Create Connection
python <<'EOF'
import urllib.request, os, json
data = json.dumps({'app': 'hubspot'}).encode()
req = urllib.request.Request('https://ctrl.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
Get Connection
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://ctrl.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": "21fd90f9-5935-43cd-b6c8-bde9d915ca80",
"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=...",
"app": "hubspot",
"metadata": {}
}
}
Open the returned url in a browser to complete OAuth authorization.
Delete Connection
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://ctrl.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
Specifying Connection
If you have multiple HubSpot connections, specify which one to use with the Maton-Connection header:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://gateway.maton.ai/hubspot/crm/v3/objects/contacts')
req.add_header('Authorization', f'Bearer {os.environ["MATON_API_KEY"]}')
req.add_header('Maton-Connection', '21fd90f9-5935-43cd-b6c8-bde9d915ca80')
print(json.dumps(json.load(urllib.request.urlopen(req)), indent=2))
EOF
If omitted, the gateway uses the default (oldest) active connection.
API Reference
Contacts
List Contacts
GET /hubspot/crm/v3/objects/contacts?limit=100&properties=email,firstname,lastname,phone
With pagination:
GET /hubspot/crm/v3/objects/contacts?limit=100&properties=email,firstname&after={cursor}
Get Contact
GET /hubspot/crm/v3/objects/contacts/{contactId}?properties=email,firstname,lastname
Create Contact
POST /hubspot/crm/v3/objects/contacts
Content-Type: application/json
{
"properties": {
"email": "john@example.com",
"firstname": "John",
"lastname": "Doe",
"phone": "+1234567890"
}
}
Update Contact
PATCH /hubspot/crm/v3/objects/contacts/{contactId}
Content-Type: application/json
{
"properties": {
"phone": "+0987654321"
}
}
Delete Contact
DELETE /hubspot/crm/v3/objects/contacts/{contactId}
Search Contacts
POST /hubspot/crm/v3/objects/contacts/search
Content-Type: application/json
{
"filterGroups": [{
"filters": [{
"propertyName": "email",
"operator": "EQ",
"value": "john@example.com"
}]
}],
"properties": ["email", "firstname", "lastname"]
}
Companies
List Companies
GET /hubspot/crm/v3/objects/companies?limit=100&properties=name,domain,industry
Get Company
GET /hubspot/crm/v3/objects/companies/{companyId}?properties=name,domain,industry
Create Company
POST /hubspot/crm/v3/objects/companies
Content-Type: application/json
{
"properties": {
"name": "Acme Corp",
"domain": "acme.com",
"industry": "COMPUTER_SOFTWARE"
}
}
Note: The industry property requires specific enum values (e.g., COMPUTER_SOFTWARE, FINANCE, HEALTHCARE). Use the List Properties endpoint to get valid values.
Update Company
PATCH /hubspot/crm/v3/objects/companies/{companyId}
Content-Type: application/json
{
"properties": {
"industry": "COMPUTER_SOFTWARE",
"numberofemployees": "50"
}
}
Delete Company
DELETE /hubspot/crm/v3/objects/companies/{companyId}
Search Companies
POST /hubspot/crm/v3/objects/companies/search
Content-Type: application/json
{
"filterGroups": [{
"filters": [{
"propertyName": "domain",
"operator": "CONTAINS_TOKEN",
"value": "*"
}]
}],
"properties": ["name", "domain"],
"limit": 10
}
Deals
List Deals
GET /hubspot/crm/v3/objects/deals?limit=100&properties=dealname,amount,dealstage
Get Deal
GET /hubspot/crm/v3/objects/deals/{dealId}?properties=dealname,amount,dealstage
Create Deal
POST /hubspot/crm/v3/objects/deals
Content-Type: application/json
{
"properties": {
"dealname": "New Deal",
"amount": "10000",
"dealstage": "appointmentscheduled"
}
}
Update Deal
PATCH /hubspot/crm/v3/objects/deals/{dealId}
Content-Type: application/json
{
"properties": {
"amount": "15000",
"dealstage": "qualifiedtobuy"
}
}
Delete Deal
DELETE /hubspot/crm/v3/objects/deals/{dealId}
Associations (v4 API)
Associate Objects
PUT /hubspot/crm/v4/objects/{fromObjectType}/{fromObjectId}/associations/{toObjectType}/{toObjectId}
Content-Type: application/json
[{"associationCategory": "HUBSPOT_DEFINED", "associationTypeId": 279}]
Common association type IDs:
279- Contact to Company3- Deal to Contact341- Deal to Company
List Associations
GET /hubspot/crm/v4/objects/{objectType}/{objectId}/associations/{toObjectType}
Batch Operations
Batch Read
POST /hubspot/crm/v3/objects/{objectType}/batch/read
Content-Type: application/json
{
"properties": ["email", "firstname"],
"inputs": [{"id": "123"}, {"id": "456"}]
}
Batch Create
POST /hubspot/crm/v3/objects/{objectType}/batch/create
Content-Type: application/json
{
"inputs": [
{"properties": {"email": "one@example.com", "firstname": "One"}},
{"properties": {"email": "two@example.com", "firstname": "Two"}}
]
}
Batch Update
POST /hubspot/crm/v3/objects/{objectType}/batch/update
Content-Type: application/json
{
"inputs": [
{"id": "123", "properties": {"firstname": "Updated"}},
{"id": "456", "properties": {"firstname": "Also Updated"}}
]
}
Batch Archive
POST /hubspot/crm/v3/objects/{objectType}/batch/archive
Content-Type: application/json
{
"inputs": [{"id": "123"}, {"id": "456"}]
}
Properties
List Properties
GET /hubspot/crm/v3/properties/{objectType}
Search Operators
EQ- Equal toNEQ- Not equal toLT/LTE- Less than / Less than or equalGT/GTE- Greater than / Greater than or equalCONTAINS_TOKEN- Contains tokenNOT_CONTAINS_TOKEN- Does not contain token
Pagination
List endpoints return a paging.next.after cursor:
{
"results": [...],
"paging": {
"next": {
"after": "12345"
}
}
}
Use the after query parameter to fetch the next page:
GET /hubspot/crm/v3/objects/contacts?limit=100&after=12345
Code Examples
JavaScript
const response = await fetch('https://gateway.maton.ai/hubspot/crm/v3/objects/contacts', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'Authorization': `Bearer ${process.env.MATON_API_KEY}`
},
body: JSON.stringify({
properties: { email: 'john@example.com', firstname: 'John' }
})
});
Python
import os
import requests
response = requests.post(
'https://gateway.maton.ai/hubspot/crm/v3/objects/contacts',
headers={'Authorization': f'Bearer {os.environ["MATON_API_KEY"]}'},
json={'properties': {'email': 'john@example.com', 'firstname': 'John'}}
)
Notes
- Batch operations support up to 100 records per request
- Archive/Delete is a soft delete - records can be restored within 90 days
- Delete endpoints return HTTP 204 (No Content) on success
- The
industryproperty on companies requires specific enum values - IMPORTANT: When using curl commands, use
curl -gwhen URLs contain brackets (fields[],sort[],records[]) to disable glob parsing - IMPORTANT: When piping curl output to
jqor other commands, environment variables like$MATON_API_KEYmay not expand correctly in some shell environments. You may get "Invalid API key" errors when piping.
Error Handling
| Status | Meaning |
|---|---|
| 400 | Missing HubSpot connection |
| 401 | Invalid or missing Maton API key |
| 429 | Rate limited (10 req/sec per account) |
| 4xx/5xx | Passthrough error from HubSpot API |
Troubleshooting: API Key Issues
- Check that the
MATON_API_KEYenvironment variable is set:
echo $MATON_API_KEY
- Verify the API key is valid by listing connections:
python <<'EOF'
import urllib.request, os, json
req = urllib.request.Request('https://ctrl.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
Troubleshooting: Invalid App Name
- Ensure your URL path starts with
hubspot. For example:
- Correct:
https://gateway.maton.ai/hubspot/crm/v3/objects/contacts - Incorrect:
https://gateway.maton.ai/crm/v3/objects/contacts
Resources
Overview
This skill enables creating, updating, searching, and syncing HubSpot CRM records (contacts, companies, deals) via a managed OAuth gateway. It streamlines data workflows by handling token injection and record associations through the gateway proxy.
How This Skill Works
Requests are routed through the Maton gateway at https://gateway.maton.ai/hubspot/{native-api-path}, which injects the OAuth token and forwards to HubSpot. All calls require the MATON_API_KEY in the Authorization header, and you can target a specific HubSpot connection using the Maton-Connection header when multiple connections exist.
When to Use It
- When creating or updating HubSpot contacts, companies, or deals from another system
- When searching for contacts by email or name in HubSpot
- When syncing data between your app and HubSpot to maintain CRM accuracy
- When managing HubSpot OAuth connections (list, create, or fetch a connection)
- When calling HubSpot endpoints through the gateway and targeting a specific connection
Quick Start
- Step 1: Set MATON_API_KEY in your environment (export MATON_API_KEY="YOUR_API_KEY")
- Step 2: Make a request to the gateway at https://gateway.maton.ai/hubspot/{native-api-path} with header Authorization: Bearer $MATON_API_KEY
- Step 3: For example, list contacts at /hubspot/crm/v3/objects/contacts?limit=10&properties=email,firstname,lastname
Best Practices
- Specify explicit properties in queries (e.g., email, firstname, lastname) to minimize payload
- Use a single HubSpot connection per integration where possible for consistency
- Never expose MATON_API_KEY in client-side code; keep it in secure server-side contexts
- Respect HubSpot API rate limits and gateway latency; implement retry/backoff
- Test changes in a sandbox or with a dedicated test connection before production
Example Use Cases
- Create a new contact with email, firstname, and lastname via hubspot crm objects
- Update a deal's stage and amount using the deals endpoint
- Search for a contact by email and retrieve associated companies
- List active HubSpot connections via the Connection API and verify status
- Proxy a call to hubspot crm/v3/objects/contacts through the gateway using a specific Maton-Connection