Rewrite tests/browser/test_webelem.py for pytest.

This commit is contained in:
Florian Bruhin 2015-08-02 22:10:00 +02:00
parent d23096f898
commit 982a6eccfb

View File

@ -31,7 +31,7 @@ import pytest
from qutebrowser.browser import webelem from qutebrowser.browser import webelem
def get_webelem(geometry=None, frame=None, null=False, visibility='', def get_webelem(geometry=None, frame=None, null=False, style=None,
display='', attributes=None, tagname=None, classes=None): display='', attributes=None, tagname=None, classes=None):
"""Factory for WebElementWrapper objects based on a mock. """Factory for WebElementWrapper objects based on a mock.
@ -39,8 +39,7 @@ def get_webelem(geometry=None, frame=None, null=False, visibility='',
geometry: The geometry of the QWebElement as QRect. geometry: The geometry of the QWebElement as QRect.
frame: The QWebFrame the element is in. frame: The QWebFrame the element is in.
null: Whether the element is null or not. null: Whether the element is null or not.
visibility: The CSS visibility style property value. style: A dict with the styleAttributes of the element.
display: The CSS display style property value.
attributes: Boolean HTML attributes to be added. attributes: Boolean HTML attributes to be added.
tagname: The tag name. tagname: The tag name.
classes: HTML classes to be added. classes: HTML classes to be added.
@ -51,6 +50,7 @@ def get_webelem(geometry=None, frame=None, null=False, visibility='',
elem.webFrame.return_value = frame elem.webFrame.return_value = frame
elem.tagName.return_value = tagname elem.tagName.return_value = tagname
elem.toOuterXml.return_value = '<fakeelem/>' elem.toOuterXml.return_value = '<fakeelem/>'
if attributes is not None: if attributes is not None:
if not isinstance(attributes, collections.abc.Mapping): if not isinstance(attributes, collections.abc.Mapping):
attributes = {e: None for e in attributes} attributes = {e: None for e in attributes}
@ -61,23 +61,22 @@ def get_webelem(geometry=None, frame=None, null=False, visibility='',
elem.hasAttribute.return_value = False elem.hasAttribute.return_value = False
elem.attribute.return_value = '' elem.attribute.return_value = ''
elem.attributeNames.return_value = [] elem.attributeNames.return_value = []
if classes is not None: if classes is not None:
elem.classes.return_value = classes.split(' ') elem.classes.return_value = classes.split(' ')
else: else:
elem.classes.return_value = [] elem.classes.return_value = []
style_dict = {'visibility': '', 'display': ''}
if style is not None:
style_dict.update(style)
def _style_property(name, strategy): def _style_property(name, strategy):
"""Helper function to act as styleProperty method.""" """Helper function to act as styleProperty method."""
if strategy != QWebElement.ComputedStyle: if strategy != QWebElement.ComputedStyle:
raise ValueError("styleProperty called with strategy != " raise ValueError("styleProperty called with strategy != "
"ComputedStyle ({})!".format(strategy)) "ComputedStyle ({})!".format(strategy))
if name == 'visibility': return style_dict[name]
return visibility
elif name == 'display':
return display
else:
raise ValueError("styleProperty called with unknown name "
"'{}'".format(name))
elem.styleProperty.side_effect = _style_property elem.styleProperty.side_effect = _style_property
wrapped = webelem.WebElementWrapper(elem) wrapped = webelem.WebElementWrapper(elem)
@ -96,19 +95,13 @@ class TestWebElementWrapper:
get_webelem(null=True) get_webelem(null=True)
class TestIsVisibleInvalid: class TestIsVisible:
"""Tests for is_visible with invalid elements. @pytest.fixture
def frame(self, stubs):
return stubs.FakeWebFrame(QRect(0, 0, 100, 100))
Attributes: def test_nullelem(self, frame):
frame: The FakeWebFrame we're using to test.
"""
@pytest.fixture(autouse=True)
def setup(self, stubs):
self.frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100))
def test_nullelem(self):
"""Passing an element with isNull() == True. """Passing an element with isNull() == True.
geometry() and webFrame() should not be called, and ValueError should geometry() and webFrame() should not be called, and ValueError should
@ -117,83 +110,46 @@ class TestIsVisibleInvalid:
elem = get_webelem() elem = get_webelem()
elem._elem.isNull.return_value = True elem._elem.isNull.return_value = True
with pytest.raises(webelem.IsNullError): with pytest.raises(webelem.IsNullError):
elem.is_visible(self.frame) elem.is_visible(frame)
def test_invalid_invisible(self): def test_invalid_invisible(self, frame):
"""Test elements with an invalid geometry which are invisible.""" """Test elements with an invalid geometry which are invisible."""
elem = get_webelem(QRect(0, 0, 0, 0), self.frame) elem = get_webelem(QRect(0, 0, 0, 0), frame)
assert not elem.geometry().isValid() assert not elem.geometry().isValid()
assert elem.geometry().x() == 0 assert elem.geometry().x() == 0
assert not elem.is_visible(self.frame) assert not elem.is_visible(frame)
def test_invalid_visible(self): def test_invalid_visible(self, frame):
"""Test elements with an invalid geometry which are visible. """Test elements with an invalid geometry which are visible.
This seems to happen sometimes in the real world, with real elements This seems to happen sometimes in the real world, with real elements
which *are* visible, but don't have a valid geometry. which *are* visible, but don't have a valid geometry.
""" """
elem = get_webelem(QRect(10, 10, 0, 0), self.frame) elem = get_webelem(QRect(10, 10, 0, 0), frame)
assert not elem.geometry().isValid() assert not elem.geometry().isValid()
assert elem.is_visible(self.frame) assert elem.is_visible(frame)
@pytest.mark.parametrize('geometry, visible', [
class TestIsVisibleScroll: (QRect(5, 5, 4, 4), False),
(QRect(10, 10, 1, 1), True),
"""Tests for is_visible when the frame is scrolled. ])
def test_scrolled(self, geometry, visible, stubs):
Attributes: scrolled_frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100),
frame: The FakeWebFrame we're using to test.
"""
@pytest.fixture(autouse=True)
def setup(self, stubs):
self.frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100),
scroll=QPoint(10, 10)) scroll=QPoint(10, 10))
elem = get_webelem(geometry, scrolled_frame)
assert elem.is_visible(scrolled_frame) == visible
def test_invisible(self): @pytest.mark.parametrize('style, visible', [
"""Test elements which should be invisible due to scrolling.""" ({'visibility': 'visible'}, True),
elem = get_webelem(QRect(5, 5, 4, 4), self.frame) ({'visibility': 'hidden'}, False),
assert not elem.is_visible(self.frame) ({'display': 'inline'}, True),
({'display': 'none'}, False),
def test_visible(self): ({'visibility': 'visible', 'display': 'none'}, False),
"""Test elements which still should be visible after scrolling.""" ({'visibility': 'hidden', 'display': 'inline'}, False),
elem = get_webelem(QRect(10, 10, 1, 1), self.frame) ])
assert elem.is_visible(self.frame) def test_css_attributes(self, frame, style, visible):
elem = get_webelem(QRect(0, 0, 10, 10), frame, style=style)
assert elem.is_visible(frame) == visible
class TestIsVisibleCss:
"""Tests for is_visible with CSS attributes.
Attributes:
frame: The FakeWebFrame we're using to test.
"""
@pytest.fixture(autouse=True)
def setup(self, stubs):
self.frame = stubs.FakeWebFrame(QRect(0, 0, 100, 100))
def test_visibility_visible(self):
"""Check that elements with "visibility = visible" are visible."""
elem = get_webelem(QRect(0, 0, 10, 10), self.frame,
visibility='visible')
assert elem.is_visible(self.frame)
def test_visibility_hidden(self):
"""Check that elements with "visibility = hidden" are not visible."""
elem = get_webelem(QRect(0, 0, 10, 10), self.frame,
visibility='hidden')
assert not elem.is_visible(self.frame)
def test_display_inline(self):
"""Check that elements with "display = inline" are visible."""
elem = get_webelem(QRect(0, 0, 10, 10), self.frame, display='inline')
assert elem.is_visible(self.frame)
def test_display_none(self):
"""Check that elements with "display = none" are not visible."""
elem = get_webelem(QRect(0, 0, 10, 10), self.frame, display='none')
assert not elem.is_visible(self.frame)
class TestIsVisibleIframe: class TestIsVisibleIframe:
@ -206,8 +162,10 @@ class TestIsVisibleIframe:
elem1-elem4: FakeWebElements to test. elem1-elem4: FakeWebElements to test.
""" """
@pytest.fixture(autouse=True) Objects = collections.namedtuple('Objects', ['frame', 'iframe', 'elems'])
def setup(self, stubs):
@pytest.fixture
def objects(self, stubs):
"""Set up the following base situation. """Set up the following base situation.
0, 0 300, 0 0, 0 300, 0
@ -215,102 +173,90 @@ class TestIsVisibleIframe:
# # # #
0,10 # iframe 100,10 # 0,10 # iframe 100,10 #
#********** # #********** #
#*e * elem1: 0, 0 in iframe (visible) #*e * elems[0]: 0, 0 in iframe (visible)
#* * # #* * #
#* e * elem2: 20,90 in iframe (visible) #* e * elems[1]: 20,90 in iframe (visible)
#********** # #********** #
0,110 #. .100,110 # 0,110 #. .100,110 #
#. . # #. . #
#. e . elem3: 20,150 in iframe (not visible) #. e . elems[2]: 20,150 in iframe (not visible)
#.......... # #.......... #
# e elem4: 30, 180 in main frame (visible) # e elems[3]: 30, 180 in main frame (visible)
# # # #
# frame # # frame #
############################## ##############################
300, 0 300, 300 300, 0 300, 300
Returns an Objects namedtuple with frame/iframe/elems attributes.
""" """
self.frame = stubs.FakeWebFrame(QRect(0, 0, 300, 300)) frame = stubs.FakeWebFrame(QRect(0, 0, 300, 300))
self.iframe = stubs.FakeWebFrame(QRect(0, 10, 100, 100), iframe = stubs.FakeWebFrame(QRect(0, 10, 100, 100), parent=frame)
parent=self.frame) assert frame.geometry().contains(iframe.geometry())
self.elem1 = get_webelem(QRect(0, 0, 10, 10), self.iframe) elems = [
self.elem2 = get_webelem(QRect(20, 90, 10, 10), self.iframe) get_webelem(QRect(0, 0, 10, 10), iframe),
self.elem3 = get_webelem(QRect(20, 150, 10, 10), self.iframe) get_webelem(QRect(20, 90, 10, 10), iframe),
self.elem4 = get_webelem(QRect(30, 180, 10, 10), self.frame) get_webelem(QRect(20, 150, 10, 10), iframe),
get_webelem(QRect(30, 180, 10, 10), frame),
]
def test_not_scrolled(self): assert elems[0].is_visible(frame)
"""Test base situation.""" assert elems[1].is_visible(frame)
assert self.frame.geometry().contains(self.iframe.geometry()) assert not elems[2].is_visible(frame)
assert self.elem1.is_visible(self.frame) assert elems[3].is_visible(frame)
assert self.elem2.is_visible(self.frame)
assert not self.elem3.is_visible(self.frame)
assert self.elem4.is_visible(self.frame)
def test_iframe_scrolled(self): return self.Objects(frame=frame, iframe=iframe, elems=elems)
def test_iframe_scrolled(self, objects):
"""Scroll iframe down so elem3 gets visible and elem1/elem2 not.""" """Scroll iframe down so elem3 gets visible and elem1/elem2 not."""
self.iframe.scrollPosition.return_value = QPoint(0, 100) objects.iframe.scrollPosition.return_value = QPoint(0, 100)
assert not self.elem1.is_visible(self.frame) assert not objects.elems[0].is_visible(objects.frame)
assert not self.elem2.is_visible(self.frame) assert not objects.elems[1].is_visible(objects.frame)
assert self.elem3.is_visible(self.frame) assert objects.elems[2].is_visible(objects.frame)
assert self.elem4.is_visible(self.frame) assert objects.elems[3].is_visible(objects.frame)
def test_mainframe_scrolled_iframe_visible(self): def test_mainframe_scrolled_iframe_visible(self, objects):
"""Scroll mainframe down so iframe is partly visible but elem1 not.""" """Scroll mainframe down so iframe is partly visible but elem1 not."""
self.frame.scrollPosition.return_value = QPoint(0, 50) objects.frame.scrollPosition.return_value = QPoint(0, 50)
geom = self.frame.geometry().translated(self.frame.scrollPosition()) geom = objects.frame.geometry().translated(
assert not geom.contains(self.iframe.geometry()) objects.frame.scrollPosition())
assert geom.intersects(self.iframe.geometry()) assert not geom.contains(objects.iframe.geometry())
assert not self.elem1.is_visible(self.frame) assert geom.intersects(objects.iframe.geometry())
assert self.elem2.is_visible(self.frame) assert not objects.elems[0].is_visible(objects.frame)
assert not self.elem3.is_visible(self.frame) assert objects.elems[1].is_visible(objects.frame)
assert self.elem4.is_visible(self.frame) assert not objects.elems[2].is_visible(objects.frame)
assert objects.elems[3].is_visible(objects.frame)
def test_mainframe_scrolled_iframe_invisible(self): def test_mainframe_scrolled_iframe_invisible(self, objects):
"""Scroll mainframe down so iframe is invisible.""" """Scroll mainframe down so iframe is invisible."""
self.frame.scrollPosition.return_value = QPoint(0, 110) objects.frame.scrollPosition.return_value = QPoint(0, 110)
geom = self.frame.geometry().translated(self.frame.scrollPosition()) geom = objects.frame.geometry().translated(
assert not geom.contains(self.iframe.geometry()) objects.frame.scrollPosition())
assert not geom.intersects(self.iframe.geometry()) assert not geom.contains(objects.iframe.geometry())
assert not self.elem1.is_visible(self.frame) assert not geom.intersects(objects.iframe.geometry())
assert not self.elem2.is_visible(self.frame) assert not objects.elems[0].is_visible(objects.frame)
assert not self.elem3.is_visible(self.frame) assert not objects.elems[1].is_visible(objects.frame)
assert self.elem4.is_visible(self.frame) assert not objects.elems[2].is_visible(objects.frame)
assert objects.elems[3].is_visible(objects.frame)
class TestIsWritable: @pytest.mark.parametrize('attributes, writable', [
([], True),
"""Check is_writable.""" (['disabled'], False),
(['readonly'], False),
def test_writable(self): (['disabled', 'readonly'], False),
"""Test a normal element.""" ])
elem = get_webelem() def test_is_writable(attributes, writable):
assert elem.is_writable() elem = get_webelem(attributes=attributes)
assert elem.is_writable() == writable
def test_disabled(self):
"""Test a disabled element."""
elem = get_webelem(attributes=['disabled'])
assert not elem.is_writable()
def test_readonly(self):
"""Test a readonly element."""
elem = get_webelem(attributes=['readonly'])
assert not elem.is_writable()
class TestJavascriptEscape: @pytest.mark.parametrize('before, after', [
"""Check javascript_escape.
Class attributes:
STRINGS: A list of (input, output) tuples.
"""
@pytest.mark.parametrize('before, after', [
('foo\\bar', r'foo\\bar'), ('foo\\bar', r'foo\\bar'),
('foo\nbar', r'foo\nbar'), ('foo\nbar', r'foo\nbar'),
("foo'bar", r"foo\'bar"), ("foo'bar", r"foo\'bar"),
('foo"bar', r'foo\"bar'), ('foo"bar', r'foo\"bar'),
]) ])
def test_fake_escape(self, before, after): def test_fake_escape(before, after):
"""Test javascript escaping.""" """Test javascript escaping."""
assert webelem.javascript_escape(before) == after assert webelem.javascript_escape(before) == after
@ -370,13 +316,6 @@ class TestIsEditable:
"""Tests for is_editable.""" """Tests for is_editable."""
@pytest.yield_fixture(autouse=True)
def setup(self):
old_config = webelem.config
webelem.config = None
yield
webelem.config = old_config
@pytest.fixture @pytest.fixture
def stubbed_config(self, config_stub, monkeypatch): def stubbed_config(self, config_stub, monkeypatch):
"""Fixture to create a config stub with an input section.""" """Fixture to create a config stub with an input section."""
@ -384,177 +323,56 @@ class TestIsEditable:
monkeypatch.setattr('qutebrowser.browser.webelem.config', config_stub) monkeypatch.setattr('qutebrowser.browser.webelem.config', config_stub)
return config_stub return config_stub
def test_input_plain(self): @pytest.mark.parametrize('tagname, attributes, editable', [
"""Test with plain input element.""" ('input', {}, True),
elem = get_webelem(tagname='input') ('input', {'type': 'text'}, True),
assert elem.is_editable() ('INPUT', {'TYPE': 'TEXT'}, True), # caps attributes/name
('input', {'type': 'email'}, True),
('input', {'type': 'url'}, True),
('input', {'type': 'tel'}, True),
('input', {'type': 'number'}, True),
('input', {'type': 'password'}, True),
('input', {'type': 'search'}, True),
('textarea', {}, True),
def test_input_text(self): ('input', {'type': 'button'}, False),
"""Test with text input element.""" ('input', {'type': 'checkbox'}, False),
elem = get_webelem(tagname='input', attributes={'type': 'text'}) ('select', {}, False),
assert elem.is_editable()
def test_input_text_caps(self): ('input', {'disabled': None}, False),
"""Test with text input element with caps attributes.""" ('input', {'readonly': None}, False),
elem = get_webelem(tagname='INPUT', attributes={'TYPE': 'TEXT'}) ('textarea', {'disabled': None}, False),
assert elem.is_editable() ('textarea', {'readonly': None}, False),
('object', {}, False),
('object', {'type': 'image/gif'}, False),
])
def test_is_editable(self, tagname, attributes, editable):
elem = get_webelem(tagname=tagname, attributes=attributes)
assert elem.is_editable() == editable
def test_input_email(self): @pytest.mark.parametrize('classes, editable', [
"""Test with email input element.""" (None, False),
elem = get_webelem(tagname='input', attributes={'type': 'email'}) ('foo-kix-bar', False),
assert elem.is_editable() ('foo kix-foo', True),
('KIX-FOO', False),
('foo CodeMirror-foo', True),
])
def test_is_editable_div(self, tagname, classes, editable):
elem = get_webelem(tagname='div', classes=classes)
assert elem.is_editable() == editable
def test_input_url(self): @pytest.mark.parametrize('setting, tagname, attributes, editable', [
"""Test with url input element.""" (True, 'embed', {}, True),
elem = get_webelem(tagname='input', attributes={'type': 'url'}) (True, 'embed', {}, True),
assert elem.is_editable() (False, 'applet', {}, False),
(False, 'applet', {}, False),
def test_input_tel(self): (True, 'object', {'type': 'application/foo'}, True),
"""Test with tel input element.""" (False, 'object', {'type': 'application/foo'}, False),
elem = get_webelem(tagname='input', attributes={'type': 'tel'}) (True, 'object', {'type': 'foo', 'classid': 'foo'}, True),
assert elem.is_editable() (False, 'object', {'type': 'foo', 'classid': 'foo'}, False),
])
def test_input_number(self): def test_is_editable_div(self, stubbed_config, setting, tagname,
"""Test with number input element.""" attributes, editable):
elem = get_webelem(tagname='input', attributes={'type': 'number'}) stubbed_config.data['input']['insert-mode-on-plugins'] = setting
assert elem.is_editable() elem = get_webelem(tagname=tagname, attributes=attributes)
assert elem.is_editable() == editable
def test_input_password(self):
"""Test with password input element."""
elem = get_webelem(tagname='input', attributes={'type': 'password'})
assert elem.is_editable()
def test_input_search(self):
"""Test with search input element."""
elem = get_webelem(tagname='input', attributes={'type': 'search'})
assert elem.is_editable()
def test_input_button(self):
"""Button should not be editable."""
elem = get_webelem(tagname='input', attributes={'type': 'button'})
assert not elem.is_editable()
def test_input_checkbox(self):
"""Checkbox should not be editable."""
elem = get_webelem(tagname='input', attributes={'type': 'checkbox'})
assert not elem.is_editable()
def test_textarea(self):
"""Test textarea element."""
elem = get_webelem(tagname='textarea')
assert elem.is_editable()
def test_select(self):
"""Test selectbox."""
elem = get_webelem(tagname='select')
assert not elem.is_editable()
def test_input_disabled(self):
"""Test disabled input element."""
elem = get_webelem(tagname='input', attributes={'disabled': None})
assert not elem.is_editable()
def test_input_readonly(self):
"""Test readonly input element."""
elem = get_webelem(tagname='input', attributes={'readonly': None})
assert not elem.is_editable()
def test_textarea_disabled(self):
"""Test disabled textarea element."""
elem = get_webelem(tagname='textarea', attributes={'disabled': None})
assert not elem.is_editable()
def test_textarea_readonly(self):
"""Test readonly textarea element."""
elem = get_webelem(tagname='textarea', attributes={'readonly': None})
assert not elem.is_editable()
def test_embed_true(self, stubbed_config):
"""Test embed-element with insert-mode-on-plugins true."""
stubbed_config.data['input']['insert-mode-on-plugins'] = True
elem = get_webelem(tagname='embed')
assert elem.is_editable()
def test_applet_true(self, stubbed_config):
"""Test applet-element with insert-mode-on-plugins true."""
stubbed_config.data['input']['insert-mode-on-plugins'] = True
elem = get_webelem(tagname='applet')
assert elem.is_editable()
def test_embed_false(self, stubbed_config):
"""Test embed-element with insert-mode-on-plugins false."""
stubbed_config.data['input']['insert-mode-on-plugins'] = False
elem = get_webelem(tagname='embed')
assert not elem.is_editable()
def test_applet_false(self, stubbed_config):
"""Test applet-element with insert-mode-on-plugins false."""
stubbed_config.data['input']['insert-mode-on-plugins'] = False
elem = get_webelem(tagname='applet')
assert not elem.is_editable()
def test_object_no_type(self):
"""Test object-element without type."""
elem = get_webelem(tagname='object')
assert not elem.is_editable()
def test_object_image(self):
"""Test object-element with image type."""
elem = get_webelem(tagname='object', attributes={'type': 'image/gif'})
assert not elem.is_editable()
def test_object_application(self, stubbed_config):
"""Test object-element with application type."""
stubbed_config.data['input']['insert-mode-on-plugins'] = True
elem = get_webelem(tagname='object',
attributes={'type': 'application/foo'})
assert elem.is_editable()
def test_object_application_false(self, stubbed_config):
"""Test object-element with application type but not ...-on-plugins."""
stubbed_config.data['input']['insert-mode-on-plugins'] = False
elem = get_webelem(tagname='object',
attributes={'type': 'application/foo'})
assert not elem.is_editable()
def test_object_classid(self, stubbed_config):
"""Test object-element with classid."""
stubbed_config.data['input']['insert-mode-on-plugins'] = True
elem = get_webelem(tagname='object',
attributes={'type': 'foo', 'classid': 'foo'})
assert elem.is_editable()
def test_object_classid_false(self, stubbed_config):
"""Test object-element with classid but not insert-mode-on-plugins."""
stubbed_config.data['input']['insert-mode-on-plugins'] = False
elem = get_webelem(tagname='object',
attributes={'type': 'foo', 'classid': 'foo'})
assert not elem.is_editable()
def test_div_empty(self):
"""Test div-element without class."""
elem = get_webelem(tagname='div')
assert not elem.is_editable()
def test_div_noneditable(self):
"""Test div-element with non-editable class."""
elem = get_webelem(tagname='div', classes='foo-kix-bar')
assert not elem.is_editable()
def test_div_xik(self):
"""Test div-element with xik class."""
elem = get_webelem(tagname='div', classes='foo kix-foo')
assert elem.is_editable()
def test_div_xik_caps(self):
"""Test div-element with xik class in caps.
This tests if classes are case sensitive as they should.
"""
elem = get_webelem(tagname='div', classes='KIX-FOO')
assert not elem.is_editable()
def test_div_codemirror(self):
"""Test div-element with codemirror class."""
elem = get_webelem(tagname='div', classes='foo CodeMirror-foo')
assert elem.is_editable()