zoukankan      html  css  js  c++  java
  • web 自动化遇到 shadowDOM 节点你会操作吗?

    本文转载自: http://www.lemfix.com/topics/971

    近期有同学在做web自动化的时候,发现页面上有些元素,在selenium中无法通过xpath来定位,各种原因找了半天,都没找到解决方案,最后发现元素在一个叫做shadow-root的节点下面,如下所示:

     
    问题:shadow-root是什么?为什么下面的节点在selenium无法通过xapth来定位?

    接下来我们来先了解一下shawod-root到到底是什么!

    一、shadowDOM介绍

    ​ 上面所看到的shadow-root标签其实就是一个shadowDOM,那么什么是shadowDOM呢?

    它是前端的一种页面封装技术,您可以将shadow DOM视为“DOM中的DOM”(可以看成一个隐藏的DOM)。

    它是一个独立的DOM树,具有自己的元素和样式,与原始文档DOM完全隔离。

    ShadowDOM 必须附在一个HTML元素中,存放shadowDOM的元素,我们可以把它称为宿主元素。

    在HTML5中有很多的标签样式都是通过shadowDOM来实现的,比如:日期选择框,音频播放标签,视频播放标签都自带了样式;

    这边以音频播放标签audio为例:在html文件中写入一个audio标签,页面上显示出来的内容就会出现一个音频播放器,如下图:

    <audio src="file/123.mp3" controls="controls"></audio>
    
    页面效果:
     

    我们没有给aduio标签做任何的css样式设置,那么上面播放器样式是怎么实现的呢?

    答案就是浏览器在解析audio标签的时候,会自动在audio下添加一个shadowDOM(这个播放器的样式都是在这里设置的),而audio就是这个shadowDOM的宿主标签,shadowDOM中封装好了所有内容和样式,只要定义一个宿主标签就能显示shadowDOM中的所有内容。

    我们按F12打开浏览器的调试工具,点击audio标签,就能看到如下信息:

     

    注意:需要调试工具中勾选显示浏览器设置的shadowDOM节点,看能看到如上信息(默认看不到浏览器的shadowDOM),如下:

     

    通过上面的案例我们大致的了解了一下shadowDOM的作用,它其实就是浏览器提供的一种“封装”功能,提供了一种强大的技术去隐藏一些实现细节,前面列举的几个html5中的元素就是这样来做的。现在有部分项目的前端页面,开发人员也使用了这一技术来进行封装,当然自己封装的shadowDOM,在使用F12调试工具打开的时候,是可以清楚的的看到内部的节点和样式和代码的。

    ​ 这种封装对于前端开发来说虽好,但是我们测试人员在做web自动给的时候就会遇到一些问题,shadowDOM中的标签无法定位。ShadowDom 是游离在 DOM 树之外的节点树,但是他的创建基于普通 DOM 元素(并不文档的DOM树),所有没有办法基于整个文档的DOM树来直接进行操作。

    二、shawomDOM中的节点操作

    ​ 关于shawomDOM中的节点,selenium中并没有提供相关操作的方法,我们要操作的话就只能通过JS代码来实现,接下来给大家演示一个案例:

    需求:修改 shawom-root下span标签中的元素内容为 666
     

    实现步骤:

    1、先定位到shadow-root的宿主节点(此处为id=box的div)

    2、切换到shadow-root中

    3、然后再选择shadow-root下的span标签

    难点:selenuim中只能选择到宿主标签,无法选择到shadow-root

    解决思路:JS来实现

    1、js实现代码如下:

     

    2、selenium中通过js实现代码如下:

    1 import time
    2 from selenium import webdriver
    3 driver = webdriver.Chrome()
    4 driver.get(url="http:127.0.0.1:5000/test")
    5 js = 'document.getElementById("box").shadowRoot.children[0].children[0].innerText=666'
    6 res = driver.execute_script(js)
  • 相关阅读:
    Unity 5.3 Assetbundle热更资源
    自定义协同程序:CustomYieldInstruction
    C# 温故而知新: 线程篇(四)
    C# 温故而知新: 线程篇(三)
    C# 温故而知新: 线程篇(二)
    c# 温故而知新: 线程篇(一)
    C# 温故而知新:Stream篇(六)
    C# 温故而知新:Stream篇(七)
    C# 温故而知新:Stream篇(四)
    Redis高级数据类型
  • 原文地址:https://www.cnblogs.com/Simple-Small/p/12035425.html
Copyright © 2011-2022 走看看