Cypress 里的 ensureAttached 检测原理
下面是我用 Cypress 开发的端到端测试。click 调用里的 force:true 参数是我后来加上去的。
data:image/s3,"s3://crabby-images/45bb7/45bb76eed6cd13b41da823dcfbd466b34c1e4155" alt=""
如果不添加,会遇到错误消息:
data:image/s3,"s3://crabby-images/22aa7/22aa702d4246496f6e20679a74c46f587eef81fd" alt=""
在方法 $Cy.ensureAttached 里面跑出来的,这个方法位于 runner/cypress_runner.js 下面。
我想知道这个 cypress_runner.js 的本地位置:
data:image/s3,"s3://crabby-images/ccfb3/ccfb30c5618135fcdc2142465a74ff5968d80c8c" alt=""
本地请求的 url:http://localhost:4200/__cypress/runner/cypress_runner.js
远端端口:59701?
data:image/s3,"s3://crabby-images/62661/626614512f6edc3c1cf7f05658d690bf3232d2b7" alt=""
重定向到 __?
data:image/s3,"s3://crabby-images/c6bdf/c6bdf792dbe8856d7e0fe702476cf0f69965904d" alt=""
这个文件有20万行代码:
data:image/s3,"s3://crabby-images/db5c0/db5c01b8d5cba9f2ef22c715fe5af51e1e01126a" alt=""
怀疑这个庞大的文件是 merge 起来的:
data:image/s3,"s3://crabby-images/3bf04/3bf048d04aea10d44d74b164e5ff95fa4ee0da47" alt=""
这里能看到所有的 ensure 检查:
data:image/s3,"s3://crabby-images/c3e46/c3e465405bbcc681d98b124ebf22e5704e7a5b2c" alt=""
断点触发时,从 callstack 是很难发现到底是哪一行 e2e spec 代码触发的这个 ensureAttached 检查:
data:image/s3,"s3://crabby-images/aeb06/aeb061358712298602487e860d4df0ef369d53c1" alt=""
但是可以通过输入参数里指定的 select,到 e2e 代码里搜索:
data:image/s3,"s3://crabby-images/a289f/a289f96165eb8423d36d5cf1b9dd5fa1a942a8cd" alt=""
cx-hamburger-menu [aria-label="Menu"]
通过参数说明,点击了 cx-hamburge 的 button 元素:
data:image/s3,"s3://crabby-images/6adbc/6adbc7e1a71aa4db2698909cc82c941865da7ac9" alt=""
很容易就找到了:
data:image/s3,"s3://crabby-images/f7df2/f7df22a7d18b85df99a062ca38a3b2cd27b350ca" alt=""
具体逻辑就是,检查元素的 ownerDocument 属性:
data:image/s3,"s3://crabby-images/d6662/d666207b1ae1da09298fc41bad179c9cc6c33e4c" alt=""
这是 Web API 里 Node 元素的标准属性,见官网。
接口的只读 ownerDocument 属性: 返回节点的顶级文档对象。
使用例子:
// Given a node "p", get the top-level HTML
// child of the document object
const d = p.ownerDocument;
const html = d.documentElement;
然后再判断这个 document 对象,是否有 activeWindow:
return nodes.every(node => {
const doc = _document__WEBPACK_IMPORTED_MODULE_1__[/* default */ "a"].getDocumentFromElement(node);
if (!_document__WEBPACK_IMPORTED_MODULE_1__[/* default */ "a"].hasActiveWindow(doc)) {
return false;
}
return node.isConnected;
});
};
这里 Cypress 还针对 firefox 进行了不同的处理:
data:image/s3,"s3://crabby-images/fad07/fad07b40e22d5b249488d61e74af0cc3f49e231e" alt=""
因为在 firefox 里,即使 detached document, 也还保留了一个指向 window 的 reference.
const hasActiveWindow = doc => {
// in firefox, detached documents still have a reference to their window
// but document.location is null
if (Cypress.isBrowser('firefox') && !doc.location) {
return false;
}
return !!doc.defaultView;
};
defaultView 是 Web API 另一个标准的接口。
In browsers, document.defaultView returns the window object associated with a document, or null if none is available.
这个也是只读属性,返回 document 关联的 window 对象。如果元素所在的 document 对象已经 detach from DOM,则 defaultView 指向 null.
node.isConnected:
data:image/s3,"s3://crabby-images/b779e/b779e2f7b2b10eb918803b676e06cd0739404ccd" alt=""
接口 的只读isConnected属性 返回一个布尔值,指示节点是否(直接或间接)连接到上下文对象,例如Document(普通 DOM 情况下的对象),或ShadowRoot 影子 DOM 情况下的对象。
更多Jerry的原创文章,尽在:"汪子熙":
data:image/s3,"s3://crabby-images/9d5ff/9d5fffea0e5ec258def5c77c56d04b5c06480366" alt=""