修改说明:这个问题是关于为什么XMLHttpRequest/fetch等。在浏览器上,Postman不受相同访问策略限制(您会收到提到CORB或CORS的错误)。这个问题不是关于如何修复“No‘Access Control Allow Origin’…”错误。这是关于它们发生的原因。

请停止发布:阳光下每种语言/框架的CORS配置。而是找到相关语言/框架的问题。允许请求绕过CORS的第三方服务用于关闭各种浏览器的CORS的命令行选项


我试图通过连接RESTful API内置Flask来使用JavaScript进行授权。但是,当我发出请求时,我会收到以下错误:

XMLHttpRequest cannot load http://myApiUrl/login. 
No 'Access-Control-Allow-Origin' header is present on the requested resource. 
Origin 'null' is therefore not allowed access.

我知道API或远程资源必须设置标头,但当我通过Chrome扩展Postman发出请求时,为什么它会起作用?

这是请求代码:

$.ajax({
      type: 'POST',
      dataType: 'text',
      url: api,
      username: 'user',
      password: 'pass',
      crossDomain: true,
      xhrFields: {
        withCredentials: true,
      },
    })
      .done(function (data) {
        console.log('done');
      })
      .fail(function (xhr, textStatus, errorThrown) {
        alert(xhr.responseText);
        alert(textStatus);
      });

当前回答

如果网关超时太短,并且正在访问的资源的处理时间比超时时间长,则也可能会出现此错误。这可能是复杂数据库查询等的情况。因此,上述错误代码可以掩盖这个问题。只需检查错误代码是否为504,而不是Kamil的答案中的404或其他内容。如果是504,那么增加网关超时可能会解决问题。

在我的情况下,可以通过禁用Internet Explorer浏览器中的同源策略(CORS)来删除CORS错误,请参阅如何禁用同源策略Internet Explorer。这样做之后,日志中出现了纯504错误。

其他回答

如果网关超时太短,并且正在访问的资源的处理时间比超时时间长,则也可能会出现此错误。这可能是复杂数据库查询等的情况。因此,上述错误代码可以掩盖这个问题。只需检查错误代码是否为504,而不是Kamil的答案中的404或其他内容。如果是504,那么增加网关超时可能会解决问题。

在我的情况下,可以通过禁用Internet Explorer浏览器中的同源策略(CORS)来删除CORS错误,请参阅如何禁用同源策略Internet Explorer。这样做之后,日志中出现了纯504错误。

如果我理解正确的话,您正在对与页面所在的域不同的域执行XMLHttpRequest。因此浏览器会阻止它,因为出于安全原因,它通常允许来自同一源的请求。当您想要进行跨域请求时,需要做一些不同的事情。

当您使用Postman时,他们不受此策略的限制。引自跨源XMLHttpRequest:

常规网页可以使用XMLHttpRequest对象从远程服务器发送和接收数据,但它们受到同源策略的限制。扩展不受限制。只要扩展首先请求跨源权限,它就可以与源之外的远程服务器通信。

对我来说,我遇到这个问题是因为不同的原因,远程域被添加到源,部署的应用程序工作得很好,除了一个端点我遇到了这个问题:

Origin https://mai-frontend.vercel.app is not allowed by Access-Control-Allow-Origin. Status code: 500

and

Fetch API cannot load https://sciigo.herokuapp.com/recommendations/recommendationsByUser/8f1bb29e-8ce6-4df2-b138-ffe53650dbab due to access control checks.

在更新Heroku数据库表后,我发现我的Heroku数据表并不包含本地表的所有列。

Deep

在以下作为API的调查中,我使用http://example.com而不是http://myApiUrl/login从你的问题,因为这第一个工作。我想你的页面已打开http://my-site.local:8088.

注意:API和您的页面有不同的域!

你看到不同结果的原因是邮差:

set header Host=example.com(您的API)NOT设置标题原点邮递员实际上根本不使用您的网站url(您只在邮递员中键入您的API地址)-他只向API发送请求,所以他假设网站的地址与API相同(浏览器不假设这一点)

这类似于当站点和API具有相同的域时浏览器发送请求的方式(浏览器也设置了标题项Referer=http://my-site.local:8088但我在邮差身上看不到)。当未设置Origin标头时,服务器通常默认允许此类请求。

这是邮差发送请求的标准方式。但是,当您的站点和API具有不同的域时,浏览器会以不同的方式发送请求,然后CORS会发生,浏览器会自动:

sets header Host=example.com(您的API)设置标题原点=http://my-site.local:8088(您的网站)

(标头Referer的值与原点相同)。现在,在Chrome的控制台和网络选项卡中,您将看到:

当您拥有主机!=Origin这是CORS,当服务器检测到这样的请求时,它通常会默认阻止它。

当您从本地目录打开HTML内容时,Origin=null被设置,并且它会发送请求。同样的情况是,当您在<iframe>中发送请求时,如下面的代码段所示(但这里根本没有设置Host头)-通常,无论HTML规范在哪里表示不透明的起源,您都可以将其转换为origin=null。您可以在此处找到有关此的更多信息。

获取('http://example.com/api',{method:“POST”});查看chrome控制台>网络选项卡

如果您不使用简单的CORS请求,通常浏览器也会在发送主请求之前自动发送OPTIONS请求-这里有更多信息。下面的代码片段显示了它:

获取('http://example.com/api', {method:'POST',headers:{“Content Type”:“application/json”}});在chrome控制台->网络选项卡中查找“api”请求。这是OPTIONS请求(服务器不允许发送POST请求)

您可以更改服务器的配置以允许CORS请求。

下面是一个在nginx(nginx.conf文件)上打开CORS的示例配置-非常小心地为nginx设置总是/“$http_origin”,为Apache设置“*”-这将阻止CORS从任何域(在生产中,而不是使用使用api的具体页面地址)

位置~^/index\.php(/|$){...add_header“Access Control Allow Origin”“$http_Origin”始终;add_header“访问控制允许凭据”“始终为true”;if($request_method=OPTIONS){add_header'访问控制允许来源'“$http_Origin”;#请勿移除该行(用上面的外部“if”加倍)add_header'访问控制允许凭据''true';add_header'访问控制最大年龄'1728000;#缓存20天的飞行前值add_header'访问控制允许方法''GET,POST,OPTIONS';add_header'访问控制允许标头''我的第一个标头,我的第二个标头,授权,内容类型,接受,来源';add_header“内容长度”0;add_header“内容类型”“text/plain charset=UTF-8”;返回204;}}

下面是一个在Apache上打开CORS的示例配置(.htaccess文件)

# ------------------------------------------------------------------------------#|跨域Ajax请求|# ------------------------------------------------------------------------------#启用跨源Ajax请求。# http://code.google.com/p/html5security/wiki/CrossOriginRequestSecurity# http://enable-cors.org/#<IfModule mod_headers.c>#标题集访问控制允许来源“*”#</IfModule>#标题集标题集访问控制允许来源“*”#标头始终将访问控制允许凭据设置为“true”访问控制允许来源“http://your-page.com:80"标头始终设置访问控制允许方法“POST、GET、OPTIONS、DELETE、PUT”标头始终设置访问控制允许标头“My First Header,My Second Header,Authorization,content type,csrf token”

应用CORS限制是由服务器定义并由浏览器实现的安全功能。

浏览器查看服务器的CORS策略并尊重它。

然而,Postman工具并不关心服务器的CORS策略。

这就是为什么CORS错误出现在浏览器中,而不是Postman中。