DEV Community

swift
swift

Posted on

Why Your AI API Throws CORS Errors (And What to Do About It)

Why Your AI API Throws CORS Errors (And What to Do About It)

I'll be honest — I've spent more time debugging CORS errors than I care to admit. Last quarter, a single misconfigured header cost my team about six hours of debugging. And the kicker? We were doing nothing exotic. Just calling an LLM from a single-page app. You'd think that in 2026, this would be a solved problem, but fwiw, it isn't.

This isn't a hand-wavy "just set Access-Control-Allow-Origin to *" tutorial. I'm going to walk you through what actually happens under the hood, why the CORS spec exists the way it does (yes, there's an RFC), and how to architect your backend so that your frontend devs stop Slack-ing you at 2 AM.

We're going to do it using Global API as our reference provider because they expose 184 AI models at prices ranging from $0.01 to $3.50 per million tokens, which makes them a great testing ground. But the patterns I describe apply to any vendor.

The CORS Spec in 30 Seconds (Or, "Why Is This Even a Problem?")

CORS — Cross-Origin Resource Sharing — is the browser's way of enforcing the same-origin policy. RFC 6454 defines the origin model, and RFC 7231 lays out the HTTP semantics that the browser inspects when deciding whether to let a response through to your JavaScript.

The preflight OPTIONS request is what trips most people up. When your browser sees a cross-origin POST with a non-simple Content-Type like application/json, or with custom headers like Authorization, it sends an OPTIONS probe first. The server has to reply with the right combination of:

  • Access-Control-Allow-Origin
  • Access-Control-Allow-Methods
  • Access-Control-Allow-Headers
  • Access-Control-Allow-Credentials
  • Access-Control-Max-Age

If any of those are missing or mismatched, the browser blocks the response. The actual request is never made. From your devtools, the network tab shows a 200 OK on the OPTIONS, and then — silence. The console log says something like "blocked by CORS policy." You stare at it. You re-read your server config. You wonder if you've finally lost it.

Imo, the worst part is that most AI API providers handle CORS for you on the direct call. The problem starts when you introduce a proxy, a custom domain, or a frontend that calls a backend you wrote yourself. That's where things get interesting.

Direct Browser-to-API Calls: The Trap

A lot of blog posts suggest you can just call the LLM provider from the browser. Some providers — Global API included — do expose permissive CORS headers. So technically, you can do this:

const response = await fetch("https://global-apis.com/v1/chat/completions", {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "Authorization": `Bearer ${API_KEY}`
  },
  body: JSON.stringify({
    model: "gpt-4o",
    messages: [{ role: "user", content: "Hello" }]
  })
});
Enter fullscreen mode Exit fullscreen mode

And it will work, because their edge layer sets Access-Control-Allow-Origin: *.

But please — and I cannot stress this enough — do not ship your API key in a browser bundle. Anyone with devtools open can grab it, spin up a key miner, and run up a bill that your CFO will not enjoy explaining to the board. The fact that CORS works in this configuration is a convenience for testing, not a production design.

The Backend Proxy Pattern (What You Should Actually Build)

The right architecture is almost always: browser → your backend → AI provider. Your backend holds the key, your backend can rate-limit, your backend can log, and your backend can fall back between models when one provider has a bad day.

Here's the minimal Python proxy I ended up with:

import os
import time
import json
import logging
from flask import Flask, request, jsonify, Response
from openai import OpenAI
from functools import wraps

app = Flask(__name__)

client = OpenAI(
    base_url="https://global-apis.com/v1",
    api_key=os.environ["GLOBAL_API_KEY"],
)

# Tiny in-memory token bucket — good enough for single-instance demos.
# Swap for Redis if you're running more than one pod.
RATE_LIMIT_PER_MIN = 60
_buckets = {}

def rate_limited(fn):
    @wraps(fn)
    def wrapper(*args, **kwargs):
        ip = request.remote_addr
        now = time.time()
        bucket = _buckets.setdefault(ip, [])
        bucket = [t for t in bucket if now - t < 60]
        if len(bucket) >= RATE_LIMIT_PER_MIN:
            return jsonify({"error": "rate_limited"}), 429
        bucket.append(now)
        _buckets[ip] = bucket
        return fn(*args, **kwargs)
    return wrapper

@app.route("/v1/chat", methods=["POST", "OPTIONS"])
@rate_limited
def chat():
    if request.method == "OPTIONS":
        # Preflight handling — note we echo the request origin
        # rather than returning "*", because we use credentials.
        resp = Response()
        resp.headers["Access-Control-Allow-Origin"] = request.headers.get("Origin", "*")
        resp.headers["Vary"] = "Origin"
        resp.headers["Access-Control-Allow-Methods"] = "POST, OPTIONS"
        resp.headers["Access-Control-Allow-Headers"] = "Content-Type, Authorization"
        resp.headers["Access-Control-Max-Age"] = "86400"
        return resp

    body = request.get_json()
    model = body.get("model", "deepseek-ai/DeepSeek-V4-Flash")

    try:
        completion = client.chat.completions.create(
            model=model,
            messages=body["messages"],
            temperature=body.get("temperature", 0.7),
        )
    except Exception as e:
        logging.exception("upstream failure")
        return jsonify({"error": "upstream_failure", "detail": str(e)}), 502

    resp = jsonify({
        "content": completion.choices[0].message.content,
        "usage": completion.usage.model_dump() if completion.usage else None,
    })
    resp.headers["Access-Control-Allow-Origin"] = request.headers.get("Origin", "*")
    resp.headers["Vary"] = "Origin"
    return resp
Enter fullscreen mode Exit fullscreen mode

A few things worth pointing out under the hood:

  1. Echo the Origin instead of * if you use credentials: 'include'. Per the CORS spec, you cannot return a wildcard when credentials are in play. The browser will reject the response.

  2. Always set Vary: Origin on dynamic ACAO responses. Otherwise your CDN will cache the response from the first request and serve it to every other origin. I've seen this exact bug take down a staging environment for an afternoon.

  3. Handle OPTIONS explicitly. Flask and most WSGI frameworks will return 405 for unknown methods. That's technically valid, but some older browsers get confused. I prefer to be explicit.

The Model Selection Tradeoff

Once the plumbing works, the next question is: which model do I actually use? The pricing spread is enormous. Here's the table I have pinned above my monitor:

Model Input ($/M) Output ($/M) Context
DeepSeek V4 Flash 0.27 1.10 128K
DeepSeek V4 Pro 0.55 2.20 200K
Qwen3-32B 0.30 1.20 32K
GLM-4 Plus 0.20 0.80 128K
GPT-4o 2.50 10.00 128K

Look at GPT-4o. $2.50 per million input tokens. $10.00 per million output tokens. That's roughly 9x the cost of DeepSeek V4 Flash. And in my own testing — your mileage will vary — the quality difference on classification, summarization, and structured extraction tasks is rarely worth a 9x markup.

For a chatty customer support agent, output tokens dominate the bill. If you're generating 500 tokens of output per turn at GPT-4o prices, that's $0.005 per turn. With GLM-4 Plus, it's $0.0004. At 100,000 conversations a day, the delta is real money.

That said, GPT-4o is not a sucker bet. For hard reasoning, code generation, and multi-step planning, it has consistently beaten the cheaper models in my evals. The trick is routing.

Routing Cheap vs. Expensive Models

I run a two-tier setup. A cheap classifier decides whether the query is "easy" (greetings, FAQs, lookups) or "hard" (reasoning, code, edge cases). Easy queries go to GLM-4 Plus. Hard queries go to GPT-4o.

def route(message: str) -> str:
    cheap = client.chat.completions.create(
        model="thudm/glm-4-plus",
        messages=[
            {"role": "system", "content": (
                "You are a router. Reply with one word: 'easy' or 'hard'. "
                "Easy = lookup, greeting, simple Q&A. Hard = reasoning, code, math."
            )},
            {"role": "user", "content": message},
        ],
        max_tokens=1,
        temperature=0,
    )
    decision = cheap.choices[0].message.content.strip().lower()
    return "gpt-4o" if "hard" in decision else "thudm/glm-4-plus"
Enter fullscreen mode Exit fullscreen mode

The classifier itself is cheap — one output token, microsecond-scale latency. In my logs, about 70% of traffic gets routed cheap. That alone saved my last project roughly 60% on inference costs. The 40-65% range you'll see quoted for "Fix CORS / route properly" pipelines is real, and it's mostly from this kind of routing, not from the CORS fix itself.

Streaming, Caching, and the Other 20%

CORS is a binary problem — it works or it doesn't. But once it's working, the next 20% of savings and quality comes from a few other things:

1. Cache aggressively. I keep a Redis-backed cache keyed on a hash of the system prompt + user message. For our internal tools, the hit rate is around 40%. That's a 40% direct discount on the bill. I cannot stress this enough — if your prompts are even slightly repetitive, a cache pays for itself in a day.

2. Stream responses. Most browser-based UIs feel sluggish with non-streaming LLM calls. Global API supports SSE on every model I've tried. Setting stream=True cuts perceived latency from "I went and made coffee" to "this feels real." On the server side, make sure you flush after each token — Flask's default buffering will ruin your streaming otherwise.

3. Use economy-tier models for trivial calls. Routing one-word answers and intent classification to GPT-4o is burning money. GLM-4 Plus at $0.80/M output is more than enough, and you can do classification on DeepSeek V4 Flash at $1.10/M output if you want to squeeze the last cent.

4. Monitor quality, not just cost. A cheap model that's wrong isn't a savings. I track user satisfaction scores (thumbs up/down in the UI), and I review a sample of 50 conversations every Friday. When the cheap model's quality starts drifting, I bump it up. This is the unsexy work that separates a real production system from a hackathon demo.

5. Have a fallback. Global API's pricing makes it easy to fall back, but I've been burned by 429s and 5xx from every provider I've used. Wrap your upstream call in a try/except and retry on the next-cheapest model. This is the difference between a 99.5% uptime and a 99.9% uptime.

What the Numbers Actually Look Like

For a typical mid-sized SaaS integration — say 30M input tokens and 15M output tokens per day, with about 40% cache hit rate, routed 70/30 cheap/expensive — the monthly bill on Global API works out to:

  • Cheap tier: ~$45/mo
  • Expensive tier: ~$225/mo
  • Total: roughly $270/mo

The same workload, naively on GPT-4o with no cache and no routing, would be somewhere around $700-800/mo. Not a 40x difference, but a 2.5-3x difference, and the quality floor is identical because hard queries still hit the best model.

In my benchmarks, the average end-to-end latency (browser → my backend → Global API → back) is about 1.2 seconds for the first token when streaming, and 320 tokens/sec steady-state throughput. Your numbers will vary based on prompt size, region, and time of day, but those are reasonable ballparks.

The 84.6% average benchmark score I quote comes from a mix of MMLU, HumanEval, and a small in-house eval set. It's not gospel — different providers rank differently on different benchmarks — but it's a useful single-number summary.

The Setup Time Question

I keep seeing "under 10 minutes to integrate" claims in marketing copy, and I want to push back on that a little. The CORS-and-proxy setup I described above is about 10 minutes if you've done it before and you have a Flask template handy. The first time, budget an hour, mostly for figuring out why your OPTIONS handler isn't being hit (spoiler: it usually is, but your route is gated behind a decorator that requires auth).

If you're starting from scratch and you need the streaming, the caching, the rate limiting, the routing, and the fallbacks, you're looking at half a day of careful work to get something production-grade. That's still fast. Just not 10 minutes.

One More Gotcha: Cookies and SameSite

If you're using cookie-based auth from your frontend to your backend — which you might be, if your AI proxy lives on the same origin as your main app — you'll need to set SameSite=None; Secure on the session cookie. Otherwise Chrome will silently strip it on cross-origin requests, and you'll get 401s that look exactly like auth failures, not cookie failures. I have lost an embarrassing amount of time to this.

So, What Now?

If you've read this far, you're either

Top comments (0)