- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用 in this post 中列出的基本方法从客户端 docker 容器连接到多个 chrome docker 容器中的任何一个(在 docker 群/服务中,可能跨越 nginx 后面的多个服务器,使用 CapRover 部署)。
在每个 chrome 容器中,我维护一个 browser
对象的池(只是一个简单的数组),并将传入的请求定向到适当的 browser
如下(非常类似于链接的帖子):
import http from 'node:http'; // https://nodejs.org/api/http.html
import httpProxy from 'http-proxy'; // https://www.npmjs.com/package/http-proxy
const proxy = new httpProxy.createProxyServer({ ws: true });
// an array (pool) of pre-launched and managed browser objects...
const browsers = [ ... ];
http
.createServer()
.on('upgrade', (req, socket, head) => {
const browser = browsers[Math.floor(Math.random() * browsers.length)]; // in reality I don't just pick a browser at random
const target = browser.wsEndpoint();
proxy.ws(req, socket, head, { target });
})
.listen(3222);
以上是在 ws://srv-captain--chrome:3222
上监听(通信是通过容器之间的 docker 网络进行的“内部”通信)。
然后,在我的客户端容器中,我连接到公共(public)端点 ws://srv-captain--chrome:3222
,如下所示:
import puppeteer from 'puppeteer'; // https://www.npmjs.com/package/puppeteer (using version 17.1.3 at time of posting this)
try {
const browser = await puppeteer.connect({ browserWSEndpoint: 'ws://srv-captain--chrome:3222' });
} catch (err) {
console.error('error connecting to browser', err);
}
这真的很好用,只是在上面的客户端容器中调用 puppeteer.connect()
时,我遇到了类似这样的偶尔/不一致错误:
Protocol error (Emulation.setDeviceMetricsOverride): Session closed. Most likely the page has been closed.
Protocol error (Performance.enable): Target closed.
几乎总是,如果我只是简单地再次尝试连接,连接就会在没有进一步错误的情况下建立,而且是在第一次尝试时。
我不知道为什么错误是提示页面已关闭
或目标已关闭
,因为在这个过程中,我没有尝试与任何 页面
交互,我从监听 browser.on('disconnected'...)
以及监控 chromium 进程本身得知,每个 数组中的浏览器
仍然工作正常......没有崩溃。
知道这里发生了什么吗?
进一步测试后更新
当然,在客户端容器中,我们连接到浏览器
并不是像上面的片段那样,而是为了打开一个页面
并使用 页面
做一些事情。实际上,在客户端容器中它更像是下面的测试片段:
const doIteration = function (i) {
return new Promise(async (resolve, reject) => {
// mimic incoming requests coming in at random times over a short period by introducing a random initial delay...
await new Promise(resolve => setTimeout(resolve, Math.random() * 5000));
// now actually connect...
let browser;
try {
browser = await puppeteer.connect({ browserWSEndpoint: `ws://srv-captain--chrome:3222?queryParam=loop_${i}` });
} catch (err) {
reject(err);
return;
}
// now that we have a browser, open a new page...
const page = await browser.newPage();
// do something useful with the page (not shown here) and then close it..
await page.close();
// now disconnect (but don't close) the browser...
browser.disconnect();
resolve();
});
};
const promises = [];
for (let i = 0; i < 15; i++) {
promises.push( doIteration(i) );
}
try {
await Promise.all(promises);
} catch (err) {
console.error(`error doing stuff`, err);
}
上面的每个迭代都同时执行了多次...我在迭代数组上使用 Promise.all()
promise 在我的生产代码中模拟多个并发传入请求。以上足以重现问题...错误不会发生在调用 puppeteer.connect()
时 every 迭代,只是一些。
所以在一次迭代中打开/关闭一个页面
和在另一次迭代中调用puppeteer.connect()
之间似乎存在某种相互作用,尽管关闭了页面
并在每次迭代中正确断开浏览器
?这可能也解释了在调用 puppeteer.connect()
时如果有一些与 page
相关的宿醉,则 Most likely the page has been closed
错误消息> 在另一次迭代中关闭...尽管由于某种原因,调用 puppeteer.connect()
时会出现此错误?
通过使用 browsers
数组中的 browser
对象池,以及在多个服务器上具有多个容器的 docker swarm,每个升级
消息可以在不同的容器(甚至可以在不同的服务器上)接收,并且可以路由到 browsers
数组中的不同的 browser
。但我现在认为这是一个转移注意力的问题,因为在进一步的测试中,我通过将所有请求路由到 browsers[0]
并将服务缩小到一个容器来缩小问题范围......因此 upgrade
消息始终由同一服务器上的同一容器处理并路由到同一 browser
... 问题仍然存在。
上述错误的完整堆栈跟踪:
Error: Protocol error (Emulation.setDeviceMetricsOverride): Session closed. Most likely the page has been closed.
at CDPSession.send (file:///root/workspace/myclientapp/node_modules/puppeteer/lib/esm/puppeteer/common/Connection.js:281:35)
at EmulationManager.emulateViewport (file:///root/workspace/myclientapp/node_modules/puppeteer/lib/esm/puppeteer/common/EmulationManager.js:33:73)
at Page.setViewport (file:///root/workspace/myclientapp/node_modules/puppeteer/lib/esm/puppeteer/common/Page.js:1776:93)
at Function._create (file:///root/workspace/myclientapp/node_modules/puppeteer/lib/esm/puppeteer/common/Page.js:242:24)
at runMicrotasks (<anonymous>)
at processTicksAndRejections (node:internal/process/task_queues:96:5)
at async Target.page (file:///root/workspace/myclientapp/node_modules/puppeteer/lib/esm/puppeteer/common/Target.js:123:23)
at async Promise.all (index 0)
at async BrowserContext.pages (file:///root/workspace/myclientapp/node_modules/puppeteer/lib/esm/puppeteer/common/Browser.js:577:23)
at async Promise.all (index 0)
最佳答案
随着我越来越深入地研究这个问题,越来越明显的是我实际上可能没有做任何根本性的错误,这可能只是 puppeteer 本身的一个错误。所以我将这些报告为 an issue over on puppeteer ...事实上,它被认为是 15.5.0
之后的任何版本的错误,并且正在修复。与此同时,解决方法是恢复到 puppeteer 版本 15.5.0
并在使用并发连接时调用 browser.pages()
时要小心,因为这可能本身会抛出错误...但我知道这也可能是他们可以/将会修复的问题,以便 browser.pages()
对并发连接的存在更有弹性。
关于docker - 调用 puppeteer.connect() 时出现协议(protocol)错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/73670998/
internal protocol Reducer { associatedtype S : BaseState associatedtype A : BaseActi
我在考虑我的应用程序中的验证检查,我认为在任何模型上调用 ValidatorFactory,实现 Validee,这意味着说哪个类负责 ValidatorCreation 听起来不错。但是下面的代码不
我已经定义了 2 个协议(protocol)。我需要第一个 (NameProtocol) 来执行 Equatable 协议(protocol)。而另一个类 (BuilderProtocol) 有一个返
在上传方面,WebDAV 协议(protocol)在哪些方面优于 HTTP 协议(protocol)。 Socket Upload 协议(protocol)和 WebDav Upload 协议(pro
是否可以在任何版本的 Swift 中扩展具有混合类/协议(protocol)类型约束的协议(protocol)?例如,仅当 Self 是 UIViewController 的子类并且符合 Protoc
我有一个协议(protocol) (ProtocolA),其中包含符合第二个协议(protocol) (ProtocolB) 的单个属性。 public protocol ProtocolA {
NSObject 协议(protocol)带有常用的协议(protocol)模板,但它似乎并不是协议(protocol)实际实现所必需的。将其排除在外似乎完全没有任何改变。那么,协议(protocol
我想根据这两种协议(protocol)的一般特征(例如开销(数据包)、安全性、信息建模和可靠性)来比较 OPC UA 和 MQTT。我在哪里可以找到每个协议(protocol)的开销和其他特性的一些示
使用 Swift 4,我正在尝试编写一个自定义协议(protocol),它提供对 @objc 协议(protocol)的一致性。 一些代码 更具体地说,我有一个自定义协议(protocol) Sear
我想定义一个在 Viper 架构中使用的协议(protocol),以使用具有弱属性的协议(protocol)在 Viper 组件之间建立连接,但我收到以下错误消息: 'weak' may only b
我在同一个网络中有 3 个 docker 容器: 存储 (golang) - 它提供了用于上传视频文件的 API。 主播 (nginx) - 它流式传输上传的文件 反向代理 (姑且称之为代理) 我有
我打算在我的项目中使用 php socket。它需要用户登录才能根据 session 填充内容。所以我的问题是,TCP/IP 协议(protocol)也像 HTTP 协议(protocol)一样为每个
目前,我的网站有两个版本。一种带有 https://-证书,一种没有。我想将我网站的 http 版本上的所有用户 301 重定向到我网站的 https://版本。 这似乎不可能,因为创建重定向将导致重
目前,我的网站有两个版本。一种带有 https://-证书,一种没有。我想将我网站的 http 版本上的所有用户 301 重定向到我网站的 https://版本。 这似乎不可能,因为创建重定向将导致重
我有一个 Swift View Controller ,它定义了一个在 Objective-C View Controller 中应该遵循的协议(protocol): ChildViewControl
我在客户那里有数百个硬件设备,需要通过telnet接口(interface)发送HTTP数据。 目标是等待数据的 Apache 2 Web 服务器和 PHP 脚本。 这已经可以正常工作了,但是我们发现
我发现如果我创建一个这样的协议(protocol): protocol MyProtocol { } 我不能这样做: weak var myVar: MyProtocol? 我找到了解决这个问题的方法
Xcode 基于模板生成了这个头文件: // this file is XYZAppDelegate.h #import @interface XYZAppDelegate : UIRespond
我在 github 中有一个公开的存储库,我正在开发一个开源应用程序,用于制作产品目录和小型 cms 内容。 我还有一个私有(private)仓库(不托管在github),它是在托管在github的开
您好,我想让别人看到私有(private) repo 代码,但不想公开我的 repo ,也不希望他们有能力更改内容。这可能吗?我查看了网站的“管理”部分,但没有找到合适的内容。谢谢大家。 最佳答案 据
我是一名优秀的程序员,十分优秀!