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': '*'
相关推荐
LongtengGensSupreme19 小时前
后端设置了跨域但是还是提示跨域问题,原因是这里有两个独立的安全策略在起作用:Chrome和Edge浏览器安全策略强制修改方案
前端·chrome·edge·浏览器·跨域
大佐不会说日语~2 天前
使用 Cloudflare平台 + Docker + Nginx 完成网站 HTTPS 部署实战记录
nginx·docker·https·部署·cloudflare
wanfeng_092 天前
nextjs cloudflare 踩坑日记
nextjs·cloudflare
bl4ckpe4ch3 天前
用可复现实验直观理解 CORS 与 CSRF 的区别与联系
前端·web安全·网络安全·csrf·cors
源代码•宸3 天前
goframe框架签到系统项目开发(补签逻辑实现、编写Lua脚本实现断签提醒功能、简历示例)
数据库·后端·中间件·go·lua·跨域·refreshtoken
勇气要爆发8 天前
跨域 (CORS) 原理:浏览器的“尽职保镖”
网络安全·跨域·cors
钦拆大仁9 天前
聊一聊跨域错误CORS
web·跨域
bkspiderx10 天前
HTTP跨域问题深度解析:4种实用解决方案与场景适配
网络·http·nginx反向代理·cors·跨域资源共享·http跨域问题
戎码江湖13 天前
前端跨域状态共享的实现方案
跨域·跨域共享·cookie跨域共享