Cloudflare Workers CORS 跨域问题排查与解决

Cloudflare Workers CORS 跨域问题排查与解决表单失败的根因是:Worker 没有正确处理 CORS 预检请求(OPTIONS)。只要把 CORS + OPTIONS 补齐,fetch failed 会立刻消失。

日志逐条解释

这个 CSP 报错【不是你的问题】

text 复制代码
Executing inline script violates the following Content Security Policy...
chrome-extension://a05f635f...

这是浏览器扩展(插件)注入脚本被 CSP 拦截。

  • 来自 chrome-extension://...
  • 和你的 Astro / Worker / 表单 无关
  • 不影响接口请求

可以完全忽略。

这个才是致命错误【真正原因】

text 复制代码
Access to fetch at 'https://xxx.workers.dev/api/contact'
from origin 'http://localhost:4321'
has been blocked by CORS policy:

Response to preflight request doesn't pass access control check:
No 'Access-Control-Allow-Origin' header

关键信息:

  • 浏览器发起了 OPTIONS 预检请求
  • Worker 没有返回 CORS 头
  • 浏览器直接拦截 → ERR_FAILED

为什么会有 OPTIONS 预检?

因为你满足了以下任意条件(你全中了):

  • Content-Type: application/json
  • POST 请求
  • 跨域(localhost → workers.dev)

浏览器 一定会先发 OPTIONS

你现在的 Worker 一定缺了这段逻辑

js 复制代码
if (request.method === 'OPTIONS') {
  // 你现在没有正确处理
}

直接给你一份生产可用的 Worker 模板

你可以整体替换当前 Worker 代码。

js 复制代码
const corsHeaders = {
  'Access-Control-Allow-Origin': 'http://localhost:4321', // 本地开发
  // 生产环境可以换成你的域名
  // 'https://yourdomain.com'

  'Access-Control-Allow-Methods': 'POST, OPTIONS',
  'Access-Control-Allow-Headers': 'Content-Type',
}

export default {
  async fetch(request, env) {
    // 1️⃣ 处理 CORS 预检
    if (request.method === 'OPTIONS') {
      return new Response(null, {
        status: 204,
        headers: corsHeaders,
      })
    }

    // 2️⃣ 只允许 POST
    if (request.method !== 'POST') {
      return new Response('Method Not Allowed', {
        status: 405,
        headers: corsHeaders,
      })
    }

    // 3️⃣ 解析数据
    const { name, email, phone, message } = await request.json()

    if (!name || !email || !message) {
      return new Response(
        JSON.stringify({ error: 'Missing fields' }),
        {
          status: 400,
          headers: {
            ...corsHeaders,
            'Content-Type': 'application/json',
          },
        }
      )
    }

    // 4️⃣ 调用 Resend
    const res = await fetch('https://api.resend.com/emails', {
      method: 'POST',
      headers: {
        'Authorization': `Bearer ${env.RESEND_API_KEY}`,
        'Content-Type': 'application/json',
      },
      body: JSON.stringify({
        from: 'Website Contact <no-reply@YOUR_VERIFIED_DOMAIN.com>',
        to: ['your@email.com'],
        subject: 'New Contact Message',
        html: `
          <p><b>Name:</b> ${name}</p>
          <p><b>Email:</b> ${email}</p>
          <p><b>Phone:</b> ${phone || '-'}</p>
          <p><b>Message:</b> ${message}</p>
        `,
      }),
    })

    if (!res.ok) {
      const err = await res.text()
      return new Response(err, {
        status: 500,
        headers: corsHeaders,
      })
    }

    // 5️⃣ 成功返回
    return new Response(
      JSON.stringify({ success: true }),
      {
        headers: {
          ...corsHeaders,
          'Content-Type': 'application/json',
        },
      }
    )
  },
}

Astro 前端这边你不用改(基本是对的)

js 复制代码
await fetch('https://icy-truth-83e1.bennyxqg.workers.dev/api/contact', {
  method: 'POST',
  headers: {
    'Content-Type': 'application/json',
  },
  body: JSON.stringify(formData),
})

没问题,CORS 是后端的责任。

常见优化(等你跑通后再做)

允许多个来源(开发 + 生产)

js 复制代码
const origin = request.headers.get('Origin')
const allowed = ['http://localhost:4321', 'https://yourdomain.com']

if (allowed.includes(origin)) {
  corsHeaders['Access-Control-Allow-Origin'] = origin
}

临时调试(不推荐生产)

js 复制代码
'Access-Control-Allow-Origin': '*'
相关推荐
炸裂狸花猫4 天前
开源域名代理与流量限制方案 - Cloudflare + Ingress + 自签名证书
运维·云原生·容器·kubernetes·cloudflare·waf·免费域名证书
哆啦code梦5 天前
网关统一配置CORS:现代架构最佳实践
cors
Irene19916 天前
前端开发中支持跨域的HTML标签和属性(附:前端常见缓存机制跨域性和实际跨域需求总结)
cors·html标签·前端跨域·缓存方式
曲幽7 天前
FastAPI部署实战:聊聊CORS跨域那些坑
python·fastapi·web·cors·corsmiddleware·origins
曲幽8 天前
FastAPI部署中间件实战:从CORS到自定义,让你的API更健壮
python·fastapi·web·cors·starlette·middleware·call_next
海棠AI实验室14 天前
第 6 篇:访问控制与零信任策略
mac·cloudflare
海棠AI实验室21 天前
第 5 篇:Cloudflare Tunnel 多服务路由模板
cloudflare
Benny的老巢25 天前
Wrangler CLI 工具完整使用指南
cloudflare·workers·wrangler·cloudflare cli·cloudflare 部署
Java程序员-小白25 天前
Sa-Token过滤器引发的CORS误判问题
vue.js·elementui·axios·cors