6

我一直在我的 Chrome 控制台中使用 Backbone 并遇到了一个我无法弄清楚的跨域问题。

我连接的主机可能正确实现了 CORS,因为原始 XHR 请求返回预期的 JSON:

var http = new XMLHttpRequest();
http.open('GET', 'http://example.com:3000/entities/item/15.json', true);
http.onreadystatechange = function(evt) { console.log(evt); }
http.send();

(在控制台上记录 3 个 XHR 进度事件,并在响应中包含正确的数据)

但是当我使用 Backbone 执行以下操作时,浏览器不喜欢它:

var Item = Backbone.Model.extend({});
var ItemsCollection = Backbone.Collection.extend({
  model: Item,
  url: 'http://example.com:3000/entities/item/'
});
var items = new ItemsCollection();
items.fetch();

(返回XMLHttpRequest cannot load http://example.com:3000/entities/item/. Origin http://localhost:8000 is not allowed by Access-Control-Allow-Origin.

我需要做些什么来告诉 Backbone 使用 CORS 吗?这个错误似乎在浏览器发出请求之前就已经出现了,所以我认为这不是服务器配置错误。

4

3 回答 3

8

我希望其中一个有帮助(我还没有尝试过):
1.覆盖 Backbone.js 同步以允许跨域

(function() {
  var proxiedSync = Backbone.sync;
  Backbone.sync = function(method, model, options) {
    options || (options = {});
    if (!options.crossDomain) {
      options.crossDomain = true;
    }
    if (!options.xhrFields) {
      options.xhrFields = {withCredentials:true};
    }
    return proxiedSync(method, model, options);
  };
})();

2.backbone.js的跨域CORS支持

$.ajaxPrefilter(function(options, originalOptions, jqXHR) {
    options.crossDomain ={
        crossDomain: true
    };
    options.xhrFields = {
        withCredentials: true
    };
});
于 2013-04-16T17:31:31.623 回答
6

嘿,你可以使用类似的东西:

 var PostsCollection = Backbone.Collection.extend({
   initialize: function(models, options) {
     //this.id = options.id;
   },
   url: 'http://myapi/api/get_posts/?count=8',
 });

 posts = new PostsCollection();
 posts.fetch({
   dataType: 'jsonp',
   success : function (data) {
     console.log(data);
   }
 });

关键是我们需要使用'jsonp'

于 2014-01-11T15:42:53.513 回答
0

事实证明,我使用 Backbone 请求的 URL 与通过 XHR 请求的 URL 略有不同(它们缺少查询参数)。这导致服务器为 500,它没有 CORS 标头。Chrome 在调试面板的网络选项卡中根本没有显示 HTTP 请求,所以我非常困惑。

修复服务器上的 500 使这项工作再次发生。

于 2013-04-16T19:31:42.523 回答