Atera API Patterns
Overview
The Atera REST API (v3) provides access to all major entities in the RMM/PSA platform. This skill covers authentication, pagination, rate limiting, error handling, and performance optimization patterns.
Authentication
X-API-KEY Header Authentication
Atera uses a simple API key authentication via the X-API-KEY header:
GET /api/v3/tickets
X-API-KEY: YOUR_API_KEY
Content-Type: application/json
Accept: application/json
Required Headers:
| Header | Value | Description |
|---|---|---|
X-API-KEY | {your_api_key} | API key from Atera portal |
Content-Type | application/json | For POST/PUT requests |
Accept | application/json | Optional, for explicit format |
Obtaining API Key
- Log into Atera portal
- Navigate to Admin > API
- Generate or copy your API key
- Store securely (treat as a password)
Environment Variable Setup
export ATERA_API_KEY="your-api-key-here"
Security Best Practices
- Never commit API keys - Use environment variables
- Rotate keys periodically - Generate new keys regularly
- Use HTTPS only - Atera requires HTTPS
- Limit key access - Only share with necessary services
- Monitor usage - Watch for unauthorized access
Base URL
All API requests use the following base URL:
https://app.atera.com/api/v3
Pagination
OData-Style Pagination
Atera uses OData-style pagination with page and itemsInPage parameters:
GET /api/v3/tickets?page=1&itemsInPage=50
X-API-KEY: {api_key}
Pagination Parameters:
| Parameter | Type | Default | Max | Description |
|---|---|---|---|---|
page | int | 1 | - | Page number (1-indexed) |
itemsInPage | int | 20 | 50 | Items per page |
Response Structure
{
"items": [...],
"totalItems": 2847,
"page": 1,
"itemsInPage": 50,
"totalPages": 57
}
Response Fields:
| Field | Type | Description |
|---|---|---|
items | array | Array of entities |
totalItems | int | Total count across all pages |
page | int | Current page number |
itemsInPage | int | Items in current page |
totalPages | int | Total number of pages |
Efficient Pagination Pattern
async function fetchAllItems(endpoint) {
const allItems = [];
let page = 1;
let hasMore = true;
while (hasMore) {
const response = await fetch(
`https://app.atera.com/api/v3/${endpoint}?page=${page}&itemsInPage=50`,
{
headers: {
'X-API-KEY': process.env.ATERA_API_KEY
}
}
);
const data = await response.json();
allItems.push(...data.items);
hasMore = page < data.totalPages;
page++;
// Respect rate limits
if (hasMore) {
await sleep(100); // 100ms between requests
}
}
return allItems;
}
Pagination Best Practices
- Use maximum page size - 50 items reduces API calls
- Add delays between pages - Avoid hitting rate limits
- Cache total counts - Don't re-fetch unnecessarily
- Implement retry logic - Handle transient failures
Rate Limiting
Rate Limit: 700 Requests per Minute
Atera enforces a rate limit of 700 requests per minute per API key.
Rate Limit Headers
Atera may return rate limit information in response headers:
| Header | Description |
|---|---|
X-RateLimit-Limit | Maximum requests per window |
X-RateLimit-Remaining | Remaining requests |
X-RateLimit-Reset | Seconds until reset |
Rate Limit Response
When rate limited (HTTP 429):
{
"Message": "Rate limit exceeded. Please wait before making more requests."
}
Retry Strategy with Exponential Backoff
async function requestWithRetry(url, options, maxRetries = 5) {
for (let attempt = 0; attempt < maxRetries; attempt++) {
try {
const response = await fetch(url, options);
if (response.status === 429) {
// Rate limited - wait and retry
const retryAfter = parseInt(response.headers.get('Retry-After')) || 30;
const jitter = Math.random() * 1000;
console.log(`Rate limited. Waiting ${retryAfter}s...`);
await sleep(retryAfter * 1000 + jitter);
continue;
}
if (!response.ok) {
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
}
return await response.json();
} catch (error) {
if (attempt === maxRetries - 1) throw error;
// Exponential backoff with jitter
const delay = Math.pow(2, attempt) * 1000 + Math.random() * 1000;
console.log(`Attempt ${attempt + 1} failed. Retrying in ${delay}ms...`);
await sleep(delay);
}
}
}
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
Request Throttling
To stay within limits, implement request throttling:
class RateLimiter {
constructor(maxRequests = 700, windowMs = 60000) {
this.maxRequests = maxRequests;
this.windowMs = windowMs;
this.requests = [];
}
async throttle() {
const now = Date.now();
// Remove requests outside window
this.requests = this.requests.filter(t => t > now - this.windowMs);
if (this.requests.length >= this.maxRequests) {
// Wait until oldest request expires
const waitTime = this.requests[0] - (now - this.windowMs) + 100;
await sleep(waitTime);
}
this.requests.push(Date.now());
}
}
const limiter = new RateLimiter();
async function makeRequest(endpoint) {
await limiter.throttle();
return fetch(`https://app.atera.com/api/v3/${endpoint}`, {
headers: { 'X-API-KEY': process.env.ATERA_API_KEY }
});
}
Error Handling
HTTP Status Codes
| Code | Meaning | Action |
|---|---|---|
| 200 | Success | Process response |
| 201 | Created | Entity created successfully |
| 400 | Bad Request | Check request format/values |
| 401 | Unauthorized | Verify API key |
| 403 | Forbidden | Check permissions |
| 404 | Not Found | Entity doesn't exist |
| 429 | Rate Limited | Implement backoff |
| 500 | Server Error | Retry with backoff |
Error Response Format
{
"Message": "Error description here",
"ErrorCode": "SPECIFIC_ERROR_CODE"
}
Error Handling Pattern
async function handleAteraRequest(endpoint, options = {}) {
const response = await fetch(
`https://app.atera.com/api/v3/${endpoint}`,
{
...options,
headers: {
'X-API-KEY': process.env.ATERA_API_KEY,
'Content-Type': 'application/json',
...options.headers
}
}
);
if (!response.ok) {
const error = await response.json().catch(() => ({}));
switch (response.status) {
case 401:
throw new Error('Invalid API key. Check ATERA_API_KEY.');
case 403:
throw new Error('Permission denied. Check API key permissions.');
case 404:
throw new Error(`Resource not found: ${endpoint}`);
case 429:
throw new Error('Rate limit exceeded. Implement backoff.');
default:
throw new Error(error.Message || `API error: ${response.status}`);
}
}
return response.json();
}
CRUD Operations
Create (POST)
POST /api/v3/tickets
X-API-KEY: {api_key}
Content-Type: application/json
{
"TicketTitle": "New ticket",
"Description": "Issue description",
"EndUserID": 12345,
"TicketPriority": "Medium"
}
Response:
{
"ActionID": 54321,
"TicketID": 54321
}
Read (GET)
Single entity:
GET /api/v3/tickets/54321
X-API-KEY: {api_key}
List with pagination:
GET /api/v3/tickets?page=1&itemsInPage=50
X-API-KEY: {api_key}
Update (POST to specific ID)
POST /api/v3/tickets/54321
X-API-KEY: {api_key}
Content-Type: application/json
{
"TicketStatus": "Resolved",
"TicketPriority": "Low"
}