草庐IT

javascript - axios onUploadProgress 和 onDownloadProgress 不适用于 CORS

coder 2024-07-20 原文

我有一个用 Node.js 编写的服务器,以及一个在浏览器中运行的 Web 客户端。客户端应从服务器上传和下载一些文件。服务器不是原来交付客户端的那个,所以我们这里就出现了跨域的情况。

服务器使用 cors实现跨域请求的中间件。由于我还使用了一些自定义 header ,因此我将其与如下配置一起使用:

api.use(cors({
  origin: '*',
  allowedHeaders: [ 'content-type', 'authorization', 'x-metadata', 'x-to' ],
  exposedHeaders: [ 'content-type', 'content-disposition', 'x-metadata' ]
}));

相比之下,客户端使用的是axios ,上传文件的代码如下所示:

await axios({
  method: 'post',
  url: 'https://localhost:3000/api/v1/add-file',
  data: content,
  headers: {
    // ...
  }
});

到目前为止,一切都按预期进行,尤其是在 CORS 方面。

现在我想跟踪上传的进度,因此我将 onUploadProgress 回调添加到 axios 调用中,如 suggested by its documentation :

await axios({
  method: 'post',
  url: 'https://localhost:3000/api/v1/add-file',
  data: content,
  headers: {
    // ...
  },
  onUploadProgress (progressEvent) {
    console.log({ progressEvent });
  }
});

如果我现在运行客户端,上传仍然有效——但永远不会调用 onUploadProgress 回调。我读到并不是所有的浏览器都支持这个(在内部这只是绑定(bind)到底层 XmlHttpRequest 对象的 onprogress 事件),但最新的 Chrome 应该能够做到这一点(并且如果我尝试不同的设置,其中不涉及 CORS,一切似乎都有效)。所以我认为这是一个 CORS 问题。

我有read一旦您将监听器添加到 onprogress 事件,就会发送预检请求。这反过来意味着服务器必须接受 OPTIONS 请求。为此,我在服务器上添加了以下代码,上述对 api.use 的调用之前:

api.options('*', cors({
  origin: '*',
  methods: [ 'GET', 'POST' ],
  allowedHeaders: [ 'content-type', 'authorization', 'x-metadata', 'x-to' ],
  exposedHeaders: [ 'content-type', 'content-disposition', 'x-metadata' ],
  optionsSuccessStatus: 200
}));

结果:上传仍然有效,但仍然没有引发 onUploadProgress 事件。所以我假设我遗漏了一些与 CORS 相关的东西。问题只是:我错过了什么?

我也曾尝试使用更大的文件(将近 2 GB,而不是几 KB)重新运行相同的设置,但结果是一样的。有谁知道我在这里做错了什么?

更新

回复一些评论:首先,我的axios版本是0.18.0,所以我使用的是可用的最新稳定版本。

服务器的源代码可以在存储库中找到 thenativeweb/wolkenkit-depot . CORS部分配置here .请注意,这与我上面发布的版本略有不同,因为我做了一些本地更改。然而,这是你需要处理的地方。

要构建服务器,您需要安装 Node.js 和 Docker。然后,运行:

$ npm install
$ npx roboter build

这将生成一个新的 Docker 镜像,其中包含服务器的代码(您需要此 Docker 镜像才能运行下面的客户端测试)。请注意,如果您在服务器上更改任何内容,则必须重新构建此 Docker 镜像。

客户端可以在存储库中找到 thenativeweb/wolkenkit-depot-client-js , 在分行 issue-145-notifiy-about-progress-when-uploading-or-downloading-files .

在文件中DepotClient.js我在 onUploadProgress 中添加了一个事件监听器,它应该只是抛出一个异常(这反过来应该非常明显地表明事件已引发)。

此代码随后由 this test 运行(对此有更多测试,但这是其中之一),这应该会导致异常,但事实并非如此。

要再次运行测试,您必须安装 Node.js 和 Docker,并且必须按照之前的描述构建服务器。然后使用以下命令代表您运行测试:

$ npm install
$ npx roboter test --type integration

我的期望是至少有一个 addFile 测试应该归档。事实上,它们都变成了绿色,这意味着上传本身可以完美运行,但永远不会引发 onUploadProgress 事件。以防万一你想知道执行测试的环境:我在这里使用 puppeteer,它是一个 headless 的 Chrome。

最佳答案

由于该代码运行良好(无论是否使用 cors),我假设您使用的是旧的 axios 版本。 onUploadProgress添加到版本 0.14.0 .

如果您使用的是旧版本,您可以使用:progress而不是 onUploadProgress ,或者只更新到最新版本。

0.14.0(2016 年 8 月 27 日)

BREAKING Splitting progress event handlers into onUploadProgress and onDownloadProgress (#423)

await axios({
  method: 'post',
  url: 'https://localhost:3000/api/v1/add-file',
  data: content,
  headers: {
    // ...
  },
  // Old version
  progress (progressEvent) {
    console.log({ progressEvent });
  },
  // New version
  onUploadProgress (progressEvent) {
    console.log({ progressEvent });
  }
});

更新

提供的代码在浏览器上完美运行,但从 Node 运行它时不起作用,因为在 Node.js 中运行 axios 时它使用 /lib/adapters/http.js 而不是 /lib/adapters/xhr.js

如果您阅读该代码,您将在 http 上看到它适配器没有任何 onUploadProgress选项。

我从浏览器测试了您的代码,访问了您的端点并且运行良好。 问题是当您运行从 Node.js 运行的集成测试时。

这里你有一个问题:https://github.com/axios/axios/issues/1966他们推荐使用的地方:progress-stream如果你想取得进展,但不会在 onUploadProress 上触发功能。

更新 2

我可以确认,在 puppeteer 上运行测试时,它使用的是 XMLHttpRequest 处理程序,而不是 Node.js。问题是您在异步回调中抛出错误,该错误未被 try/catch 捕获。 .

try {
  await request({
    method: 'post',
    url: `${protocol}://${host}:${port}/api/v1/add-file`,
    data: content,
    headers,
    onUploadProgress (progress) {
      // This error occurs in a different tick of the event loop
      // It's not catched by any of the try/catchs that you have in here
      // Pass a callback function, or emit an event. And check that on your test
      throw new Error(JSON.stringify(progress));
    }
  });

  return id;
} catch (ex) {
  if (!ex.response) {
    throw ex;
  }

  switch (ex.response.status) {
    case 401:
      throw new Error('Authentication required.');
    default:
      throw ex;
  }
}

如果你想捕捉到它,你必须将 axios 调用包装在 Promise 中。并在那里拒绝(这没有意义,因为那仅用于测试)。

所以我的建议是传递 onUploadProgress addFile 的参数, 这样你就可以检查它是否到达了 onUploadProgress来自你的测试。

为了看到:throw new Error(JSON.stringify(progress));被调用,使用 { headless: false } 启动 puppeteer .

这样你会看到错误被正确触发。

XMLHttpRequestUpload.onUploadProgress

关于javascript - axios onUploadProgress 和 onDownloadProgress 不适用于 CORS,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55295036/

有关javascript - axios onUploadProgress 和 onDownloadProgress 不适用于 CORS的更多相关文章

  1. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  2. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  3. ruby - inverse_of 是否适用于 has_many? - 2

    当我使用has_one时,它​​工作得很好,但在has_many上却不行。在这里您可以看到object_id不同,因为它运行了另一个SQL来再次获取它。ruby-1.9.2-p290:001>e=Employee.create(name:'rafael',active:false)ruby-1.9.2-p290:002>b=Badge.create(number:1,employee:e)ruby-1.9.2-p290:003>a=Address.create(street:"123MarketSt",city:"SanDiego",employee:e)ruby-1.9.2-p290

  4. ruby - "undefined method"用于 rails 模型 - 2

    我正在使用带有Rails的Devise,我想添加一个方法“getAllComments”,所以我这样写:classUser在我的Controller中:defdashboard@user=current_user@comments=@user.getAllComments();end当我访问我的url时,我得到了undefinedmethod`getAllComments'for#我做错了什么?谢谢 最佳答案 因为getAllComments是一个类方法,而您正试图将其作为实例方法访问。您要么需要访问它:User.getAllCom

  5. ruby-on-rails - 使用 javascript 更改数据方法不会更改 ajax 调用用户的什么方法? - 2

    我遇到了一个非常奇怪的问题,我很难解决。在我看来,我有一个与data-remote="true"和data-method="delete"的链接。当我单击该链接时,我可以看到对我的Rails服务器的DELETE请求。返回的JS代码会更改此链接的属性,其中包括href和data-method。再次单击此链接后,我的服务器收到了对新href的请求,但使用的是旧的data-method,即使我已将其从DELETE到POST(它仍然发送一个DELETE请求)。但是,如果我刷新页面,HTML与"new"HTML相同(随返回的JS发生变化),但它实际上发送了正确的请求类型。这就是这个问题令我困惑的

  6. Ruby on Rails regexp equals-tilde 与 array include 用于检查选项列表 - 2

    我正在使用Rails3.2.3和Ruby1.9.3p0。我发现我经常需要确定某个字符串是否出现在选项列表中。看来我可以使用Ruby数组.includemethod:或正则表达式equals-tildematchshorthand用竖线分隔选项:就性能而言,一个比另一个好吗?还有更好的方法吗? 最佳答案 总结:Array#include?包含String元素,在接受和拒绝输入时均胜出,对于您的示例只有三个可接受的值。对于要检查的更大的集合,看起来Set#include?和String元素可能会获胜。如何测试我们应该根据经验对此进行测试

  7. ruby-on-rails - Ruby "Undefined Method"用于类方法 - 2

    Ruby初学者努力简单地将这个@@people散列的值打印到控制台classPerson#haveafirst_nameandlast_nameattributewithpublicaccessorsattr_accessor:first_nameattr_accessor:last_name#haveaclassattributecalled`people`thatholdsanarrayofobjects@@people=[]#havean`initialize`methodtoinitializeeachinstancedefinitialize(first_name,last_

  8. python - 用于 Python 或 Ruby 的 Amazon Book API? - 2

    这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:AmazonAPIlibraryforPython?我正在寻找一个AmazonAPI,它可以让我:按书名或作者查找书籍显示书籍封面获取有关每本书的信息(价格、评级、评论数、格式、页数等)Python或Ruby库都可以(我只想要最容易使用的库)。有什么建议么?我知道在SO上还有其他一些关于此的帖子,但这些API似乎很快就过时了。[几个月前我尝试了几个建议的Ruby库,但无法让它们中的任何一个工作。]

  9. ruby-on-rails - 用于门户的 Ruby 技术 - 2

    我刚刚看到whitehouse.gov正在使用drupal作为CMS和门户技术。drupal的优点之一似乎是很容易添加插件,而且编程最少,即重新发明轮子最少。这实际上正是Ruby-on-Rails的DRY理念。所以:drupal的缺点是什么?Rails或其他基于Ruby的技术有哪些不符合whitehouse.org(或其他CMS门户)门户技术的资格? 最佳答案 Whatarethedrawbacksofdrupal?对于Ruby和Rails,这确实是一个相当主观的问题。Drupal是一个可靠的内容管理选项,非常适合面向社区的站点。它

  10. ruby - 在 Mechanize 中使用 JavaScript 单击链接 - 2

    我有这个:AccountSummary我想单击该链接,但在使用link_to时出现错误。我试过:bot.click(page.link_with(:href=>/menu_home/))bot.click(page.link_with(:class=>'top_level_active'))bot.click(page.link_with(:href=>/AccountSummary/))我得到的错误是:NoMethodError:nil:NilClass的未定义方法“[]” 最佳答案 那是一个javascript链接。Mechan

随机推荐