A production-ready TypeScript-first drop-in replacement for native fetch, or any fetch-compatible implementation.
ffetch can wrap any fetch-compatible implementation (native fetch, node-fetch, undici, or framework-provided fetch), making it flexible for SSR, edge, and custom environments.
ffetch uses a plugin architecture for optional features, so you only include what you need.
- Keep native fetch ergonomics, add production safety (timeouts, retries, error strategy).
- Keep your runtime flexibility (use global fetch or any fetch-compatible handler).
- Keep your bundle lean – ~3kb minified (optional plugins, zero runtime dependencies).
- @fetchkit/ffetch
- Timeouts – per-request or global
- Retries – exponential backoff + jitter
- Abort-aware retries – aborting during backoff cancels immediately
- Plugin architecture – extensible lifecycle-based plugins for optional behavior
- Hooks – logging, auth, metrics, request/response transformation
- Pending requests – real-time monitoring of active requests
- Per-request overrides – customize behavior on a per-request basis
- Universal – Node.js, Browser, Cloudflare Workers, React Native
- Zero runtime deps – ships as dual ESM/CJS
- Configurable error handling – custom error types and
throwOnHttpErrorflag to throw on HTTP errors - Circuit breaker plugin (optional, prebuilt) – automatic failure protection
- Deduplication plugin (optional, prebuilt) – automatic deduping of in-flight identical requests
- Request shortcuts plugin (optional, prebuilt) – call
client.get(url)/.post()/.put()/.patch()/.delete()directly on the client - Response shortcuts plugin (optional, prebuilt) – call
client(url).json()/.text()/.blob()directly on the request promise
Built-in error classes: TimeoutError, RetryLimitError, CircuitOpenError, HttpError, NetworkError, AbortError
All plugins are tree-shakeable — import only what you use.
- dedupePlugin (optional): dedupe in-flight identical requests.
- circuitPlugin (optional): fail fast after repeated failures.
- requestShortcutsPlugin (optional): HTTP method shortcuts on the client (
.get()/.post()/.put()/.patch()/.delete()/.head()/.options()). - responseShortcutsPlugin (optional): use
client(url).json()/.text()/.blob()style parsing.
ffetch is ideal for:
- Microservices and REST APIs with retry requirements and timeout control
- High-traffic client applications that need in-flight deduplication and circuit breaker protection
- SSR and metaframework apps that require runtime flexibility (custom fetch handlers for different environments)
- Type-safe request handling with strong TypeScript support and zero runtime dependencies
# npm
npm install @fetchkit/ffetch
# yarn
yarn add @fetchkit/ffetch
# pnpm
pnpm add @fetchkit/ffetch
# bun
bun add @fetchkit/ffetchimport { createClient } from '@fetchkit/ffetch'
type User = { id: number; name: string }
const api = createClient({ timeout: 5000, retries: 2 })
const response = await api('https://api.example.com/users')
if (!response.ok) {
throw new Error(`Request failed: ${response.status}`)
}
const users = (await response.json()) as User[]import { createClient } from '@fetchkit/ffetch'
import { dedupePlugin } from '@fetchkit/ffetch/plugins/dedupe'
import { circuitPlugin } from '@fetchkit/ffetch/plugins/circuit'
import { requestShortcutsPlugin } from '@fetchkit/ffetch/plugins/request-shortcuts'
import { responseShortcutsPlugin } from '@fetchkit/ffetch/plugins/response-shortcuts'
const api = createClient({
timeout: 10_000,
retries: 2,
plugins: [
// 1) Optional: dedupe identical in-flight requests
dedupePlugin({ ttl: 30_000, sweepInterval: 5_000 }),
// 2) Optional: open the circuit after repeated failures
circuitPlugin({ threshold: 5, reset: 30_000 }),
// 3) Optional: enable request-promise parsing shortcuts
responseShortcutsPlugin(),
// 4) Optional: enable client HTTP method shortcuts
requestShortcutsPlugin(),
],
})
const users = await api
.get('https://api.example.com/users')
.json<Array<{ id: number; name: string }>>()
const p1 = api('https://api.example.com/data')
const p2 = api('https://api.example.com/data')
const [res1, res2] = await Promise.all([p1, p2])What this setup gives you:
- Operational safety: retries with timeout defaults.
- Lower duplicate traffic (optional): concurrent identical requests share one in-flight call.
- Faster failure recovery (optional): circuit breaker blocks repeated failing calls.
- Cleaner request ergonomics (optional):
client.get(url)/.post(url, init)style shortcuts. - Cleaner parsing (optional):
client(url).json()style shortcuts.
- Native fetch is a great baseline, but production apps usually need retries and timeout control.
- ffetch keeps the fetch model and adds optional resilience features.
- You can keep strict native behavior and only opt into plugins you need.
// Throw on non-2xx/429 once retries are exhausted
const strict = createClient({ throwOnHttpError: true })
// Use a custom fetch implementation (SSR/framework/runtime)
import nodeFetch from 'node-fetch'
const apiWithCustomHandler = createClient({ fetchHandler: nodeFetch })
// Keep native Response flow (works with or without plugins)
const plainApi = createClient({ timeout: 5000 })
const response = await plainApi('https://api.example.com/health')
const text = await response.text()// Why this exists:
// ffetch wraps whatever fetch-compatible function you provide.
// This is useful when your runtime has a scoped/framework fetch,
// or when Node needs an explicit fetch implementation.
import { createClient } from '@fetchkit/ffetch'
import nodeFetch from 'node-fetch'
// Node.js example: provide node-fetch explicitly
const apiNode = createClient({
fetchHandler: nodeFetch,
timeout: 5000,
})
const nodeResponse = await apiNode('https://api.example.com/data')
// Framework example: pass the framework-scoped fetch
// (e.g. the fetch passed into a request handler)
async function loadData(frameworkFetch: typeof fetch) {
const api = createClient({
fetchHandler: frameworkFetch,
timeout: 5000,
})
const response = await api('/internal/data')
return response.json()
}All ffetch features (timeouts, retries, plugins, hooks) behave the same with a custom fetchHandler.
With responseShortcutsPlugin() enabled, request-promise shortcuts like api(url).json() also work the same.
// Production-ready client with error handling and monitoring
import { createClient } from '@fetchkit/ffetch'
import { dedupePlugin } from '@fetchkit/ffetch/plugins/dedupe'
import { circuitPlugin } from '@fetchkit/ffetch/plugins/circuit'
const client = createClient({
timeout: 10000,
retries: 2,
fetchHandler: fetch, // Use custom fetch if needed
plugins: [
dedupePlugin({
hashFn: (params) => `${params.method}|${params.url}|${params.body}`,
ttl: 30_000,
sweepInterval: 5_000,
}),
circuitPlugin({
threshold: 5,
reset: 30_000,
onCircuitOpen: (req) => console.warn('Circuit opened due to:', req.url),
onCircuitClose: (req) => console.info('Circuit closed after:', req.url),
}),
],
hooks: {
before: async (req) => console.log('→', req.url),
after: async (req, res) => console.log('←', res.status),
onError: async (req, err) => console.error('Error:', err.message),
},
})
try {
const response = await client('/api/data')
// Check HTTP status manually (like native fetch)
if (!response.ok) {
console.log('HTTP error:', response.status)
return
}
const data = await response.json()
console.log('Active requests:', client.pendingRequests.length)
} catch (err) {
if (err instanceof TimeoutError) {
console.log('Request timed out')
} else if (err instanceof RetryLimitError) {
console.log('Request failed after retries')
}
}Native fetch's controversial behavior of not throwing errors for HTTP error status codes (4xx, 5xx) can lead to overlooked errors in applications. By default, ffetch follows this same pattern, returning a Response object regardless of the HTTP status code. However, with the throwOnHttpError flag, developers can configure ffetch to throw an HttpError for HTTP error responses, making error handling more explicit and robust. Note that this behavior is affected by retries and the circuit breaker - full details are explained in the Error Handling documentation.
| Topic | Description |
|---|---|
| Complete Documentation | Start here - Documentation index and overview |
| API Reference | Complete API documentation and configuration options |
| Plugin Architecture | Plugin lifecycle, custom plugin authoring, and integration patterns |
| Deduplication | How deduplication works, hash config, optional TTL cleanup, limitations |
| Error Handling | Strategies for managing errors, including throwOnHttpError |
| Advanced Features | Per-request overrides, pending requests, circuit breakers, custom errors |
| Hooks & Transformation | Lifecycle hooks, authentication, logging, request/response transformation |
| Usage Examples | Real-world patterns: REST clients, GraphQL, file uploads, microservices |
| Compatibility | Browser/Node.js support, polyfills, framework integration |
ffetch works best with native AbortSignal.any support:
- Node.js 20.6+ (native
AbortSignal.any) - Modern browsers with
AbortSignal.any(for example: Chrome 117+, Firefox 117+, Safari 17+, Edge 117+)
If your environment does not support AbortSignal.any (Node.js < 20.6, older browsers), you can still use ffetch by installing an AbortSignal.any polyfill. AbortSignal.timeout is optional because ffetch includes an internal timeout fallback. See the compatibility guide for instructions.
Custom fetch support:
You can pass any fetch-compatible implementation (native fetch, node-fetch, undici, SvelteKit, Next.js, Nuxt, or a polyfill) via the fetchHandler option. This makes ffetch fully compatible with SSR, edge, metaframework environments, custom backends, and test runners.
Solution: Install a polyfill for AbortSignal.any
npm install abort-controller-x<script type="module">
import { createClient } from 'https://unpkg.com/@fetchkit/ffetch/dist/index.min.js'
const api = createClient({ timeout: 5000 })
const data = await api('/api/data').then((r) => r.json())
</script>- Deduplication is off by default. Enable it via
plugins: [dedupePlugin()]. - The default hash function is
dedupeRequestHash, which handles common body types and skips deduplication for streams and FormData. - Optional stale-entry cleanup:
dedupePlugin({ ttl, sweepInterval })enables map-entry eviction. TTL eviction only removes dedupe keys; it does not reject already in-flight promises. - Stream bodies (
ReadableStream,FormData): Deduplication is skipped for requests with these body types, as they cannot be reliably hashed or replayed. - Non-idempotent requests: Use deduplication with caution for non-idempotent methods (e.g., POST), as it may suppress multiple intended requests.
- Custom hash function: Ensure your hash function uniquely identifies requests to avoid accidental deduplication.
See deduplication.md for full details.
| Feature | Native Fetch | Axios | ky | ffetch |
|---|---|---|---|---|
| Timeouts | ❌ Manual AbortController | ✅ Built-in | ✅ Built-in | ✅ Built-in with fallbacks |
| Retries | ❌ Manual implementation | ❌ Manual or plugins | ✅ Built-in | ✅ Smart exponential backoff |
| Response Parsing DX | await fetch(...).then(...)) |
✅ response.data convenience |
✅ .json()/.text()/.blob() on request chain |
✅ Optional responseShortcutsPlugin() (.json()/.text()/.blob() on request chain) |
| Plugin Architecture | ❌ Not available | ✅ First-class plugin pipeline (optional built-in + custom plugins) | ||
| Circuit Breaker | ❌ Not available | ❌ Manual or plugins | ❌ Manual | ✅ Automatic failure protection |
| Deduplication | ❌ Not available | ❌ Not available | ❌ Not available | ✅ Optional via dedupePlugin() |
| Request Monitoring | ❌ Manual tracking | ❌ Manual tracking | ❌ Manual tracking | ✅ Built-in pending requests |
| Error Types | ❌ Generic errors | ✅ Specific error classes | ✅ Specific error classes | |
| TypeScript | ✅ Strong types | ✅ Full type safety | ||
| Hooks/Middleware | ❌ Not available | ✅ Interceptors | ✅ Hooks | ✅ Comprehensive lifecycle hooks |
| Bundle Size | ✅ Native (0kb) | ❌ ~13kb minified | ✅ Lightweight (fetch-based) | ✅ ~3kb minified |
| Modern APIs | ✅ Web standards | ❌ XMLHttpRequest | ✅ Fetch + modern APIs | ✅ Fetch + modern features |
| Custom Fetch Support | ❌ No (global only) | ❌ No | ❌ No | ✅ Yes (wrap any fetch-compatible implementation, including framework or custom fetch) |
Note: built-in plugins in ffetch are opt-in. Use dedupePlugin() for deduplication, circuitPlugin() for circuit breaking, requestShortcutsPlugin() for client HTTP method shortcuts, and responseShortcutsPlugin() for request-promise parsing shortcuts. Bundle size: ~3kb core, additional optional plugin imports are tree-shakeable.
Want to see these clients in practice? Check out ffetch-demo for working examples and side-by-side comparisons of how ffetch simplifies common fetch patterns.
Got questions, want to discuss features, or share examples? Join the Fetch-Kit Discord server:
- Issues: GitHub Issues
- Pull Requests: GitHub PRs
- Documentation: Found in
./docs/- PRs welcome!
MIT © 2025 gkoos