Skip to main content

Overview

Bifrost provides three levels of tool filtering to control which MCP tools are available:
  1. Client Configuration - Set which tools a client can execute (tools_to_execute)
  2. Request Headers - Filter tools per-request via HTTP headers or context
  3. Virtual Key Configuration - Control tools per-VK (Gateway only)
These levels stack: a tool must pass all applicable filters to be available.

Level 1: Client Configuration

The tools_to_execute field on each MCP client config defines the baseline of available tools.

Semantics

ValueBehavior
["*"]All tools from this client are available
[] or omittedNo tools available (deny-by-default)
["tool1", "tool2"]Only specified tools are available

Configuration

curl -X POST http://localhost:8080/api/mcp/client \
  -H "Content-Type: application/json" \
  -d '{
    "name": "filesystem",
    "connection_type": "stdio",
    "stdio_config": {
      "command": "npx",
      "args": ["-y", "@anthropic/mcp-filesystem"]
    },
    "tools_to_execute": ["read_file", "list_directory"]
  }'

Level 2: Request-Level Filtering

Filter tools dynamically on a per-request basis using headers (Gateway) or context values (SDK).

Available Filters

FilterPurpose
mcp-include-clientsOnly include tools from specified clients
mcp-include-toolsOnly include specified tools (format: clientName/toolName)

Gateway Headers

# Include only specific clients
curl -X POST http://localhost:8080/v1/chat/completions \
  -H "x-bf-mcp-include-clients: filesystem,web_search" \
  -d '...'

# Include only specific tools
curl -X POST http://localhost:8080/v1/chat/completions \
  -H "x-bf-mcp-include-tools: filesystem/read_file,web_search/search" \
  -d '...'

# Include all tools from one client, specific tools from another
curl -X POST http://localhost:8080/v1/chat/completions \
  -H "x-bf-mcp-include-tools: filesystem/*,web_search/search" \
  -d '...'

Go SDK Context Values

// Include only specific clients
ctx := context.WithValue(context.Background(),
    schemas.BifrostContextKey("mcp-include-clients"),
    []string{"filesystem", "web_search"})

// Include only specific tools
ctx = context.WithValue(ctx,
    schemas.BifrostContextKey("mcp-include-tools"),
    []string{"filesystem/read_file", "web_search/search"})

// Wildcard for all tools from a client
ctx = context.WithValue(ctx,
    schemas.BifrostContextKey("mcp-include-tools"),
    []string{"filesystem/*", "web_search/search"})

response, err := client.ChatCompletionRequest(ctx, request)

Wildcard Support

PatternMeaning
* (in include-clients)Include all clients
clientName/* (in include-tools)Include all tools from that client
clientName/toolNameInclude specific tool

Level 3: Virtual Key Filtering (Gateway Only)

Virtual Keys can have their own MCP tool access configuration, which takes precedence over request-level headers.
When a Virtual Key has MCP configurations, it generates the x-bf-mcp-include-tools header automatically, overriding any manually sent header.

Configuration

  1. Navigate to Virtual Keys in the governance section
  2. Create or edit a Virtual Key
  3. In MCP Client Configurations, add the clients and tools this VK can access
Virtual Key MCP Configuration

Virtual Key MCP Config Semantics

ConfigurationResult
tools_to_execute: ["*"]All tools from this client
tools_to_execute: []No tools from this client
tools_to_execute: ["a", "b"]Only specified tools
Client not configuredAll tools blocked from that client
Learn more in MCP Tool Filtering for Virtual Keys.

Filtering Logic

How Filters Combine

  1. Client config is the baseline (must include the tool)
  2. Request filters further narrow down (if specified)
  3. VK filters override request filters (if VK has MCP configs)

Example Scenario

Setup:
  • Client filesystem has tools_to_execute: ["read_file", "write_file", "delete_file"]
  • Virtual Key prod-key has mcp_configs: [{ mcp_client_name: "filesystem", tools_to_execute: ["read_file"] }]
Request with prod-key:
curl -X POST http://localhost:8080/v1/chat/completions \
  -H "Authorization: Bearer vk_prod_key" \
  -H "x-bf-mcp-include-tools: filesystem/write_file" \  # This is IGNORED
  -d '...'
Result: Only read_file is available (VK config overrides request header) Request without VK (if allowed):
curl -X POST http://localhost:8080/v1/chat/completions \
  -H "x-bf-mcp-include-tools: filesystem/write_file" \
  -d '...'
Result: Only write_file is available (request header applies)

Common Patterns

Read-Only Access

Allow only read operations:
{
  "tools_to_execute": ["read_file", "list_directory", "get_file_info"]
}

Environment-Based Filtering

Use different VKs for different environments:
{
  "virtual_keys": [
    {
      "name": "development",
      "mcp_configs": [
        { "mcp_client_name": "filesystem", "tools_to_execute": ["*"] },
        { "mcp_client_name": "database", "tools_to_execute": ["*"] }
      ]
    },
    {
      "name": "production",
      "mcp_configs": [
        { "mcp_client_name": "filesystem", "tools_to_execute": ["read_file"] },
        { "mcp_client_name": "database", "tools_to_execute": ["query"] }
      ]
    }
  ]
}

Per-User Tool Access

Create VKs for different user roles:
{
  "virtual_keys": [
    {
      "name": "viewer-role",
      "mcp_configs": [
        { "mcp_client_name": "documents", "tools_to_execute": ["view", "search"] }
      ]
    },
    {
      "name": "editor-role",
      "mcp_configs": [
        { "mcp_client_name": "documents", "tools_to_execute": ["view", "search", "edit", "create"] }
      ]
    },
    {
      "name": "admin-role",
      "mcp_configs": [
        { "mcp_client_name": "documents", "tools_to_execute": ["*"] }
      ]
    }
  ]
}

Advanced: Context-Based Filtering

For SDK users, filtering can be applied at the context level, enabling per-request tool customization:

Go SDK Context Filtering

import (
    "context"
    "github.com/maximhq/bifrost/core/schemas"
)

// Filter to specific clients
ctx := context.WithValue(
    context.Background(),
    schemas.BifrostContextKey("mcp-include-clients"),
    []string{"filesystem", "web_search"},
)

// Or filter to specific tools
ctx = context.WithValue(
    ctx,
    schemas.BifrostContextKey("mcp-include-tools"),
    []string{"filesystem/read_file", "web_search/search"},
)

// Request will only see filtered tools
response, _ := client.ChatCompletionRequest(ctx, request)

Filter Precedence

When multiple filters apply, they combine as an intersection (AND logic):
Client Config Tools ∩ Request Filters ∩ VK Filters = Available Tools
Example:
  • Client config allows: [read_file, write_file, delete_file]
  • Request header specifies: [read_file, write_file]
  • VK config restricts to: [read_file]
  • Result: Only [read_file] available

Debugging Tool Availability

Check Available Tools

Gateway API:
curl http://localhost:8080/api/mcp/clients
Response shows tools per client:
[
  {
    "config": { "name": "filesystem", "tools_to_execute": ["read_file", "write_file"] },
    "tools": [
      { "name": "read_file", "description": "Read file contents" },
      { "name": "write_file", "description": "Write to file" }
    ],
    "state": "connected"
  }
]

Check What LLM Receives

The tools included in a chat request depend on all active filters. To see what tools are available for a specific request, check the request body sent to the LLM provider in your logs or observability platform.

Next Steps