Credits & Rate Limits
AstroWay uses a credit-based model instead of a simple request counter.
This is fairer: lightweight endpoints (/planets) cost 10 credits, heavy ones (/rectification) — 500.
Two kinds of limits
Section titled “Two kinds of limits”1. Monthly credit budget
Section titled “1. Monthly credit budget”Credits per month depend on your plan:
| Plan | Credits/mo | Rate limit |
|---|---|---|
| Free | 10,000 | 10 req/min |
| Indie | 50,000 | 30 req/min |
| Starter | 200,000 | 120 req/min |
| Pro | 800,000 | 400 req/min |
| Business | 3,500,000 | 1,000 req/min |
| Enterprise | Custom | Custom |
When credits run out — subsequent requests return 402 Payment Required (Free) or trigger overage billing (Indie+).
2. Rate limit (requests per minute)
Section titled “2. Rate limit (requests per minute)”Rate limit uses a sliding window. If you fire 120 requests in 10 seconds, the next 50 seconds on that key return 429.
3. Public endpoints (no key)
Section titled “3. Public endpoints (no key)”The 14 reference dictionaries /v1/reference/* (signs, planets, houses, aspects, elements, modalities, polarities, dignities, decans, nakshatras, lots, asteroids, zodiac-systems, glyphs) are served publicly without an X-Api-Key and cost 0 credits. Limit — 30 requests / hour per IP. The same data is also exposed as MCP Resources at astroway://reference/<slug> for LLM agents.
Pricing table
Section titled “Pricing table”Full transparent per-endpoint table with all 705+ endpoints and their costs — at Per-endpoint Cost. DivineAPI / AstrologyAPI / Prokerala don’t publish anything similar. Quick reference:
| Tier | Credits | Typical time | Examples |
|---|---|---|---|
| 1 | 10 | < 50 ms | /planets, /moon-voc, /iching |
| 2 | 20 | 50–200 ms | /chart, /harmonics, /horary |
| 3 | 50 | 200–500 ms | /synastry, /progressions, /acg |
| 4 | 100 | > 500 ms | /transit-calendar, /forecast-calendar |
| 5 | 250 | seconds | /rectification/trutine |
| 6 | 500 | up to 120s | /rectification |
Response headers
Section titled “Response headers”Every response includes:
X-Credits-Used: 20X-Credits-Remaining: 9980X-Credits-Reset: 2026-05-01T00:00:00ZX-RateLimit-Limit: 120X-RateLimit-Remaining: 118X-RateLimit-Reset: 1714521600X-Request-Id: 01HXY2A7ZM...X-Credits-Used— credits charged for this requestX-Credits-Remaining— remainder of monthly budgetX-Credits-Reset— ISO-8601 balance reset dateX-RateLimit-*— current sliding window state (standard format)X-Request-Id— ULID for this request, use in support requests
Caching
Section titled “Caching”Identical requests within 5 minutes return from cache without consuming credits. This is a separate header:
X-Cache: HITX-Credits-Used: 0What counts as “identical”:
- Same endpoint
- Same JSON body (byte-for-byte after whitespace normalization)
- Same API key
This is safe because all endpoints are deterministic — same input always produces same output.
What to do when limits are hit
Section titled “What to do when limits are hit”402 Payment Required — out of credits
Section titled “402 Payment Required — out of credits”{ "error": { "code": "credits_exhausted", "message": "Monthly credit budget exhausted. Upgrade plan or wait for reset.", "credits_remaining": 0, "credits_reset": "2026-05-01T00:00:00Z", "upgrade_url": "https://api.astroway.info/dashboard/billing" }}Options:
- Upgrade plan — immediate new budget
- Overage (Starter/Pro) — auto-enabled unless disabled in settings
- Wait for reset (Free) — 1st of next month
429 Too Many Requests — RPM exceeded
Section titled “429 Too Many Requests — RPM exceeded”{ "error": { "code": "rate_limit_exceeded", "message": "Too many requests. Retry after 12 seconds.", "retry_after_seconds": 12 }}Retry-After: 12 header — standard HTTP, your HTTP client should understand it.
Correct handling:
async function callWithRetry(endpoint: string, body: object) { const response = await fetch(endpoint, { method: 'POST', headers: { 'X-Api-Key': key, 'Content-Type': 'application/json' }, body: JSON.stringify(body), });
if (response.status === 429) { const retryAfter = Number(response.headers.get('Retry-After') ?? 1); await new Promise((r) => setTimeout(r, retryAfter * 1000)); return callWithRetry(endpoint, body); }
return response.json();}Budget estimation in advance
Section titled “Budget estimation in advance”Before integrating, estimate monthly credits:
monthly_credits = daily_active_users × calls_per_user × avg_cost_per_call × 30Example: app with 500 DAU, each making 3 chart calculations + 1 synastry per day:
500 × (3 × 20 + 1 × 50) × 30 = 500 × 110 × 30 = 1,650,000 credits/mo→ Enterprise (custom)Budget calculator — on the Pricing page.
- Errors — full error code reference
- Idempotency — how to retry safely