Custom Providers
Custom Providers allow you to integrate AI providers that are not natively supported by AI Gateway. This feature enables you to use AI Gateway's observability, caching, rate limiting, and other features with any AI provider that has an HTTPS API endpoint.
- Internal AI models: Connect to your organization's self-hosted AI models
- Regional providers: Integrate with AI providers specific to your region
- Specialized models: Use domain-specific AI services not available through standard providers
- Custom endpoints: Route requests to your own AI infrastructure
- An active Cloudflare account with AI Gateway access
- A valid API key from your custom AI provider
- The HTTPS base URL for your provider's API
The API endpoints for creating, reading, updating, or deleting custom providers require authentication. You need to create a Cloudflare API token with the appropriate permissions.
To create an API token:
- Go to the Cloudflare dashboard API tokens page ↗
- Click Create Token
- Select Custom Token and add the following permissions:
AI Gateway - Edit
- Click Continue to summary and then Create Token
- Copy the token - you'll use it in the
Authorization: Bearer $CLOUDFLARE_API_TOKENheader
To create a new custom provider using the API:
-
Get your Account ID and Account Tag.
-
Send a
POSTrequest to create a new custom provider:
curl -X POST "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN" \ -H "Content-Type: application/json" \ -d '{ "name": "My Custom Provider", "slug": "some-provider", "base_url": "https://api.myprovider.com", "description": "Custom AI provider for internal models", "enable": true }'Required fields:
name(string): Display name for your providerslug(string): Unique identifier (alphanumeric with hyphens). Must be unique within your account.base_url(string): HTTPS URL for your provider's API endpoint. Must start withhttps://.
Optional fields:
description(string): Description of the providerlink(string): URL to provider documentationenable(boolean): Whether the provider is active (default:false)beta(boolean): Mark as beta feature (default:false)curl_example(string): Example cURL command for using the providerjs_example(string): Example JavaScript code for using the provider
Response:
{ "success": true, "result": { "id": "550e8400-e29b-41d4-a716-446655440000", "account_id": "abc123def456", "account_tag": "my-account", "name": "My Custom Provider", "slug": "some-provider", "base_url": "https://api.myprovider.com", "description": "Custom AI provider for internal models", "enable": true, "beta": false, "logo": "Base64 encoded SVG logo", "link": null, "curl_example": null, "js_example": null, "created_at": 1700000000, "modified_at": 1700000000 }}To create a new custom provider using the dashboard:
- Log in to the Cloudflare dashboard ↗ and select your account.
- Go to Compute & AI > AI Gateway > Custom Providers ↗.
- Select Add Custom Provider.
- Enter the following information:
- Provider Name: Display name for your provider
- Provider Slug: Unique identifier (alphanumeric with hyphens)
- Base URL: HTTPS URL for your provider's API endpoint (e.g.,
https://api.myprovider.com/v1)
- Select Save to create your custom provider.
Retrieve all custom providers with optional filtering and pagination:
curl "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN"Query parameters:
page(number): Page number (default:1)per_page(number): Items per page (default:20, max:100)enable(boolean): Filter by enabled statusbeta(boolean): Filter by beta statussearch(string): Search in id, name, or slug fieldsorder_by(string): Sort field and direction (default:"name ASC")
Examples:
List only enabled providers:
curl "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers?enable=true" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN"Search for specific providers:
curl "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers?search=custom" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN"Response:
{ "success": true, "result": [ { "id": "550e8400-e29b-41d4-a716-446655440000", "name": "My Custom Provider", "slug": "some-provider", "base_url": "https://api.myprovider.com", "enable": true, "created_at": 1700000000, "modified_at": 1700000000 } ], "result_info": { "page": 1, "per_page": 20, "total_count": 1, "total_pages": 1 }}To view all your custom providers:
- Log in to the Cloudflare dashboard ↗ and select your account.
- Go to Compute & AI > AI Gateway > Custom Providers ↗.
- You will see a list of all your custom providers with their names, slugs, base URLs, and status.
Retrieve details for a specific custom provider by its ID:
curl "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers/{provider_id}" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN"Response:
{ "success": true, "result": { "id": "550e8400-e29b-41d4-a716-446655440000", "account_id": "abc123def456", "account_tag": "my-account", "name": "My Custom Provider", "slug": "some-provider", "base_url": "https://api.myprovider.com", "description": "Custom AI provider for internal models", "enable": true, "beta": false, "logo": "Base64 encoded SVG logo", "link": "https://docs.myprovider.com", "curl_example": "curl -X POST https://api.myprovider.com/v1/chat ...", "js_example": "fetch('https://api.myprovider.com/v1/chat', {...})", "created_at": 1700000000, "modified_at": 1700000000 }}Update an existing custom provider. All fields are optional - only include the fields you want to change:
curl -X PATCH "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers/{provider_id}" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN" \ -H "Content-Type: application/json" \ -d '{ "name": "Updated Provider Name", "enable": true, "description": "Updated description" }'Updatable fields:
name(string): Provider display nameslug(string): Provider identifierbase_url(string): API endpoint URL (must be HTTPS)description(string): Provider descriptionlink(string): Documentation URLenable(boolean): Active statusbeta(boolean): Beta flagcurl_example(string): Example cURL commandjs_example(string): Example JavaScript code
Examples:
Enable a provider:
curl -X PATCH "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers/{provider_id}" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN" \ -H "Content-Type: application/json" \ -d '{"enable": true}'Update provider URL:
curl -X PATCH "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers/{provider_id}" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN" \ -H "Content-Type: application/json" \ -d '{"base_url": "https://api.newprovider.com"}'To update an existing custom provider:
- Log in to the Cloudflare dashboard ↗ and select your account.
- Go to Compute & AI > AI Gateway > Custom Providers ↗.
- Find the custom provider you want to update and select Edit.
- Update the fields you want to change (name, slug, base URL, etc.).
- Select Save to apply your changes.
Delete a custom provider:
curl -X DELETE "https://api.cloudflare.com/client/v4/accounts/$ACCOUNT_ID/ai-gateway/custom-providers/{provider_id}" \ -H "Authorization: Bearer $CLOUDFLARE_API_TOKEN"Response:
{ "success": true, "result": { "id": "550e8400-e29b-41d4-a716-446655440000", "name": "My Custom Provider", "slug": "some-provider" }}To delete a custom provider:
- Log in to the Cloudflare dashboard ↗ and select your account.
- Go to Compute & AI > AI Gateway > Custom Providers ↗.
- Find the custom provider you want to delete and select Delete.
- Confirm the deletion when prompted.
Once you've created a custom provider, you can route requests through AI Gateway using one of two approaches: the Unified API or the provider-specific endpoint. When referencing your custom provider with either approach, you must prefix the slug with custom-.
When AI Gateway receives a request for a custom provider, it constructs the upstream URL by combining the provider's configured base_url with the path that comes after custom-{slug}/ in the gateway URL.
The base_url field should contain only the root domain (or domain with a fixed prefix) of the provider's API. Any API-specific path segments (like /v1/chat/completions) go in the request URL, not in base_url.
The formula is:
Gateway URL: https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-{slug}/{provider-path}Upstream URL: {base_url}/{provider-path}Everything after custom-{slug}/ in your request URL is appended directly to the base_url to form the final upstream URL. This means {provider-path} can include multiple path segments, query parameters, or any path structure your provider requires.
Unified API (/compat) | Provider-specific endpoint | |
|---|---|---|
| Best for | Providers with OpenAI-compatible APIs | Providers with any API structure |
| Request format | Must follow the OpenAI /chat/completions schema | Uses the provider's native request format |
| Path control | Fixed to /compat/chat/completions | Full control over the upstream path |
| How to specify the provider | model field: custom-{slug}/{model-name} | URL path: /custom-{slug}/{path} |
Use the Unified API when your custom provider accepts the OpenAI-compatible /chat/completions request format. This is the simplest option and works well with OpenAI SDKs.
Use the provider-specific endpoint when your custom provider uses a non-standard API path or request format. This gives you full control over both the URL path and the request body sent to the upstream provider.
The Unified API sends requests to the provider's chat completions endpoint using the OpenAI-compatible format. Specify the model using the format custom-{slug}/{model-name}.
curl https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/compat/chat/completions \ -H "Authorization: Bearer $PROVIDER_API_KEY" \ -H "cf-aig-authorization: Bearer $CF_AIG_TOKEN" \ -H "Content-Type: application/json" \ -d '{ "model": "custom-some-provider/model-name", "messages": [{"role": "user", "content": "Hello!"}] }'The provider-specific endpoint gives you full control over the upstream path. Everything after custom-{slug}/ in the URL is appended to the base_url.
curl https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-some-provider/v1/chat/completions \ -H "Authorization: Bearer $PROVIDER_API_KEY" \ -H "cf-aig-authorization: Bearer $CF_AIG_TOKEN" \ -H "Content-Type: application/json" \ -d '{ "model": "model-name", "messages": [{"role": "user", "content": "Hello!"}] }'If base_url is https://api.myprovider.com, this request is proxied to: https://api.myprovider.com/v1/chat/completions
The following examples show how to configure base_url and construct request URLs for different types of providers.
Many providers follow the OpenAI convention of hosting their API at {domain}/v1/chat/completions.
Configuration:
slug:my-openai-compatbase_url:https://api.example-provider.com
Provider-specific endpoint:
curl https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-my-openai-compat/v1/chat/completions \ -H "Authorization: Bearer $PROVIDER_API_KEY" \ -H "Content-Type: application/json" \ -d '{ "model": "example-model", "messages": [{"role": "user", "content": "Hello!"}] }'URL mapping:
| Component | Value |
|---|---|
| Gateway URL | https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-my-openai-compat/v1/chat/completions |
base_url | https://api.example-provider.com |
| Provider path | /v1/chat/completions |
| Upstream URL | https://api.example-provider.com/v1/chat/completions |
Since this provider is OpenAI-compatible, you could also use the Unified API:
curl https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/compat/chat/completions \ -H "Authorization: Bearer $PROVIDER_API_KEY" \ -H "Content-Type: application/json" \ -d '{ "model": "custom-my-openai-compat/example-model", "messages": [{"role": "user", "content": "Hello!"}] }'Some providers use API paths that don't follow the /v1/ convention. For example, a provider whose chat endpoint is at https://api.custom-ai.com/api/coding/paas/v4/chat/completions.
Configuration:
slug:custom-aibase_url:https://api.custom-ai.com
Provider-specific endpoint:
curl https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-custom-ai/api/coding/paas/v4/chat/completions \ -H "Authorization: Bearer $PROVIDER_API_KEY" \ -H "Content-Type: application/json" \ -d '{ "model": "custom-ai-model", "messages": [{"role": "user", "content": "Hello!"}] }'URL mapping:
| Component | Value |
|---|---|
| Gateway URL | https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-custom-ai/api/coding/paas/v4/chat/completions |
base_url | https://api.custom-ai.com |
| Provider path | /api/coding/paas/v4/chat/completions |
| Upstream URL | https://api.custom-ai.com/api/coding/paas/v4/chat/completions |
If you host your own model behind a reverse proxy or on a platform that adds a path prefix, include only the fixed prefix portion in base_url if all your endpoints share it. Otherwise, keep base_url as just the domain.
Configuration (domain-only base_url):
slug:internal-llmbase_url:https://ml.internal.example.com
Provider-specific endpoint:
curl https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-internal-llm/serving/models/my-model:predict \ -H "Authorization: Bearer $INTERNAL_API_KEY" \ -H "Content-Type: application/json" \ -d '{ "instances": [{"prompt": "Summarize the following text:"}] }'URL mapping:
| Component | Value |
|---|---|
| Gateway URL | https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-internal-llm/serving/models/my-model:predict |
base_url | https://ml.internal.example.com |
| Provider path | /serving/models/my-model:predict |
| Upstream URL | https://ml.internal.example.com/serving/models/my-model:predict |
When using the OpenAI SDK to connect to a custom provider through AI Gateway, set the SDK's base_url to the gateway's provider-specific endpoint path (up to and including the API version prefix that your provider expects).
Configuration:
slug:alt-providerbase_url:https://api.alt-provider.com
Python (OpenAI SDK):
from openai import OpenAI
client = OpenAI( api_key="your-provider-api-key", base_url="https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-alt-provider/v1", default_headers={ "cf-aig-authorization": "Bearer {cf_aig_token}", },)
# The SDK appends /chat/completions to the base_url automatically.# Final upstream URL: https://api.alt-provider.com/v1/chat/completionsresponse = client.chat.completions.create( model="alt-model-v2", messages=[{"role": "user", "content": "Hello!"}],)URL mapping:
| Component | Value |
|---|---|
SDK base_url | https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-alt-provider/v1 |
| SDK appends | /chat/completions |
| Full gateway URL | https://gateway.ai.cloudflare.com/v1/{account_id}/{gateway_id}/custom-alt-provider/v1/chat/completions |
Provider base_url | https://api.alt-provider.com |
| Provider path | /v1/chat/completions |
| Upstream URL | https://api.alt-provider.com/v1/chat/completions |
{ "success": false, "errors": [ { "code": 1003, "message": "A custom provider with this slug already exists", "path": ["body", "slug"] } ]}Each custom provider slug must be unique within your account. Choose a different slug or update the existing provider.
{ "success": false, "errors": [ { "code": 1004, "message": "Custom Provider not found" } ]}The specified provider ID does not exist or you don't have access to it. Verify the provider ID and your authentication credentials.
{ "success": false, "errors": [ { "code": 1002, "message": "base_url must be a valid HTTPS URL starting with https://", "path": ["body", "base_url"] } ]}The base_url field must be a valid HTTPS URL. HTTP URLs are not supported for security reasons.
If you receive a 404 from the upstream provider, the most common cause is an incorrect path mapping. Verify that:
- Your
base_urlis set to the provider's root domain (for example,https://api.provider.com) rather than including API path segments. - Your request URL includes the full API path after
custom-{slug}/. For example, if the upstream endpoint ishttps://api.provider.com/api/v2/chat, your gateway URL should end in/custom-{slug}/api/v2/chat. - There is no duplicate or missing path segment. A common mistake is including
/v1in bothbase_urland the request path, resulting in the upstream receiving/v1/v1/chat/completions.
- Use descriptive slugs: Choose slugs that clearly identify the provider (e.g.,
internal-gpt,regional-ai) - Document your integrations: Use the
curl_exampleandjs_examplefields to provide usage examples - Enable gradually: Test with
enable: falsebefore making the provider active - Monitor usage: Use AI Gateway's analytics to track requests to your custom providers
- Secure your endpoints: Ensure your custom provider's base URL implements proper authentication and authorization
- Use BYOK: Store provider API keys securely using BYOK instead of including them in every request
- Custom providers are account-specific and not shared across Cloudflare accounts
- The
base_urlmust use HTTPS (HTTP is not supported) - Provider slugs must be unique within each account
- Cache and rate limiting settings apply globally to the provider, not per-model