DEV Community

Cover image for The Concept of Automatic Fallbacks And How Bifrost Implements It
Anthony Max
Anthony Max Subscriber

Posted on

The Concept of Automatic Fallbacks And How Bifrost Implements It

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.

Bifrost


πŸ’» 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!" }]
});
Enter fullscreen mode Exit fullscreen mode

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({...});
}
Enter fullscreen mode Exit fullscreen mode

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
      }
    ]
  }'
Enter fullscreen mode Exit fullscreen mode

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!' }]
  })
});
Enter fullscreen mode Exit fullscreen mode

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
Enter fullscreen mode Exit fullscreen mode

Key behaviors:

  1. Weighted Selection: Your primary request goes to the provider with the highest weight
  2. Automatic Retry: If that provider fails, Bifrost automatically retries the next provider in line
  3. Weight-Ordered Chain: Fallbacks are sorted by weight providers get priority
  4. 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
    }
  ]
}
Enter fullscreen mode Exit fullscreen mode

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)
Enter fullscreen mode Exit fullscreen mode

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
  })
});
Enter fullscreen mode Exit fullscreen mode

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
    }
  ]
}
Enter fullscreen mode Exit fullscreen mode

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 }
  ]
}
Enter fullscreen mode Exit fullscreen mode

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 }
  ]
}
Enter fullscreen mode Exit fullscreen mode

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 }
  ]
}
Enter fullscreen mode Exit fullscreen mode

βš™οΈ 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
Enter fullscreen mode Exit fullscreen mode

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 fallbacks array
  • βœ“ 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
Enter fullscreen mode Exit fullscreen mode

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
}
Enter fullscreen mode Exit fullscreen mode

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:

  1. Deploy or access Bifrost (self-hosted or managed service)
  2. Add multiple providers to your infrastructure (OpenAI, Anthropic, etc.)
  3. Create a Virtual Key with weighted provider configs
  4. Replace your direct API calls with Bifrost-routed requests
  5. 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:

πŸ’Ž Star Bifrost β˜†

Top comments (1)

Collapse
 
anthonymax profile image
Anthony Max • Edited

The examples use JavaScript, but you can also use the terminal.

const response = await fetch('http://localhost:..., {...} )
Enter fullscreen mode Exit fullscreen mode
Collapse
 
marvin_p profile image
Marvin Poole • Edited

Does Bifrost have a GitHub Repository?