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': '*'
相关推荐
海棠AI实验室4 天前
第 6 篇:访问控制与零信任策略
mac·cloudflare
海棠AI实验室10 天前
第 5 篇:Cloudflare Tunnel 多服务路由模板
cloudflare
Benny的老巢14 天前
Wrangler CLI 工具完整使用指南
cloudflare·workers·wrangler·cloudflare cli·cloudflare 部署
Java程序员-小白14 天前
Sa-Token过滤器引发的CORS误判问题
vue.js·elementui·axios·cors
tang7778915 天前
爬虫如何绕过绕过“5秒盾”Cloudflare:从浏览器指纹模拟到Rust求解之不完全指南
开发语言·爬虫·rust·cloudflare
大飞哥~BigFei15 天前
新版chrome浏览器安全限制及解决办法
java·前端·chrome·安全·跨域
我的golang之路果然有问题19 天前
实习中遇到的 CORS 同源策略自己的理解分析
前端·javascript·vue·reactjs·同源策略·cors
Benny的老巢20 天前
Cloudflare Workers 实现 Resend 邮件发送接口
cloudflare·邮件服务·workers·resend服务
Benny的老巢21 天前
Cloudflare Workers 接口服务能力详解
cloudflare·服务端·workers·接口服务·d1数据库