The promise of AI is transformative. The reality is distributed, fragile, and increasingly complex. Production LLM applications need more than just a single provider. They need reliability by design.
If you've deployed ML workloads at scale, you know the pain: OpenAI goes down, your app goes down. Anthropic is overloaded, requests queue indefinitely.
This is where Bifrost automatic fallback mechanism comes in. It is a routing layer that transforms single points of failure into resilient, changing request chains.
π» A small example of work
Traditional LLM integrations look like this:
// Traditional approach - brittle
const response = await openai.chat.completions.create({
model: "gpt-4o",
messages: [{ role: "user", content: "Hello!" }]
});
What happens when OpenAI is down? Your entire application fails. No graceful degradation. No fallback. Just errors.
Even with try-catch, you're manually writing fallback logic:
let response;
try {
response = await openai.chat.completions.create({...});
} catch (error) {
console.log("OpenAI failed, trying Anthropic...");
response = await anthropic.chat.completions.create({...});
}
This approach has real problems:
- Boilerplate everywhere: Every API call needs fallback logic
- Hard to maintain: Adding a third provider means refactoring all your code
- Inconsistent behavior: Different services handle timeouts differently
βοΈ Declarative Resilience
Bifrost flips this model. Instead of embedding fallback logic in your application, you declare your resilience strategy once, at the Virtual Key level:
# Configure your Virtual Key with multiple providers
curl -X POST https://api.bifrost.example.com/virtual-keys \
-H "Authorization: Bearer $TOKEN" \
-d '{
"name": "vk-prod-main",
"provider_configs": [
{
"provider": "openai",
"allowed_models": ["gpt-4o", "gpt-4o-mini"],
"weight": 0.6
},
{
"provider": "anthropic",
"allowed_models": ["gpt-4o"],
"weight": 0.4
}
]
}'
Now your application code is beautifully simple:
// With Bifrost - no fallback logic needed
const response = await fetch('http://localhost:8000/v1/chat/completions', {
method: 'POST',
headers: {
'x-bf-vk': 'vk-prod-main',
'Content-Type': 'application/json'
},
body: JSON.stringify({
model: 'gpt-4o',
messages: [{ role: 'user', content: 'Hello!' }]
})
});
Bifrost handles the rest automatically.
βοΈ How Automatic Fallbacks Work
Here's the magic: when you configure multiple providers on a Virtual Key, Bifrost creates an automatic fallback chain without any intervention from you.
The Anatomy of a Fallback Chain
Request: gpt-4o
β
[Load Balancer]
β
Route to: Anthropic (60% weight) β Primary
β
β Anthropic failed (timeout/error)
β
Fall back to: OpenAI (40% weight) β Secondary
β
β Success! Return response
Key behaviors:
- Weighted Selection: Your primary request goes to the provider with the highest weight
- Automatic Retry: If that provider fails, Bifrost automatically retries the next provider in line
- Weight-Ordered Chain: Fallbacks are sorted by weight providers get priority
- Transparent to Application: Your code never sees the fallback happen
A Real-World Example
Imagine this Virtual Key configuration:
{
"provider_configs": [
{
"provider": "openai",
"allowed_models": ["gpt-4o"],
"weight": 0.15
},
{
"provider": "anthropic",
"allowed_models": ["gpt-4o"], // via Model Catalog wildcard
"weight": 0.05
}
]
}
Request flow for a gpt-4o query:
Attempt 1: OpenAI (15% weight)
β Rate limit exceeded
Attempt 2: Anthropic (5% weight)
β β SUCCESS - Response returned to user
Total latency: ~12 seconds
User sees: Normal response (not an error)
Without Bifrost, that same request would have failed at Attempt 1, never trying the fallbacks.
π Preserving LLM Control
Automatic fallbacks are opt-in by default. If you already have fallback logic in your request, Bifrost respects it and doesn't add automatic chains:
// With explicit fallbacks - automatic chain is skipped
const response = await fetch('http://localhost:8000/v1/chat/completions', {
method: 'POST',
headers: { 'x-bf-vk': 'vk-prod-main' },
body: JSON.stringify({
model: 'gpt-4o',
messages: [{ role: 'user', content: 'Hello!' }],
fallbacks: ['anthropic/claude-3-sonnet-20240229'] // β Your custom chain
})
});
This flexibility is crucial. Sometimes you need specific fallback behavior for compliance, cost, or performance reasons.
βοΈ Combining Automatic Fallbacks with Weighted Load Balancing
The real power emerges when you combine three Bifrost features:
1. Weighted Load Balancing
Distribute traffic proportionally across providers:
- 70% to cheap OpenAI (cost optimization)
- 5% to Anthropic (bleeding-edge features)
2. Automatic Fallbacks
If OpenAI fails β try Anthropic
3. API Key Restrictions
Ensure production workloads use production keys only:
{
"provider_configs": [
{
"provider": "openai",
"key_ids": ["key-prod-001"], // Only production OpenAI key
"allowed_models": ["gpt-4o"],
"weight": 0.8
}
]
}
Together, these features create a governance layer that makes your infrastructure simultaneously:
- Resilient (automatic failover)
- Cost-effective (weighted optimization)
- Compliant (fine-grained access control)
- Simple (no application-level boilerplate)
π» Use Cases
Use Case 1
// Production VK - strict, resilient
{
"name": "vk-prod",
"provider_configs": [
{ "provider": "openai", "key_ids": ["key-prod"], "weight": 0.6 },
{ "provider": "anthropic", "key_ids": ["key-anthopic-prod"], "weight": 0.4 }
]
}
// Development VK - cheaper models
{
"name": "vk-dev",
"provider_configs": [
{ "provider": "openai", "key_ids": ["key-dev"], "allowed_models": ["gpt-4o-mini"], "weight": 1.0 }
]
}
Use Case 2
{
"name": "vk-cost-optimized",
"provider_configs": [
// Primary: cheapest provider
{ "provider": "openai", "allowed_models": ["gpt-4o-mini"], "weight": 0.85 },
{ "provider": "anthropic", "allowed_models": ["claude-3-sonnet"], "weight": 0.05 }
]
}
Use Case 3
{
"name": "vk-global",
"provider_configs": [
// Primary: lowest latency for your region
{ "provider": "anthopic", "key_ids": ["key-anthopic-eu"], "weight": 0.7 },
// Fallback: global provider
{ "provider": "openai", "weight": 0.3 }
]
}
βοΈ Implementation Details You Should Know
How Bifrost Determines Fallback Order
Bifrost sorts fallback providers by weight, descending:
Weight 0.15 (OpenAI) β tried first
Weight 0.05 (Anthropic) β tried last
This means your "best" provider (highest weight) is also your primary, and your fallbacks gracefully degrade through less-preferred options.
When Automatic Fallbacks DON'T Trigger
Automatic fallback chains are only created if:
- β Your request has no existing
fallbacksarray - β You have multiple providers configured on the Virtual Key
If you've manually specified fallbacks, Bifrost respects your configuration and doesn't add automatic chains. This prevents surprising behavior for applications that have custom fallback strategies.
Model Validation Across Providers
Bifrost doesn't blindly route requests. It validates that the requested model is actually supported by the provider:
// β This works - both Anthropic and OpenAI support gpt-4o
curl -H "x-bf-vk: vk-prod-main" \
-d '{"model": "gpt-4o"}' \
http://localhost:8000/v1/chat/completions
// β This fails - only OpenAI supports gpt-4o-mini
curl -H "x-bf-vk: vk-prod-main" \
-d '{"model": "gpt-4o-mini"}' \
http://localhost:8000/v1/chat/completions
# Error: Model not available on configured providers
The validation happens via Bifrost's Model Catalog, which syncs with each provider's actual supported models on startup and during updates.
π Observability
With automatic fallbacks, visibility becomes critical. You need to know:
- Which requests are using fallbacks?
- Which providers are failing?
- What's the fallback success rate?
Bifrost exposes this through:
- Structured logs identifying which fallback was used
- Metrics on fallback frequency and success rates
- Distributed tracing showing the full request chain
{
"timestamp": "2026-01-15",
"request_id": "req-req1-123",
"model": "gpt-4o",
"primary_provider": "anthropic",
"fallback_used": true,
"fallback_provider": "openai",
"total_latency_ms": 8420,
"primary_latency_ms": 5000,
"fallback_latency_ms": 3420
}
With this data, you can:
- Detect outages early (spike in fallback usage)
- Optimize weights (if fallbacks are used 50% of the time, reweight providers)
- Cost allocation (track which provider actually fulfilled each request)
β Getting Started with Bifrost
To start building resilient AI applications:
- Deploy or access Bifrost (self-hosted or managed service)
- Add multiple providers to your infrastructure (OpenAI, Anthropic, etc.)
- Create a Virtual Key with weighted provider configs
- Replace your direct API calls with Bifrost-routed requests
- Monitor and optimize based on fallback metrics
Your application code doesn't need to change. Just point it at the Bifrost endpoint instead of OpenAI directly.
ποΈ Conclusion
Building resilient AI systems has traditionally meant building resilience into your application code. Automatic fallbacks flip this model: resilience is declared once, at the infrastructure layer, and automatically enforced for all requests.
π Resources:
- Bifrost GitHub: https://github.com/maximhq/bifrost
- Bifrost Docs: https://docs.getbifrost.ai
-
Bifrost CLI:
npx -y @maximhq/bifrost-cli

Top comments (1)
The examples use JavaScript, but you can also use the terminal.
Does Bifrost have a GitHub Repository?