@intentsolutionsio/vercel-pack 1.0.0 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +1 -1
- package/README.md +67 -44
- package/package.json +4 -4
- package/skills/vercel-advanced-troubleshooting/SKILL.md +185 -195
- package/skills/vercel-advanced-troubleshooting/references/errors.md +11 -0
- package/skills/vercel-advanced-troubleshooting/references/evidence-collection-framework.md +34 -0
- package/skills/vercel-advanced-troubleshooting/references/examples.md +11 -0
- package/skills/vercel-advanced-troubleshooting/references/systematic-isolation.md +56 -0
- package/skills/vercel-advanced-troubleshooting/references/timing-analysis.md +35 -0
- package/skills/vercel-architecture-variants/SKILL.md +227 -216
- package/skills/vercel-architecture-variants/references/errors.md +11 -0
- package/skills/vercel-architecture-variants/references/examples.md +12 -0
- package/skills/vercel-architecture-variants/references/variant-a-monolith-(simple).md +44 -0
- package/skills/vercel-architecture-variants/references/variant-b-service-layer-(moderate).md +72 -0
- package/skills/vercel-architecture-variants/references/variant-c-microservice-(complex).md +81 -0
- package/skills/vercel-ci-integration/SKILL.md +183 -73
- package/skills/vercel-ci-integration/references/errors.md +10 -0
- package/skills/vercel-ci-integration/references/examples.md +36 -0
- package/skills/vercel-ci-integration/references/implementation.md +54 -0
- package/skills/vercel-common-errors/SKILL.md +164 -60
- package/skills/vercel-common-errors/references/errors.md +53 -0
- package/skills/vercel-common-errors/references/examples.md +23 -0
- package/skills/vercel-cost-tuning/SKILL.md +158 -145
- package/skills/vercel-cost-tuning/references/cost-estimation.md +34 -0
- package/skills/vercel-cost-tuning/references/cost-reduction-strategies.md +40 -0
- package/skills/vercel-cost-tuning/references/errors.md +11 -0
- package/skills/vercel-cost-tuning/references/examples.md +15 -0
- package/skills/vercel-data-handling/SKILL.md +202 -155
- package/skills/vercel-data-handling/references/errors.md +11 -0
- package/skills/vercel-data-handling/references/examples.md +27 -0
- package/skills/vercel-data-handling/references/implementation.md +223 -0
- package/skills/vercel-debug-bundle/SKILL.md +163 -67
- package/skills/vercel-debug-bundle/references/errors.md +12 -0
- package/skills/vercel-debug-bundle/references/examples.md +24 -0
- package/skills/vercel-debug-bundle/references/implementation.md +54 -0
- package/skills/vercel-deploy-integration/SKILL.md +163 -156
- package/skills/vercel-deploy-integration/references/errors.md +11 -0
- package/skills/vercel-deploy-integration/references/examples.md +21 -0
- package/skills/vercel-deploy-integration/references/google-cloud-run.md +36 -0
- package/skills/vercel-deploy-integration/references/vercel-deployment.md +35 -0
- package/skills/vercel-deploy-preview/SKILL.md +164 -39
- package/skills/vercel-edge-functions/SKILL.md +185 -37
- package/skills/vercel-enterprise-rbac/SKILL.md +185 -170
- package/skills/vercel-enterprise-rbac/references/errors.md +11 -0
- package/skills/vercel-enterprise-rbac/references/examples.md +12 -0
- package/skills/vercel-enterprise-rbac/references/role-implementation.md +33 -0
- package/skills/vercel-enterprise-rbac/references/sso-integration.md +35 -0
- package/skills/vercel-hello-world/SKILL.md +141 -55
- package/skills/vercel-incident-runbook/SKILL.md +186 -138
- package/skills/vercel-incident-runbook/references/errors.md +11 -0
- package/skills/vercel-incident-runbook/references/examples.md +10 -0
- package/skills/vercel-incident-runbook/references/immediate-actions-by-error-type.md +41 -0
- package/skills/vercel-install-auth/SKILL.md +130 -53
- package/skills/vercel-known-pitfalls/SKILL.md +235 -233
- package/skills/vercel-known-pitfalls/references/errors.md +11 -0
- package/skills/vercel-known-pitfalls/references/examples.md +12 -0
- package/skills/vercel-load-scale/SKILL.md +197 -204
- package/skills/vercel-load-scale/references/capacity-planning.md +47 -0
- package/skills/vercel-load-scale/references/errors.md +11 -0
- package/skills/vercel-load-scale/references/examples.md +26 -0
- package/skills/vercel-load-scale/references/load-testing-with-k6.md +59 -0
- package/skills/vercel-load-scale/references/scaling-patterns.md +65 -0
- package/skills/vercel-local-dev-loop/SKILL.md +159 -71
- package/skills/vercel-local-dev-loop/references/errors.md +11 -0
- package/skills/vercel-local-dev-loop/references/examples.md +21 -0
- package/skills/vercel-local-dev-loop/references/implementation.md +60 -0
- package/skills/vercel-migration-deep-dive/SKILL.md +202 -187
- package/skills/vercel-migration-deep-dive/references/errors.md +11 -0
- package/skills/vercel-migration-deep-dive/references/examples.md +12 -0
- package/skills/vercel-migration-deep-dive/references/implementation-plan.md +80 -0
- package/skills/vercel-migration-deep-dive/references/pre-migration-assessment.md +39 -0
- package/skills/vercel-multi-env-setup/SKILL.md +167 -164
- package/skills/vercel-multi-env-setup/references/configuration-structure.md +59 -0
- package/skills/vercel-multi-env-setup/references/errors.md +11 -0
- package/skills/vercel-multi-env-setup/references/examples.md +11 -0
- package/skills/vercel-observability/SKILL.md +205 -195
- package/skills/vercel-observability/references/alert-configuration.md +40 -0
- package/skills/vercel-observability/references/errors.md +11 -0
- package/skills/vercel-observability/references/examples.md +13 -0
- package/skills/vercel-observability/references/metrics-collection.md +65 -0
- package/skills/vercel-performance-tuning/SKILL.md +212 -156
- package/skills/vercel-performance-tuning/references/caching-strategy.md +49 -0
- package/skills/vercel-performance-tuning/references/errors.md +11 -0
- package/skills/vercel-performance-tuning/references/examples.md +13 -0
- package/skills/vercel-policy-guardrails/SKILL.md +276 -193
- package/skills/vercel-policy-guardrails/references/errors.md +11 -0
- package/skills/vercel-policy-guardrails/references/eslint-rules.md +46 -0
- package/skills/vercel-policy-guardrails/references/examples.md +10 -0
- package/skills/vercel-prod-checklist/SKILL.md +219 -94
- package/skills/vercel-prod-checklist/references/errors.md +11 -0
- package/skills/vercel-prod-checklist/references/examples.md +25 -0
- package/skills/vercel-prod-checklist/references/implementation.md +60 -0
- package/skills/vercel-rate-limits/SKILL.md +187 -100
- package/skills/vercel-rate-limits/references/errors.md +11 -0
- package/skills/vercel-rate-limits/references/examples.md +46 -0
- package/skills/vercel-rate-limits/references/implementation.md +66 -0
- package/skills/vercel-reference-architecture/SKILL.md +226 -180
- package/skills/vercel-reference-architecture/references/errors.md +11 -0
- package/skills/vercel-reference-architecture/references/examples.md +13 -0
- package/skills/vercel-reference-architecture/references/key-components.md +65 -0
- package/skills/vercel-reference-architecture/references/project-structure.md +40 -0
- package/skills/vercel-reliability-patterns/SKILL.md +272 -211
- package/skills/vercel-reliability-patterns/references/circuit-breaker.md +36 -0
- package/skills/vercel-reliability-patterns/references/dead-letter-queue.md +48 -0
- package/skills/vercel-reliability-patterns/references/errors.md +11 -0
- package/skills/vercel-reliability-patterns/references/examples.md +11 -0
- package/skills/vercel-reliability-patterns/references/idempotency-keys.md +36 -0
- package/skills/vercel-sdk-patterns/SKILL.md +264 -92
- package/skills/vercel-sdk-patterns/references/errors.md +11 -0
- package/skills/vercel-sdk-patterns/references/examples.md +45 -0
- package/skills/vercel-sdk-patterns/references/implementation.md +67 -0
- package/skills/vercel-security-basics/SKILL.md +186 -96
- package/skills/vercel-security-basics/references/errors.md +10 -0
- package/skills/vercel-security-basics/references/examples.md +70 -0
- package/skills/vercel-security-basics/references/implementation.md +39 -0
- package/skills/vercel-upgrade-migration/SKILL.md +167 -67
- package/skills/vercel-upgrade-migration/references/errors.md +10 -0
- package/skills/vercel-upgrade-migration/references/examples.md +51 -0
- package/skills/vercel-upgrade-migration/references/implementation.md +29 -0
- package/skills/vercel-webhooks-events/SKILL.md +208 -132
- package/skills/vercel-webhooks-events/references/errors.md +11 -0
- package/skills/vercel-webhooks-events/references/event-handler-pattern.md +37 -0
- package/skills/vercel-webhooks-events/references/examples.md +16 -0
- package/skills/vercel-webhooks-events/references/signature-verification.md +33 -0
|
@@ -1,149 +1,236 @@
|
|
|
1
1
|
---
|
|
2
2
|
name: vercel-rate-limits
|
|
3
|
-
description:
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
3
|
+
description: 'Handle Vercel API rate limits, implement retry logic, and configure
|
|
4
|
+
WAF rate limiting.
|
|
5
|
+
|
|
6
|
+
Use when hitting 429 errors, implementing retry logic,
|
|
7
|
+
|
|
8
|
+
or setting up rate limiting for your Vercel-deployed API endpoints.
|
|
9
|
+
|
|
7
10
|
Trigger with phrases like "vercel rate limit", "vercel throttling",
|
|
8
|
-
|
|
11
|
+
|
|
12
|
+
"vercel 429", "vercel retry", "vercel backoff", "vercel WAF rate limit".
|
|
13
|
+
|
|
14
|
+
'
|
|
9
15
|
allowed-tools: Read, Write, Edit
|
|
10
16
|
version: 1.0.0
|
|
11
17
|
license: MIT
|
|
12
18
|
author: Jeremy Longshore <jeremy@intentsolutions.io>
|
|
19
|
+
tags:
|
|
20
|
+
- saas
|
|
21
|
+
- vercel
|
|
22
|
+
- api
|
|
23
|
+
- rate-limiting
|
|
24
|
+
- security
|
|
25
|
+
compatibility: Designed for Claude Code, also compatible with Codex and OpenClaw
|
|
13
26
|
---
|
|
14
|
-
|
|
15
27
|
# Vercel Rate Limits
|
|
16
28
|
|
|
17
29
|
## Overview
|
|
18
|
-
|
|
30
|
+
|
|
31
|
+
Handle Vercel REST API rate limits with proper retry logic, and configure Vercel's WAF rate limiting SDK to protect your deployed API endpoints from abuse. Covers both consuming the Vercel API (outbound) and protecting your own functions (inbound).
|
|
19
32
|
|
|
20
33
|
## Prerequisites
|
|
21
|
-
|
|
22
|
-
-
|
|
23
|
-
-
|
|
34
|
+
|
|
35
|
+
- Vercel CLI installed and authenticated
|
|
36
|
+
- Understanding of HTTP 429 status codes
|
|
37
|
+
- For WAF rate limiting: Vercel Pro or Enterprise plan
|
|
24
38
|
|
|
25
39
|
## Instructions
|
|
26
40
|
|
|
27
|
-
### Step 1:
|
|
41
|
+
### Step 1: Vercel REST API Rate Limits
|
|
42
|
+
|
|
43
|
+
The Vercel REST API enforces rate limits per endpoint. When exceeded, the API returns HTTP 429 with rate limit headers:
|
|
44
|
+
|
|
45
|
+
```
|
|
46
|
+
HTTP/1.1 429 Too Many Requests
|
|
47
|
+
X-RateLimit-Limit: 100
|
|
48
|
+
X-RateLimit-Remaining: 0
|
|
49
|
+
X-RateLimit-Reset: 1711152000
|
|
50
|
+
Retry-After: 60
|
|
51
|
+
```
|
|
52
|
+
|
|
53
|
+
**Known API limits:**
|
|
28
54
|
|
|
29
|
-
|
|
|
30
|
-
|
|
31
|
-
|
|
|
32
|
-
|
|
|
33
|
-
|
|
|
55
|
+
| Endpoint Category | Rate Limit |
|
|
56
|
+
|-------------------|-----------|
|
|
57
|
+
| Deployments (create) | 100/hour per project |
|
|
58
|
+
| Deployments (list/get) | 500/min |
|
|
59
|
+
| Projects (CRUD) | 200/min |
|
|
60
|
+
| Environment variables | 200/min |
|
|
61
|
+
| Domains | 200/min |
|
|
62
|
+
| Teams | 200/min |
|
|
63
|
+
| DNS records | 200/min |
|
|
64
|
+
| General API | 120 requests/min (default) |
|
|
34
65
|
|
|
35
|
-
### Step 2: Implement
|
|
66
|
+
### Step 2: Implement Retry with Backoff for Vercel API
|
|
36
67
|
|
|
37
68
|
```typescript
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
69
|
+
// lib/rate-limit-handler.ts
|
|
70
|
+
interface RateLimitInfo {
|
|
71
|
+
limit: number;
|
|
72
|
+
remaining: number;
|
|
73
|
+
reset: number; // Unix timestamp
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function parseRateLimitHeaders(headers: Headers): RateLimitInfo {
|
|
77
|
+
return {
|
|
78
|
+
limit: Number(headers.get('X-RateLimit-Limit') ?? 100),
|
|
79
|
+
remaining: Number(headers.get('X-RateLimit-Remaining') ?? 100),
|
|
80
|
+
reset: Number(headers.get('X-RateLimit-Reset') ?? 0),
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
async function vercelFetchWithRetry(
|
|
85
|
+
url: string,
|
|
86
|
+
options: RequestInit,
|
|
87
|
+
maxRetries = 3
|
|
88
|
+
): Promise<Response> {
|
|
89
|
+
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
90
|
+
const res = await fetch(url, options);
|
|
91
|
+
|
|
92
|
+
if (res.status !== 429) return res;
|
|
93
|
+
|
|
94
|
+
if (attempt === maxRetries) {
|
|
95
|
+
throw new Error(`Rate limited after ${maxRetries} retries: ${url}`);
|
|
57
96
|
}
|
|
97
|
+
|
|
98
|
+
// Use Retry-After header if present, otherwise exponential backoff
|
|
99
|
+
const retryAfter = res.headers.get('Retry-After');
|
|
100
|
+
const waitMs = retryAfter
|
|
101
|
+
? Number(retryAfter) * 1000
|
|
102
|
+
: Math.min(1000 * Math.pow(2, attempt) + Math.random() * 1000, 30000);
|
|
103
|
+
|
|
104
|
+
console.warn(`Rate limited (attempt ${attempt + 1}/${maxRetries}). Waiting ${Math.round(waitMs)}ms...`);
|
|
105
|
+
await new Promise(r => setTimeout(r, waitMs));
|
|
58
106
|
}
|
|
59
107
|
throw new Error('Unreachable');
|
|
60
108
|
}
|
|
61
109
|
```
|
|
62
110
|
|
|
63
|
-
### Step 3:
|
|
111
|
+
### Step 3: Proactive Rate Limit Avoidance
|
|
64
112
|
|
|
65
113
|
```typescript
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
114
|
+
// lib/rate-limiter.ts
|
|
115
|
+
// Track remaining quota and slow down before hitting the wall
|
|
116
|
+
class VercelRateLimiter {
|
|
117
|
+
private remaining = 100;
|
|
118
|
+
private resetAt = 0;
|
|
119
|
+
|
|
120
|
+
async throttle(): Promise<void> {
|
|
121
|
+
// If near the limit, wait until reset
|
|
122
|
+
if (this.remaining < 5) {
|
|
123
|
+
const waitMs = Math.max(0, this.resetAt * 1000 - Date.now()) + 1000;
|
|
124
|
+
console.warn(`Near rate limit (${this.remaining} remaining). Waiting ${waitMs}ms...`);
|
|
125
|
+
await new Promise(r => setTimeout(r, waitMs));
|
|
126
|
+
}
|
|
127
|
+
}
|
|
74
128
|
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
): Promise<T> {
|
|
80
|
-
// Use provided key (for retries) or generate deterministic key from params
|
|
81
|
-
const key = idempotencyKey || generateIdempotencyKey(params.method || 'POST', params);
|
|
82
|
-
return client.request({
|
|
83
|
-
...params,
|
|
84
|
-
headers: { 'Idempotency-Key': key, ...params.headers },
|
|
85
|
-
});
|
|
129
|
+
update(headers: Headers): void {
|
|
130
|
+
this.remaining = Number(headers.get('X-RateLimit-Remaining') ?? this.remaining);
|
|
131
|
+
this.resetAt = Number(headers.get('X-RateLimit-Reset') ?? this.resetAt);
|
|
132
|
+
}
|
|
86
133
|
}
|
|
87
134
|
```
|
|
88
135
|
|
|
89
|
-
|
|
90
|
-
- Reliable API calls with automatic retry
|
|
91
|
-
- Idempotent requests preventing duplicates
|
|
92
|
-
- Rate limit headers properly handled
|
|
93
|
-
|
|
94
|
-
## Error Handling
|
|
95
|
-
| Header | Description | Action |
|
|
96
|
-
|--------|-------------|--------|
|
|
97
|
-
| X-RateLimit-Limit | Max requests | Monitor usage |
|
|
98
|
-
| X-RateLimit-Remaining | Remaining requests | Throttle if low |
|
|
99
|
-
| X-RateLimit-Reset | Reset timestamp | Wait until reset |
|
|
100
|
-
| Retry-After | Seconds to wait | Honor this value |
|
|
136
|
+
### Step 4: Protect Your Own Endpoints — Vercel WAF Rate Limiting
|
|
101
137
|
|
|
102
|
-
|
|
138
|
+
Vercel's WAF provides built-in rate limiting for your deployed functions:
|
|
103
139
|
|
|
104
|
-
### Queue-Based Rate Limiting
|
|
105
140
|
```typescript
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
141
|
+
// middleware.ts — WAF rate limiting via Vercel Firewall SDK
|
|
142
|
+
import { ipAddress } from '@vercel/functions';
|
|
143
|
+
import { checkRateLimit } from '@vercel/firewall';
|
|
144
|
+
|
|
145
|
+
export async function middleware(request: Request) {
|
|
146
|
+
const ip = ipAddress(request) ?? '127.0.0.1';
|
|
147
|
+
|
|
148
|
+
// Rate limit: 100 requests per 60 seconds per IP
|
|
149
|
+
const { rateLimited } = await checkRateLimit('api-limit', {
|
|
150
|
+
key: ip,
|
|
151
|
+
limit: 100,
|
|
152
|
+
window: '60s',
|
|
153
|
+
});
|
|
113
154
|
|
|
114
|
-
|
|
115
|
-
|
|
155
|
+
if (rateLimited) {
|
|
156
|
+
return new Response(
|
|
157
|
+
JSON.stringify({ error: 'Too many requests. Please try again later.' }),
|
|
158
|
+
{ status: 429, headers: { 'Content-Type': 'application/json', 'Retry-After': '60' } }
|
|
159
|
+
);
|
|
160
|
+
}
|
|
116
161
|
}
|
|
162
|
+
|
|
163
|
+
export const config = {
|
|
164
|
+
matcher: '/api/:path*',
|
|
165
|
+
};
|
|
117
166
|
```
|
|
118
167
|
|
|
119
|
-
|
|
168
|
+
Install: `npm install @vercel/firewall @vercel/functions`
|
|
169
|
+
|
|
170
|
+
### Step 5: Custom Rate Limiting with Edge Config
|
|
171
|
+
|
|
120
172
|
```typescript
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
173
|
+
// api/rate-limited-endpoint.ts
|
|
174
|
+
import { get } from '@vercel/edge-config';
|
|
175
|
+
|
|
176
|
+
export const config = { runtime: 'edge' };
|
|
177
|
+
|
|
178
|
+
// Simple in-memory sliding window (per-isolate, not global)
|
|
179
|
+
const windowMs = 60_000;
|
|
180
|
+
const maxRequests = 50;
|
|
181
|
+
const requests = new Map<string, number[]>();
|
|
182
|
+
|
|
183
|
+
function isRateLimited(key: string): boolean {
|
|
184
|
+
const now = Date.now();
|
|
185
|
+
const timestamps = (requests.get(key) ?? []).filter(t => now - t < windowMs);
|
|
186
|
+
timestamps.push(now);
|
|
187
|
+
requests.set(key, timestamps);
|
|
188
|
+
return timestamps.length > maxRequests;
|
|
189
|
+
}
|
|
132
190
|
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
return this.remaining < 5 && new Date() < this.resetAt;
|
|
136
|
-
}
|
|
191
|
+
export default async function handler(request: Request): Promise<Response> {
|
|
192
|
+
const ip = request.headers.get('x-forwarded-for') ?? 'unknown';
|
|
137
193
|
|
|
138
|
-
|
|
139
|
-
return
|
|
194
|
+
if (isRateLimited(ip)) {
|
|
195
|
+
return Response.json({ error: 'Rate limit exceeded' }, { status: 429 });
|
|
140
196
|
}
|
|
197
|
+
|
|
198
|
+
return Response.json({ data: 'ok' });
|
|
141
199
|
}
|
|
142
200
|
```
|
|
143
201
|
|
|
202
|
+
## Platform Concurrency Limits
|
|
203
|
+
|
|
204
|
+
| Plan | Concurrent Executions | Builds/Hour |
|
|
205
|
+
|------|-----------------------|-------------|
|
|
206
|
+
| Hobby | 10 | 32 |
|
|
207
|
+
| Pro | 1,000 | 6,000/day |
|
|
208
|
+
| Enterprise | 100,000 | Custom |
|
|
209
|
+
|
|
210
|
+
## Output
|
|
211
|
+
|
|
212
|
+
- Vercel API calls wrapped with automatic retry and backoff
|
|
213
|
+
- Rate limit headers parsed and monitored proactively
|
|
214
|
+
- WAF rate limiting protecting deployed API endpoints
|
|
215
|
+
- Custom per-IP rate limiting for fine-grained control
|
|
216
|
+
|
|
217
|
+
## Error Handling
|
|
218
|
+
|
|
219
|
+
| Error | Cause | Solution |
|
|
220
|
+
|-------|-------|----------|
|
|
221
|
+
| `429 Too Many Requests` | API rate limit exceeded | Use `vercelFetchWithRetry()` wrapper |
|
|
222
|
+
| `FUNCTION_THROTTLED` | Concurrent execution limit hit | Reduce parallelism or upgrade plan |
|
|
223
|
+
| Rate limit not applied | Middleware not matching routes | Check `config.matcher` pattern |
|
|
224
|
+
| In-memory rate limit resets | Edge function isolate recycled | Use Redis or Vercel KV for persistent state |
|
|
225
|
+
|
|
144
226
|
## Resources
|
|
145
|
-
|
|
146
|
-
- [
|
|
227
|
+
|
|
228
|
+
- [Vercel Limits](https://vercel.com/docs/limits)
|
|
229
|
+
- [WAF Rate Limiting](https://vercel.com/docs/vercel-firewall/vercel-waf/rate-limiting)
|
|
230
|
+
- [Rate Limiting SDK](https://vercel.com/docs/vercel-firewall/vercel-waf/rate-limiting-sdk)
|
|
231
|
+
- [FUNCTION_THROTTLED](https://vercel.com/docs/errors/FUNCTION_THROTTLED)
|
|
232
|
+
- [Concurrency Scaling](https://vercel.com/docs/functions/concurrency-scaling)
|
|
147
233
|
|
|
148
234
|
## Next Steps
|
|
149
|
-
|
|
235
|
+
|
|
236
|
+
For security best practices, see `vercel-security-basics`.
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
# Error Handling Reference
|
|
2
|
+
|
|
3
|
+
| Header | Description | Action |
|
|
4
|
+
|--------|-------------|--------|
|
|
5
|
+
| X-RateLimit-Limit | Max requests | Monitor usage |
|
|
6
|
+
| X-RateLimit-Remaining | Remaining requests | Throttle if low |
|
|
7
|
+
| X-RateLimit-Reset | Reset timestamp | Wait until reset |
|
|
8
|
+
| Retry-After | Seconds to wait | Honor this value |
|
|
9
|
+
|
|
10
|
+
---
|
|
11
|
+
*[Tons of Skills](https://tonsofskills.com) by [Intent Solutions](https://intentsolutions.io) | [jeremylongshore.com](https://jeremylongshore.com)*
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
## Examples
|
|
2
|
+
|
|
3
|
+
### Queue-Based Rate Limiting
|
|
4
|
+
|
|
5
|
+
```typescript
|
|
6
|
+
import PQueue from 'p-queue';
|
|
7
|
+
|
|
8
|
+
const queue = new PQueue({
|
|
9
|
+
concurrency: 5,
|
|
10
|
+
interval: 1000,
|
|
11
|
+
intervalCap: 10,
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
async function queuedRequest<T>(operation: () => Promise<T>): Promise<T> {
|
|
15
|
+
return queue.add(operation);
|
|
16
|
+
}
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
### Monitor Rate Limit Usage
|
|
20
|
+
|
|
21
|
+
```typescript
|
|
22
|
+
class RateLimitMonitor {
|
|
23
|
+
private remaining: number = 60;
|
|
24
|
+
private resetAt: Date = new Date();
|
|
25
|
+
|
|
26
|
+
updateFromHeaders(headers: Headers) {
|
|
27
|
+
this.remaining = parseInt(headers.get('X-RateLimit-Remaining') || '60');
|
|
28
|
+
const resetTimestamp = headers.get('X-RateLimit-Reset');
|
|
29
|
+
if (resetTimestamp) {
|
|
30
|
+
this.resetAt = new Date(parseInt(resetTimestamp) * 1000);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
shouldThrottle(): boolean {
|
|
35
|
+
// Only throttle if low remaining AND reset hasn't happened yet
|
|
36
|
+
return this.remaining < 5 && new Date() < this.resetAt;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
getWaitTime(): number {
|
|
40
|
+
return Math.max(0, this.resetAt.getTime() - Date.now());
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
---
|
|
46
|
+
*[Tons of Skills](https://tonsofskills.com) by [Intent Solutions](https://intentsolutions.io) | [jeremylongshore.com](https://jeremylongshore.com)*
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
## Implementation Guide
|
|
2
|
+
|
|
3
|
+
### Step 1: Understand Rate Limit Tiers
|
|
4
|
+
|
|
5
|
+
| Tier | Requests/min | Requests/day | Burst |
|
|
6
|
+
|------|-------------|--------------|-------|
|
|
7
|
+
| Hobby | 100 | 100,000 | 10 |
|
|
8
|
+
| Pro | 1,000 | 1,000,000 | 50 |
|
|
9
|
+
| Enterprise | 10,000 | Unlimited | 200 |
|
|
10
|
+
|
|
11
|
+
### Step 2: Implement Exponential Backoff with Jitter
|
|
12
|
+
|
|
13
|
+
```typescript
|
|
14
|
+
async function withExponentialBackoff<T>(
|
|
15
|
+
operation: () => Promise<T>,
|
|
16
|
+
config = { maxRetries: 5, baseDelayMs: 1000, maxDelayMs: 32000, jitterMs: 500 }
|
|
17
|
+
): Promise<T> {
|
|
18
|
+
for (let attempt = 0; attempt <= config.maxRetries; attempt++) {
|
|
19
|
+
try {
|
|
20
|
+
return await operation();
|
|
21
|
+
} catch (error: any) {
|
|
22
|
+
if (attempt === config.maxRetries) throw error;
|
|
23
|
+
const status = error.status || error.response?.status;
|
|
24
|
+
if (status !== 429 && (status < 500 || status >= 600)) throw error;
|
|
25
|
+
|
|
26
|
+
// Exponential delay with jitter to prevent thundering herd
|
|
27
|
+
const exponentialDelay = config.baseDelayMs * Math.pow(2, attempt);
|
|
28
|
+
const jitter = Math.random() * config.jitterMs;
|
|
29
|
+
const delay = Math.min(exponentialDelay + jitter, config.maxDelayMs);
|
|
30
|
+
|
|
31
|
+
console.log(`Rate limited. Retrying in ${delay.toFixed(0)}ms...`);
|
|
32
|
+
await new Promise(r => setTimeout(r, delay));
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
throw new Error('Unreachable');
|
|
36
|
+
}
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
### Step 3: Add Idempotency Keys
|
|
40
|
+
|
|
41
|
+
```typescript
|
|
42
|
+
import { v4 as uuidv4 } from 'uuid';
|
|
43
|
+
import crypto from 'crypto';
|
|
44
|
+
|
|
45
|
+
// Generate deterministic key from operation params (for safe retries)
|
|
46
|
+
function generateIdempotencyKey(operation: string, params: Record<string, any>): string {
|
|
47
|
+
const data = JSON.stringify({ operation, params });
|
|
48
|
+
return crypto.createHash('sha256').update(data).digest('hex');
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
async function idempotentRequest<T>(
|
|
52
|
+
client: VercelClient,
|
|
53
|
+
params: Record<string, any>,
|
|
54
|
+
idempotencyKey?: string // Pass existing key for retries
|
|
55
|
+
): Promise<T> {
|
|
56
|
+
// Use provided key (for retries) or generate deterministic key from params
|
|
57
|
+
const key = idempotencyKey || generateIdempotencyKey(params.method || 'POST', params);
|
|
58
|
+
return client.request({
|
|
59
|
+
...params,
|
|
60
|
+
headers: { 'Idempotency-Key': key, ...params.headers },
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
---
|
|
66
|
+
*[Tons of Skills](https://tonsofskills.com) by [Intent Solutions](https://intentsolutions.io) | [jeremylongshore.com](https://jeremylongshore.com)*
|