chromium/third_party/blink/web_tests/external/wpt/webdriver/tests/bidi/script/call_function/remote_reference.py

import pytest
import webdriver.bidi.error as error
from webdriver.bidi.modules.script import ContextTarget, SerializationOptions

from ... import any_string, recursive_compare


@pytest.mark.asyncio
@pytest.mark.parametrize(
    "setup_expression, function_declaration, expected",
    [
        (
            "Symbol('foo')",
            "(symbol) => symbol.toString()",
            {"type": "string", "value": "Symbol(foo)"},
        ),
        ("[1,2]", "(array) => array[0]", {"type": "number", "value": 1}),
        (
            "new RegExp('foo')",
            "(regexp) => regexp.source",
            {"type": "string", "value": "foo"},
        ),
        (
            "new Date(1654004849000)",
            "(date) => date.toISOString()",
            {"type": "string", "value": "2022-05-31T13:47:29.000Z"},
        ),
        (
            "new Map([['foo', 'bar']])",
            "(map) => map.get('foo')",
            {"type": "string", "value": "bar"},
        ),
        (
            "new Set(['foo'])",
            "(set) => set.has('foo')",
            {"type": "boolean", "value": True},
        ),
        (
            "{const weakMap = new WeakMap(); weakMap.set(weakMap, 'foo')}",
            "(weakMap)=> weakMap.get(weakMap)",
            {"type": "string", "value": "foo"},
        ),
        (
            "{const weakSet = new WeakSet(); weakSet.add(weakSet)}",
            "(weakSet)=> weakSet.has(weakSet)",
            {"type": "boolean", "value": True},
        ),
        (
            "new Error('error message')",
            "(error) => error.message",
            {"type": "string", "value": "error message"},
        ),
        (
            "new SyntaxError('syntax error message')",
            "(error) => error.message",
            {"type": "string", "value": "syntax error message"},
        ),
        (
            "new Promise((resolve) => resolve(3))",
            "(promise) => promise",
            {"type": "number", "value": 3},
        ),
        (
            "new Int8Array(2)",
            "(int8Array) => int8Array.length",
            {"type": "number", "value": 2},
        ),
        (
            "new ArrayBuffer(8)",
            "(arrayBuffer) => arrayBuffer.byteLength",
            {"type": "number", "value": 8},
        ),
        ("() => true", "(func) => func()", {"type": "boolean", "value": True}),
        (
            "(function() {return false;})",
            "(func) => func()",
            {"type": "boolean", "value": False},
        ),
        (
            "window.foo = 3; window",
            "(window) => window.foo",
            {"type": "number", "value": 3},
        ),
        (
            "window.url = new URL('https://example.com'); window.url",
            "(url) => url.hostname",
            {"type": "string", "value": "example.com"},
        ),
        (
            "({SOME_PROPERTY:'SOME_VALUE'})",
            "(obj) => obj.SOME_PROPERTY",
            {"type": "string", "value": "SOME_VALUE"},
        ),
    ],
)
async def test_remote_reference_argument(
    bidi_session, top_context, setup_expression, function_declaration, expected
):
    remote_value_result = await bidi_session.script.evaluate(
        expression=setup_expression,
        await_promise=False,
        result_ownership="root",
        target=ContextTarget(top_context["context"]),
    )
    remote_value_handle = remote_value_result.get("handle")

    assert isinstance(remote_value_handle, str)

    result = await bidi_session.script.call_function(
        function_declaration=function_declaration,
        arguments=[{"handle": remote_value_handle}],
        await_promise=True if remote_value_result["type"] == "promise" else False,
        target=ContextTarget(top_context["context"]),
    )

    assert result == expected


@pytest.mark.asyncio
@pytest.mark.parametrize(
    "value_fn, function_declaration",
    [
        (
            lambda value: value,
            "function(arg) { return arg === window.SOME_OBJECT; }",
        ),
        (
            lambda value: ({"type": "object", "value": [["nested", value]]}),
            "function(arg) { return arg.nested === window.SOME_OBJECT; }",
        ),
        (
            lambda value: ({"type": "array", "value": [value]}),
            "function(arg) { return arg[0] === window.SOME_OBJECT; }",
        ),
        (
            lambda value: ({"type": "map", "value": [["foobar", value]]}),
            "function(arg) { return arg.get('foobar') === window.SOME_OBJECT; }",
        ),
        (
            lambda value: ({"type": "set", "value": [value]}),
            "function(arg) { return arg.has(window.SOME_OBJECT); }",
        ),
    ],
)
async def test_remote_reference_deserialization(
    bidi_session, top_context, call_function, evaluate, value_fn, function_declaration
):
    remote_value = await evaluate(
        "window.SOME_OBJECT = { SOME_PROPERTY: 'SOME_VALUE' }; window.SOME_OBJECT",
        result_ownership="root",
    )

    # Check that a remote value can be successfully deserialized as an "argument"
    # parameter and compared against the original object in the page.
    result = await call_function(
        function_declaration=function_declaration,
        arguments=[value_fn(remote_value)],
    )
    assert result == {"type": "boolean", "value": True}

    # Reload the page to cleanup the state
    await bidi_session.browsing_context.navigate(
        context=top_context["context"], url=top_context["url"], wait="complete"
    )


@pytest.mark.asyncio
@pytest.mark.parametrize(
    "setup_expression, expected_node_type",
    [
        ("document.querySelector('img')", 1),
        ("document.querySelector('input#button').attributes[0]", 2),
        ("document.querySelector('#with-text-node').childNodes[0]", 3),
        ("""document.createProcessingInstruction("xml-stylesheet", "href='foo.css'")""", 7),
        ("document.querySelector('#with-comment').childNodes[0]", 8),
        ("document", 9),
        ("document.doctype", 10),
        ("document.createDocumentFragment()", 11),
        ("document.querySelector('#custom-element').shadowRoot", 11),
    ],
    ids=[
        "element",
        "attribute",
        "text node",
        "processing instruction",
        "comment",
        "document",
        "doctype",
        "document fragment",
        "shadow root",
    ]
)
async def test_remote_reference_node_argument(
    bidi_session, get_test_page, top_context, setup_expression, expected_node_type
):
    await bidi_session.browsing_context.navigate(
        context=top_context['context'], url=get_test_page(), wait="complete"
    )

    remote_reference = await bidi_session.script.evaluate(
        expression=setup_expression,
        await_promise=False,
        target=ContextTarget(top_context["context"]),
    )

    result = await bidi_session.script.call_function(
        function_declaration="(node) => node.nodeType",
        arguments=[remote_reference],
        await_promise=False,
        target=ContextTarget(top_context["context"]),
    )

    assert result == {"type": "number", "value": expected_node_type}


@pytest.mark.asyncio
async def test_remote_reference_node_cdata(bidi_session, inline, top_context):
    xml_page = inline("""<foo>CDATA section: <![CDATA[ < > & ]]>.</foo>""", doctype="xml")

    await bidi_session.browsing_context.navigate(
        context=top_context['context'], url=xml_page, wait="complete"
    )

    remote_reference = await bidi_session.script.evaluate(
        expression="document.querySelector('foo').childNodes[1]",
        await_promise=False,
        target=ContextTarget(top_context["context"]),
    )

    result = await bidi_session.script.call_function(
        function_declaration="(node) => node.nodeType",
        arguments=[remote_reference],
        await_promise=False,
        target=ContextTarget(top_context["context"]),
    )

    assert result == {"type": "number", "value": 4}


@pytest.mark.asyncio
async def test_remote_reference_sharedId_precedence_over_handle(
    bidi_session, get_test_page, top_context
):
    await bidi_session.browsing_context.navigate(
        context=top_context['context'], url=get_test_page(), wait="complete"
    )

    remote_reference = await bidi_session.script.evaluate(
        expression="document.querySelector('img')",
        await_promise=False,
        result_ownership="root",
        target=ContextTarget(top_context["context"]),
    )

    assert "handle" in remote_reference
    # Invalidate shared reference to trigger a "no such node" error
    remote_reference["sharedId"] = "foo"

    with pytest.raises(error.NoSuchNodeException):
        await bidi_session.script.call_function(
            function_declaration="(node) => node.nodeType",
            arguments=[remote_reference],
            await_promise=False,
            target=ContextTarget(top_context["context"]),
        )


@pytest.mark.asyncio
@pytest.mark.parametrize(
    "expression, function_declaration, expected",
    [
        (
            "document.getElementsByTagName('span')",
            "(collection) => collection.item(0)",
            {
                "type": "node",
                "sharedId": any_string,
                "value": {
                    "attributes": {},
                    "childNodeCount": 0,
                    "children": [],
                    "localName": "span",
                    "namespaceURI": "http://www.w3.org/1999/xhtml",
                    "nodeType": 1
                }
            }
        ),
        (
            "document.querySelectorAll('span')",
            "(nodeList) => nodeList.item(0)",
            {
                "type": "node",
                "sharedId": any_string,
                "value": {
                    "attributes": {},
                    "childNodeCount": 0,
                    "children": [],
                    "localName": "span",
                    "namespaceURI": "http://www.w3.org/1999/xhtml",
                    "nodeType": 1
                }
            }
        ),
    ], ids=[
        "htmlcollection",
        "nodelist"
    ]
)
async def test_remote_reference_dom_collection(
    bidi_session,
    inline,
    top_context,
    call_function,
    expression,
    function_declaration,
    expected
):
    page_url = inline("""<p><span>""")
    await bidi_session.browsing_context.navigate(
        context=top_context['context'], url=page_url, wait="complete"
    )

    remote_value = await bidi_session.script.evaluate(
        expression=expression,
        result_ownership="root",
        target=ContextTarget(top_context["context"]),
        await_promise=False,
    )

    # Check that a remote value can be successfully deserialized as an "argument"
    # parameter and the first element be extracted.
    result = await call_function(
        function_declaration=function_declaration,
        arguments=[remote_value],
        serialization_options=SerializationOptions(max_dom_depth=1),
    )

    recursive_compare(expected, result)