htmx-Request header field hx-target/hx-request/hx-current-url is not allowed by Access-Control-Allow

问题描述

体验htmx hx-get 属性时,控制台报错:Access to XMLHttpRequest at 'http://localhost:8080/clicked' from origin 'http://127.0.0.1:5500' has been blocked by CORS policy: Request header field hx-request is not allowed by Access-Control-Allow-Headers in preflight response.

或者存在以下描述:Request header field hx-target/hx-request/hx-current-url is not allowed by Access-Control-Allow-Headers in preflight response.

发生原因

说明通过htmx中利用hx-get发送请求时,请求头携带了hx-target/hx-request/hx-current-url,如下图所示。但后端并未允许携带额外请求头的请求,所以请求失败。

hx-get

解决方法

方法1:后端接口新增允许的请求头Header

我用的是express,其他后端也类似,通过中间件增加允许的请求头:res.header('Access-Control-Allow-Headers', 'Content-Type, hx-target, hx-current-url, hx-request');完成示例代码如下:

const express = require('express');

const router = express.Router();
const app = express();

// 设置允许跨域访问的域名和端口
const allowedOrigins = [
    'http://127.0.0.1:5500',
    'http://localhost:5500'
];

// 中间件函数,用于检查请求来源是否在允许的域名列表中
const allowCrossDomain = function(req, res, next) {
    const origin = req.headers.origin;
    if (allowedOrigins.includes(origin)) {
        res.setHeader('Access-Control-Allow-Origin', origin);
    }

    res.header('Access-Control-Allow-Methods', 'GET, POST, PUT');
    res.header('Access-Control-Allow-Headers', 'Content-Type, hx-target, hx-current-url, hx-request');
    next();
};

// 使用自定义的跨域中间件
app.use(allowCrossDomain);

方法2:htmx 发起请求时改写请求头

对于第三方API,你可能控制不了它到底允许哪些请求头header,那么这种方式更合适。

由于htmx发起请求时会额外增加hx-target/hx-request/hx-current-url请求头,那么思路打开:监听htmx:configRequest事件,发起请求的header先将header置为空,在加上通用的Content-Type请求头,根据请求响应类型设置对应的值,我这儿返回的是html文本,所以设置为text/html

<script>
    document.body.addEventListener('htmx:configRequest', function (event) {
        event.detail.headers = ''
        event.detail.headers['Content-Type'] = "text/html; charset=UTF-8"
    });
</script>

可以看到请求能正常返回了,下面是完整的请求header,可以看到已经不包含hx-target/hx-request/hx-current-url

htmx

其他方式

在htmx的hx-on增加如下代码:

<button hx-get="http://localhost:8080/clicked" hx-trigger="click" hx-target="#parent-div" hx-swap="outerHTML"
    hx-on="htmx:configRequest: event.detail.headers=''; event.detail.headers['Content-Type']='application/x-www-form-urlencoded; charset=UTF-8'"
    >

扩展阅读:Content-Type 含义、常见类型

Content-Type 含义

在请求头和响应头header中携带,用于指示资源的 MIME 类型(media type)。比如application/x-www-form-urlencoded; charset=UTF-8,通常由两部分组成:媒体类型和字符集。

  • 媒体类型:制定要传输的数据类型,例如文本、图像、音频或视频,分别为 text/plain、audio/aac、video/x-msvideo。

  • 字符集:指定用于编码数据的字符集,常见的为 utf-8。

Content-Type 常见类型

文档类型媒体类型(MIME 类型)
Windows OS/2 位图image/bmp
JPEG 图像image/jpeg
层叠样式表(CSS)text/css
GZip 压缩归档application/gzip
超文本标记语言(HTML)text/html
JavaScripttext/javascript
MP4 视频video/mp4
OpenType 字体font/otf
文本text/plain

参考资料:

1、常见 MIME 类型列表

2、CORS user experience: htmx request headers hx-* not allowed by Access-Control-Allow-Headers in preflight response


这里是松桑的前端后花园,目前坚持在写前端周刊:前端后花园周刊,全网同名。专注前端基础领域,分享前端最新动态、开源项目、工具库等内容,致力于构建属于自己的前端知识网络!这里是我精选的往期文章,希望对你有所帮助。

Node.js常用命令及学习资料

18个宝藏技术周刊,涵盖前端、产品、UX、AIGC、独立开发、科技(偷偷收藏)

为什么每个自媒体人都需要个人网站?(附十分钟快速建站教程)

我用 Nextjs 十分钟搭建了一个展示955公司网站

Gitbook 小书 - 快速创建你的个人专栏

  • 6
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

前端后花园

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值