Nodejs + 自定义 CORS

nodejs + 自定义 cors

cors(跨源资源共享) 是一种允许一个域上的 web 应用程序访问另一个域上的资源的机制。当开发前端和后端分离并通过 api 进行通信的应用程序时,这一点至关重要。

这里有一篇文章解释了在 node.js 和 express 中不使用外部库的 cors 实现:

"use strict";/*jshint node:true */var simplemethods, simplerequestheaders, simpleresponseheaders, tolowercase, checkoriginmatch, origin;object.defineproperty(exports, "simplemethods", {    get: function () {        return [            "get",            "head",            "post",            "put",            "delete"        ];    }});simplemethods = exports.simplemethods;object.defineproperty(exports, "origin", {    get: function () {        return ["http://localhost:3000"];    }});origin = exports.origin;

export simplemethods:定义 cors 请求允许的 http 方法(例如 get、post、put 等)。

导出来源:指定允许访问的来源列表。在此示例中,允许使用 http://localhost:3000。

object.defineproperty(exports, "simplerequestheaders", {    get: function () {        return ["accept", "accept-language", "content-language", "content-type", "authorization", "token"];    }});simplerequestheaders = exports.simplerequestheaders;object.defineproperty(exports, "simpleresponseheaders", {    get: function () {        return ["cache-control", "content-language", "content-type", "expires", "last-modified", "pragma"];    }});simpleresponseheaders = exports.simpleresponseheaders;

导出 simplerequestheaders:定义跨域请求中客户端允许的请求标头。

导出 simpleresponseheaders:定义从服务器到客户端允许的响应标头。

checkoriginmatch = function (originheader, origins, callback) {    if (typeof origins === "function") {        origins(originheader, function (err, allow) {            callback(err, allow);        });    } else if (origins.length > 0) {        callback(null, origins.some(function (origin) {            return origin === originheader;        }));    } else {        callback(null, true);    }};

函数 checkoriginmatch:检查请求来源是否与允许的来源列表匹配。如果匹配,则允许请求。

exports.create = function (options) {    options = options || {};    options.origins = options.origins || origin;    options.methods = options.methods || simplemethods;

来源和方法选项的初始化,如果未提供,则使用来自 origin 和 simplemethods 的默认值。

设置请求和响应标头

 if (options.hasownproperty("requestheaders") === true) {        options.requestheaders = tolowercase(options.requestheaders);    } else {        options.requestheaders = simplerequestheaders;    }    if (options.hasownproperty("responseheaders") === true) {        options.responseheaders = tolowercase(options.responseheaders);    } else {        options.responseheaders = simpleresponseheaders;    }

设置允许的请求(requestheaders)和响应(responseheaders)标头。将任何给定的请求或响应标头转换为小写。

附加中间件配置

 options.maxage = options.maxage || null;    options.supportscredentials = options.supportscredentials || false;    if (options.hasownproperty("endpreflightrequests") === false) {        options.endpreflightrequests = true;    }

maxage:指定 cors 预检的最大缓存期限。 supportcredentials:确定服务器是否支持跨域请求中的凭据(cookie 或令牌)。 endpreflightrequests:决定服务器是否应终止预检请求(选项)或继续执行下一个中间件。

 return function (req, res, next) {        if (!req.headers.hasownproperty("origin")) {            next();        } else {            checkoriginmatch(req.headers.origin, options.origins, function (err, originmatches) {                if (err !== null) {                    next(err);                } else {                    var endpreflight = function () {                        if (options.endpreflightrequests === true) {                            res.writehead(204);                            res.end();                        } else {                            next();                        }                    };

函数 endpreflight:如果 endpreflightrequests 设置为 true,则结束预检(options)请求。来源检查:使用 checkoriginmatch 来验证请求来源是否与允许的来源匹配。

处理预检请求(选项)

 if (req.method === "options") {                        if (!req.headers.hasownproperty("access-control-request-method")) {                            endpreflight();                        } else {                            requestmethod = req.headers["access-control-request-method"];                            if (req.headers.hasownproperty("access-control-request-headers")) {                                requestheaders = tolowercase(req.headers["access-control-request-headers"].split(/,s*/));                            } else {                                requestheaders = [];                            }                            methodmatches = options.methods.indexof(requestmethod) !== -1;                            if (!methodmatches) {                                endpreflight();                            } else {                                headersmatch = requestheaders.every(function (requestheader) {                                    return options.requestheaders.includes(requestheader);                                });                                if (!headersmatch) {                                    endpreflight();                                } else {                                    if (options.supportscredentials) {                                        res.setheader("access-control-allow-origin", req.headers.origin);                                        res.setheader("access-control-allow-credentials", "true");                                    } else {                                        res.setheader("access-control-allow-origin", "*");                                    }                                    if (options.maxage !== null) {                                        res.setheader("access-control-max-age", options.maxage);                                    }                                    res.setheader("access-control-allow-methods", options.methods.join(","));                                    res.setheader("access-control-allow-headers", options.requestheaders.join(","));                                    endpreflight();                                }                            }                        }                    }

请求方法和标头匹配:检查请求方法和标头是否与允许的匹配。 cors 响应标头:设置 cors 标头,例如 access-control-allow-origin、access-control-allow-credentials、access-control-allow-methods 等

在响应中公开标头
} 其他 {
if (options.supportscredentials) {
res.setheader(“access-control-allow-origin”, req.headers.origin);
res.setheader(“access-control-allow-credentials”, “true”);
} 其他 {
res.setheader(“access-control-allow-origin”, “*”);
}

                    exposedheaders = options.responseheaders.filter(function (header) {                        return !simpleresponseheaders.includes(header);                    });                    if (exposedheaders.length > 0) {                        res.setheader("access-control-expose-headers", exposedheaders.join(","));                    }                    next();                }            }        });    }};
 } else {                        if (options.supportsCredentials) {                            res.setHeader("Access-Control-Allow-Origin", req.headers.origin);                            res.setHeader("Access-Control-Allow-Credentials", "true");                        } else {                            res.setHeader("Access-Control-Allow-Origin", "*");                        }                        exposedHeaders = options.responseHeaders.filter(function (header) {                            return !simpleResponseHeaders.includes(header);                        });                        if (exposedHeaders.length > 0) {                            res.setHeader("Access-Control-Expose-Headers", exposedHeaders.join(","));                        }                        next();                    }                }            });        }    };

access-control-expose-headers:如果 simpleresponseheaders 中未包含自定义标头,则设置客户端可访问的响应标头。

这就是如何在 node.js 中实现自定义 cors,而无需使用任何库。完整的脚本可以参考这个例子

以上就是Nodejs + 自定义 CORS的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1497477.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小程序WebView嵌套Vue页面:如何实现页面截图?
上一篇 2025年12月19日 19:33:28
JavaScript 中链式调用是如何实现的?
下一篇 2025年12月19日 19:33:44

相关推荐

  • PHP多维数组到复杂XML结构的SOAP序列化实践

    本文旨在解决php多维数组向复杂soap xml结构序列化时遇到的“无法序列化结果”问题。通过深入理解soap xml的结构要求,包括命名空间和类型属性,文章将指导您如何构建符合特定xml schema的php关联数组。我们将利用`spatie/array-to-xml`库,详细演示其安装与使用方法…

    2026年5月10日
    100
  • CodeIgniter在IIS环境下实现URL重写与index.php移除指南

    本教程详细指导如何在IIS服务器上部署的CodeIgniter应用中,移除URL中不必要的index.php。核心解决方案涉及修改CodeIgniter的config.php文件,将$config[‘index_page’]设置为空,并辅以正确的IIS web.config重…

    2026年5月10日
    100
  • PHP代码注入检测日志分析_PHP代码注入日志检测方法详解

    答案:日志分析是发现PHP代码注入的关键手段,主要通过Web服务器访问日志、PHP错误日志、PHP-FPM日志及应用自定义日志等多源数据,结合grep、ELK、WAF等工具识别含eval()、system()、Base64编码、目录遍历等特征的异常请求,并建立基线、设置检测规则与自动化告警,配合事件…

    2026年5月10日
    000
  • Go语言与Microsoft SharePoint集成指南

    Go语言可以有效集成Microsoft SharePoint,主要通过两种途径:一是利用SharePoint提供的RESTful API进行数据交互,Go的标准HTTP客户端库即可轻松实现;二是通过SharePoint应用模型开发自托管应用,这种模型支持使用包括Go在内的任何语言编写后端逻辑。 1.…

    2026年5月10日
    000
  • Python继承中父类属性的初始化与访问策略

    本文深入探讨python面向对象编程中,子类如何正确初始化和访问父类属性。重点分析`super().__init__()`的工作原理,解释在继承链中参数传递的重要性,并提供通过子类构造函数传递参数的解决方案。此外,针对子类需要与特定父类实例交互的场景,文章还介绍了组合(composition)模式的…

    2026年5月10日
    000
  • 如何用Golang构建无状态微服务 分享Session管理最佳实践

    如何用Golang构建无状态微服务 分享Session管理最佳实践如何用Golang构建无状态微服务 分享Session管理最佳实践如何用Golang构建无状态微服务 分享Session管理最佳实践如何用Golang构建无状态微服务 分享Session管理最佳实践

    构建无状态微服务时,session管理可通过jwt、redis和统一认证中心实现。①使用jwt作为token,客户端存储,服务端无状态;②结合redis记录session元数据,支持主动失效;③设立统一认证中心,中间件校验token;④确保https传输安全并设计token刷新机制。 用 Golan…

    2026年5月10日 用户投稿
    000
  • 怎么安装html5_HTML5开发环境安装与配置详细步骤

    答案是配置HTML5开发环境需三步:1. 安装VS Code等编辑器并配置插件;2. 使用Chrome或Firefox测试页面;3. 可选搭建本地服务器,如Live Server或http-server;最后创建index.html文件验证环境。 安装HTML5开发环境其实并不复杂,因为HTML5本…

    2026年5月10日
    000
  • Go语言中通过字符串动态创建类型实例的实践指南

    本文探讨了在Go语言中如何通过字符串动态创建类型实例。由于Go的静态类型特性和编译优化,直接实现此功能具有挑战性。文章详细介绍了两种主要方法:一是利用reflect包手动维护类型注册表并通过反射创建实例,并提供了示例代码和注意事项;二是推荐使用工厂模式或函数映射等更符合Go惯用法的替代方案,以提高代…

    2026年5月10日
    000
  • C#如何处理异常?C# try-catch-finally最佳实践与常见错误规避

    正确使用 try-catch-finally 应捕获具体异常、用 finally 或 using 释放资源、避免空 catch 和裸抛异常,确保异常日志记录并保留堆栈跟踪,提升代码健壮性与可维护性。 在C#中,异常处理是保障程序稳定运行的重要机制。正确使用 try-catch-finally 结构不…

    2026年5月10日
    000
  • PHP处理大型文本文件转JSON:内存溢出诊断与优化实践

    本文深入探讨了PHP在将大型文本文件转换为结构化JSON时可能遇到的内存溢出问题。文章详细指导读者如何通过phpinfo()诊断并正确配置PHP的memory_limit,包括检查php.ini和.htaccess的潜在冲突,并提供了逐步增加内存限制的建议。同时,文章也分析了特定数据格式下内存消耗的…

    2026年5月10日
    100
  • Nginx 子目录应用URI重写与参数传递教程

    本教程详细阐述了如何在Nginx中为PHP应用实现子目录URI重写,特别是如何从请求URI中剥离子目录路径并将其余部分作为参数传递给主入口文件。通过try_files和rewrite指令的组合,本教程提供了一种高效且准确的解决方案,以替代Apache .htaccess的RewriteRule功能,…

    2026年5月10日
    000
  • JavaScript中如何确保IoT安全?

    在javascript中确保iot安全可以通过以下步骤实现:1) 使用https协议进行安全通信;2) 实施oauth 2.0或jwt进行身份验证和授权;3) 避免使用不安全的javascript功能并验证输入;4) 使用异步编程优化性能;5) 定期更新和修补软件。 在JavaScript中确保Io…

    2026年5月10日
    000
  • 在R Markdown中运行JavaScript并导入库的正确姿势

    本文旨在解决在R Markdown文档中运行JavaScript代码并成功导入外部库(如MSAL)时遇到的常见问题。通过详细的代码示例和步骤说明,帮助读者掌握在R Markdown环境中集成JavaScript库的正确方法,实现更强大的交互式数据分析和可视化功能。 在R Markdown文档中集成J…

    2026年5月10日
    100
  • 使用PHP FirestoreClient发送自定义头部认证令牌的最佳实践

    本文旨在解决php firestoreclient在启用安全规则后遇到的“权限不足”错误。核心内容是,对于服务器端应用,应通过服务账户进行身份验证,并推荐在`firestoreclient`构造函数中使用`keyfilepath`参数明确指定服务账户密钥文件路径,以确保请求能够正确通过firesto…

    2026年5月10日
    000
  • php 收集哪些日志

    PHP 收集广泛类型的日志,包括错误、警告、通知、调试、HTTP 和事件日志。PHP 提供了几种方法来收集日志:使用内置函数、第三方库和 Web 服务器配置。对于最佳实践,建议启用日志记录、选择适当的日志级别、定期审查日志、使用日志文件轮换并保护日志文件。 PHP 日志收集 PHP 收集哪些日志? …

    2026年5月10日
    100
  • 实现平滑连续滑动但仅输出离散值的HTML Range Slider教程

    本教程详细介绍了如何创建一种HTML范围滑块(input type=”range”),使其在用户拖动时呈现出平滑连续的视觉效果,但实际输出的值却是预设的离散整数。核心方法是通过将滑块的step属性设置为一个很小的浮点数,从而实现细致的滑动,然后利用JavaScript将获取到…

    2026年5月10日
    000
  • 为什么未使用特定指令的输入框也会受到Vue自定义指令的影响?

    Vue自定义指令意外生效之谜:深入探讨 本文探讨一个常见的Vue.js开发问题:自定义指令在未绑定目标元素上生效的原因。我们分析一个案例,解释这种现象背后的机制,并提供解决方案。 案例描述 我们创建了一个全局自定义指令 validateNumber,用于限制输入框只能输入数字: Vue.direct…

    2026年5月10日
    000
  • JavaScript实现可折叠图片显示/隐藏功能教程

    JavaScript实现可折叠图片显示/隐藏功能教程JavaScript实现可折叠图片显示/隐藏功能教程JavaScript实现可折叠图片显示/隐藏功能教程JavaScript实现可折叠图片显示/隐藏功能教程

    本教程详细介绍了如何使用JavaScript和HTML创建一个可折叠的图片显示/隐藏功能。通过引入一个状态变量来管理图片当前是展开还是折叠,结合按钮点击事件动态切换图片的可见性及按钮文本,实现用户友好的交互式内容展示,适用于在网页中按需显示或隐藏图片资源。 1. 功能概述与核心思路 在网页开发中,有…

    2026年5月10日 用户投稿
    000
  • 在点击图片时动态显示其替代文本(Alt Text)的JavaScript教程

    在点击图片时动态显示其替代文本(Alt Text)的JavaScript教程在点击图片时动态显示其替代文本(Alt Text)的JavaScript教程在点击图片时动态显示其替代文本(Alt Text)的JavaScript教程在点击图片时动态显示其替代文本(Alt Text)的JavaScript教程

    本教程详细介绍了如何利用JavaScript在用户点击缩略图时,动态地在大图下方显示其对应的替代文本(Alt Text)。通过修改现有函数,我们能够获取图像的alt属性,并将其内容插入到指定的HTML元素中,从而提升用户体验和信息传达效率。 引言 在网页开发中,图片是不可或缺的元素。为了提升用户体验…

    2026年5月10日 用户投稿
    000
  • HTML表单如何实现白名单功能?怎样只允许授权用户?

    要实现%ignore_a_1%的白名单功能并确保只有授权用户操作,核心答案是必须依赖后端服务器进行严格的身份认证、会话管理、授权检查和数据验证,前端仅能提供用户体验层面的初步提示而不能保障安全;具体而言,首先通过用户身份认证(如用户名/密码或oauth)确认用户身份,服务器创建会话并返回标识符,后续…

    2026年5月10日
    800

发表回复

登录后才能评论
关注微信